From 803fb21aa34454a1e1d8a7a5a83e87f975c33011 Mon Sep 17 00:00:00 2001 From: CJ_Clippy Date: Sat, 11 Jan 2025 04:47:23 -0800 Subject: [PATCH] burn it all down --- .dockerignore | 2 + .gitignore | 1 + Makefile | 61 - README.md | 17 +- Tiltfile | 646 - certs/letsencrypt-stg-root-x1.pem | 32 - certs/letsencrypt-stg-root-x2.pem | 15 - charts/README.md | 49 - charts/cert-manager/cert-manager/Chart.yaml | 26 - charts/cert-manager/cert-manager/README.md | 1884 -- .../cert-manager/templates/NOTES.txt | 18 - .../cert-manager/templates/_helpers.tpl | 202 - .../templates/cainjector-config.yaml | 18 - .../templates/cainjector-deployment.yaml | 140 - .../cainjector-poddisruptionbudget.yaml | 29 - .../templates/cainjector-psp-clusterrole.yaml | 20 - .../cainjector-psp-clusterrolebinding.yaml | 22 - .../templates/cainjector-psp.yaml | 51 - .../templates/cainjector-rbac.yaml | 103 - .../templates/cainjector-serviceaccount.yaml | 27 - .../templates/controller-config.yaml | 18 - .../cert-manager/templates/crds.yaml | 8179 ------ .../cert-manager/templates/deployment.yaml | 231 - .../templates/extras-objects.yaml | 4 - .../templates/networkpolicy-egress.yaml | 19 - .../templates/networkpolicy-webhooks.yaml | 21 - .../templates/poddisruptionbudget.yaml | 29 - .../cert-manager/templates/podmonitor.yaml | 50 - .../templates/psp-clusterrole.yaml | 18 - .../templates/psp-clusterrolebinding.yaml | 20 - .../cert-manager/templates/psp.yaml | 49 - .../cert-manager/templates/rbac.yaml | 574 - .../cert-manager/templates/service.yaml | 37 - .../templates/serviceaccount.yaml | 25 - .../templates/servicemonitor.yaml | 50 - .../templates/startupapicheck-job.yaml | 87 - .../startupapicheck-psp-clusterrole.yaml | 24 - ...tartupapicheck-psp-clusterrolebinding.yaml | 26 - .../templates/startupapicheck-psp.yaml | 51 - .../templates/startupapicheck-rbac.yaml | 48 - .../startupapicheck-serviceaccount.yaml | 27 - .../templates/webhook-config.yaml | 18 - .../templates/webhook-deployment.yaml | 192 - .../templates/webhook-mutating-webhook.yaml | 48 - .../webhook-poddisruptionbudget.yaml | 29 - .../templates/webhook-psp-clusterrole.yaml | 18 - .../webhook-psp-clusterrolebinding.yaml | 20 - .../cert-manager/templates/webhook-psp.yaml | 54 - .../cert-manager/templates/webhook-rbac.yaml | 83 - .../templates/webhook-service.yaml | 38 - .../templates/webhook-serviceaccount.yaml | 25 - .../templates/webhook-validating-webhook.yaml | 49 - charts/cert-manager/cert-manager/values.yaml | 1349 - charts/cert-manager/values-overrides.yaml | 3 - .../chisel-operator/.helmignore | 23 - .../chisel-operator/Chart.yaml | 6 - .../chisel-operator/templates/_helpers.tpl | 62 - .../templates/crds/exit-node-provisioner.yaml | 101 - .../templates/crds/exit-node.yaml | 84 - .../chisel-operator/templates/deployment.yaml | 49 - .../templates/serviceaccount.yaml | 58 - .../chisel-operator/values.yaml | 51 - charts/chisel-operator/values-overrides.yaml | 1 - charts/drupal/README.md | 3 - charts/drupal/drupal/.helmignore | 25 - charts/drupal/drupal/Chart.lock | 9 - charts/drupal/drupal/Chart.yaml | 42 - charts/drupal/drupal/README.md | 586 - charts/drupal/drupal/templates/NOTES.txt | 88 - charts/drupal/drupal/templates/_helpers.tpl | 144 - .../drupal/drupal/templates/deployment.yaml | 384 - .../drupal/templates/externaldb-secrets.yaml | 19 - .../drupal/drupal/templates/extra-list.yaml | 9 - charts/drupal/drupal/templates/ingress.yaml | 66 - .../drupal/drupal/templates/metrics-svc.yaml | 39 - .../drupal/templates/networkpolicy.yaml | 80 - charts/drupal/drupal/templates/pdb.yaml | 26 - .../drupal/templates/prometheusrule.yaml | 23 - charts/drupal/drupal/templates/pv.yaml | 23 - charts/drupal/drupal/templates/pvc.yaml | 33 - charts/drupal/drupal/templates/secrets.yaml | 26 - .../drupal/templates/serviceaccount.yaml | 18 - .../drupal/templates/servicemonitor.yaml | 43 - charts/drupal/drupal/templates/svc.yaml | 57 - .../drupal/drupal/templates/tls-secrets.yaml | 22 - charts/drupal/drupal/values.schema.json | 233 - charts/drupal/drupal/values.yaml | 982 - charts/drupal/values-overrides.yaml | 15 - .../external-secrets/Chart.lock | 6 - .../external-secrets/Chart.yaml | 20 - .../external-secrets/README.md | 225 - .../external-secrets/templates/NOTES.txt | 7 - .../external-secrets/templates/_helpers.tpl | 198 - .../templates/cert-controller-deployment.yaml | 124 - .../cert-controller-poddisruptionbudget.yaml | 19 - .../templates/cert-controller-rbac.yaml | 86 - .../templates/cert-controller-service.yaml | 28 - .../cert-controller-serviceaccount.yaml | 16 - .../templates/crds/acraccesstoken.yaml | 203 - .../templates/crds/clusterexternalsecret.yaml | 666 - .../templates/crds/clustersecretstore.yaml | 4601 ---- .../templates/crds/ecrauthorizationtoken.yaml | 177 - .../templates/crds/externalsecret.yaml | 820 - .../external-secrets/templates/crds/fake.yaml | 86 - .../templates/crds/gcraccesstoken.yaml | 138 - .../templates/crds/githubaccesstoken.yaml | 112 - .../templates/crds/password.yaml | 108 - .../templates/crds/pushsecret.yaml | 386 - .../templates/crds/secretstore.yaml | 4601 ---- .../templates/crds/vaultdynamicsecret.yaml | 707 - .../templates/crds/webhook.yaml | 157 - .../templates/deployment.yaml | 146 - .../templates/extra-manifests.yaml | 4 - .../templates/poddisruptionbudget.yaml | 19 - .../external-secrets/templates/rbac.yaml | 301 - .../external-secrets/templates/service.yaml | 28 - .../templates/serviceaccount.yaml | 16 - .../templates/servicemonitor.yaml | 164 - .../templates/validatingwebhook.yaml | 78 - .../templates/webhook-certificate.yaml | 30 - .../templates/webhook-deployment.yaml | 128 - .../webhook-poddisruptionbudget.yaml | 20 - .../templates/webhook-secret.yaml | 14 - .../templates/webhook-service.yaml | 37 - .../templates/webhook-serviceaccount.yaml | 16 - .../external-secrets/values.yaml | 532 - charts/external-secrets/values.yaml | 5 - charts/fp/Chart.yaml | 18 - charts/fp/README.md | 5 - charts/fp/templates-staging/README.md | 3 - charts/fp/templates-staging/bot.yaml | 69 - charts/fp/templates-staging/capture.yaml | 108 - charts/fp/templates-staging/chihaya.yaml | 39 - charts/fp/templates-staging/factory.yaml | 87 - charts/fp/templates-staging/mailbox.yaml | 59 - .../fp/templates-staging/migrations-data.yaml | 21 - .../templates-staging/migrations-schema.yaml | 21 - charts/fp/templates-staging/next.yaml | 84 - charts/fp/templates-staging/postgres.yaml | 71 - charts/fp/templates-staging/postgresql.yaml | 53 - charts/fp/templates-staging/postgrest.yaml | 81 - charts/fp/templates-staging/redis.yaml | 48 - charts/fp/templates-staging/scout.yaml | 67 - charts/fp/templates-staging/supertokens.yaml | 141 - charts/fp/templates-staging/trigger.yaml | 110 - charts/fp/templates/.gitignore | 1 - charts/fp/templates/bright.yaml | 87 - charts/fp/templates/cert-manager.yaml | 101 - charts/fp/templates/chisel.yaml | 14 - charts/fp/templates/exoscale-webhook.yaml | 321 - charts/fp/templates/external-dns.yaml | 87 - charts/fp/templates/gateway.yaml | 12 - charts/fp/templates/pgadmin4.yaml | 55 - charts/fp/templates/roles.yaml | 28 - charts/fp/templates/superstreamer.yaml | 339 - charts/fp/templates/traefik.yaml | 115 - charts/fp/templates/whoami.yaml | 70 - charts/fp/values.yaml | 138 - .../kubernetes-ingress-controller/.helmignore | 25 - .../CHANGELOG.md | 156 - .../kubernetes-ingress-controller/Chart.lock | 6 - .../kubernetes-ingress-controller/Chart.yaml | 21 - .../kubernetes-ingress-controller/README.md | 93 - .../templates/NOTES.txt | 53 - .../templates/_helpers.tpl | 87 - .../templates/controller-cm.yaml | 16 - .../templates/controller-deployment.yaml | 143 - .../templates/controller-pdb.yaml | 26 - .../templates/controller-rbac.yaml | 96 - .../templates/controller-serviceaccount.yaml | 15 - .../crds/ingress.k8s.ngrok.com_domains.yaml | 105 - .../ingress.k8s.ngrok.com_httpsedges.yaml | 1062 - .../ingress.k8s.ngrok.com_ippolicies.yaml | 109 - ...ingress.k8s.ngrok.com_ngrokmodulesets.yaml | 1008 - .../crds/ingress.k8s.ngrok.com_tcpedges.yaml | 132 - .../crds/ingress.k8s.ngrok.com_tlsedges.yaml | 166 - .../crds/ingress.k8s.ngrok.com_tunnels.yaml | 78 - ...ok.k8s.ngrok.com_ngroktrafficpolicies.yaml | 62 - .../templates/credentials-secret.yaml | 11 - .../templates/ingress-class.yaml | 15 - .../templates/rbac/domain_editor_role.yaml | 27 - .../templates/rbac/domain_viewer_role.yaml | 23 - .../templates/rbac/httpsedge_editor_role.yaml | 31 - .../templates/rbac/httpsedge_viewer_role.yaml | 27 - .../templates/rbac/ippolicy_editor_role.yaml | 31 - .../templates/rbac/ippolicy_viewer_role.yaml | 27 - .../rbac/ngrokmoduleset_editor_role.yaml | 31 - .../rbac/ngrokmoduleset_viewer_role.yaml | 27 - .../templates/rbac/role.yaml | 330 - .../templates/rbac/tcpedge_editor_role.yaml | 31 - .../templates/rbac/tcpedge_viewer_role.yaml | 27 - .../templates/rbac/tlsedge_editor_role.yaml | 31 - .../templates/rbac/tlsedge_viewer_role.yaml | 27 - .../templates/rbac/tunnel_editor_role.yaml | 27 - .../templates/rbac/tunnel_viewer_role.yaml | 23 - .../kubernetes-ingress-controller/values.yaml | 202 - .../values-overrides.yaml | 7 - charts/mariadb/mariadb/.helmignore | 25 - charts/mariadb/mariadb/Chart.lock | 6 - charts/mariadb/mariadb/Chart.yaml | 36 - charts/mariadb/mariadb/README.md | 650 - charts/mariadb/mariadb/templates/NOTES.txt | 77 - charts/mariadb/mariadb/templates/_helpers.tpl | 164 - .../mariadb/mariadb/templates/extra-list.yaml | 9 - .../mariadb/templates/networkpolicy.yaml | 76 - .../mariadb/templates/primary/configmap.yaml | 20 - .../primary/initialization-configmap.yaml | 19 - .../mariadb/templates/primary/pdb.yaml | 28 - .../templates/primary/statefulset.yaml | 443 - .../mariadb/templates/primary/svc.yaml | 62 - .../mariadb/templates/prometheusrules.yaml | 24 - charts/mariadb/mariadb/templates/role.yaml | 23 - .../mariadb/templates/rolebinding.yaml | 23 - .../templates/secondary/configmap.yaml | 20 - .../mariadb/templates/secondary/pdb.yaml | 28 - .../templates/secondary/statefulset.yaml | 414 - .../mariadb/templates/secondary/svc.yaml | 64 - charts/mariadb/mariadb/templates/secrets.yaml | 98 - .../mariadb/templates/serviceaccount.yaml | 18 - .../mariadb/templates/servicemonitor.yaml | 44 - charts/mariadb/mariadb/values.schema.json | 176 - charts/mariadb/mariadb/values.yaml | 1373 - charts/mariadb/values-overrides.yaml | 8 - charts/phpmyadmin/phpmyadmin/.helmignore | 25 - charts/phpmyadmin/phpmyadmin/Chart.lock | 9 - charts/phpmyadmin/phpmyadmin/Chart.yaml | 38 - charts/phpmyadmin/phpmyadmin/README.md | 588 - .../phpmyadmin/phpmyadmin/templates/NOTES.txt | 69 - .../phpmyadmin/templates/_helpers.tpl | 86 - .../phpmyadmin/templates/certs.yaml | 27 - .../phpmyadmin/templates/deployment.yaml | 306 - .../phpmyadmin/templates/extra-list.yaml | 9 - .../phpmyadmin/templates/ingress.yaml | 65 - .../phpmyadmin/templates/metrics-svc.yaml | 45 - .../phpmyadmin/templates/networkpolicy.yaml | 80 - .../phpmyadmin/phpmyadmin/templates/pdb.yaml | 26 - .../phpmyadmin/templates/service.yaml | 59 - .../phpmyadmin/templates/serviceaccount.yaml | 18 - .../phpmyadmin/templates/servicemonitor.yaml | 46 - .../phpmyadmin/templates/tls-secrets.yaml | 43 - charts/phpmyadmin/phpmyadmin/values.yaml | 803 - charts/phpmyadmin/values-overrides.yaml | 6 - charts/postgresql/postgresql/.helmignore | 25 - charts/postgresql/postgresql/Chart.lock | 6 - charts/postgresql/postgresql/Chart.yaml | 37 - charts/postgresql/postgresql/README.md | 1021 - .../postgresql/postgresql/templates/NOTES.txt | 117 - .../postgresql/templates/_helpers.tpl | 417 - .../postgresql/templates/backup/cronjob.yaml | 138 - .../templates/backup/networkpolicy.yaml | 32 - .../postgresql/templates/backup/pvc.yaml | 34 - .../postgresql/templates/extra-list.yaml | 9 - .../templates/primary/configmap.yaml | 26 - .../templates/primary/extended-configmap.yaml | 20 - .../primary/initialization-configmap.yaml | 17 - .../templates/primary/metrics-configmap.yaml | 18 - .../templates/primary/metrics-svc.yaml | 31 - .../templates/primary/networkpolicy.yaml | 78 - .../postgresql/templates/primary/pdb.yaml | 29 - .../primary/preinitialization-configmap.yaml | 17 - .../templates/primary/servicemonitor.yaml | 46 - .../templates/primary/statefulset.yaml | 703 - .../templates/primary/svc-headless.yaml | 31 - .../postgresql/templates/primary/svc.yaml | 54 - .../postgresql/templates/prometheusrule.yaml | 22 - .../postgresql/postgresql/templates/psp.yaml | 42 - .../templates/read/extended-configmap.yaml | 20 - .../templates/read/metrics-configmap.yaml | 18 - .../templates/read/metrics-svc.yaml | 31 - .../templates/read/networkpolicy.yaml | 80 - .../postgresql/templates/read/pdb.yaml | 29 - .../templates/read/servicemonitor.yaml | 46 - .../templates/read/statefulset.yaml | 578 - .../templates/read/svc-headless.yaml | 33 - .../postgresql/templates/read/svc.yaml | 56 - .../postgresql/postgresql/templates/role.yaml | 32 - .../postgresql/templates/rolebinding.yaml | 24 - .../postgresql/templates/secrets.yaml | 99 - .../postgresql/templates/serviceaccount.yaml | 18 - .../postgresql/templates/tls-secrets.yaml | 30 - .../postgresql/postgresql/values.schema.json | 156 - charts/postgresql/postgresql/values.yaml | 1780 -- charts/postgresql/values-overrides.yaml | 14 - charts/redis/redis/.helmignore | 25 - charts/redis/redis/Chart.lock | 6 - charts/redis/redis/Chart.yaml | 38 - charts/redis/redis/README.md | 1280 - charts/redis/redis/templates/NOTES.txt | 213 - charts/redis/redis/templates/_helpers.tpl | 346 - charts/redis/redis/templates/configmap.yaml | 64 - charts/redis/redis/templates/extra-list.yaml | 9 - .../redis/redis/templates/headless-svc.yaml | 35 - .../redis/templates/health-configmap.yaml | 194 - .../redis/templates/master/application.yaml | 546 - charts/redis/redis/templates/master/pdb.yaml | 27 - charts/redis/redis/templates/master/psp.yaml | 47 - charts/redis/redis/templates/master/pvc.yaml | 33 - .../redis/redis/templates/master/service.yaml | 62 - .../templates/master/serviceaccount.yaml | 18 - charts/redis/redis/templates/metrics-svc.yaml | 44 - .../redis/redis/templates/networkpolicy.yaml | 108 - charts/redis/redis/templates/podmonitor.yaml | 81 - .../redis/redis/templates/prometheusrule.yaml | 23 - .../redis/templates/replicas/application.yaml | 561 - .../redis/redis/templates/replicas/hpa.yaml | 49 - .../redis/redis/templates/replicas/pdb.yaml | 28 - .../redis/templates/replicas/service.yaml | 59 - .../templates/replicas/serviceaccount.yaml | 18 - charts/redis/redis/templates/role.yaml | 34 - charts/redis/redis/templates/rolebinding.yaml | 23 - .../redis/templates/scripts-configmap.yaml | 791 - .../redis/redis/templates/secret-svcbind.yaml | 37 - charts/redis/redis/templates/secret.yaml | 25 - .../redis/redis/templates/sentinel/hpa.yaml | 49 - .../templates/sentinel/node-services.yaml | 67 - .../redis/redis/templates/sentinel/pdb.yaml | 27 - .../templates/sentinel/ports-configmap.yaml | 102 - .../redis/templates/sentinel/service.yaml | 160 - .../redis/templates/sentinel/statefulset.yaml | 831 - .../redis/redis/templates/serviceaccount.yaml | 18 - .../redis/redis/templates/servicemonitor.yaml | 82 - charts/redis/redis/templates/tls-secret.yaml | 31 - charts/redis/redis/values.schema.json | 163 - charts/redis/redis/values.yaml | 2237 -- charts/redis/values-overrides.yaml | 8 - charts/traefik/traefik/.helmignore | 2 - charts/traefik/traefik/Changelog.md | 10505 -------- charts/traefik/traefik/Chart.yaml | 39 - charts/traefik/traefik/EXAMPLES.md | 1014 - charts/traefik/traefik/Guidelines.md | 34 - charts/traefik/traefik/LICENSE | 202 - charts/traefik/traefik/README.md | 158 - charts/traefik/traefik/VALUES.md | 321 - .../crds/gateway-standard-install-v1.2.0.yaml | 10345 -------- .../hub.traefik.io_accesscontrolpolicies.yaml | 368 - .../crds/hub.traefik.io_apiaccesses.yaml | 188 - .../crds/hub.traefik.io_apibundles.yaml | 125 - .../traefik/crds/hub.traefik.io_apiplans.yaml | 103 - .../crds/hub.traefik.io_apiportals.yaml | 139 - .../crds/hub.traefik.io_apiratelimits.yaml | 166 - .../traefik/crds/hub.traefik.io_apis.yaml | 190 - .../crds/hub.traefik.io_apiversions.yaml | 194 - .../crds/traefik.io_ingressroutes.yaml | 366 - .../crds/traefik.io_ingressroutetcps.yaml | 247 - .../crds/traefik.io_ingressrouteudps.yaml | 111 - .../traefik/crds/traefik.io_middlewares.yaml | 1126 - .../crds/traefik.io_middlewaretcps.yaml | 87 - .../crds/traefik.io_serverstransports.yaml | 139 - .../crds/traefik.io_serverstransporttcps.yaml | 120 - .../traefik/crds/traefik.io_tlsoptions.yaml | 114 - .../traefik/crds/traefik.io_tlsstores.yaml | 97 - .../crds/traefik.io_traefikservices.yaml | 644 - charts/traefik/traefik/templates/NOTES.txt | 36 - charts/traefik/traefik/templates/_helpers.tpl | 178 - .../traefik/templates/_podtemplate.tpl | 850 - .../traefik/templates/_service-metrics.tpl | 25 - charts/traefik/traefik/templates/_service.tpl | 84 - .../traefik/traefik/templates/daemonset.yaml | 58 - .../traefik/traefik/templates/deployment.yaml | 58 - .../traefik/templates/extra-objects.yaml | 4 - charts/traefik/traefik/templates/gateway.yaml | 62 - .../traefik/templates/gatewayclass.yaml | 14 - charts/traefik/traefik/templates/hpa.yaml | 35 - .../templates/hub-admission-controller.yaml | 198 - .../traefik/templates/hub-apiportal.yaml | 19 - .../traefik/templates/ingressclass.yaml | 12 - .../traefik/templates/ingressroute.yaml | 43 - .../templates/poddisruptionbudget.yaml | 23 - .../traefik/templates/prometheusrules.yaml | 28 - .../traefik/templates/provider-file-cm.yaml | 12 - charts/traefik/traefik/templates/pvc.yaml | 26 - .../traefik/templates/rbac/clusterrole.yaml | 271 - .../templates/rbac/clusterrolebinding.yaml | 17 - .../templates/rbac/podsecuritypolicy.yaml | 68 - .../traefik/traefik/templates/rbac/role.yaml | 143 - .../traefik/templates/rbac/rolebinding.yaml | 25 - .../templates/rbac/serviceaccount.yaml | 14 - .../traefik/templates/requirements.yaml | 33 - .../traefik/templates/service-metrics.yaml | 33 - charts/traefik/traefik/templates/service.yaml | 86 - .../traefik/templates/servicemonitor.yaml | 69 - .../traefik/traefik/templates/tlsoption.yaml | 39 - .../traefik/traefik/templates/tlsstore.yaml | 12 - charts/traefik/traefik/values.schema.json | 1676 -- charts/traefik/traefik/values.yaml | 951 - charts/traefik/values-overrides.yaml | 39 - charts/velero/values.yaml | 34 - charts/velero/velero/.helmignore | 21 - charts/velero/velero/Chart.yaml | 19 - charts/velero/velero/OWNERS | 10 - charts/velero/velero/README.md | 173 - charts/velero/velero/ci/test-values.yaml | 119 - .../velero/crds/backuprepositories.yaml | 106 - charts/velero/velero/crds/backups.yaml | 662 - .../velero/crds/backupstoragelocations.yaml | 190 - charts/velero/velero/crds/datadownloads.yaml | 191 - charts/velero/velero/crds/datauploads.yaml | 216 - .../velero/crds/deletebackuprequests.yaml | 80 - .../velero/velero/crds/downloadrequests.yaml | 100 - .../velero/velero/crds/podvolumebackups.yaml | 225 - .../velero/velero/crds/podvolumerestores.yaml | 211 - charts/velero/velero/crds/restores.yaml | 558 - charts/velero/velero/crds/schedules.yaml | 599 - .../velero/crds/serverstatusrequests.yaml | 86 - .../velero/crds/volumesnapshotlocations.yaml | 94 - charts/velero/velero/templates/NOTES.txt | 79 - charts/velero/velero/templates/_helpers.tpl | 96 - .../templates/backupstoragelocation.yaml | 56 - .../velero/velero/templates/cleanup-crds.yaml | 85 - .../velero/templates/clusterrolebinding.yaml | 20 - .../velero/velero/templates/configmaps.yaml | 18 - .../velero/velero/templates/deployment.yaml | 315 - .../velero/templates/extra-manifests.yaml | 4 - .../label-namespace/labelnamespace.yaml | 39 - .../templates/node-agent-daemonset.yaml | 204 - .../velero/velero/templates/podmonitor.yaml | 44 - .../velero/templates/prometheusrule.yaml | 21 - charts/velero/velero/templates/role.yaml | 21 - .../velero/velero/templates/rolebinding.yaml | 21 - charts/velero/velero/templates/schedule.yaml | 34 - charts/velero/velero/templates/secret.yaml | 24 - charts/velero/velero/templates/service.yaml | 29 - .../templates/serviceaccount-server.yaml | 25 - .../velero/templates/servicemonitor.yaml | 46 - .../upgrade-crds/clusterrole-upgrade.yaml | 27 - .../clusterrolebinding-upgrade.yaml | 25 - .../upgrade-crds/serviceaccount-upgrade.yaml | 22 - .../templates/upgrade-crds/upgrade-crds.yaml | 127 - .../templates/volumesnapshotlocation.yaml | 41 - charts/velero/velero/values.yaml | 656 - config/deploy.yml | 45 +- contrib/superstreamer | 1 - devbox.json | 13 +- devbox.lock | 456 +- dockerfiles/bright.dockerfile | 74 +- flux/README.md | 3 - flux/apps/base/bitnami/kustomization.yaml | 5 - flux/apps/base/bitnami/repository.yaml | 8 - flux/apps/base/drupal/kustomization.yaml | 5 - flux/apps/base/drupal/release.yaml | 25 - flux/apps/base/fp/kustomization.yaml | 6 - flux/apps/base/fp/release.yaml | 57 - flux/apps/base/fp/repository.yaml | 15 - flux/apps/base/mariadb/kustomization.yaml | 5 - flux/apps/base/mariadb/release.yaml | 23 - flux/apps/base/podinfo/kustomization.yaml | 7 - flux/apps/base/podinfo/namespace.yaml | 4 - flux/apps/base/podinfo/release.yaml | 39 - flux/apps/base/podinfo/repository.yaml | 8 - flux/apps/base/postgresql/kustomization.yaml | 5 - flux/apps/base/postgresql/release.yaml | 36 - flux/apps/base/redis/kustomization.yaml | 5 - flux/apps/base/redis/release.yaml | 30 - flux/apps/development/README.md | 10 - flux/apps/development/fp-values.yaml | 20 - flux/apps/development/kustomization.yaml | 19 - flux/apps/development/podinfo-values.yaml | 19 - flux/apps/production/fp-values.yaml | 18 - flux/apps/production/kustomization.yaml | 13 - flux/apps/staging/fp-values.yaml | 20 - flux/apps/staging/kustomization.yaml | 18 - flux/apps/staging/podinfo-values.yaml | 18 - flux/clusters/production/apps.yaml | 17 - .../flux-system/gotk-components.yaml | 12385 --------- .../production/flux-system/gotk-sync.yaml | 27 - .../production/flux-system/kustomization.yaml | 19 - flux/clusters/production/infrastructure.yaml | 56 - flux/clusters/staging/apps.yaml | 16 - .../staging/flux-system/gotk-components.yaml | 12385 --------- .../staging/flux-system/gotk-sync.yaml | 27 - .../staging/flux-system/kustomization.yaml | 5 - flux/clusters/staging/infrastructure.yaml | 56 - .../configs/cluster-issuers.yaml | 31 - .../infrastructure/configs/kustomization.yaml | 4 - .../controllers/cert-manager.yaml | 36 - .../controllers/grafana-k8s-dashboards.yaml | 21313 ---------------- .../controllers/ingress-nginx.yaml | 32 - .../controllers/kube-prometheus-stack.yaml | 56 - .../controllers/kustomization.yaml | 8 - package.json | 2 +- packages/fetchers/README.md | 5 - packages/fetchers/package.json | 26 - packages/fetchers/pnpm-lock.yaml | 221 - packages/fetchers/src/config.ts | 27 - packages/fetchers/src/createRecording.ts | 29 - packages/fetchers/src/createS3File.ts | 31 - packages/fetchers/src/createSegment.ts | 30 - .../fetchers/src/createSegmentsVodLink.ts | 36 - .../fetchers/src/createStreamInDatabase.ts | 30 - packages/fetchers/src/createVod.ts | 27 - packages/fetchers/src/findOrCreateStream.ts | 83 - .../fetchers/src/findOrCreateVtuber.spec.ts | 20 - packages/fetchers/src/findOrCreateVtuber.ts | 111 - packages/fetchers/src/findVod.ts | 25 - packages/fetchers/src/getPlaylistUrl.ts | 23 - packages/fetchers/src/getRecording.ts | 45 - .../fetchers/src/getSegmentsFromDatabase.ts | 37 - .../fetchers/src/getStreamFromDatabase.ts | 35 - .../fetchers/src/getStreamIdFromMessage.ts | 9 - packages/fetchers/src/getUrlFromMessage.ts | 36 - packages/fetchers/src/getVod.ts | 23 - packages/fetchers/src/patchRecording.ts | 28 - packages/fetchers/src/patchVod.ts | 28 - packages/fetchers/src/patchVodInDatabase.ts | 23 - .../fetchers/src/updateSegmentInDatabase.ts | 52 - packages/infra/README.md | 3 - packages/infra/package.json | 16 - packages/infra/pnpm-lock.yaml | 23 - .../infra/vultr-delete-orphaned-resources.js | 84 - packages/storage/.mocharc.json | 3 - packages/storage/package.json | 26 - packages/storage/pnpm-lock.yaml | 1362 - packages/storage/src/mux.ts | 0 packages/storage/src/s3.spec.ts | 19 - packages/storage/src/s3.ts | 106 - packages/storage/tsconfig.json | 29 - packages/types/.eslintignore | 1 - packages/types/.gitignore | 1 - packages/types/package.json | 18 - packages/types/pnpm-lock.yaml | 24 - packages/types/src/index.ts | 449 - packages/types/tsconfig.json | 36 - packages/utils/src/image.spec.ts | 2 +- scripts/build-test.sh | 21 - scripts/capture-integration.sh | 25 - scripts/drupal-init-wrapper.sh | 8 - scripts/drupal-init.sh | 24 - scripts/flux-bootstrap.sh | 26 - scripts/k8s-metrics.sh | 12 - scripts/k8s-namespaces.sh | 7 - scripts/k8s-secrets.sh | 210 - scripts/kind-load.sh | 16 - scripts/kind-with-local-registry.sh | 81 - scripts/pgadmin-connection-profile.json | 16 - scripts/pgadmin-import-connection.sh | 20 - scripts/postgres-backup.sh | 23 - scripts/postgres-create.sh | 107 - scripts/postgres-drop.sh | 16 - scripts/postgres-migrations.sh | 13 - scripts/postgres-refresh.sh | 9 - scripts/postgres-restore.sh | 32 - scripts/velero-create.sh | 11 - services/bot/.mocharc.json | 6 - services/bot/README.md | 28 - services/bot/crontab | 18 - services/bot/graphile.config.ts | 12 - services/bot/package.json | 56 - services/bot/pnpm-lock.yaml | 2101 -- services/bot/src/bot.ts | 57 - services/bot/src/collector.ts | 14 - services/bot/src/commands.ts | 22 - services/bot/src/commands/cancel.ts | 53 - services/bot/src/commands/donger.ts | 49 - services/bot/src/commands/ping.ts | 13 - services/bot/src/commands/process.ts | 31 - services/bot/src/commands/record.ts | 148 - services/bot/src/commands/yeah.ts | 20 - services/bot/src/config.ts | 47 - services/bot/src/events/interactionCreate.ts | 51 - services/bot/src/events/ready.ts | 20 - services/bot/src/index.ts | 63 - services/bot/src/middlewares/rbac.ts | 28 - services/bot/src/register-commands.ts | 7 - services/bot/src/tasks/README.md | 30 - .../src/tasks/restart_failed_recordings.ts | 85 - .../bot/src/tasks/update_discord_message.ts | 226 - .../bot/src/tasks/update_vod_statuses.test.ts | 11 - services/bot/src/tasks/update_vod_statuses.ts | 127 - services/bot/src/utils/loader.ts | 23 - services/bot/src/utils/update-commands.ts | 8 - services/bot/tsconfig.json | 32 - services/bright/config/dev.exs | 2 +- .../lib/bright/jobs/create_hls_playlist.ex | 1 + .../bright/lib/bright/platforms/platform.ex | 11 - .../bright_web/components/core_components.ex | 2 +- .../components/layouts/root.html.heex | 42 - .../components/navigation_components.ex | 14 +- .../bright_web/controllers/page_controller.ex | 7 +- .../controllers/page_html/about.html.heex | 22 +- .../controllers/page_html/home.html.heex | 40 +- .../controllers/platform_html/index.html.heex | 2 +- .../controllers/platform_html/show.html.heex | 2 +- .../platform_session_controller.ex | 42 - .../vtuber_html/vtuber_form.html.heex | 3 +- ...platform_confirmation_instructions_live.ex | 51 - .../live/platform_confirmation_live.ex | 58 - .../live/platform_forgot_password_live.ex | 50 - .../bright_web/live/platform_login_live.ex | 43 - .../live/platform_registration_live.ex | 87 - .../live/platform_reset_password_live.ex | 89 - .../bright_web/live/platform_settings_live.ex | 167 - services/bright/lib/bright_web/router.ex | 73 +- services/bright/mix.exs | 6 - services/bright/mix.lock | 3 + ...109070300_create_platforms_auth_tables.exs | 29 - .../bright/test/bright/platforms_test.exs | 567 +- ...rm_confirmation_instructions_live_test.exs | 67 - .../live/platform_confirmation_live_test.exs | 89 - .../platform_forgot_password_live_test.exs | 63 - .../live/platform_login_live_test.exs | 87 - .../live/platform_registration_live_test.exs | 87 - .../platform_reset_password_live_test.exs | 118 - .../live/platform_settings_live_test.exs | 210 - .../test/bright_web/platform_auth_test.exs | 272 - services/capture/.gitignore | 153 - services/capture/.mocharc.json | 6 - services/capture/README.md | 48 - services/capture/package.json | 87 - services/capture/pnpm-lock.yaml | 5836 ----- services/capture/src/Capture.js | 121 - services/capture/src/Capture.spec.js | 147 - services/capture/src/Ipfs.js | 57 - services/capture/src/Record.spec.ts | 97 - services/capture/src/Record.ts | 269 - .../capture/src/RecordNextGeneration.spec.ts | 15 - services/capture/src/RecordNextGeneration.ts | 570 - services/capture/src/Video.js | 68 - services/capture/src/Video.spec.js | 86 - services/capture/src/Voddo.js | 243 - services/capture/src/Voddo.spec.js | 490 - services/capture/src/app.spec.ts | 89 - services/capture/src/app.ts | 59 - services/capture/src/blah.ts | 31 - services/capture/src/cb.js | 18 - services/capture/src/config.ts | 49 - services/capture/src/disk.js | 33 - .../src/fastify-graphile-worker-plugin.ts | 22 - services/capture/src/fastify-pgboss-plugin.ts | 23 - .../capture/src/fixtures/just-a-text-file.txt | 0 services/capture/src/fixtures/mime.types | 2 - .../capture/src/fixtures/mock-stream0.mp4 | Bin 192627 -> 0 bytes .../capture/src/fixtures/mock-stream1.mp4 | Bin 173979 -> 0 bytes .../capture/src/fixtures/mock-stream2.mp4 | Bin 165027 -> 0 bytes services/capture/src/fixtures/my.abc | 1 - services/capture/src/index.ts | 114 - .../capture/src/integration/Capture.spec.js | 123 - services/capture/src/integration/Ipfs.spec.js | 18 - .../capture/src/integration/Voddo.spec.js | 62 - .../capture/src/integration/record.spec.js | 35 - .../capture/src/integration/video.spec.js | 33 - services/capture/src/logger.js | 25 - services/capture/src/poc-lite.js | 25 - services/capture/src/poc-s3-alt.ts | 135 - services/capture/src/poc-s3.ts | 122 - services/capture/src/poc.ts | 177 - services/capture/src/record.js.old | 117 - services/capture/src/s3.ts | 90 - services/capture/src/sample.ts | 28 - services/capture/src/server.ts.old | 154 - services/capture/src/tasks/record.ts | 73 - services/capture/tsconfig.json | 32 - services/capture/tsup.config.ts | 30 - services/factory/.mocharc.json | 6 - services/factory/README.md | 17 - services/factory/crontab | 17 - services/factory/package.json | 47 - services/factory/pnpm-lock.yaml | 2974 --- services/factory/src/config.ts | 70 - services/factory/src/index.ts | 45 - .../src/tasks/combine_video_segments.ts | 337 - .../factory/src/tasks/generate_thumbnail.ts | 92 - services/factory/src/tasks/process_video.ts | 72 - services/factory/src/tasks/remux_video.ts | 83 - .../src/tasks/synchronize_patrons_list.ts | 140 - services/factory/src/utils/importDirectory.ts | 22 - services/factory/tsconfig.json | 33 - services/mailbox/README.md | 17 - services/mailbox/package.json | 43 - services/mailbox/pnpm-lock.yaml | 3207 --- services/mailbox/src/imap.ts | 131 - services/mailbox/src/index.ts | 70 - services/mailbox/src/parsers.spec.ts | 34 - services/mailbox/src/parsers.ts | 99 - services/mailbox/tsconfig.json | 29 - services/mailbox/tsup.config.js | 38 - services/migrations-data/README.md | 27 - services/migrations-data/index.js | 68 - .../0001_from-strapi-to-postgrest-mk2.sql | 980 - .../0002_relate-vtubers-to-vods.sql | 17 - .../0003_relate-mux_asset_id-to-vods.sql | 9 - .../0004_relate-thumbnails-to-vods.sql | 9 - .../0005_rename-vtuber_num-to-vtuber_id.sql | 9 - .../0006_relate-vods-to-streams.sql | 9 - services/migrations-data/migrations/README.md | 17 - services/migrations-data/package.json | 20 - services/migrations-data/pnpm-lock.yaml | 186 - services/migrations-schema/README.md | 27 - services/migrations-schema/index.js | 32 - .../migrations/00001_create.sql | 26 - .../migrations/00002_add-records-table.sql | 13 - .../00003_create-graphile-worker-schema.sql | 4 - .../00004_create-trigger-function.sql | 29 - .../00005_add-trigger-for-record-update.sql | 21 - .../00006_add-updated-at-to-records.sql | 5 - .../00007_add-default-records-timestamps.sql | 7 - .../00008_add-default-records-timestamp.sql | 26 - .../00009_add-streams-vods-vtubers.sql | 140 - .../migrations/00010_record-segments.sql | 7 - .../00011_use-composite-primary-keys.sql | 28 - .../00012_order-records_segments.sql | 2 - .../00013_rename-segments-order.sql | 5 - .../00014_create-segments-stream-links.sql | 2 - .../00015_create-segments-stream-links-2.sql | 16 - .../00016_remove-unecessary-columns.sql | 8 - .../00017_add-stream-status-col.sql | 5 - .../00018_add-stream-status-default.sql | 5 - .../00019_drop-discord-interactions.sql | 1 - .../00020_add-streams-update-trigger.sql | 38 - ...021_add-foreign-key-to-segments_stream.sql | 9 - ...-permissions-for-segments_stream_links.sql | 2 - .../00023_drop-capture_job_id-column.sql | 3 - .../00024_add-updated_at-for-segments.sql | 39 - ...25_add-is_recording_aborted-to-streams.sql | 2 - .../migrations/00026_use-moddatetime.sql | 2 - .../00027_create-triggers-for-moddatetime.sql | 49 - .../00028_remove-moddate-on-insert.sql | 9 - .../00029_add-discord-message-id.sql | 5 - .../00030_update-update_discord_message.sql | 14 - .../00031_recreate-update_stream-trigger.sql | 5 - ...032_update-stream-when-segment-updates.sql | 23 - .../migrations/00033_create-vods.sql | 46 - .../00034_move-segments-to-vods.sql | 24 - .../migrations/00035_seed-vtubers.sql | 159 - ...0036_update_discord_message-use-vod_id.sql | 14 - .../00037_drop-oudated-streamscolumns.sql | 16 - .../00038_add-vods-update-trigger.sql | 41 - .../migrations/00039_create-limiter-table.sql | 7 - .../00040_theme_color-is-optional.sql | 9 - ...00041_add-is_recording_aborted-to-vods.sql | 3 - .../migrations/00042_segments-vods-fk.sql | 6 - .../00043_update_vod_on_segment_update.sql | 23 - .../migrations/00044_segments-uuid.sql | 6 - .../00045_recreate-segments_vod_links.sql | 10 - .../00046_segments_vod_links-many-to-one.sql | 10 - ...ecording_aborted-to-vods-default-false.sql | 5 - .../00048_segments-vod-links-perms.sql | 3 - .../00049_add-discord_message_id-on-vods.sql | 4 - .../migrations/00050_add-status-to-vod.sql | 3 - .../migrations/00051_set-vod-status.sql | 21 - .../00052_correct-typo-filesize-to-bytes.sql | 21 - .../00053_add-default-status-on-vods.sql | 5 - ...0054_remove-bytes-from-vod-tg-function.sql | 27 - ...55_trigger_update_vod-switch-to-before.sql | 7 - .../00056_add_job-retry-only-6-times.sql | 15 - .../migrations/00057_recreate-vod_create.sql | 8 - .../00058_conditionally-change-vod-status.sql | 19 - .../migrations/00059_fix-filesize-typo.sql | 20 - .../00060_create-recordings-table.sql | 35 - ...defaults-for-created_at-and-updated_at.sql | 6 - .../00062_create-discord_interactions.sql | 8 - ...e-discord_interactions-with-recordings.sql | 14 - ..._execute-procedure-instead-of-function.sql | 8 - .../00065_move-is_aborted-to-recordings.sql | 6 - .../00066_add-segments-fk-to-vods.sql | 3 - .../00067_remove-duplicate-vods-segments.sql | 29 - .../00068_remove-vod_id-from-recordings.sql | 27 - ...9_recreate-recordings-vod-relationship.sql | 3 - .../00070_relate-vod-to-recording.sql | 3 - .../00071_remove-recordings_vod_id_fkey.sql | 4 - .../migrations/00072_create-builds-table.sql | 33 - .../00073_update-builds-table-timestamps.sql | 14 - ...74_switch-after-update-to-after-insert.sql | 7 - .../00075_add-checksum-to-segments.sql | 2 - ...00076_builds-use-vod_id-instead-of-vod.sql | 8 - ...77_builds-trigger-change-vod-to-vod_id.sql | 22 - .../00078_add-bytes_uploaded-to-segments.sql | 2 - .../migrations/00079_stream_id-optional.sql | 9 - .../00080_add-created_at-to-vtubers.sql | 11 - .../00081_add-id_deprecated-to-vods.sql | 3 - .../00082_add-id_deprecated-to-s3_files.sql | 3 - .../migrations/00083_create_vods_s3_join.sql | 5 - .../00084_add-file_id-to-s3_files.sql | 2 - .../migrations/00085_sync-s3_files.sql | 9 - .../00086_add-cdn_url-to-s3_files.sql | 2 - .../migrations/00087_remove-id_deprecated.sql | 2 - ...0088_remove-id_deprecated-from-vtubers.sql | 2 - .../00089_create-vods_s3_files_join.sql | 10 - .../00090_rename-vods-s3-files-joins.sql | 19 - .../00091_drop-vods_s3_file_join.sql | 2 - .../migrations/00092_use-s3_file_id.sql | 5 - .../migrations/00093_enable-pg_trgm.sql | 2 - .../migrations/00094_add-patrons-table.sql | 7 - .../migrations/00095_create-contributors.sql | 7 - .../00096_add-id_num-to-s3_files.sql | 4 - .../00097_add-id_num-to-mux-assets.sql | 4 - .../00098_add-id_num-to-streams.sql | 6 - .../migrations/00099_add-vtuber_num.sql | 6 - .../migrations/00100_add-id_num-to-tags.sql | 2 - .../00101_add-missing-cols-to-tags.sql | 11 - .../00102_add-toy_id_num-to-tags.sql | 3 - .../migrations/00103_add-toy_num.sql | 21 - .../migrations/00104_add-toy_num-fix.sql | 7 - .../00105_add-id_num-to-vods_links.sql | 18 - .../migrations/00106_add-id_num-to-vods.sql | 2 - .../migrations/00107_remove-uuid.sql | 125 - .../migrations/00108_add-strapi-tables.sql | 47 - .../00109_use-consistent-description_1.sql | 5 - .../migrations/00110_unseed-vtubers.sql | 5 - .../00111_create-strapi-tables-tvr.sql | 33 - .../migrations/00112_create-tvr_tag_links.sql | 38 - .../migrations/00113_create-tvr_vod_links.sql | 47 - .../00114_create-tags_toy_links.sql | 47 - .../00115_create-tags_vods_links.sql | 55 - .../migrations/00116_create-timestamps.sql | 210 - .../00117_create-timestamps_tag_links.sql | 41 - .../00118_create-timestamps_vod_links.sql | 50 - .../00119_create-toys_link_tag_links.sql | 41 - .../00120_create-vods_mux_asset_links.sql | 41 - .../00121_create-vods_stream_links.sql | 50 - .../00122_create-vods_thumbnail_links.sql | 87 - .../00123_create-vods_uploader_links.sql | 37 - .../00124_create-vods_video_src_b_2_links.sql | 41 - .../00125_create-vods_vtuber_links.sql | 50 - .../00126_create-vtubers_toy_links.sql | 105 - ...rant-permissions-for-vods_vtuber_links.sql | 2 - .../00128_add-vtuber-fk-to-vods.sql | 6 - .../00129_add-vtuber_id-to-vods.sql | 9 - .../00130_add-s3_files-fk-to-vods.sql | 6 - .../00131_grant-perms-on-s3_files.sql | 7 - .../00132_rename-vtuber_num-to-vtuber_id.sql | 7 - .../00133_add-mux_assets-fk-to-vods.sql | 6 - .../00134_add-thumbnail-fk-to-vods.sql | 6 - .../00135_add-b2_files-permissions.sql | 2 - .../migrations/00136_add-uuid-to-streams.sql | 8 - .../00137_add-vods-fk-to-streams.sql | 4 - .../00138_add-date_2-to-streams-and-vods.sql | 7 - .../00139_add-timestamps-permissions.sql | 2 - .../00140_add-permissions-for-links.sql | 15 - ...1_add-permissions-timestamps_vod_links.sql | 2 - ..._add-permissions-for-tag_vod_relations.sql | 4 - ...ssions-for-tag_vod_relations_vod_links.sql | 6 - ...44_add-permissions-for-tags_vods_links.sql | 2 - services/migrations-schema/package.json | 19 - services/migrations-schema/pnpm-lock.yaml | 183 - services/next/.dockerignore | 3 - services/next/.eslintrc.json | 3 - services/next/.gitignore | 48 - services/next/.nvmrc | 1 - services/next/CHECKS | 1 - services/next/LICENSE | 21 - services/next/README.md | 3 - services/next/app.json | 14 - services/next/app/404.tsx | 3 - services/next/app/about/page.tsx | 37 - .../next/app/api/auth/[...nextauth]/route.ts | 17 - services/next/app/api/blogs/route.ts | 10 - services/next/app/api/page.tsx | 58 - .../patreon/currently-entitled-tiers/route.ts | 65 - .../next/app/api/patreon/session/route.ts | 18 - services/next/app/api/profile/route.ts | 51 - services/next/app/api/revalidate/route.ts | 26 - services/next/app/api/service.json/route.ts | 139 - services/next/app/api/session/route.ts | 12 - services/next/app/api/user/route.ts | 3 - services/next/app/api/v1.json/route.ts | 91 - .../next/app/archive/[uuid]/not-found.tsx | 12 - services/next/app/archive/[uuid]/page.tsx | 21 - services/next/app/archive/page.tsx | 31 - .../app/components/access-denied-screen.tsx | 24 - .../next/app/components/archive-progress.tsx | 36 - services/next/app/components/auth-buttons.tsx | 41 - .../next/app/components/auth-provider.tsx | 11 - services/next/app/components/contributors.tsx | 40 - services/next/app/components/error-card.tsx | 9 - services/next/app/components/footer.tsx | 117 - services/next/app/components/funding-goal.tsx | 88 - services/next/app/components/icons/carrd.tsx | 8 - .../next/app/components/icons/chaturbate.tsx | 14 - services/next/app/components/icons/fansly.tsx | 20 - .../next/app/components/icons/linktree.tsx | 30 - .../next/app/components/icons/onlyfans.tsx | 55 - .../next/app/components/icons/pornhub.tsx | 23 - services/next/app/components/icons/throne.tsx | 31 - services/next/app/components/ipfs-cid.tsx | 42 - services/next/app/components/ipfs-logo.tsx | 18 - services/next/app/components/ipfs.tsx | 39 - .../next/app/components/linkable-heading.tsx | 28 - .../next/app/components/localized-date.tsx | 26 - services/next/app/components/navbar.tsx | 93 - .../app/components/notification-center.tsx | 13 - .../next/app/components/notifications.tsx | 10 - services/next/app/components/pager.tsx | 92 - services/next/app/components/patron-perks.tsx | 51 - services/next/app/components/patrons-list.tsx | 52 - .../next/app/components/permissions-table.tsx | 8 - .../next/app/components/protected-route.tsx | 53 - .../next/app/components/query-provider.tsx | 27 - .../next/app/components/sortable-tags.tsx | 70 - services/next/app/components/spinner.tsx | 6 - .../next/app/components/stream-button.tsx | 17 - services/next/app/components/stream-page.tsx | 203 - services/next/app/components/stream.tsx | 86 - services/next/app/components/streams-list.tsx | 58 - .../next/app/components/streams-table.tsx | 287 - services/next/app/components/tag-button.tsx | 8 - services/next/app/components/tag.tsx | 56 - services/next/app/components/tagger.tsx | 241 - services/next/app/components/thumbnail.tsx | 24 - .../next/app/components/timestamps-list.tsx | 68 - services/next/app/components/toys.tsx | 80 - services/next/app/components/upload-form.tsx | 457 - .../next/app/components/user-controls.tsx | 230 - .../next/app/components/user-metadata.tsx | 37 - services/next/app/components/vibrate-test.tsx | 29 - .../next/app/components/video-context.tsx | 55 - .../next/app/components/video-interactive.tsx | 149 - services/next/app/components/video-player.tsx | 145 - .../app/components/video-source-selector.tsx | 142 - services/next/app/components/vod-card.tsx | 70 - services/next/app/components/vod-nav.tsx | 76 - services/next/app/components/vod-page-2.tsx | 61 - services/next/app/components/vod-page.tsx | 106 - services/next/app/components/vods-list.tsx | 66 - .../next/app/components/vtuber-button.tsx | 29 - services/next/app/components/vtuber-card.tsx | 44 - services/next/app/config/clientConfigs.ts | 21 - services/next/app/config/configs.ts | 39 - services/next/app/config/server.ts | 20 - .../app/connect/patreon/redirect/page.tsx | 122 - services/next/app/faq/page.tsx | 65 - services/next/app/favicon.ico | Bin 318 -> 0 bytes services/next/app/feed/feed.json/route.ts | 11 - services/next/app/feed/feed.xml/route.ts | 11 - services/next/app/feed/page.tsx | 28 - services/next/app/feed/rss.xml/route.ts | 11 - services/next/app/goals/page.tsx | 82 - services/next/app/health/page.tsx | 10 - services/next/app/latest-vods/[page]/page.tsx | 29 - services/next/app/latest-vods/page.tsx | 25 - services/next/app/layout.tsx | 58 - services/next/app/lib/auth.ts | 173 - services/next/app/lib/b2File.ts | 2 - services/next/app/lib/constants.ts | 27 - services/next/app/lib/contributors.ts | 22 - services/next/app/lib/dates.ts | 59 - services/next/app/lib/fetch-api.ts | 35 - services/next/app/lib/fetchers.ts | 42 - services/next/app/lib/ipfs.ts | 7 - services/next/app/lib/keycloak.ts | 210 - services/next/app/lib/patreon.ts | 277 - services/next/app/lib/pm.ts | 139 - services/next/app/lib/retry.ts | 13 - services/next/app/lib/rss.ts | 51 - services/next/app/lib/shareRef.ts | 16 - services/next/app/lib/streams.ts | 415 - services/next/app/lib/tag-vod-relations.ts | 174 - services/next/app/lib/tags.ts | 116 - services/next/app/lib/timestamps.ts | 125 - services/next/app/lib/toys.ts | 19 - services/next/app/lib/tweets.ts | 28 - services/next/app/lib/useForwardRef.ts | 27 - services/next/app/lib/users.ts | 1 - services/next/app/lib/vods.ts | 414 - services/next/app/lib/vtubers.ts | 95 - services/next/app/page.tsx | 81 - services/next/app/patrons/page.tsx | 46 - .../hooks/patreonCurrentlyEntitledTiers.ts | 16 - .../next/app/profile/hooks/updateProfile.ts | 21 - .../next/app/profile/hooks/useMetadata.ts | 52 - services/next/app/profile/page.tsx | 242 - services/next/app/sign-in.tsx | 19 - services/next/app/sign-out.tsx | 19 - services/next/app/tags/[slug]/page.tsx | 35 - services/next/app/tags/page.tsx | 15 - services/next/app/upload/page.tsx | 30 - services/next/app/uppy.tsx | 57 - .../next/app/vods/[safeDateOrUUID]/page.tsx | 16 - services/next/app/vods/page.tsx | 6 - services/next/app/vt/[slug]/not-found.tsx | 11 - services/next/app/vt/[slug]/page.tsx | 221 - .../app/vt/[slug]/stream/[safeDate]/page.tsx | 31 - services/next/app/vt/[slug]/streams/page.tsx | 23 - .../next/app/vt/[slug]/toys/[page]/page.tsx | 33 - services/next/app/vt/[slug]/toys/page.tsx | 33 - .../vt/[slug]/vod/[safeDateOrUUID]/page.tsx | 19 - services/next/app/vt/[slug]/vod/page.tsx | 15 - .../next/app/vt/[slug]/vods/[page]/page.tsx | 46 - services/next/app/vt/[slug]/vods/page.tsx | 32 - services/next/app/vt/page.tsx | 36 - services/next/assets/styles/cid.module.css | 19 - services/next/assets/styles/fp.module.css | 28 - services/next/assets/styles/global.css | 58 - services/next/assets/styles/icon.module.css | 20 - services/next/assets/styles/player.module.css | 4 - services/next/assets/svg/README.md | 5 - services/next/assets/svg/carrd.svg | 1 - services/next/assets/svg/chaturbate.svg | 15 - services/next/assets/svg/checkmark.svg | 5 - services/next/assets/svg/fansly.tsx | 20 - services/next/assets/svg/ipfs.svg | 1 - services/next/assets/svg/linktree.svg | 64 - .../assets/svg/noun-adult-content-1731184.svg | 1 - .../next/assets/svg/noun-anime-3890912.svg | 1 - .../next/assets/svg/noun-avatar-3546974.svg | 1 - services/next/assets/svg/noun-girl-842331.svg | 4 - .../next/assets/svg/noun-network-1603820.svg | 1 - services/next/assets/svg/onlyfans.svg | 9 - services/next/assets/svg/pornhub.svg | 5 - services/next/assets/svg/throne.svg | 12 - services/next/next.config.js | 28 - services/next/package.json | 86 - services/next/pnpm-lock.yaml | 5293 ---- .../fonts/nunito-v26-latin-regular.woff2 | Bin 16292 -> 0 bytes .../fonts/overpass-v13-latin-regular.woff2 | Bin 16996 -> 0 bytes services/next/public/futureporn-icon.png | Bin 11599 -> 0 bytes services/next/public/images/.keep | 0 services/next/public/images/cj_clippy.jpg | Bin 301528 -> 0 bytes .../next/public/images/default-thumbnail.webp | Bin 27268 -> 0 bytes .../next/public/images/projekt-melody.jpg | Bin 134526 -> 0 bytes .../images/projektmelody-thumbnail.webp | Bin 1356 -> 0 bytes services/next/public/images/vercel.svg | 3 - services/next/test-jwe-decrypt.js | 18 - services/next/tsconfig.json | 41 - services/next/tsconfig.json.old | 51 - services/scout/.dockerignore | 4 - services/scout/.gitignore | 146 - services/scout/.mocharc.json | 6 - services/scout/README.md | 47 - services/scout/node.d.ts | 4 - services/scout/package.json | 85 - services/scout/pnpm-lock.yaml | 8278 ------ services/scout/src/api.yaml | 55 - services/scout/src/cb.spec.ts | 45 - services/scout/src/cb.ts | 233 - services/scout/src/config.ts | 35 - services/scout/src/css/SwaggerDark.css | 851 - services/scout/src/fansly.spec.ts | 47 - services/scout/src/fansly.ts | 171 - services/scout/src/fastify.ts | 165 - services/scout/src/fixtures/.gitignore | 1 - .../fixtures/cb.projektmelody.fixture.html | 718 - services/scout/src/hello.ts | 9 - services/scout/src/icons.ts | 4 - services/scout/src/index.browser.js | 174 - services/scout/src/index.ts | 6 - services/scout/src/loader.js | 13 - services/scout/src/schemas.ts | 28 - services/scout/src/scrapeVtuberData.spec.ts | 14 - services/scout/src/scrapeVtuberData.ts | 51 - services/scout/src/scrapingFetch.spec.ts | 31 - services/scout/src/scrapingFetch.ts | 75 - services/scout/src/signals.js | 297 - services/scout/src/spawnWrapper.spec.ts | 30 - services/scout/src/spawnWrapper.ts | 34 - services/scout/src/twitter.js | 65 - services/scout/src/twitter.spec.js | 28 - services/scout/src/ua.ts | 1 - services/scout/src/vtuber.spec.js | 36 - services/scout/src/vtuber.ts | 44 - services/scout/src/ytdlp.spec.ts | 53 - services/scout/src/ytdlp.ts | 43 - services/scout/tsconfig.json | 33 - services/scout/tsup.config.ts | 18 - services/strapi/.editorconfig | 16 - services/strapi/.eslintignore | 3 - services/strapi/.eslintrc | 27 - services/strapi/.gitignore | 115 - services/strapi/README.md | 57 - services/strapi/config/admin.js | 17 - services/strapi/config/api.js | 7 - services/strapi/config/database.js | 92 - services/strapi/config/middlewares.js | 12 - services/strapi/config/plugins.js | 45 - services/strapi/config/server.js | 12 - services/strapi/database/daily-backup.sh | 13 - services/strapi/database/devDb.sh | 21 - services/strapi/database/migrations/.gitkeep | 0 ...2023-08-01-relate-vods-to-vtubers-part2.js | 25 - .../migrations/2023-08-17-reformat-cdnUrl.js | 18 - .../2023-08-20-strip-query-string-from-cid.js | 23 - .../2023-08-30-remove-cloudinary.js | 13 - .../2023-08-30-toy-image-field-simplify.js | 33 - .../2023-09-08-change-date-to-string.js | 23 - .../migrations/2023-09-08-drop-toys-image.js | 11 - .../2023-09-08-drop-vod-videosrc.js | 11 - .../migrations/2023-12-24-add-cuid-to-vods.js | 31 - .../2023-12-26-add-cuid-to-streams.js | 33 - .../2023-12-27-relate-vods-to-streams.js | 35 - .../2023.05.09-video-src-sanity.js.noexec | 26 - ...12.32.00.convert-to-video-src-b2.js.noexec | 98 - ....000Z.migrate-tags-to-tag-vod-relations.js | 43 - .../2023.05.15T02.44.00.000Z.drop-vod-tags.js | 12 - .../2023.05.25-gimme-the-tags.js.noexec | 110 - ...023.05.25T20.44.00.000Z.get-the-og-tags.js | 124 - .../2023.07.17.relate-vods-to-vtubers.js | 70 - .../2023.07.31.add-b2-file-cdnUrl.js | 18 - .../2024-01-08-add-streams.js.noexec | 30 - .../2024-01-14-add-date2-to-streams.js | 29 - .../2024-01-15-add-platform-to-streams.js | 49 - .../2024-07-04-add-fansly-id-to-vtuber.js | 28 - services/strapi/database/og-tags.json | 1009 - services/strapi/favicon.png | Bin 497 -> 0 bytes services/strapi/jsconfig.json | 9 - services/strapi/misc/generateCuid.js | 7 - services/strapi/package.json | 51 - services/strapi/pnpm-lock.yaml | 14433 ----------- services/strapi/public/robots.txt | 3 - services/strapi/public/uploads/.gitkeep | 0 services/strapi/src/admin/app.example.js | 39 - .../src/admin/webpack.config.example.js | 9 - services/strapi/src/api/.gitkeep | 0 .../b2-file/content-types/b2-file/schema.json | 36 - .../src/api/b2-file/controllers/b2-file.js | 9 - .../strapi/src/api/b2-file/routes/b2-file.js | 9 - .../src/api/b2-file/services/b2-file.js | 9 - .../content-types/contributor/schema.json | 30 - .../contributor/controllers/contributor.js | 9 - .../src/api/contributor/routes/contributor.js | 9 - .../api/contributor/services/contributor.js | 9 - .../discord-interaction/schema.json | 26 - .../controllers/discord-interaction.js | 9 - .../routes/discord-interaction.js | 9 - .../services/discord-interaction.js | 9 - .../api/goal/content-types/goal/schema.json | 30 - .../strapi/src/api/goal/controllers/goal.js | 9 - services/strapi/src/api/goal/routes/goal.js | 9 - services/strapi/src/api/goal/services/goal.js | 11 - .../api/gogs/content-types/gogs/schema.json | 24 - .../strapi/src/api/gogs/controllers/gogs.js | 29 - services/strapi/src/api/gogs/routes/gogs.js | 34 - services/strapi/src/api/gogs/services/gogs.js | 42 - .../api/issue/content-types/issue/schema.json | 36 - .../strapi/src/api/issue/controllers/issue.js | 9 - services/strapi/src/api/issue/routes/issue.js | 9 - .../strapi/src/api/issue/services/issue.js | 9 - .../content-types/mux-asset/schema.json | 28 - .../api/mux-asset/controllers/mux-asset.js | 56 - .../src/api/mux-asset/routes/mux-asset.js | 33 - .../src/api/mux-asset/services/mux-asset.js | 78 - .../patreon/content-types/patreon/schema.json | 39 - .../src/api/patreon/controllers/patreon.js | 37 - .../strapi/src/api/patreon/routes/patreon.js | 37 - .../src/api/patreon/services/patreon.js | 45 - .../platform-notification/schema.json | 56 - .../controllers/platform-notification.js | 9 - .../routes/platform-notification.js | 9 - .../services/platform-notification.js | 9 - .../src/api/profile/controllers/profile.js | 21 - .../strapi/src/api/profile/routes/profile.js | 23 - .../src/api/profile/services/profile.js | 7 - .../stream/content-types/stream/lifecycles.js | 119 - .../stream/content-types/stream/schema.json | 79 - .../src/api/stream/controllers/stream.js | 9 - .../strapi/src/api/stream/routes/stream.js | 9 - .../strapi/src/api/stream/services/stream.js | 9 - .../tag-vod-relation/schema.json | 39 - .../controllers/tag-vod-relation.js | 222 - .../routes/tag-vod-relation.js | 51 - .../services/tag-vod-relation.js | 69 - .../src/api/tag/content-types/tag/schema.json | 38 - .../strapi/src/api/tag/controllers/tag.js | 87 - services/strapi/src/api/tag/routes/tag.js | 37 - services/strapi/src/api/tag/services/tag.js | 46 - .../content-types/timestamp/schema.json | 47 - .../api/timestamp/controllers/timestamp.js | 166 - .../src/api/timestamp/routes/timestamp.js | 53 - .../src/api/timestamp/services/timestamp.js | 44 - .../src/api/toy/content-types/toy/schema.json | 51 - .../strapi/src/api/toy/controllers/toy.js | 9 - services/strapi/src/api/toy/routes/toy.js | 9 - services/strapi/src/api/toy/services/toy.js | 9 - .../tweet/content-types/tweet/lifecycles.js | 136 - .../api/tweet/content-types/tweet/schema.json | 49 - .../strapi/src/api/tweet/controllers/tweet.js | 9 - services/strapi/src/api/tweet/routes/tweet.js | 9 - .../strapi/src/api/tweet/services/tweet.js | 9 - .../user-submitted-content/lifecycles.js | 57 - .../user-submitted-content/schema.json | 57 - .../controllers/user-submitted-content.js | 64 - .../routes/user-submitted-content.js | 33 - .../services/user-submitted-content.js | 9 - .../api/vod/content-types/vod/lifecycles.js | 27 - .../src/api/vod/content-types/vod/schema.json | 142 - .../strapi/src/api/vod/controllers/vod.js | 85 - services/strapi/src/api/vod/routes/vod.js | 38 - services/strapi/src/api/vod/services/vod.js | 9 - .../vtuber/content-types/vtuber/lifecycles.js | 23 - .../vtuber/content-types/vtuber/schema.json | 127 - .../src/api/vtuber/controllers/vtuber.js | 9 - .../strapi/src/api/vtuber/routes/vtuber.js | 9 - .../strapi/src/api/vtuber/services/vtuber.js | 9 - services/strapi/src/extensions/.gitkeep | 0 services/strapi/src/extensions/README.md | 13 - services/strapi/src/index.js | 20 - .../strapi/src/policies/updateOwnerOnly.js | 34 - services/uppy/.gitignore | 145 - services/uppy/README.md | 16 - services/uppy/apply-backblaze-cors-rules.sh | 8 - services/uppy/index.js | 125 - services/uppy/package.json | 28 - services/uppy/pnpm-lock.yaml | 3177 --- 1191 files changed, 142 insertions(+), 228439 deletions(-) delete mode 100644 Makefile delete mode 100644 Tiltfile delete mode 100644 certs/letsencrypt-stg-root-x1.pem delete mode 100644 certs/letsencrypt-stg-root-x2.pem delete mode 100644 charts/README.md delete mode 100644 charts/cert-manager/cert-manager/Chart.yaml delete mode 100644 charts/cert-manager/cert-manager/README.md delete mode 100644 charts/cert-manager/cert-manager/templates/NOTES.txt delete mode 100644 charts/cert-manager/cert-manager/templates/_helpers.tpl delete mode 100644 charts/cert-manager/cert-manager/templates/cainjector-config.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/cainjector-deployment.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/cainjector-poddisruptionbudget.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/cainjector-psp-clusterrole.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/cainjector-psp-clusterrolebinding.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/cainjector-psp.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/cainjector-rbac.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/cainjector-serviceaccount.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/controller-config.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/crds.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/deployment.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/extras-objects.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/networkpolicy-egress.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/networkpolicy-webhooks.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/poddisruptionbudget.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/podmonitor.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/psp-clusterrole.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/psp-clusterrolebinding.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/psp.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/rbac.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/service.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/serviceaccount.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/servicemonitor.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/startupapicheck-job.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/startupapicheck-psp-clusterrole.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/startupapicheck-psp-clusterrolebinding.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/startupapicheck-psp.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/startupapicheck-rbac.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/startupapicheck-serviceaccount.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/webhook-config.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/webhook-deployment.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/webhook-mutating-webhook.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/webhook-poddisruptionbudget.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/webhook-psp-clusterrole.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/webhook-psp-clusterrolebinding.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/webhook-psp.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/webhook-rbac.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/webhook-service.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/webhook-serviceaccount.yaml delete mode 100644 charts/cert-manager/cert-manager/templates/webhook-validating-webhook.yaml delete mode 100644 charts/cert-manager/cert-manager/values.yaml delete mode 100644 charts/cert-manager/values-overrides.yaml delete mode 100644 charts/chisel-operator/chisel-operator/.helmignore delete mode 100644 charts/chisel-operator/chisel-operator/Chart.yaml delete mode 100644 charts/chisel-operator/chisel-operator/templates/_helpers.tpl delete mode 100644 charts/chisel-operator/chisel-operator/templates/crds/exit-node-provisioner.yaml delete mode 100644 charts/chisel-operator/chisel-operator/templates/crds/exit-node.yaml delete mode 100644 charts/chisel-operator/chisel-operator/templates/deployment.yaml delete mode 100644 charts/chisel-operator/chisel-operator/templates/serviceaccount.yaml delete mode 100644 charts/chisel-operator/chisel-operator/values.yaml delete mode 100644 charts/chisel-operator/values-overrides.yaml delete mode 100644 charts/drupal/README.md delete mode 100644 charts/drupal/drupal/.helmignore delete mode 100644 charts/drupal/drupal/Chart.lock delete mode 100644 charts/drupal/drupal/Chart.yaml delete mode 100644 charts/drupal/drupal/README.md delete mode 100644 charts/drupal/drupal/templates/NOTES.txt delete mode 100644 charts/drupal/drupal/templates/_helpers.tpl delete mode 100644 charts/drupal/drupal/templates/deployment.yaml delete mode 100644 charts/drupal/drupal/templates/externaldb-secrets.yaml delete mode 100644 charts/drupal/drupal/templates/extra-list.yaml delete mode 100644 charts/drupal/drupal/templates/ingress.yaml delete mode 100644 charts/drupal/drupal/templates/metrics-svc.yaml delete mode 100644 charts/drupal/drupal/templates/networkpolicy.yaml delete mode 100644 charts/drupal/drupal/templates/pdb.yaml delete mode 100644 charts/drupal/drupal/templates/prometheusrule.yaml delete mode 100644 charts/drupal/drupal/templates/pv.yaml delete mode 100644 charts/drupal/drupal/templates/pvc.yaml delete mode 100644 charts/drupal/drupal/templates/secrets.yaml delete mode 100644 charts/drupal/drupal/templates/serviceaccount.yaml delete mode 100644 charts/drupal/drupal/templates/servicemonitor.yaml delete mode 100644 charts/drupal/drupal/templates/svc.yaml delete mode 100644 charts/drupal/drupal/templates/tls-secrets.yaml delete mode 100644 charts/drupal/drupal/values.schema.json delete mode 100644 charts/drupal/drupal/values.yaml delete mode 100644 charts/drupal/values-overrides.yaml delete mode 100644 charts/external-secrets/external-secrets/Chart.lock delete mode 100644 charts/external-secrets/external-secrets/Chart.yaml delete mode 100644 charts/external-secrets/external-secrets/README.md delete mode 100644 charts/external-secrets/external-secrets/templates/NOTES.txt delete mode 100644 charts/external-secrets/external-secrets/templates/_helpers.tpl delete mode 100644 charts/external-secrets/external-secrets/templates/cert-controller-deployment.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/cert-controller-poddisruptionbudget.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/cert-controller-rbac.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/cert-controller-service.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/cert-controller-serviceaccount.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/crds/acraccesstoken.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/crds/clusterexternalsecret.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/crds/clustersecretstore.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/crds/ecrauthorizationtoken.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/crds/externalsecret.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/crds/fake.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/crds/gcraccesstoken.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/crds/githubaccesstoken.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/crds/password.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/crds/pushsecret.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/crds/secretstore.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/crds/vaultdynamicsecret.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/crds/webhook.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/deployment.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/extra-manifests.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/poddisruptionbudget.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/rbac.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/service.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/serviceaccount.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/servicemonitor.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/validatingwebhook.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/webhook-certificate.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/webhook-deployment.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/webhook-poddisruptionbudget.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/webhook-secret.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/webhook-service.yaml delete mode 100644 charts/external-secrets/external-secrets/templates/webhook-serviceaccount.yaml delete mode 100644 charts/external-secrets/external-secrets/values.yaml delete mode 100644 charts/external-secrets/values.yaml delete mode 100644 charts/fp/Chart.yaml delete mode 100644 charts/fp/README.md delete mode 100644 charts/fp/templates-staging/README.md delete mode 100644 charts/fp/templates-staging/bot.yaml delete mode 100644 charts/fp/templates-staging/capture.yaml delete mode 100644 charts/fp/templates-staging/chihaya.yaml delete mode 100644 charts/fp/templates-staging/factory.yaml delete mode 100644 charts/fp/templates-staging/mailbox.yaml delete mode 100644 charts/fp/templates-staging/migrations-data.yaml delete mode 100644 charts/fp/templates-staging/migrations-schema.yaml delete mode 100644 charts/fp/templates-staging/next.yaml delete mode 100644 charts/fp/templates-staging/postgres.yaml delete mode 100644 charts/fp/templates-staging/postgresql.yaml delete mode 100644 charts/fp/templates-staging/postgrest.yaml delete mode 100644 charts/fp/templates-staging/redis.yaml delete mode 100644 charts/fp/templates-staging/scout.yaml delete mode 100644 charts/fp/templates-staging/supertokens.yaml delete mode 100644 charts/fp/templates-staging/trigger.yaml delete mode 100644 charts/fp/templates/.gitignore delete mode 100644 charts/fp/templates/bright.yaml delete mode 100644 charts/fp/templates/cert-manager.yaml delete mode 100644 charts/fp/templates/chisel.yaml delete mode 100644 charts/fp/templates/exoscale-webhook.yaml delete mode 100644 charts/fp/templates/external-dns.yaml delete mode 100644 charts/fp/templates/gateway.yaml delete mode 100644 charts/fp/templates/pgadmin4.yaml delete mode 100644 charts/fp/templates/roles.yaml delete mode 100644 charts/fp/templates/superstreamer.yaml delete mode 100644 charts/fp/templates/traefik.yaml delete mode 100644 charts/fp/templates/whoami.yaml delete mode 100644 charts/fp/values.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/.helmignore delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/CHANGELOG.md delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/Chart.lock delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/Chart.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/README.md delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/NOTES.txt delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/_helpers.tpl delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-cm.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-deployment.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-pdb.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-rbac.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-serviceaccount.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_domains.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_httpsedges.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_ippolicies.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_ngrokmodulesets.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_tcpedges.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_tlsedges.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_tunnels.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ngrok.k8s.ngrok.com_ngroktrafficpolicies.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/credentials-secret.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/ingress-class.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/domain_editor_role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/domain_viewer_role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/httpsedge_editor_role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/httpsedge_viewer_role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ippolicy_editor_role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ippolicy_viewer_role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ngrokmoduleset_editor_role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ngrokmoduleset_viewer_role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tcpedge_editor_role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tcpedge_viewer_role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tlsedge_editor_role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tlsedge_viewer_role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tunnel_editor_role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tunnel_viewer_role.yaml delete mode 100644 charts/kubernetes-ingress-controller/kubernetes-ingress-controller/values.yaml delete mode 100644 charts/kubernetes-ingress-controller/values-overrides.yaml delete mode 100644 charts/mariadb/mariadb/.helmignore delete mode 100644 charts/mariadb/mariadb/Chart.lock delete mode 100644 charts/mariadb/mariadb/Chart.yaml delete mode 100644 charts/mariadb/mariadb/README.md delete mode 100644 charts/mariadb/mariadb/templates/NOTES.txt delete mode 100644 charts/mariadb/mariadb/templates/_helpers.tpl delete mode 100644 charts/mariadb/mariadb/templates/extra-list.yaml delete mode 100644 charts/mariadb/mariadb/templates/networkpolicy.yaml delete mode 100644 charts/mariadb/mariadb/templates/primary/configmap.yaml delete mode 100644 charts/mariadb/mariadb/templates/primary/initialization-configmap.yaml delete mode 100644 charts/mariadb/mariadb/templates/primary/pdb.yaml delete mode 100644 charts/mariadb/mariadb/templates/primary/statefulset.yaml delete mode 100644 charts/mariadb/mariadb/templates/primary/svc.yaml delete mode 100644 charts/mariadb/mariadb/templates/prometheusrules.yaml delete mode 100644 charts/mariadb/mariadb/templates/role.yaml delete mode 100644 charts/mariadb/mariadb/templates/rolebinding.yaml delete mode 100644 charts/mariadb/mariadb/templates/secondary/configmap.yaml delete mode 100644 charts/mariadb/mariadb/templates/secondary/pdb.yaml delete mode 100644 charts/mariadb/mariadb/templates/secondary/statefulset.yaml delete mode 100644 charts/mariadb/mariadb/templates/secondary/svc.yaml delete mode 100644 charts/mariadb/mariadb/templates/secrets.yaml delete mode 100644 charts/mariadb/mariadb/templates/serviceaccount.yaml delete mode 100644 charts/mariadb/mariadb/templates/servicemonitor.yaml delete mode 100644 charts/mariadb/mariadb/values.schema.json delete mode 100644 charts/mariadb/mariadb/values.yaml delete mode 100644 charts/mariadb/values-overrides.yaml delete mode 100644 charts/phpmyadmin/phpmyadmin/.helmignore delete mode 100644 charts/phpmyadmin/phpmyadmin/Chart.lock delete mode 100644 charts/phpmyadmin/phpmyadmin/Chart.yaml delete mode 100644 charts/phpmyadmin/phpmyadmin/README.md delete mode 100644 charts/phpmyadmin/phpmyadmin/templates/NOTES.txt delete mode 100644 charts/phpmyadmin/phpmyadmin/templates/_helpers.tpl delete mode 100644 charts/phpmyadmin/phpmyadmin/templates/certs.yaml delete mode 100644 charts/phpmyadmin/phpmyadmin/templates/deployment.yaml delete mode 100644 charts/phpmyadmin/phpmyadmin/templates/extra-list.yaml delete mode 100644 charts/phpmyadmin/phpmyadmin/templates/ingress.yaml delete mode 100644 charts/phpmyadmin/phpmyadmin/templates/metrics-svc.yaml delete mode 100644 charts/phpmyadmin/phpmyadmin/templates/networkpolicy.yaml delete mode 100644 charts/phpmyadmin/phpmyadmin/templates/pdb.yaml delete mode 100644 charts/phpmyadmin/phpmyadmin/templates/service.yaml delete mode 100644 charts/phpmyadmin/phpmyadmin/templates/serviceaccount.yaml delete mode 100644 charts/phpmyadmin/phpmyadmin/templates/servicemonitor.yaml delete mode 100644 charts/phpmyadmin/phpmyadmin/templates/tls-secrets.yaml delete mode 100644 charts/phpmyadmin/phpmyadmin/values.yaml delete mode 100644 charts/phpmyadmin/values-overrides.yaml delete mode 100644 charts/postgresql/postgresql/.helmignore delete mode 100644 charts/postgresql/postgresql/Chart.lock delete mode 100644 charts/postgresql/postgresql/Chart.yaml delete mode 100644 charts/postgresql/postgresql/README.md delete mode 100644 charts/postgresql/postgresql/templates/NOTES.txt delete mode 100644 charts/postgresql/postgresql/templates/_helpers.tpl delete mode 100644 charts/postgresql/postgresql/templates/backup/cronjob.yaml delete mode 100644 charts/postgresql/postgresql/templates/backup/networkpolicy.yaml delete mode 100644 charts/postgresql/postgresql/templates/backup/pvc.yaml delete mode 100644 charts/postgresql/postgresql/templates/extra-list.yaml delete mode 100644 charts/postgresql/postgresql/templates/primary/configmap.yaml delete mode 100644 charts/postgresql/postgresql/templates/primary/extended-configmap.yaml delete mode 100644 charts/postgresql/postgresql/templates/primary/initialization-configmap.yaml delete mode 100644 charts/postgresql/postgresql/templates/primary/metrics-configmap.yaml delete mode 100644 charts/postgresql/postgresql/templates/primary/metrics-svc.yaml delete mode 100644 charts/postgresql/postgresql/templates/primary/networkpolicy.yaml delete mode 100644 charts/postgresql/postgresql/templates/primary/pdb.yaml delete mode 100644 charts/postgresql/postgresql/templates/primary/preinitialization-configmap.yaml delete mode 100644 charts/postgresql/postgresql/templates/primary/servicemonitor.yaml delete mode 100644 charts/postgresql/postgresql/templates/primary/statefulset.yaml delete mode 100644 charts/postgresql/postgresql/templates/primary/svc-headless.yaml delete mode 100644 charts/postgresql/postgresql/templates/primary/svc.yaml delete mode 100644 charts/postgresql/postgresql/templates/prometheusrule.yaml delete mode 100644 charts/postgresql/postgresql/templates/psp.yaml delete mode 100644 charts/postgresql/postgresql/templates/read/extended-configmap.yaml delete mode 100644 charts/postgresql/postgresql/templates/read/metrics-configmap.yaml delete mode 100644 charts/postgresql/postgresql/templates/read/metrics-svc.yaml delete mode 100644 charts/postgresql/postgresql/templates/read/networkpolicy.yaml delete mode 100644 charts/postgresql/postgresql/templates/read/pdb.yaml delete mode 100644 charts/postgresql/postgresql/templates/read/servicemonitor.yaml delete mode 100644 charts/postgresql/postgresql/templates/read/statefulset.yaml delete mode 100644 charts/postgresql/postgresql/templates/read/svc-headless.yaml delete mode 100644 charts/postgresql/postgresql/templates/read/svc.yaml delete mode 100644 charts/postgresql/postgresql/templates/role.yaml delete mode 100644 charts/postgresql/postgresql/templates/rolebinding.yaml delete mode 100644 charts/postgresql/postgresql/templates/secrets.yaml delete mode 100644 charts/postgresql/postgresql/templates/serviceaccount.yaml delete mode 100644 charts/postgresql/postgresql/templates/tls-secrets.yaml delete mode 100644 charts/postgresql/postgresql/values.schema.json delete mode 100644 charts/postgresql/postgresql/values.yaml delete mode 100644 charts/postgresql/values-overrides.yaml delete mode 100644 charts/redis/redis/.helmignore delete mode 100644 charts/redis/redis/Chart.lock delete mode 100644 charts/redis/redis/Chart.yaml delete mode 100644 charts/redis/redis/README.md delete mode 100644 charts/redis/redis/templates/NOTES.txt delete mode 100644 charts/redis/redis/templates/_helpers.tpl delete mode 100644 charts/redis/redis/templates/configmap.yaml delete mode 100644 charts/redis/redis/templates/extra-list.yaml delete mode 100644 charts/redis/redis/templates/headless-svc.yaml delete mode 100644 charts/redis/redis/templates/health-configmap.yaml delete mode 100644 charts/redis/redis/templates/master/application.yaml delete mode 100644 charts/redis/redis/templates/master/pdb.yaml delete mode 100644 charts/redis/redis/templates/master/psp.yaml delete mode 100644 charts/redis/redis/templates/master/pvc.yaml delete mode 100644 charts/redis/redis/templates/master/service.yaml delete mode 100644 charts/redis/redis/templates/master/serviceaccount.yaml delete mode 100644 charts/redis/redis/templates/metrics-svc.yaml delete mode 100644 charts/redis/redis/templates/networkpolicy.yaml delete mode 100644 charts/redis/redis/templates/podmonitor.yaml delete mode 100644 charts/redis/redis/templates/prometheusrule.yaml delete mode 100644 charts/redis/redis/templates/replicas/application.yaml delete mode 100644 charts/redis/redis/templates/replicas/hpa.yaml delete mode 100644 charts/redis/redis/templates/replicas/pdb.yaml delete mode 100644 charts/redis/redis/templates/replicas/service.yaml delete mode 100644 charts/redis/redis/templates/replicas/serviceaccount.yaml delete mode 100644 charts/redis/redis/templates/role.yaml delete mode 100644 charts/redis/redis/templates/rolebinding.yaml delete mode 100644 charts/redis/redis/templates/scripts-configmap.yaml delete mode 100644 charts/redis/redis/templates/secret-svcbind.yaml delete mode 100644 charts/redis/redis/templates/secret.yaml delete mode 100644 charts/redis/redis/templates/sentinel/hpa.yaml delete mode 100644 charts/redis/redis/templates/sentinel/node-services.yaml delete mode 100644 charts/redis/redis/templates/sentinel/pdb.yaml delete mode 100644 charts/redis/redis/templates/sentinel/ports-configmap.yaml delete mode 100644 charts/redis/redis/templates/sentinel/service.yaml delete mode 100644 charts/redis/redis/templates/sentinel/statefulset.yaml delete mode 100644 charts/redis/redis/templates/serviceaccount.yaml delete mode 100644 charts/redis/redis/templates/servicemonitor.yaml delete mode 100644 charts/redis/redis/templates/tls-secret.yaml delete mode 100644 charts/redis/redis/values.schema.json delete mode 100644 charts/redis/redis/values.yaml delete mode 100644 charts/redis/values-overrides.yaml delete mode 100644 charts/traefik/traefik/.helmignore delete mode 100644 charts/traefik/traefik/Changelog.md delete mode 100644 charts/traefik/traefik/Chart.yaml delete mode 100644 charts/traefik/traefik/EXAMPLES.md delete mode 100644 charts/traefik/traefik/Guidelines.md delete mode 100644 charts/traefik/traefik/LICENSE delete mode 100644 charts/traefik/traefik/README.md delete mode 100644 charts/traefik/traefik/VALUES.md delete mode 100644 charts/traefik/traefik/crds/gateway-standard-install-v1.2.0.yaml delete mode 100644 charts/traefik/traefik/crds/hub.traefik.io_accesscontrolpolicies.yaml delete mode 100644 charts/traefik/traefik/crds/hub.traefik.io_apiaccesses.yaml delete mode 100644 charts/traefik/traefik/crds/hub.traefik.io_apibundles.yaml delete mode 100644 charts/traefik/traefik/crds/hub.traefik.io_apiplans.yaml delete mode 100644 charts/traefik/traefik/crds/hub.traefik.io_apiportals.yaml delete mode 100644 charts/traefik/traefik/crds/hub.traefik.io_apiratelimits.yaml delete mode 100644 charts/traefik/traefik/crds/hub.traefik.io_apis.yaml delete mode 100644 charts/traefik/traefik/crds/hub.traefik.io_apiversions.yaml delete mode 100644 charts/traefik/traefik/crds/traefik.io_ingressroutes.yaml delete mode 100644 charts/traefik/traefik/crds/traefik.io_ingressroutetcps.yaml delete mode 100644 charts/traefik/traefik/crds/traefik.io_ingressrouteudps.yaml delete mode 100644 charts/traefik/traefik/crds/traefik.io_middlewares.yaml delete mode 100644 charts/traefik/traefik/crds/traefik.io_middlewaretcps.yaml delete mode 100644 charts/traefik/traefik/crds/traefik.io_serverstransports.yaml delete mode 100644 charts/traefik/traefik/crds/traefik.io_serverstransporttcps.yaml delete mode 100644 charts/traefik/traefik/crds/traefik.io_tlsoptions.yaml delete mode 100644 charts/traefik/traefik/crds/traefik.io_tlsstores.yaml delete mode 100644 charts/traefik/traefik/crds/traefik.io_traefikservices.yaml delete mode 100644 charts/traefik/traefik/templates/NOTES.txt delete mode 100644 charts/traefik/traefik/templates/_helpers.tpl delete mode 100644 charts/traefik/traefik/templates/_podtemplate.tpl delete mode 100644 charts/traefik/traefik/templates/_service-metrics.tpl delete mode 100644 charts/traefik/traefik/templates/_service.tpl delete mode 100644 charts/traefik/traefik/templates/daemonset.yaml delete mode 100644 charts/traefik/traefik/templates/deployment.yaml delete mode 100644 charts/traefik/traefik/templates/extra-objects.yaml delete mode 100644 charts/traefik/traefik/templates/gateway.yaml delete mode 100644 charts/traefik/traefik/templates/gatewayclass.yaml delete mode 100644 charts/traefik/traefik/templates/hpa.yaml delete mode 100644 charts/traefik/traefik/templates/hub-admission-controller.yaml delete mode 100644 charts/traefik/traefik/templates/hub-apiportal.yaml delete mode 100644 charts/traefik/traefik/templates/ingressclass.yaml delete mode 100644 charts/traefik/traefik/templates/ingressroute.yaml delete mode 100644 charts/traefik/traefik/templates/poddisruptionbudget.yaml delete mode 100644 charts/traefik/traefik/templates/prometheusrules.yaml delete mode 100644 charts/traefik/traefik/templates/provider-file-cm.yaml delete mode 100644 charts/traefik/traefik/templates/pvc.yaml delete mode 100644 charts/traefik/traefik/templates/rbac/clusterrole.yaml delete mode 100644 charts/traefik/traefik/templates/rbac/clusterrolebinding.yaml delete mode 100644 charts/traefik/traefik/templates/rbac/podsecuritypolicy.yaml delete mode 100644 charts/traefik/traefik/templates/rbac/role.yaml delete mode 100644 charts/traefik/traefik/templates/rbac/rolebinding.yaml delete mode 100644 charts/traefik/traefik/templates/rbac/serviceaccount.yaml delete mode 100644 charts/traefik/traefik/templates/requirements.yaml delete mode 100644 charts/traefik/traefik/templates/service-metrics.yaml delete mode 100644 charts/traefik/traefik/templates/service.yaml delete mode 100644 charts/traefik/traefik/templates/servicemonitor.yaml delete mode 100644 charts/traefik/traefik/templates/tlsoption.yaml delete mode 100644 charts/traefik/traefik/templates/tlsstore.yaml delete mode 100644 charts/traefik/traefik/values.schema.json delete mode 100644 charts/traefik/traefik/values.yaml delete mode 100644 charts/traefik/values-overrides.yaml delete mode 100644 charts/velero/values.yaml delete mode 100644 charts/velero/velero/.helmignore delete mode 100644 charts/velero/velero/Chart.yaml delete mode 100644 charts/velero/velero/OWNERS delete mode 100644 charts/velero/velero/README.md delete mode 100644 charts/velero/velero/ci/test-values.yaml delete mode 100644 charts/velero/velero/crds/backuprepositories.yaml delete mode 100644 charts/velero/velero/crds/backups.yaml delete mode 100644 charts/velero/velero/crds/backupstoragelocations.yaml delete mode 100644 charts/velero/velero/crds/datadownloads.yaml delete mode 100644 charts/velero/velero/crds/datauploads.yaml delete mode 100644 charts/velero/velero/crds/deletebackuprequests.yaml delete mode 100644 charts/velero/velero/crds/downloadrequests.yaml delete mode 100644 charts/velero/velero/crds/podvolumebackups.yaml delete mode 100644 charts/velero/velero/crds/podvolumerestores.yaml delete mode 100644 charts/velero/velero/crds/restores.yaml delete mode 100644 charts/velero/velero/crds/schedules.yaml delete mode 100644 charts/velero/velero/crds/serverstatusrequests.yaml delete mode 100644 charts/velero/velero/crds/volumesnapshotlocations.yaml delete mode 100644 charts/velero/velero/templates/NOTES.txt delete mode 100644 charts/velero/velero/templates/_helpers.tpl delete mode 100644 charts/velero/velero/templates/backupstoragelocation.yaml delete mode 100644 charts/velero/velero/templates/cleanup-crds.yaml delete mode 100644 charts/velero/velero/templates/clusterrolebinding.yaml delete mode 100644 charts/velero/velero/templates/configmaps.yaml delete mode 100644 charts/velero/velero/templates/deployment.yaml delete mode 100644 charts/velero/velero/templates/extra-manifests.yaml delete mode 100644 charts/velero/velero/templates/label-namespace/labelnamespace.yaml delete mode 100644 charts/velero/velero/templates/node-agent-daemonset.yaml delete mode 100644 charts/velero/velero/templates/podmonitor.yaml delete mode 100644 charts/velero/velero/templates/prometheusrule.yaml delete mode 100644 charts/velero/velero/templates/role.yaml delete mode 100644 charts/velero/velero/templates/rolebinding.yaml delete mode 100644 charts/velero/velero/templates/schedule.yaml delete mode 100644 charts/velero/velero/templates/secret.yaml delete mode 100644 charts/velero/velero/templates/service.yaml delete mode 100644 charts/velero/velero/templates/serviceaccount-server.yaml delete mode 100644 charts/velero/velero/templates/servicemonitor.yaml delete mode 100644 charts/velero/velero/templates/upgrade-crds/clusterrole-upgrade.yaml delete mode 100644 charts/velero/velero/templates/upgrade-crds/clusterrolebinding-upgrade.yaml delete mode 100644 charts/velero/velero/templates/upgrade-crds/serviceaccount-upgrade.yaml delete mode 100644 charts/velero/velero/templates/upgrade-crds/upgrade-crds.yaml delete mode 100644 charts/velero/velero/templates/volumesnapshotlocation.yaml delete mode 100644 charts/velero/velero/values.yaml delete mode 160000 contrib/superstreamer delete mode 100644 flux/README.md delete mode 100644 flux/apps/base/bitnami/kustomization.yaml delete mode 100644 flux/apps/base/bitnami/repository.yaml delete mode 100644 flux/apps/base/drupal/kustomization.yaml delete mode 100644 flux/apps/base/drupal/release.yaml delete mode 100644 flux/apps/base/fp/kustomization.yaml delete mode 100644 flux/apps/base/fp/release.yaml delete mode 100644 flux/apps/base/fp/repository.yaml delete mode 100644 flux/apps/base/mariadb/kustomization.yaml delete mode 100644 flux/apps/base/mariadb/release.yaml delete mode 100644 flux/apps/base/podinfo/kustomization.yaml delete mode 100644 flux/apps/base/podinfo/namespace.yaml delete mode 100644 flux/apps/base/podinfo/release.yaml delete mode 100644 flux/apps/base/podinfo/repository.yaml delete mode 100644 flux/apps/base/postgresql/kustomization.yaml delete mode 100644 flux/apps/base/postgresql/release.yaml delete mode 100644 flux/apps/base/redis/kustomization.yaml delete mode 100644 flux/apps/base/redis/release.yaml delete mode 100644 flux/apps/development/README.md delete mode 100644 flux/apps/development/fp-values.yaml delete mode 100644 flux/apps/development/kustomization.yaml delete mode 100644 flux/apps/development/podinfo-values.yaml delete mode 100644 flux/apps/production/fp-values.yaml delete mode 100644 flux/apps/production/kustomization.yaml delete mode 100644 flux/apps/staging/fp-values.yaml delete mode 100644 flux/apps/staging/kustomization.yaml delete mode 100644 flux/apps/staging/podinfo-values.yaml delete mode 100644 flux/clusters/production/apps.yaml delete mode 100644 flux/clusters/production/flux-system/gotk-components.yaml delete mode 100644 flux/clusters/production/flux-system/gotk-sync.yaml delete mode 100644 flux/clusters/production/flux-system/kustomization.yaml delete mode 100644 flux/clusters/production/infrastructure.yaml delete mode 100644 flux/clusters/staging/apps.yaml delete mode 100644 flux/clusters/staging/flux-system/gotk-components.yaml delete mode 100644 flux/clusters/staging/flux-system/gotk-sync.yaml delete mode 100644 flux/clusters/staging/flux-system/kustomization.yaml delete mode 100644 flux/clusters/staging/infrastructure.yaml delete mode 100644 flux/infrastructure/configs/cluster-issuers.yaml delete mode 100644 flux/infrastructure/configs/kustomization.yaml delete mode 100644 flux/infrastructure/controllers/cert-manager.yaml delete mode 100644 flux/infrastructure/controllers/grafana-k8s-dashboards.yaml delete mode 100644 flux/infrastructure/controllers/ingress-nginx.yaml delete mode 100644 flux/infrastructure/controllers/kube-prometheus-stack.yaml delete mode 100644 flux/infrastructure/controllers/kustomization.yaml delete mode 100644 packages/fetchers/README.md delete mode 100644 packages/fetchers/package.json delete mode 100644 packages/fetchers/pnpm-lock.yaml delete mode 100644 packages/fetchers/src/config.ts delete mode 100644 packages/fetchers/src/createRecording.ts delete mode 100644 packages/fetchers/src/createS3File.ts delete mode 100644 packages/fetchers/src/createSegment.ts delete mode 100644 packages/fetchers/src/createSegmentsVodLink.ts delete mode 100644 packages/fetchers/src/createStreamInDatabase.ts delete mode 100644 packages/fetchers/src/createVod.ts delete mode 100644 packages/fetchers/src/findOrCreateStream.ts delete mode 100644 packages/fetchers/src/findOrCreateVtuber.spec.ts delete mode 100644 packages/fetchers/src/findOrCreateVtuber.ts delete mode 100644 packages/fetchers/src/findVod.ts delete mode 100644 packages/fetchers/src/getPlaylistUrl.ts delete mode 100644 packages/fetchers/src/getRecording.ts delete mode 100644 packages/fetchers/src/getSegmentsFromDatabase.ts delete mode 100644 packages/fetchers/src/getStreamFromDatabase.ts delete mode 100644 packages/fetchers/src/getStreamIdFromMessage.ts delete mode 100644 packages/fetchers/src/getUrlFromMessage.ts delete mode 100644 packages/fetchers/src/getVod.ts delete mode 100644 packages/fetchers/src/patchRecording.ts delete mode 100644 packages/fetchers/src/patchVod.ts delete mode 100644 packages/fetchers/src/patchVodInDatabase.ts delete mode 100644 packages/fetchers/src/updateSegmentInDatabase.ts delete mode 100644 packages/infra/README.md delete mode 100644 packages/infra/package.json delete mode 100644 packages/infra/pnpm-lock.yaml delete mode 100755 packages/infra/vultr-delete-orphaned-resources.js delete mode 100644 packages/storage/.mocharc.json delete mode 100644 packages/storage/package.json delete mode 100644 packages/storage/pnpm-lock.yaml delete mode 100644 packages/storage/src/mux.ts delete mode 100644 packages/storage/src/s3.spec.ts delete mode 100644 packages/storage/src/s3.ts delete mode 100644 packages/storage/tsconfig.json delete mode 100644 packages/types/.eslintignore delete mode 100644 packages/types/.gitignore delete mode 100644 packages/types/package.json delete mode 100644 packages/types/pnpm-lock.yaml delete mode 100644 packages/types/src/index.ts delete mode 100644 packages/types/tsconfig.json delete mode 100755 scripts/build-test.sh delete mode 100755 scripts/capture-integration.sh delete mode 100755 scripts/drupal-init-wrapper.sh delete mode 100755 scripts/drupal-init.sh delete mode 100755 scripts/flux-bootstrap.sh delete mode 100755 scripts/k8s-metrics.sh delete mode 100755 scripts/k8s-namespaces.sh delete mode 100755 scripts/k8s-secrets.sh delete mode 100755 scripts/kind-load.sh delete mode 100755 scripts/kind-with-local-registry.sh delete mode 100644 scripts/pgadmin-connection-profile.json delete mode 100755 scripts/pgadmin-import-connection.sh delete mode 100644 scripts/postgres-backup.sh delete mode 100755 scripts/postgres-create.sh delete mode 100644 scripts/postgres-drop.sh delete mode 100755 scripts/postgres-migrations.sh delete mode 100644 scripts/postgres-refresh.sh delete mode 100755 scripts/postgres-restore.sh delete mode 100755 scripts/velero-create.sh delete mode 100644 services/bot/.mocharc.json delete mode 100644 services/bot/README.md delete mode 100644 services/bot/crontab delete mode 100644 services/bot/graphile.config.ts delete mode 100644 services/bot/package.json delete mode 100644 services/bot/pnpm-lock.yaml delete mode 100644 services/bot/src/bot.ts delete mode 100644 services/bot/src/collector.ts delete mode 100644 services/bot/src/commands.ts delete mode 100644 services/bot/src/commands/cancel.ts delete mode 100644 services/bot/src/commands/donger.ts delete mode 100644 services/bot/src/commands/ping.ts delete mode 100644 services/bot/src/commands/process.ts delete mode 100644 services/bot/src/commands/record.ts delete mode 100644 services/bot/src/commands/yeah.ts delete mode 100644 services/bot/src/config.ts delete mode 100644 services/bot/src/events/interactionCreate.ts delete mode 100644 services/bot/src/events/ready.ts delete mode 100644 services/bot/src/index.ts delete mode 100644 services/bot/src/middlewares/rbac.ts delete mode 100644 services/bot/src/register-commands.ts delete mode 100644 services/bot/src/tasks/README.md delete mode 100644 services/bot/src/tasks/restart_failed_recordings.ts delete mode 100644 services/bot/src/tasks/update_discord_message.ts delete mode 100644 services/bot/src/tasks/update_vod_statuses.test.ts delete mode 100644 services/bot/src/tasks/update_vod_statuses.ts delete mode 100644 services/bot/src/utils/loader.ts delete mode 100644 services/bot/src/utils/update-commands.ts delete mode 100644 services/bot/tsconfig.json delete mode 100644 services/bright/lib/bright_web/controllers/platform_session_controller.ex delete mode 100644 services/bright/lib/bright_web/live/platform_confirmation_instructions_live.ex delete mode 100644 services/bright/lib/bright_web/live/platform_confirmation_live.ex delete mode 100644 services/bright/lib/bright_web/live/platform_forgot_password_live.ex delete mode 100644 services/bright/lib/bright_web/live/platform_login_live.ex delete mode 100644 services/bright/lib/bright_web/live/platform_registration_live.ex delete mode 100644 services/bright/lib/bright_web/live/platform_reset_password_live.ex delete mode 100644 services/bright/lib/bright_web/live/platform_settings_live.ex delete mode 100644 services/bright/priv/repo/migrations/20250109070300_create_platforms_auth_tables.exs delete mode 100644 services/bright/test/bright_web/live/platform_confirmation_instructions_live_test.exs delete mode 100644 services/bright/test/bright_web/live/platform_confirmation_live_test.exs delete mode 100644 services/bright/test/bright_web/live/platform_forgot_password_live_test.exs delete mode 100644 services/bright/test/bright_web/live/platform_login_live_test.exs delete mode 100644 services/bright/test/bright_web/live/platform_registration_live_test.exs delete mode 100644 services/bright/test/bright_web/live/platform_reset_password_live_test.exs delete mode 100644 services/bright/test/bright_web/live/platform_settings_live_test.exs delete mode 100644 services/bright/test/bright_web/platform_auth_test.exs delete mode 100644 services/capture/.gitignore delete mode 100644 services/capture/.mocharc.json delete mode 100644 services/capture/README.md delete mode 100644 services/capture/package.json delete mode 100644 services/capture/pnpm-lock.yaml delete mode 100644 services/capture/src/Capture.js delete mode 100644 services/capture/src/Capture.spec.js delete mode 100644 services/capture/src/Ipfs.js delete mode 100644 services/capture/src/Record.spec.ts delete mode 100644 services/capture/src/Record.ts delete mode 100644 services/capture/src/RecordNextGeneration.spec.ts delete mode 100644 services/capture/src/RecordNextGeneration.ts delete mode 100644 services/capture/src/Video.js delete mode 100644 services/capture/src/Video.spec.js delete mode 100644 services/capture/src/Voddo.js delete mode 100644 services/capture/src/Voddo.spec.js delete mode 100644 services/capture/src/app.spec.ts delete mode 100644 services/capture/src/app.ts delete mode 100644 services/capture/src/blah.ts delete mode 100644 services/capture/src/cb.js delete mode 100644 services/capture/src/config.ts delete mode 100644 services/capture/src/disk.js delete mode 100644 services/capture/src/fastify-graphile-worker-plugin.ts delete mode 100644 services/capture/src/fastify-pgboss-plugin.ts delete mode 100644 services/capture/src/fixtures/just-a-text-file.txt delete mode 100644 services/capture/src/fixtures/mime.types delete mode 100644 services/capture/src/fixtures/mock-stream0.mp4 delete mode 100644 services/capture/src/fixtures/mock-stream1.mp4 delete mode 100644 services/capture/src/fixtures/mock-stream2.mp4 delete mode 100644 services/capture/src/fixtures/my.abc delete mode 100644 services/capture/src/index.ts delete mode 100644 services/capture/src/integration/Capture.spec.js delete mode 100644 services/capture/src/integration/Ipfs.spec.js delete mode 100644 services/capture/src/integration/Voddo.spec.js delete mode 100644 services/capture/src/integration/record.spec.js delete mode 100644 services/capture/src/integration/video.spec.js delete mode 100644 services/capture/src/logger.js delete mode 100644 services/capture/src/poc-lite.js delete mode 100644 services/capture/src/poc-s3-alt.ts delete mode 100644 services/capture/src/poc-s3.ts delete mode 100644 services/capture/src/poc.ts delete mode 100644 services/capture/src/record.js.old delete mode 100644 services/capture/src/s3.ts delete mode 100644 services/capture/src/sample.ts delete mode 100755 services/capture/src/server.ts.old delete mode 100644 services/capture/src/tasks/record.ts delete mode 100644 services/capture/tsconfig.json delete mode 100644 services/capture/tsup.config.ts delete mode 100644 services/factory/.mocharc.json delete mode 100644 services/factory/README.md delete mode 100644 services/factory/crontab delete mode 100644 services/factory/package.json delete mode 100644 services/factory/pnpm-lock.yaml delete mode 100644 services/factory/src/config.ts delete mode 100644 services/factory/src/index.ts delete mode 100644 services/factory/src/tasks/combine_video_segments.ts delete mode 100644 services/factory/src/tasks/generate_thumbnail.ts delete mode 100644 services/factory/src/tasks/process_video.ts delete mode 100644 services/factory/src/tasks/remux_video.ts delete mode 100644 services/factory/src/tasks/synchronize_patrons_list.ts delete mode 100644 services/factory/src/utils/importDirectory.ts delete mode 100644 services/factory/tsconfig.json delete mode 100644 services/mailbox/README.md delete mode 100644 services/mailbox/package.json delete mode 100644 services/mailbox/pnpm-lock.yaml delete mode 100644 services/mailbox/src/imap.ts delete mode 100644 services/mailbox/src/index.ts delete mode 100644 services/mailbox/src/parsers.spec.ts delete mode 100644 services/mailbox/src/parsers.ts delete mode 100644 services/mailbox/tsconfig.json delete mode 100644 services/mailbox/tsup.config.js delete mode 100644 services/migrations-data/README.md delete mode 100644 services/migrations-data/index.js delete mode 100644 services/migrations-data/migrations/0001_from-strapi-to-postgrest-mk2.sql delete mode 100644 services/migrations-data/migrations/0002_relate-vtubers-to-vods.sql delete mode 100644 services/migrations-data/migrations/0003_relate-mux_asset_id-to-vods.sql delete mode 100644 services/migrations-data/migrations/0004_relate-thumbnails-to-vods.sql delete mode 100644 services/migrations-data/migrations/0005_rename-vtuber_num-to-vtuber_id.sql delete mode 100644 services/migrations-data/migrations/0006_relate-vods-to-streams.sql delete mode 100644 services/migrations-data/migrations/README.md delete mode 100644 services/migrations-data/package.json delete mode 100644 services/migrations-data/pnpm-lock.yaml delete mode 100644 services/migrations-schema/README.md delete mode 100644 services/migrations-schema/index.js delete mode 100644 services/migrations-schema/migrations/00001_create.sql delete mode 100644 services/migrations-schema/migrations/00002_add-records-table.sql delete mode 100644 services/migrations-schema/migrations/00003_create-graphile-worker-schema.sql delete mode 100644 services/migrations-schema/migrations/00004_create-trigger-function.sql delete mode 100644 services/migrations-schema/migrations/00005_add-trigger-for-record-update.sql delete mode 100644 services/migrations-schema/migrations/00006_add-updated-at-to-records.sql delete mode 100644 services/migrations-schema/migrations/00007_add-default-records-timestamps.sql delete mode 100644 services/migrations-schema/migrations/00008_add-default-records-timestamp.sql delete mode 100644 services/migrations-schema/migrations/00009_add-streams-vods-vtubers.sql delete mode 100644 services/migrations-schema/migrations/00010_record-segments.sql delete mode 100644 services/migrations-schema/migrations/00011_use-composite-primary-keys.sql delete mode 100644 services/migrations-schema/migrations/00012_order-records_segments.sql delete mode 100644 services/migrations-schema/migrations/00013_rename-segments-order.sql delete mode 100644 services/migrations-schema/migrations/00014_create-segments-stream-links.sql delete mode 100644 services/migrations-schema/migrations/00015_create-segments-stream-links-2.sql delete mode 100644 services/migrations-schema/migrations/00016_remove-unecessary-columns.sql delete mode 100644 services/migrations-schema/migrations/00017_add-stream-status-col.sql delete mode 100644 services/migrations-schema/migrations/00018_add-stream-status-default.sql delete mode 100644 services/migrations-schema/migrations/00019_drop-discord-interactions.sql delete mode 100644 services/migrations-schema/migrations/00020_add-streams-update-trigger.sql delete mode 100644 services/migrations-schema/migrations/00021_add-foreign-key-to-segments_stream.sql delete mode 100644 services/migrations-schema/migrations/00022_add-permissions-for-segments_stream_links.sql delete mode 100644 services/migrations-schema/migrations/00023_drop-capture_job_id-column.sql delete mode 100644 services/migrations-schema/migrations/00024_add-updated_at-for-segments.sql delete mode 100644 services/migrations-schema/migrations/00025_add-is_recording_aborted-to-streams.sql delete mode 100644 services/migrations-schema/migrations/00026_use-moddatetime.sql delete mode 100644 services/migrations-schema/migrations/00027_create-triggers-for-moddatetime.sql delete mode 100644 services/migrations-schema/migrations/00028_remove-moddate-on-insert.sql delete mode 100644 services/migrations-schema/migrations/00029_add-discord-message-id.sql delete mode 100644 services/migrations-schema/migrations/00030_update-update_discord_message.sql delete mode 100644 services/migrations-schema/migrations/00031_recreate-update_stream-trigger.sql delete mode 100644 services/migrations-schema/migrations/00032_update-stream-when-segment-updates.sql delete mode 100644 services/migrations-schema/migrations/00033_create-vods.sql delete mode 100644 services/migrations-schema/migrations/00034_move-segments-to-vods.sql delete mode 100644 services/migrations-schema/migrations/00035_seed-vtubers.sql delete mode 100644 services/migrations-schema/migrations/00036_update_discord_message-use-vod_id.sql delete mode 100644 services/migrations-schema/migrations/00037_drop-oudated-streamscolumns.sql delete mode 100644 services/migrations-schema/migrations/00038_add-vods-update-trigger.sql delete mode 100644 services/migrations-schema/migrations/00039_create-limiter-table.sql delete mode 100644 services/migrations-schema/migrations/00040_theme_color-is-optional.sql delete mode 100644 services/migrations-schema/migrations/00041_add-is_recording_aborted-to-vods.sql delete mode 100644 services/migrations-schema/migrations/00042_segments-vods-fk.sql delete mode 100644 services/migrations-schema/migrations/00043_update_vod_on_segment_update.sql delete mode 100644 services/migrations-schema/migrations/00044_segments-uuid.sql delete mode 100644 services/migrations-schema/migrations/00045_recreate-segments_vod_links.sql delete mode 100644 services/migrations-schema/migrations/00046_segments_vod_links-many-to-one.sql delete mode 100644 services/migrations-schema/migrations/00047_add-is_recording_aborted-to-vods-default-false.sql delete mode 100644 services/migrations-schema/migrations/00048_segments-vod-links-perms.sql delete mode 100644 services/migrations-schema/migrations/00049_add-discord_message_id-on-vods.sql delete mode 100644 services/migrations-schema/migrations/00050_add-status-to-vod.sql delete mode 100644 services/migrations-schema/migrations/00051_set-vod-status.sql delete mode 100644 services/migrations-schema/migrations/00052_correct-typo-filesize-to-bytes.sql delete mode 100644 services/migrations-schema/migrations/00053_add-default-status-on-vods.sql delete mode 100644 services/migrations-schema/migrations/00054_remove-bytes-from-vod-tg-function.sql delete mode 100644 services/migrations-schema/migrations/00055_trigger_update_vod-switch-to-before.sql delete mode 100644 services/migrations-schema/migrations/00056_add_job-retry-only-6-times.sql delete mode 100644 services/migrations-schema/migrations/00057_recreate-vod_create.sql delete mode 100644 services/migrations-schema/migrations/00058_conditionally-change-vod-status.sql delete mode 100644 services/migrations-schema/migrations/00059_fix-filesize-typo.sql delete mode 100644 services/migrations-schema/migrations/00060_create-recordings-table.sql delete mode 100644 services/migrations-schema/migrations/00061_defaults-for-created_at-and-updated_at.sql delete mode 100644 services/migrations-schema/migrations/00062_create-discord_interactions.sql delete mode 100644 services/migrations-schema/migrations/00063_relate-discord_interactions-with-recordings.sql delete mode 100644 services/migrations-schema/migrations/00064_execute-procedure-instead-of-function.sql delete mode 100644 services/migrations-schema/migrations/00065_move-is_aborted-to-recordings.sql delete mode 100644 services/migrations-schema/migrations/00066_add-segments-fk-to-vods.sql delete mode 100644 services/migrations-schema/migrations/00067_remove-duplicate-vods-segments.sql delete mode 100644 services/migrations-schema/migrations/00068_remove-vod_id-from-recordings.sql delete mode 100644 services/migrations-schema/migrations/00069_recreate-recordings-vod-relationship.sql delete mode 100644 services/migrations-schema/migrations/00070_relate-vod-to-recording.sql delete mode 100644 services/migrations-schema/migrations/00071_remove-recordings_vod_id_fkey.sql delete mode 100644 services/migrations-schema/migrations/00072_create-builds-table.sql delete mode 100644 services/migrations-schema/migrations/00073_update-builds-table-timestamps.sql delete mode 100644 services/migrations-schema/migrations/00074_switch-after-update-to-after-insert.sql delete mode 100644 services/migrations-schema/migrations/00075_add-checksum-to-segments.sql delete mode 100644 services/migrations-schema/migrations/00076_builds-use-vod_id-instead-of-vod.sql delete mode 100644 services/migrations-schema/migrations/00077_builds-trigger-change-vod-to-vod_id.sql delete mode 100644 services/migrations-schema/migrations/00078_add-bytes_uploaded-to-segments.sql delete mode 100644 services/migrations-schema/migrations/00079_stream_id-optional.sql delete mode 100644 services/migrations-schema/migrations/00080_add-created_at-to-vtubers.sql delete mode 100644 services/migrations-schema/migrations/00081_add-id_deprecated-to-vods.sql delete mode 100644 services/migrations-schema/migrations/00082_add-id_deprecated-to-s3_files.sql delete mode 100644 services/migrations-schema/migrations/00083_create_vods_s3_join.sql delete mode 100644 services/migrations-schema/migrations/00084_add-file_id-to-s3_files.sql delete mode 100644 services/migrations-schema/migrations/00085_sync-s3_files.sql delete mode 100644 services/migrations-schema/migrations/00086_add-cdn_url-to-s3_files.sql delete mode 100644 services/migrations-schema/migrations/00087_remove-id_deprecated.sql delete mode 100644 services/migrations-schema/migrations/00088_remove-id_deprecated-from-vtubers.sql delete mode 100644 services/migrations-schema/migrations/00089_create-vods_s3_files_join.sql delete mode 100644 services/migrations-schema/migrations/00090_rename-vods-s3-files-joins.sql delete mode 100644 services/migrations-schema/migrations/00091_drop-vods_s3_file_join.sql delete mode 100644 services/migrations-schema/migrations/00092_use-s3_file_id.sql delete mode 100644 services/migrations-schema/migrations/00093_enable-pg_trgm.sql delete mode 100644 services/migrations-schema/migrations/00094_add-patrons-table.sql delete mode 100644 services/migrations-schema/migrations/00095_create-contributors.sql delete mode 100644 services/migrations-schema/migrations/00096_add-id_num-to-s3_files.sql delete mode 100644 services/migrations-schema/migrations/00097_add-id_num-to-mux-assets.sql delete mode 100644 services/migrations-schema/migrations/00098_add-id_num-to-streams.sql delete mode 100644 services/migrations-schema/migrations/00099_add-vtuber_num.sql delete mode 100644 services/migrations-schema/migrations/00100_add-id_num-to-tags.sql delete mode 100644 services/migrations-schema/migrations/00101_add-missing-cols-to-tags.sql delete mode 100644 services/migrations-schema/migrations/00102_add-toy_id_num-to-tags.sql delete mode 100644 services/migrations-schema/migrations/00103_add-toy_num.sql delete mode 100644 services/migrations-schema/migrations/00104_add-toy_num-fix.sql delete mode 100644 services/migrations-schema/migrations/00105_add-id_num-to-vods_links.sql delete mode 100644 services/migrations-schema/migrations/00106_add-id_num-to-vods.sql delete mode 100644 services/migrations-schema/migrations/00107_remove-uuid.sql delete mode 100644 services/migrations-schema/migrations/00108_add-strapi-tables.sql delete mode 100644 services/migrations-schema/migrations/00109_use-consistent-description_1.sql delete mode 100644 services/migrations-schema/migrations/00110_unseed-vtubers.sql delete mode 100644 services/migrations-schema/migrations/00111_create-strapi-tables-tvr.sql delete mode 100644 services/migrations-schema/migrations/00112_create-tvr_tag_links.sql delete mode 100644 services/migrations-schema/migrations/00113_create-tvr_vod_links.sql delete mode 100644 services/migrations-schema/migrations/00114_create-tags_toy_links.sql delete mode 100644 services/migrations-schema/migrations/00115_create-tags_vods_links.sql delete mode 100644 services/migrations-schema/migrations/00116_create-timestamps.sql delete mode 100644 services/migrations-schema/migrations/00117_create-timestamps_tag_links.sql delete mode 100644 services/migrations-schema/migrations/00118_create-timestamps_vod_links.sql delete mode 100644 services/migrations-schema/migrations/00119_create-toys_link_tag_links.sql delete mode 100644 services/migrations-schema/migrations/00120_create-vods_mux_asset_links.sql delete mode 100644 services/migrations-schema/migrations/00121_create-vods_stream_links.sql delete mode 100644 services/migrations-schema/migrations/00122_create-vods_thumbnail_links.sql delete mode 100644 services/migrations-schema/migrations/00123_create-vods_uploader_links.sql delete mode 100644 services/migrations-schema/migrations/00124_create-vods_video_src_b_2_links.sql delete mode 100644 services/migrations-schema/migrations/00125_create-vods_vtuber_links.sql delete mode 100644 services/migrations-schema/migrations/00126_create-vtubers_toy_links.sql delete mode 100644 services/migrations-schema/migrations/00127_grant-permissions-for-vods_vtuber_links.sql delete mode 100644 services/migrations-schema/migrations/00128_add-vtuber-fk-to-vods.sql delete mode 100644 services/migrations-schema/migrations/00129_add-vtuber_id-to-vods.sql delete mode 100644 services/migrations-schema/migrations/00130_add-s3_files-fk-to-vods.sql delete mode 100644 services/migrations-schema/migrations/00131_grant-perms-on-s3_files.sql delete mode 100644 services/migrations-schema/migrations/00132_rename-vtuber_num-to-vtuber_id.sql delete mode 100644 services/migrations-schema/migrations/00133_add-mux_assets-fk-to-vods.sql delete mode 100644 services/migrations-schema/migrations/00134_add-thumbnail-fk-to-vods.sql delete mode 100644 services/migrations-schema/migrations/00135_add-b2_files-permissions.sql delete mode 100644 services/migrations-schema/migrations/00136_add-uuid-to-streams.sql delete mode 100644 services/migrations-schema/migrations/00137_add-vods-fk-to-streams.sql delete mode 100644 services/migrations-schema/migrations/00138_add-date_2-to-streams-and-vods.sql delete mode 100644 services/migrations-schema/migrations/00139_add-timestamps-permissions.sql delete mode 100644 services/migrations-schema/migrations/00140_add-permissions-for-links.sql delete mode 100644 services/migrations-schema/migrations/00141_add-permissions-timestamps_vod_links.sql delete mode 100644 services/migrations-schema/migrations/00142_add-permissions-for-tag_vod_relations.sql delete mode 100644 services/migrations-schema/migrations/00143_add-permissions-for-tag_vod_relations_vod_links.sql delete mode 100644 services/migrations-schema/migrations/00144_add-permissions-for-tags_vods_links.sql delete mode 100644 services/migrations-schema/package.json delete mode 100644 services/migrations-schema/pnpm-lock.yaml delete mode 100644 services/next/.dockerignore delete mode 100644 services/next/.eslintrc.json delete mode 100644 services/next/.gitignore delete mode 100644 services/next/.nvmrc delete mode 100644 services/next/CHECKS delete mode 100644 services/next/LICENSE delete mode 100644 services/next/README.md delete mode 100644 services/next/app.json delete mode 100644 services/next/app/404.tsx delete mode 100644 services/next/app/about/page.tsx delete mode 100644 services/next/app/api/auth/[...nextauth]/route.ts delete mode 100644 services/next/app/api/blogs/route.ts delete mode 100644 services/next/app/api/page.tsx delete mode 100644 services/next/app/api/patreon/currently-entitled-tiers/route.ts delete mode 100644 services/next/app/api/patreon/session/route.ts delete mode 100644 services/next/app/api/profile/route.ts delete mode 100644 services/next/app/api/revalidate/route.ts delete mode 100644 services/next/app/api/service.json/route.ts delete mode 100644 services/next/app/api/session/route.ts delete mode 100644 services/next/app/api/user/route.ts delete mode 100644 services/next/app/api/v1.json/route.ts delete mode 100644 services/next/app/archive/[uuid]/not-found.tsx delete mode 100644 services/next/app/archive/[uuid]/page.tsx delete mode 100644 services/next/app/archive/page.tsx delete mode 100644 services/next/app/components/access-denied-screen.tsx delete mode 100644 services/next/app/components/archive-progress.tsx delete mode 100644 services/next/app/components/auth-buttons.tsx delete mode 100644 services/next/app/components/auth-provider.tsx delete mode 100644 services/next/app/components/contributors.tsx delete mode 100644 services/next/app/components/error-card.tsx delete mode 100644 services/next/app/components/footer.tsx delete mode 100644 services/next/app/components/funding-goal.tsx delete mode 100644 services/next/app/components/icons/carrd.tsx delete mode 100644 services/next/app/components/icons/chaturbate.tsx delete mode 100644 services/next/app/components/icons/fansly.tsx delete mode 100644 services/next/app/components/icons/linktree.tsx delete mode 100644 services/next/app/components/icons/onlyfans.tsx delete mode 100644 services/next/app/components/icons/pornhub.tsx delete mode 100644 services/next/app/components/icons/throne.tsx delete mode 100644 services/next/app/components/ipfs-cid.tsx delete mode 100644 services/next/app/components/ipfs-logo.tsx delete mode 100644 services/next/app/components/ipfs.tsx delete mode 100644 services/next/app/components/linkable-heading.tsx delete mode 100644 services/next/app/components/localized-date.tsx delete mode 100644 services/next/app/components/navbar.tsx delete mode 100644 services/next/app/components/notification-center.tsx delete mode 100644 services/next/app/components/notifications.tsx delete mode 100644 services/next/app/components/pager.tsx delete mode 100644 services/next/app/components/patron-perks.tsx delete mode 100644 services/next/app/components/patrons-list.tsx delete mode 100644 services/next/app/components/permissions-table.tsx delete mode 100644 services/next/app/components/protected-route.tsx delete mode 100644 services/next/app/components/query-provider.tsx delete mode 100644 services/next/app/components/sortable-tags.tsx delete mode 100644 services/next/app/components/spinner.tsx delete mode 100644 services/next/app/components/stream-button.tsx delete mode 100644 services/next/app/components/stream-page.tsx delete mode 100644 services/next/app/components/stream.tsx delete mode 100644 services/next/app/components/streams-list.tsx delete mode 100644 services/next/app/components/streams-table.tsx delete mode 100644 services/next/app/components/tag-button.tsx delete mode 100644 services/next/app/components/tag.tsx delete mode 100644 services/next/app/components/tagger.tsx delete mode 100644 services/next/app/components/thumbnail.tsx delete mode 100644 services/next/app/components/timestamps-list.tsx delete mode 100644 services/next/app/components/toys.tsx delete mode 100644 services/next/app/components/upload-form.tsx delete mode 100644 services/next/app/components/user-controls.tsx delete mode 100644 services/next/app/components/user-metadata.tsx delete mode 100644 services/next/app/components/vibrate-test.tsx delete mode 100644 services/next/app/components/video-context.tsx delete mode 100644 services/next/app/components/video-interactive.tsx delete mode 100644 services/next/app/components/video-player.tsx delete mode 100644 services/next/app/components/video-source-selector.tsx delete mode 100644 services/next/app/components/vod-card.tsx delete mode 100644 services/next/app/components/vod-nav.tsx delete mode 100644 services/next/app/components/vod-page-2.tsx delete mode 100644 services/next/app/components/vod-page.tsx delete mode 100644 services/next/app/components/vods-list.tsx delete mode 100644 services/next/app/components/vtuber-button.tsx delete mode 100644 services/next/app/components/vtuber-card.tsx delete mode 100644 services/next/app/config/clientConfigs.ts delete mode 100644 services/next/app/config/configs.ts delete mode 100644 services/next/app/config/server.ts delete mode 100644 services/next/app/connect/patreon/redirect/page.tsx delete mode 100644 services/next/app/faq/page.tsx delete mode 100644 services/next/app/favicon.ico delete mode 100644 services/next/app/feed/feed.json/route.ts delete mode 100644 services/next/app/feed/feed.xml/route.ts delete mode 100644 services/next/app/feed/page.tsx delete mode 100644 services/next/app/feed/rss.xml/route.ts delete mode 100644 services/next/app/goals/page.tsx delete mode 100644 services/next/app/health/page.tsx delete mode 100644 services/next/app/latest-vods/[page]/page.tsx delete mode 100644 services/next/app/latest-vods/page.tsx delete mode 100644 services/next/app/layout.tsx delete mode 100644 services/next/app/lib/auth.ts delete mode 100644 services/next/app/lib/b2File.ts delete mode 100644 services/next/app/lib/constants.ts delete mode 100644 services/next/app/lib/contributors.ts delete mode 100644 services/next/app/lib/dates.ts delete mode 100644 services/next/app/lib/fetch-api.ts delete mode 100644 services/next/app/lib/fetchers.ts delete mode 100644 services/next/app/lib/ipfs.ts delete mode 100644 services/next/app/lib/keycloak.ts delete mode 100644 services/next/app/lib/patreon.ts delete mode 100644 services/next/app/lib/pm.ts delete mode 100644 services/next/app/lib/retry.ts delete mode 100644 services/next/app/lib/rss.ts delete mode 100644 services/next/app/lib/shareRef.ts delete mode 100644 services/next/app/lib/streams.ts delete mode 100644 services/next/app/lib/tag-vod-relations.ts delete mode 100644 services/next/app/lib/tags.ts delete mode 100644 services/next/app/lib/timestamps.ts delete mode 100644 services/next/app/lib/toys.ts delete mode 100644 services/next/app/lib/tweets.ts delete mode 100644 services/next/app/lib/useForwardRef.ts delete mode 100644 services/next/app/lib/users.ts delete mode 100644 services/next/app/lib/vods.ts delete mode 100644 services/next/app/lib/vtubers.ts delete mode 100644 services/next/app/page.tsx delete mode 100644 services/next/app/patrons/page.tsx delete mode 100644 services/next/app/profile/hooks/patreonCurrentlyEntitledTiers.ts delete mode 100644 services/next/app/profile/hooks/updateProfile.ts delete mode 100644 services/next/app/profile/hooks/useMetadata.ts delete mode 100644 services/next/app/profile/page.tsx delete mode 100644 services/next/app/sign-in.tsx delete mode 100644 services/next/app/sign-out.tsx delete mode 100644 services/next/app/tags/[slug]/page.tsx delete mode 100644 services/next/app/tags/page.tsx delete mode 100644 services/next/app/upload/page.tsx delete mode 100644 services/next/app/uppy.tsx delete mode 100644 services/next/app/vods/[safeDateOrUUID]/page.tsx delete mode 100644 services/next/app/vods/page.tsx delete mode 100644 services/next/app/vt/[slug]/not-found.tsx delete mode 100644 services/next/app/vt/[slug]/page.tsx delete mode 100644 services/next/app/vt/[slug]/stream/[safeDate]/page.tsx delete mode 100644 services/next/app/vt/[slug]/streams/page.tsx delete mode 100644 services/next/app/vt/[slug]/toys/[page]/page.tsx delete mode 100644 services/next/app/vt/[slug]/toys/page.tsx delete mode 100644 services/next/app/vt/[slug]/vod/[safeDateOrUUID]/page.tsx delete mode 100644 services/next/app/vt/[slug]/vod/page.tsx delete mode 100644 services/next/app/vt/[slug]/vods/[page]/page.tsx delete mode 100644 services/next/app/vt/[slug]/vods/page.tsx delete mode 100644 services/next/app/vt/page.tsx delete mode 100644 services/next/assets/styles/cid.module.css delete mode 100644 services/next/assets/styles/fp.module.css delete mode 100644 services/next/assets/styles/global.css delete mode 100644 services/next/assets/styles/icon.module.css delete mode 100644 services/next/assets/styles/player.module.css delete mode 100644 services/next/assets/svg/README.md delete mode 100644 services/next/assets/svg/carrd.svg delete mode 100644 services/next/assets/svg/chaturbate.svg delete mode 100644 services/next/assets/svg/checkmark.svg delete mode 100644 services/next/assets/svg/fansly.tsx delete mode 100644 services/next/assets/svg/ipfs.svg delete mode 100644 services/next/assets/svg/linktree.svg delete mode 100644 services/next/assets/svg/noun-adult-content-1731184.svg delete mode 100644 services/next/assets/svg/noun-anime-3890912.svg delete mode 100644 services/next/assets/svg/noun-avatar-3546974.svg delete mode 100644 services/next/assets/svg/noun-girl-842331.svg delete mode 100644 services/next/assets/svg/noun-network-1603820.svg delete mode 100644 services/next/assets/svg/onlyfans.svg delete mode 100644 services/next/assets/svg/pornhub.svg delete mode 100644 services/next/assets/svg/throne.svg delete mode 100644 services/next/next.config.js delete mode 100644 services/next/package.json delete mode 100644 services/next/pnpm-lock.yaml delete mode 100644 services/next/public/fonts/nunito-v26-latin-regular.woff2 delete mode 100644 services/next/public/fonts/overpass-v13-latin-regular.woff2 delete mode 100644 services/next/public/futureporn-icon.png delete mode 100644 services/next/public/images/.keep delete mode 100644 services/next/public/images/cj_clippy.jpg delete mode 100644 services/next/public/images/default-thumbnail.webp delete mode 100644 services/next/public/images/projekt-melody.jpg delete mode 100644 services/next/public/images/projektmelody-thumbnail.webp delete mode 100644 services/next/public/images/vercel.svg delete mode 100644 services/next/test-jwe-decrypt.js delete mode 100644 services/next/tsconfig.json delete mode 100644 services/next/tsconfig.json.old delete mode 100644 services/scout/.dockerignore delete mode 100644 services/scout/.gitignore delete mode 100644 services/scout/.mocharc.json delete mode 100644 services/scout/README.md delete mode 100644 services/scout/node.d.ts delete mode 100644 services/scout/package.json delete mode 100644 services/scout/pnpm-lock.yaml delete mode 100644 services/scout/src/api.yaml delete mode 100644 services/scout/src/cb.spec.ts delete mode 100644 services/scout/src/cb.ts delete mode 100644 services/scout/src/config.ts delete mode 100644 services/scout/src/css/SwaggerDark.css delete mode 100644 services/scout/src/fansly.spec.ts delete mode 100644 services/scout/src/fansly.ts delete mode 100644 services/scout/src/fastify.ts delete mode 100644 services/scout/src/fixtures/.gitignore delete mode 100644 services/scout/src/fixtures/cb.projektmelody.fixture.html delete mode 100644 services/scout/src/hello.ts delete mode 100644 services/scout/src/icons.ts delete mode 100644 services/scout/src/index.browser.js delete mode 100644 services/scout/src/index.ts delete mode 100644 services/scout/src/loader.js delete mode 100644 services/scout/src/schemas.ts delete mode 100644 services/scout/src/scrapeVtuberData.spec.ts delete mode 100644 services/scout/src/scrapeVtuberData.ts delete mode 100644 services/scout/src/scrapingFetch.spec.ts delete mode 100644 services/scout/src/scrapingFetch.ts delete mode 100644 services/scout/src/signals.js delete mode 100644 services/scout/src/spawnWrapper.spec.ts delete mode 100644 services/scout/src/spawnWrapper.ts delete mode 100644 services/scout/src/twitter.js delete mode 100644 services/scout/src/twitter.spec.js delete mode 100644 services/scout/src/ua.ts delete mode 100644 services/scout/src/vtuber.spec.js delete mode 100644 services/scout/src/vtuber.ts delete mode 100644 services/scout/src/ytdlp.spec.ts delete mode 100644 services/scout/src/ytdlp.ts delete mode 100644 services/scout/tsconfig.json delete mode 100644 services/scout/tsup.config.ts delete mode 100644 services/strapi/.editorconfig delete mode 100644 services/strapi/.eslintignore delete mode 100644 services/strapi/.eslintrc delete mode 100644 services/strapi/.gitignore delete mode 100644 services/strapi/README.md delete mode 100644 services/strapi/config/admin.js delete mode 100644 services/strapi/config/api.js delete mode 100644 services/strapi/config/database.js delete mode 100644 services/strapi/config/middlewares.js delete mode 100644 services/strapi/config/plugins.js delete mode 100644 services/strapi/config/server.js delete mode 100644 services/strapi/database/daily-backup.sh delete mode 100755 services/strapi/database/devDb.sh delete mode 100644 services/strapi/database/migrations/.gitkeep delete mode 100644 services/strapi/database/migrations/2023-08-01-relate-vods-to-vtubers-part2.js delete mode 100644 services/strapi/database/migrations/2023-08-17-reformat-cdnUrl.js delete mode 100644 services/strapi/database/migrations/2023-08-20-strip-query-string-from-cid.js delete mode 100644 services/strapi/database/migrations/2023-08-30-remove-cloudinary.js delete mode 100644 services/strapi/database/migrations/2023-08-30-toy-image-field-simplify.js delete mode 100644 services/strapi/database/migrations/2023-09-08-change-date-to-string.js delete mode 100644 services/strapi/database/migrations/2023-09-08-drop-toys-image.js delete mode 100644 services/strapi/database/migrations/2023-09-08-drop-vod-videosrc.js delete mode 100644 services/strapi/database/migrations/2023-12-24-add-cuid-to-vods.js delete mode 100644 services/strapi/database/migrations/2023-12-26-add-cuid-to-streams.js delete mode 100644 services/strapi/database/migrations/2023-12-27-relate-vods-to-streams.js delete mode 100644 services/strapi/database/migrations/2023.05.09-video-src-sanity.js.noexec delete mode 100644 services/strapi/database/migrations/2023.05.11T12.32.00.convert-to-video-src-b2.js.noexec delete mode 100644 services/strapi/database/migrations/2023.05.14T00.42.00.000Z.migrate-tags-to-tag-vod-relations.js delete mode 100644 services/strapi/database/migrations/2023.05.15T02.44.00.000Z.drop-vod-tags.js delete mode 100644 services/strapi/database/migrations/2023.05.25-gimme-the-tags.js.noexec delete mode 100644 services/strapi/database/migrations/2023.05.25T20.44.00.000Z.get-the-og-tags.js delete mode 100644 services/strapi/database/migrations/2023.07.17.relate-vods-to-vtubers.js delete mode 100644 services/strapi/database/migrations/2023.07.31.add-b2-file-cdnUrl.js delete mode 100644 services/strapi/database/migrations/2024-01-08-add-streams.js.noexec delete mode 100644 services/strapi/database/migrations/2024-01-14-add-date2-to-streams.js delete mode 100644 services/strapi/database/migrations/2024-01-15-add-platform-to-streams.js delete mode 100644 services/strapi/database/migrations/2024-07-04-add-fansly-id-to-vtuber.js delete mode 100644 services/strapi/database/og-tags.json delete mode 100644 services/strapi/favicon.png delete mode 100644 services/strapi/jsconfig.json delete mode 100644 services/strapi/misc/generateCuid.js delete mode 100644 services/strapi/package.json delete mode 100644 services/strapi/pnpm-lock.yaml delete mode 100644 services/strapi/public/robots.txt delete mode 100644 services/strapi/public/uploads/.gitkeep delete mode 100644 services/strapi/src/admin/app.example.js delete mode 100644 services/strapi/src/admin/webpack.config.example.js delete mode 100644 services/strapi/src/api/.gitkeep delete mode 100644 services/strapi/src/api/b2-file/content-types/b2-file/schema.json delete mode 100644 services/strapi/src/api/b2-file/controllers/b2-file.js delete mode 100644 services/strapi/src/api/b2-file/routes/b2-file.js delete mode 100644 services/strapi/src/api/b2-file/services/b2-file.js delete mode 100644 services/strapi/src/api/contributor/content-types/contributor/schema.json delete mode 100644 services/strapi/src/api/contributor/controllers/contributor.js delete mode 100644 services/strapi/src/api/contributor/routes/contributor.js delete mode 100644 services/strapi/src/api/contributor/services/contributor.js delete mode 100644 services/strapi/src/api/discord-interaction/content-types/discord-interaction/schema.json delete mode 100644 services/strapi/src/api/discord-interaction/controllers/discord-interaction.js delete mode 100644 services/strapi/src/api/discord-interaction/routes/discord-interaction.js delete mode 100644 services/strapi/src/api/discord-interaction/services/discord-interaction.js delete mode 100644 services/strapi/src/api/goal/content-types/goal/schema.json delete mode 100644 services/strapi/src/api/goal/controllers/goal.js delete mode 100644 services/strapi/src/api/goal/routes/goal.js delete mode 100644 services/strapi/src/api/goal/services/goal.js delete mode 100644 services/strapi/src/api/gogs/content-types/gogs/schema.json delete mode 100644 services/strapi/src/api/gogs/controllers/gogs.js delete mode 100644 services/strapi/src/api/gogs/routes/gogs.js delete mode 100644 services/strapi/src/api/gogs/services/gogs.js delete mode 100644 services/strapi/src/api/issue/content-types/issue/schema.json delete mode 100644 services/strapi/src/api/issue/controllers/issue.js delete mode 100644 services/strapi/src/api/issue/routes/issue.js delete mode 100644 services/strapi/src/api/issue/services/issue.js delete mode 100644 services/strapi/src/api/mux-asset/content-types/mux-asset/schema.json delete mode 100644 services/strapi/src/api/mux-asset/controllers/mux-asset.js delete mode 100644 services/strapi/src/api/mux-asset/routes/mux-asset.js delete mode 100644 services/strapi/src/api/mux-asset/services/mux-asset.js delete mode 100644 services/strapi/src/api/patreon/content-types/patreon/schema.json delete mode 100644 services/strapi/src/api/patreon/controllers/patreon.js delete mode 100644 services/strapi/src/api/patreon/routes/patreon.js delete mode 100644 services/strapi/src/api/patreon/services/patreon.js delete mode 100644 services/strapi/src/api/platform-notification/content-types/platform-notification/schema.json delete mode 100644 services/strapi/src/api/platform-notification/controllers/platform-notification.js delete mode 100644 services/strapi/src/api/platform-notification/routes/platform-notification.js delete mode 100644 services/strapi/src/api/platform-notification/services/platform-notification.js delete mode 100644 services/strapi/src/api/profile/controllers/profile.js delete mode 100644 services/strapi/src/api/profile/routes/profile.js delete mode 100644 services/strapi/src/api/profile/services/profile.js delete mode 100644 services/strapi/src/api/stream/content-types/stream/lifecycles.js delete mode 100644 services/strapi/src/api/stream/content-types/stream/schema.json delete mode 100644 services/strapi/src/api/stream/controllers/stream.js delete mode 100644 services/strapi/src/api/stream/routes/stream.js delete mode 100644 services/strapi/src/api/stream/services/stream.js delete mode 100644 services/strapi/src/api/tag-vod-relation/content-types/tag-vod-relation/schema.json delete mode 100644 services/strapi/src/api/tag-vod-relation/controllers/tag-vod-relation.js delete mode 100644 services/strapi/src/api/tag-vod-relation/routes/tag-vod-relation.js delete mode 100644 services/strapi/src/api/tag-vod-relation/services/tag-vod-relation.js delete mode 100644 services/strapi/src/api/tag/content-types/tag/schema.json delete mode 100644 services/strapi/src/api/tag/controllers/tag.js delete mode 100644 services/strapi/src/api/tag/routes/tag.js delete mode 100644 services/strapi/src/api/tag/services/tag.js delete mode 100644 services/strapi/src/api/timestamp/content-types/timestamp/schema.json delete mode 100644 services/strapi/src/api/timestamp/controllers/timestamp.js delete mode 100644 services/strapi/src/api/timestamp/routes/timestamp.js delete mode 100644 services/strapi/src/api/timestamp/services/timestamp.js delete mode 100644 services/strapi/src/api/toy/content-types/toy/schema.json delete mode 100644 services/strapi/src/api/toy/controllers/toy.js delete mode 100644 services/strapi/src/api/toy/routes/toy.js delete mode 100644 services/strapi/src/api/toy/services/toy.js delete mode 100644 services/strapi/src/api/tweet/content-types/tweet/lifecycles.js delete mode 100644 services/strapi/src/api/tweet/content-types/tweet/schema.json delete mode 100644 services/strapi/src/api/tweet/controllers/tweet.js delete mode 100644 services/strapi/src/api/tweet/routes/tweet.js delete mode 100644 services/strapi/src/api/tweet/services/tweet.js delete mode 100644 services/strapi/src/api/user-submitted-content/content-types/user-submitted-content/lifecycles.js delete mode 100644 services/strapi/src/api/user-submitted-content/content-types/user-submitted-content/schema.json delete mode 100644 services/strapi/src/api/user-submitted-content/controllers/user-submitted-content.js delete mode 100644 services/strapi/src/api/user-submitted-content/routes/user-submitted-content.js delete mode 100644 services/strapi/src/api/user-submitted-content/services/user-submitted-content.js delete mode 100644 services/strapi/src/api/vod/content-types/vod/lifecycles.js delete mode 100644 services/strapi/src/api/vod/content-types/vod/schema.json delete mode 100644 services/strapi/src/api/vod/controllers/vod.js delete mode 100644 services/strapi/src/api/vod/routes/vod.js delete mode 100644 services/strapi/src/api/vod/services/vod.js delete mode 100644 services/strapi/src/api/vtuber/content-types/vtuber/lifecycles.js delete mode 100644 services/strapi/src/api/vtuber/content-types/vtuber/schema.json delete mode 100644 services/strapi/src/api/vtuber/controllers/vtuber.js delete mode 100644 services/strapi/src/api/vtuber/routes/vtuber.js delete mode 100644 services/strapi/src/api/vtuber/services/vtuber.js delete mode 100644 services/strapi/src/extensions/.gitkeep delete mode 100644 services/strapi/src/extensions/README.md delete mode 100644 services/strapi/src/index.js delete mode 100644 services/strapi/src/policies/updateOwnerOnly.js delete mode 100644 services/uppy/.gitignore delete mode 100644 services/uppy/README.md delete mode 100755 services/uppy/apply-backblaze-cors-rules.sh delete mode 100644 services/uppy/index.js delete mode 100644 services/uppy/package.json delete mode 100644 services/uppy/pnpm-lock.yaml diff --git a/.dockerignore b/.dockerignore index 6c20d71..885ab8c 100644 --- a/.dockerignore +++ b/.dockerignore @@ -1,3 +1,5 @@ +backups + **/Dockerfile **/*.dockerfile **/.dockerignore diff --git a/.gitignore b/.gitignore index 7bbd68f..f212a25 100644 --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,4 @@ +backups .kamal/secrets* diff --git a/Makefile b/Makefile deleted file mode 100644 index 3b8b6f8..0000000 --- a/Makefile +++ /dev/null @@ -1,61 +0,0 @@ -ifeq ($(ENV),) -$(error ENV variable is not defined. Please set it to one of development|staging|production) -endif - -.PHONY: flux - -namespaces: - ./scripts/k8s-namespaces.sh - -secrets: - dotenvx run -f .env.${ENV} -- ./scripts/k8s-secrets.sh - -flux: - ./scripts/flux-bootstrap.sh - -cluster: echoenv kind namespaces gateway secrets velero flux metrics chisel - -gateway: - kubectl apply -f https://github.com/kubernetes-sigs/gateway-api/releases/download/v1.2.0/standard-install.yaml - -metrics: - ./scripts/k8s-metrics.sh - -echoenv: - echo "[echoenv] Using ${ENV} environment. If this is not what you want, export ENV=development|staging|production" - -scrap: namespaces secrets velero flux - -velero: - ./scripts/velero-create.sh - -tilt: - kind get kubeconfig > ~/.kube/futureporn.yaml - KUBECONFIG=~/.kube/futureporn.yaml tilt up -f ./Tiltfile - -cert-manager: - kubectl apply -f https://github.com/cert-manager/cert-manager/releases/download/v1.15.1/cert-manager.yaml - -exoscale: - kubectl apply -f https://raw.githubusercontent.com/exoscale/cert-manager-webhook-exoscale/master/deploy/exoscale-webhook-kustomize/deploy.yaml - -kind: - ./scripts/kind-with-local-registry.sh - -kindload: - ./scripts/kind-load.sh - -clean: - kind delete cluster - dotenvx run -f .env.${ENV} -- node ./packages/infra/vultr-delete-orphaned-resources.js - -deps: - echo "use `devbox install`" - -# A gitea act runner which runs locally -# https://docs.gitea.com/next/usage/actions/overview -# this doesnt work because of missing docker in docker -# I'm running this using systemd instead -#runner: -# docker run -d --rm -e GITEA_INSTANCE_URL=https://gitea.futureporn.net -e GITEA_RUNNER_REGISTRATION_TOKEN=${GITEA_RUNNER_REGISTRATION_TOKEN} -v /var/run/docker.sock:/var/run/docker.sock -v $$HOME/.local/share/applications/fp/act-runner-data:/data --privileged --name fp-gitea-act-runner gitea/act_runner - diff --git a/README.md b/README.md index fcf6396..b1175b9 100644 --- a/README.md +++ b/README.md @@ -17,15 +17,22 @@ The main gist is as follows devbox install -3. Start a local KIND Kubernetes cluster +3. - make cluster +## backup/restore dev database -4. Start Tilt +@see https://stackoverflow.com/a/29913462/1004931 - make tilt +### backup + +Use devbox helper script + + devbox run backup + +### restore + + cat ./backups/your-backup.sql | docker exec -i postgres_db psql -U postgres -Tilt will manage the KIND cluster, downloading necessary docker containers and building the containers listed in the fp helm chart at ./Charts/fp. Making changes to these charts or the application code will update or re-build the images as necessary. ## Metrics Notes diff --git a/Tiltfile b/Tiltfile deleted file mode 100644 index 22df057..0000000 --- a/Tiltfile +++ /dev/null @@ -1,646 +0,0 @@ -## Tiltfile for working with Futureporn cluster - -## remote development settings -# allow_k8s_contexts('vke-e01a95c7-aa18-45a9-b8c2-ca36b6bb33f3') -# default_registry('ttl.sh/cjfuturepornnet-98ajf9iwejf9iupawh4efu8hawe') - -## don't scrub secrets so we can more easily debug -secret_settings( - disable_scrub=True -) - -## @warning Don't use kubefwd, it's unreliable. In my testing, it deleted important /etc/hosts entries. -## Our workaround for SSL in dev is to use a VPS with caddy and chisel. -## Caddy handles SSL certs/termination and chisel proxies into our cluster. -## This means that cert-manager is only used in staging and production env (not development.) -## This also means Gateway and HTTPRoute is only used in staging and production. -## @todo Caddy/Chisel is not ideal since that setup is out-of-scope of the monorepo. For now it's the best solution because it unblocks our DX -## -## @see https://blog.tilt.dev/2021/09/09/kubefwd-operator.html -# v1alpha1.extension_repo(name='default', url='https://github.com/tilt-dev/tilt-extensions') -# v1alpha1.extension(name='kubefwd:config', repo_name='default', repo_path='kubefwd') - - -## helm_remote and deploy_cert_manager are BANNED because they use too much bandwidth and crash my computer -## -## cert-manager slows down Tilt updates so I prefer to keep it commented unless I specifically need to test certs -## cert-manager loaded using this extension is PAINFULLY SLOW, and it must re-install and re-test every time the Tiltfile changes. -## additionally, it is SYNCRHONOUS, which means nothing else can update until cert-manager is updated. @see https://github.com/tilt-dev/tilt-extensions/pull/90#issuecomment-704381205 -## TL;DR: This is convenient, but it's much faster to use a helm chart for working with cert-manager. -# load('ext://cert_manager', 'deploy_cert_manager') -# deploy_cert_manager( -# load_to_kind=True, -# version='v1.15.1', -# ) -# load('ext://helm_remote', 'helm_remote') - -load('ext://dotenv', 'dotenv') -dotenv(fn='.env.development') - - -## Right now we use Tilt/Helm in dev and Flux/Kustomize/Helm in production. -## It is a pipedream to use the same deployment/templating tool in development as we do in production. This vastly simplifies deployment. -## We can't use Flux in development unless we figure out a way for flux/kustomize to reference our fp Helm chart as a relative directory. -## Right now, we reference gitea.futureporn.net (kind: GitRepository) where Kustomize downloads the fp Helm chart. -## We could possibly rewrite our fp Helm chart as a Kustomization and then deprecate Helm in development. -## k8s_yaml(kustomize('./flux/apps/development')) -## We are constrained to CrossNamespaceObjectReference kind list -## @see https://fluxcd.io/flux/components/helm/api/v2/#helm.toolkit.fluxcd.io/v2.CrossNamespaceObjectReference -## @see https://github.com/fluxcd/helm-controller/blob/c8ae4b6ad225d37b19bacb634db784d6096908ac/api/v2beta2/reference_types.go#L53 - - - - -# helm_remote( -# 'nitter', -# repo_name='truecharts', -# repo_url='https://charts.truecharts.org', -# namespace='futureporn', -# version='7.1.4', -# ) - - -# helm_remote( -# 'kubernetes-ingress-controller', -# repo_name='kubernetes-ingress-controller', -# repo_url='https://ngrok.github.io/kubernetes-ingress-controller', -# namespace='futureporn', -# create_namespace='false', -# set=[ -# 'credentials.apiKey=%s' % os.getenv('NGROK_API_KEY'), -# 'credentials.authtoken=%s' % os.getenv('NGROK_AUTHTOKEN') -# ] -# ) - - -# k8s_yaml(helm( -# './charts/nitter', -# values=['./charts/nitter/values.yaml'], -# )) - - - -k8s_yaml(helm( - './charts/traefik/traefik', - namespace='futureporn', - values=[ - './charts/traefik/values-overrides.yaml' - ] -)) - -k8s_yaml(helm( - './charts/fp', - values=['./charts/fp/values.yaml'], -)) -# we are using a local helm chart instead of using helm_remote because that command makes the tilt builds Hella slow. -# to download this chart, we used the following commands. -# future re-pulling is needed to keep things up-to-date. -# -# helm repo add bitnami https://charts.bitnami.com/bitnami -# helm pull bitnami/postgresql --untar --destination ./charts/postgresql -k8s_yaml(helm( - './charts/postgresql/postgresql', - namespace='futureporn', - values=[ - './charts/postgresql/values-overrides.yaml' - ] -)) - - -# k8s_yaml(helm( -# './charts/velero/velero', -# namespace='velero', -# values=[ -# './charts/velero/values.yaml' -# ] -# )) - - - -# k8s_yaml(helm( -# './charts/drupal/drupal', -# namespace='futureporn', -# values=[ -# './charts/drupal/values-overrides.yaml' -# ] -# )) - -# k8s_yaml(helm( -# './charts/phpmyadmin/phpmyadmin', -# namespace='futureporn', -# values=[ -# './charts/phpmyadmin/values-overrides.yaml' -# ] -# )) - -# k8s_yaml(helm( -# './charts/mariadb/mariadb', -# namespace='futureporn', -# values=[ -# './charts/mariadb/values-overrides.yaml' -# ] -# )) - -# k8s_yaml(helm( -# './charts/external-secrets/external-secrets', -# namespace='futureporn', -# )) - - - - -# ## redis is for uppy -# ## before you think of switching to valkey, dragonfly, or one of the other redis alternatives, STOP. Uppy is picky. -# ## I tested dragonfly, valkey, and KeyDB. Uppy's ioredis client was unable to connect. "ECONNREFUSED" ... -# ## Uppy was only happy connecting to official redis. -k8s_yaml(helm( - './charts/redis/redis', - namespace='futureporn', - values=[ - './charts/redis/values-overrides.yaml' - ] -)) -k8s_resource( - workload='redis-master', - labels=['database'] -) - -k8s_yaml(helm( - './charts/cert-manager/cert-manager', - namespace='cert-manager', - values=['./charts/cert-manager/values-overrides.yaml'] -)) - -k8s_yaml(helm( - './charts/chisel-operator/chisel-operator', - namespace='futureporn', - values=['./charts/chisel-operator/values-overrides.yaml'] -)) -k8s_resource( - workload='chisel-operator', - labels=['networking'], -) - -## ngrok -# k8s_yaml(helm( -# './charts/kubernetes-ingress-controller/kubernetes-ingress-controller', -# namespace='futureporn', -# values=['./charts/kubernetes-ingress-controller/values-overrides.yaml'] -# )) - - -# docker_build('fp/link2cid', './packages/link2cid') - - -# docker_build( -# 'fp/bot', -# '.', -# only=[ -# './.npmrc', -# './package.json', -# './pnpm-lock.yaml', -# './pnpm-workspace.yaml', -# './services/bot', -# './packages/types', -# './packages/utils', -# './packages/fetchers', -# ], -# dockerfile='./dockerfiles/bot.dockerfile', -# target='dev', -# live_update=[ -# sync('./services/bot', '/app/services/bot') -# ] -# ) - -# docker_build( -# 'fp/scout', -# '.', -# only=[ -# './.npmrc', -# './package.json', -# './pnpm-lock.yaml', -# './pnpm-workspace.yaml', -# './packages/types', -# './packages/utils', -# './packages/fetchers', -# './services/scout', -# ], -# dockerfile='./dockerfiles/scout.dockerfile', -# target='dev', -# # target='prod', -# live_update=[ -# sync('./services/scout', '/app/services/scout') -# ] -# ) - - - - - - -load('ext://uibutton', 'cmd_button') - - -cmd_button('postgres:restore', - argv=['./scripts/postgres-restore.sh'], - resource='postgresql-primary', - icon_name='upload', - text='restore db from backup', -) -cmd_button('postgres:drop', - argv=['sh', './scripts/postgres-drop.sh'], - resource='postgresql-primary', - icon_name='delete', - text='DROP all databases' -) -cmd_button('postgres:create:bright', - argv=['sh', './scripts/postgres-create-bright.sh'], - resource='bright', - icon_name='star', - text='Create bright db' -) -cmd_button('postgres:create:superstreamer', - argv=['sh', './scripts/postgres-create-superstreamer.sh'], - resource='superstreamer-api', - icon_name='star', - text='Create superstreamer db' -) -# cmd_button('migrations-schema:refresh', -# argv=['echo', '@todo please restart postgrest container manually.'], -# resource='migrations-schema', -# icon_name='refresh', -# text='Refresh schema cache' -# ) -# cmd_button('migrations-data:refresh', -# argv=['echo', '@todo please restart postgrest container manually.'], -# resource='migrations-data', -# icon_name='refresh', -# text='Refresh schema cache' -# ) -## @todo let's make this get a random room from scout then use the random room to record via POST /recordings -# cmd_button('capture-worker:create', -# argv=['./scripts/capture-integration.sh'], -# resource='capture-worker', -# icon_name='send', -# text='Recording Integration Test' -# ) -# k8s_resource( -# workload='capture-worker', -# labels=['backend'], -# resource_deps=['postgrest', 'postgresql-primary'], -# ) -k8s_resource( - workload='superstreamer-app', - labels=['app'], - resource_deps=['postgresql-primary', 'redis-master'], - port_forwards=['52002'] -) -k8s_resource( - workload='superstreamer-api', - labels=['app'], - resource_deps=['postgresql-primary', 'redis-master'], - port_forwards=['52001'] -) -k8s_resource( - workload='superstreamer-stitcher', - labels=['app'], - resource_deps=['postgresql-primary', 'redis-master'], -) -k8s_resource( - workload='superstreamer-artisan', - labels=['app'], - resource_deps=['postgresql-primary', 'redis-master'], -) - - -cmd_button('pgadmin4:restore', - argv=['./scripts/pgadmin-import-connection.sh'], - resource='pgadmin4', - icon_name='hub', - text='import connection', -) -cmd_button('build:test', - argv=['./scripts/build-test.sh'], - resource='build', - icon_name='build', - text='test', -) - -## we ignore unused image warnings because we do actually use this image. -## instead of being invoked by helm, we start a container using this image manually via Tilt UI -# update_settings(suppress_unused_image_warnings=["fp/migrations"]) -# docker_build( -# 'fp/migrations-schema', -# '.', -# dockerfile='dockerfiles/migrations-schema.dockerfile', -# target='migrations-schema', -# pull=False, -# only=[ -# './.npmrc', -# './package.json', -# './pnpm-lock.yaml', -# './pnpm-workspace.yaml', -# './services/migrations-schema' -# ], -# ) -# docker_build( -# 'fp/migrations-data', -# '.', -# dockerfile='dockerfiles/migrations-data.dockerfile', -# target='migrations-data', -# pull=False, -# only=[ -# './.npmrc', -# './package.json', -# './pnpm-lock.yaml', -# './pnpm-workspace.yaml', -# './services/migrations-data' -# ], -# ) - -## Uncomment the following for fp/next in dev mode -## this is useful for changing the UI and seeing results -# docker_build( -# 'fp/next', -# '.', -# dockerfile='dockerfiles/next.dockerfile', -# target='dev', -# live_update=[ -# sync('./services/next', '/app/services/next') -# ], -# pull=False, -# ) -# docker_build( -# 'fp/superstreamer-artisan', -# './contrib/superstreamer/packages/artisan', -# dockerfile='contrib/superstreamer/packages/artisan/Dockerfile', -# ) -# docker_build( -# 'fp/superstreamer-api', -# './contrib/superstreamer/packages/api', -# dockerfile='contrib/superstreamer/packages/api/Dockerfile', -# ) -# docker_build( -# 'fp/superstreamer-app', -# './contrib/superstreamer/packages/app', -# dockerfile='contrib/superstreamer/packages/app/Dockerfile', -# ) -# docker_build( -# 'fp/superstreamer-stitcher', -# './contrib/superstreamer/packages/stitcher', -# dockerfile='contrib/superstreamer/packages/stitcher/Dockerfile', -# ) - -docker_build( - 'fp/bright', - '.', - dockerfile='dockerfiles/bright.dockerfile', - live_update=[ - sync('./services/bright', '/app') - ], - target='dev' -) - - - - - -# docker_build( -# 'fp/mailbox', -# '.', -# dockerfile='dockerfiles/mailbox.dockerfile', -# target='mailbox', -# only=[ -# './.npmrc', -# './package.json', -# './pnpm-lock.yaml', -# './pnpm-workspace.yaml', -# './services/mailbox', -# './packages/types', -# './packages/utils', -# './packages/fetchers', -# './packages/video', -# './packages/storage', -# ], -# live_update=[ -# sync('./services/mailbox', '/app'), -# run('cd /app && pnpm i', trigger=['./services/mailbox/package.json', './services/mailbox/pnpm-lock.yaml']), -# ], -# pull=False, -# # entrypoint='pnpm nodemon --ext js,ts,json,yaml --exec node --no-warnings=ExperimentalWarning --loader ts-node/esm ./src/index.ts' -# ) - - - - -# docker_build( -# 'fp/capture', -# '.', -# dockerfile='dockerfiles/capture.dockerfile', -# target='dev', -# only=[ -# './.npmrc', -# './package.json', -# './pnpm-lock.yaml', -# './pnpm-workspace.yaml', -# './packages/types', -# './packages/utils', -# './packages/fetchers', -# './services/capture', -# ], -# live_update=[ -# sync('./services/capture', '/app/services/capture'), -# ], -# pull=False, -# ) - - - - - - -# k8s_resource( -# workload='scout', -# resource_deps=['postgresql-primary'], -# # port_forwards=['8134'], -# labels=['backend'], -# ) -# k8s_resource( -# workload='uppy', -# links=[ -# link('https://uppy.fp.sbtp.xyz'), -# ], -# resource_deps=['redis-master'], -# labels=['backend'], -# ) -# k8s_resource( -# workload='next', -# links=[ -# link('https://next.fp.sbtp.xyz') -# ], -# resource_deps=['postgrest', 'postgresql-primary'], -# labels=['frontend'], -# port_forwards=['3000'], -# ) -k8s_resource( - workload='bright', - links=[ - link('https://bright.fp.sbtp.xyz') - ], - resource_deps=['postgresql-primary'], - labels=['app'], - port_forwards=['4000'], -) - -# whoami is for testing routing -k8s_resource( - workload='whoami', - labels=['app'], - links=[ - link('https://whoami.fp.sbtp.xyz/') - ] -) -k8s_resource( - workload='postgresql-primary', - # port_forwards=['5432'], - labels=['database'], -) -k8s_resource( - workload='postgresql-read', - labels=['database'] -) - - -# k8s_resource( -# workload='chart-velero', -# resource_deps=['postgresql-primary'], -# labels=['backend'], -# ) -# k8s_resource( -# workload='chart-velero-upgrade-crds', -# resource_deps=['postgresql-primary'], -# labels=['backend'], -# ) - - - - -# k8s_resource( -# workload='mailbox', -# resource_deps=['postgresql-primary', 'postgrest'], -# labels=['backend'], -# ) - -# k8s_resource( -# workload='nitter', -# port_forwards=['6060:10606'], -# ) - -# temporarily disabled to save CPU resources -# helm_remote( -# 'kube-prometheus-stack', -# repo_name='kube-prometheus-stack', -# repo_url='https://prometheus-community.github.io/helm-charts', -# namespace='futureporn', -# version='61.1.1', -# set=[ -# 'prometheus.prometheusSpec.storageSpec.volumeClaimTemplate.spec.storageClassName=vultr-block-storage', -# 'admin.existingSecret=grafana', -# 'sidecar.dashboards.enabled=true', -# 'grafana.admin.existingSecret=grafana', -# 'grafana.sidecar.dashboards.enabled=true', -# 'grafana.sidecar.dashboards.defaultFolderName=balls', -# 'grafana.sidecar.dashboards.label=grafana_dashboard', -# 'grafana.sidecar.dashboards.provider.foldersFromFileStructure=true' -# ] -# ) - - - -k8s_resource( - workload='external-dns', - labels=['networking'], -) - -k8s_resource( - workload='cert-manager-webhook-exoscale', - labels=['networking'], -) - - -# k8s_resource( -# workload='factory', -# labels=['backend'], -# ) -# docker_build( -# 'fp/factory', -# '.', -# dockerfile='./dockerfiles/factory.dockerfile', -# target='dev', -# live_update=[ -# sync('./services/factory', '/app/services/factory') -# ], -# pull=False, -# ) - - -# k8s_resource( -# workload='bot', -# labels=['backend'], -# resource_deps=['postgrest'], -# ) - -# k8s_resource( -# workload='chihaya', -# labels=['backend'] -# ) -# k8s_resource( -# workload='postgrest', -# # port_forwards=['9000'], -# labels=['database'], -# links=[ -# link('https://postgrest.fp.sbtp.xyz'), -# ], -# resource_deps=['postgresql-primary'], -# ) -k8s_resource( - workload='traefik', - links=[ - link('https://traefik.fp.sbtp.xyz/whoami'), - link('https://traefik.fp.sbtp.xyz/postgrest'), - ], - labels=['networking'], -) -k8s_resource( - workload='pgadmin4', - port_forwards=['5050:80'], - labels=['database'], -) -# k8s_resource( -# workload='migrations-schema', -# labels=['database'], -# resource_deps=['postgresql-primary'], -# ) -# k8s_resource( -# workload='migrations-data', -# labels=['database'], -# resource_deps=['postgresql-primary'], -# ) - -k8s_resource( - workload='cert-manager', - labels=['certificates'], -) -k8s_resource( - workload='cert-manager-cainjector', - labels=['certificates'], -) -k8s_resource( - workload='cert-manager-webhook', - labels=['certificates'], -) -k8s_resource( - workload='cert-manager-startupapicheck', - labels=['certificates'], -) - diff --git a/certs/letsencrypt-stg-root-x1.pem b/certs/letsencrypt-stg-root-x1.pem deleted file mode 100644 index 37655b2..0000000 --- a/certs/letsencrypt-stg-root-x1.pem +++ /dev/null @@ -1,32 +0,0 @@ ------BEGIN CERTIFICATE----- -MIIFmDCCA4CgAwIBAgIQU9C87nMpOIFKYpfvOHFHFDANBgkqhkiG9w0BAQsFADBm -MQswCQYDVQQGEwJVUzEzMDEGA1UEChMqKFNUQUdJTkcpIEludGVybmV0IFNlY3Vy -aXR5IFJlc2VhcmNoIEdyb3VwMSIwIAYDVQQDExkoU1RBR0lORykgUHJldGVuZCBQ -ZWFyIFgxMB4XDTE1MDYwNDExMDQzOFoXDTM1MDYwNDExMDQzOFowZjELMAkGA1UE -BhMCVVMxMzAxBgNVBAoTKihTVEFHSU5HKSBJbnRlcm5ldCBTZWN1cml0eSBSZXNl -YXJjaCBHcm91cDEiMCAGA1UEAxMZKFNUQUdJTkcpIFByZXRlbmQgUGVhciBYMTCC -AiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBALbagEdDTa1QgGBWSYkyMhsc -ZXENOBaVRTMX1hceJENgsL0Ma49D3MilI4KS38mtkmdF6cPWnL++fgehT0FbRHZg -jOEr8UAN4jH6omjrbTD++VZneTsMVaGamQmDdFl5g1gYaigkkmx8OiCO68a4QXg4 -wSyn6iDipKP8utsE+x1E28SA75HOYqpdrk4HGxuULvlr03wZGTIf/oRt2/c+dYmD -oaJhge+GOrLAEQByO7+8+vzOwpNAPEx6LW+crEEZ7eBXih6VP19sTGy3yfqK5tPt -TdXXCOQMKAp+gCj/VByhmIr+0iNDC540gtvV303WpcbwnkkLYC0Ft2cYUyHtkstO -fRcRO+K2cZozoSwVPyB8/J9RpcRK3jgnX9lujfwA/pAbP0J2UPQFxmWFRQnFjaq6 -rkqbNEBgLy+kFL1NEsRbvFbKrRi5bYy2lNms2NJPZvdNQbT/2dBZKmJqxHkxCuOQ -FjhJQNeO+Njm1Z1iATS/3rts2yZlqXKsxQUzN6vNbD8KnXRMEeOXUYvbV4lqfCf8 -mS14WEbSiMy87GB5S9ucSV1XUrlTG5UGcMSZOBcEUpisRPEmQWUOTWIoDQ5FOia/ -GI+Ki523r2ruEmbmG37EBSBXdxIdndqrjy+QVAmCebyDx9eVEGOIpn26bW5LKeru -mJxa/CFBaKi4bRvmdJRLAgMBAAGjQjBAMA4GA1UdDwEB/wQEAwIBBjAPBgNVHRMB -Af8EBTADAQH/MB0GA1UdDgQWBBS182Xy/rAKkh/7PH3zRKCsYyXDFDANBgkqhkiG -9w0BAQsFAAOCAgEAncDZNytDbrrVe68UT6py1lfF2h6Tm2p8ro42i87WWyP2LK8Y -nLHC0hvNfWeWmjZQYBQfGC5c7aQRezak+tHLdmrNKHkn5kn+9E9LCjCaEsyIIn2j -qdHlAkepu/C3KnNtVx5tW07e5bvIjJScwkCDbP3akWQixPpRFAsnP+ULx7k0aO1x -qAeaAhQ2rgo1F58hcflgqKTXnpPM02intVfiVVkX5GXpJjK5EoQtLceyGOrkxlM/ -sTPq4UrnypmsqSagWV3HcUlYtDinc+nukFk6eR4XkzXBbwKajl0YjztfrCIHOn5Q -CJL6TERVDbM/aAPly8kJ1sWGLuvvWYzMYgLzDul//rUF10gEMWaXVZV51KpS9DY/ -5CunuvCXmEQJHo7kGcViT7sETn6Jz9KOhvYcXkJ7po6d93A/jy4GKPIPnsKKNEmR -xUuXY4xRdh45tMJnLTUDdC9FIU0flTeO9/vNpVA8OPU1i14vCz+MU8KX1bV3GXm/ -fxlB7VBBjX9v5oUep0o/j68R/iDlCOM4VVfRa8gX6T2FU7fNdatvGro7uQzIvWof -gN9WUwCbEMBy/YhBSrXycKA8crgGg3x1mIsopn88JKwmMBa68oS7EHM9w7C4y71M -7DiA+/9Qdp9RBWJpTS9i/mDnJg1xvo8Xz49mrrgfmcAXTCJqXi24NatI3Oc= ------END CERTIFICATE----- diff --git a/certs/letsencrypt-stg-root-x2.pem b/certs/letsencrypt-stg-root-x2.pem deleted file mode 100644 index f627e1d..0000000 --- a/certs/letsencrypt-stg-root-x2.pem +++ /dev/null @@ -1,15 +0,0 @@ ------BEGIN CERTIFICATE----- -MIICTjCCAdSgAwIBAgIRAIPgc3k5LlLVLtUUvs4K/QcwCgYIKoZIzj0EAwMwaDEL -MAkGA1UEBhMCVVMxMzAxBgNVBAoTKihTVEFHSU5HKSBJbnRlcm5ldCBTZWN1cml0 -eSBSZXNlYXJjaCBHcm91cDEkMCIGA1UEAxMbKFNUQUdJTkcpIEJvZ3VzIEJyb2Nj -b2xpIFgyMB4XDTIwMDkwNDAwMDAwMFoXDTQwMDkxNzE2MDAwMFowaDELMAkGA1UE -BhMCVVMxMzAxBgNVBAoTKihTVEFHSU5HKSBJbnRlcm5ldCBTZWN1cml0eSBSZXNl -YXJjaCBHcm91cDEkMCIGA1UEAxMbKFNUQUdJTkcpIEJvZ3VzIEJyb2Njb2xpIFgy -MHYwEAYHKoZIzj0CAQYFK4EEACIDYgAEOvS+w1kCzAxYOJbA06Aw0HFP2tLBLKPo -FQqR9AMskl1nC2975eQqycR+ACvYelA8rfwFXObMHYXJ23XLB+dAjPJVOJ2OcsjT -VqO4dcDWu+rQ2VILdnJRYypnV1MMThVxo0IwQDAOBgNVHQ8BAf8EBAMCAQYwDwYD -VR0TAQH/BAUwAwEB/zAdBgNVHQ4EFgQU3tGjWWQOwZo2o0busBB2766XlWYwCgYI -KoZIzj0EAwMDaAAwZQIwRcp4ZKBsq9XkUuN8wfX+GEbY1N5nmCRc8e80kUkuAefo -uc2j3cICeXo1cOybQ1iWAjEA3Ooawl8eQyR4wrjCofUE8h44p0j7Yl/kBlJZT8+9 -vbtH7QiVzeKCOTQPINyRql6P ------END CERTIFICATE----- diff --git a/charts/README.md b/charts/README.md deleted file mode 100644 index ce99127..0000000 --- a/charts/README.md +++ /dev/null @@ -1,49 +0,0 @@ -# Futureporn Helm charts - -These charts define the bulk of Futureporn's Kubernetes (k8s) payload. (Some of the infrastrucutre-specific payload is at `/flux/`) -In production, Flux CD deploys these charts into the k8s cluster. - - -## development - -We are using a local Helm charts instead of using Tilt's `helm_remote` because that command makes the Tilt builds Hella slow. - -We override default values in the parent folder. - -/charts/postgresql/postgresql/values.yaml -/charts/postgresql/values-overrides.yaml - -### postgresql - - helm repo add bitnami https://charts.bitnami.com/bitnami --force-update - helm pull bitnami/postgresql --untar --destination ./charts/postgresql - -### cert-manager - - helm repo add jetstack https://charts.jetstack.io --force-update - helm pull jetstack/cert-manager --untar --destination ./charts/cert-manager - -### redis - - helm repo add bitnami https://charts.bitnami.com/bitnami - helm pull bitnami/redis --version 20.0.1 --untar --destination ./charts/redis - -### chisel-operator - - helm pull oci://ghcr.io/fyralabs/chisel-operator/chisel-operator --version 0.1.0 --untar --destination ./charts/chisel-operator - -### traefik - - helm repo add traefik https://traefik.github.io/charts - helm pull traefik/traefik --version 33.0.0 --untar --destination ./charts/traefik - -### velero - - helm repo add vmware-tanzu https://vmware-tanzu.github.io/helm-charts/ - helm pull vmware-tanzu/velero --version 7.2.1 --untar --destination ./charts/velero - -### external-secrets-operator - - helm repo add external-secrets https://charts.external-secrets.io - helm pull external-secrets/external-secrets --version 0.10.2 --untar --destination ./charts/external-secrets - diff --git a/charts/cert-manager/cert-manager/Chart.yaml b/charts/cert-manager/cert-manager/Chart.yaml deleted file mode 100644 index 1402a65..0000000 --- a/charts/cert-manager/cert-manager/Chart.yaml +++ /dev/null @@ -1,26 +0,0 @@ -annotations: - artifacthub.io/category: security - artifacthub.io/license: Apache-2.0 - artifacthub.io/prerelease: "false" - artifacthub.io/signKey: | - fingerprint: 1020CF3C033D4F35BAE1C19E1226061C665DF13E - url: https://cert-manager.io/public-keys/cert-manager-keyring-2021-09-20-1020CF3C033D4F35BAE1C19E1226061C665DF13E.gpg -apiVersion: v2 -appVersion: v1.15.1 -description: A Helm chart for cert-manager -home: https://cert-manager.io -icon: https://raw.githubusercontent.com/cert-manager/community/4d35a69437d21b76322157e6284be4cd64e6d2b7/logo/logo-small.png -keywords: -- cert-manager -- kube-lego -- letsencrypt -- tls -kubeVersion: '>= 1.22.0-0' -maintainers: -- email: cert-manager-maintainers@googlegroups.com - name: cert-manager-maintainers - url: https://cert-manager.io -name: cert-manager -sources: -- https://github.com/cert-manager/cert-manager -version: v1.15.1 diff --git a/charts/cert-manager/cert-manager/README.md b/charts/cert-manager/cert-manager/README.md deleted file mode 100644 index c5a5607..0000000 --- a/charts/cert-manager/cert-manager/README.md +++ /dev/null @@ -1,1884 +0,0 @@ -# cert-manager - -cert-manager is a Kubernetes addon to automate the management and issuance of -TLS certificates from various issuing sources. - -It will ensure certificates are valid and up to date periodically, and attempt -to renew certificates at an appropriate time before expiry. - -## Prerequisites - -- Kubernetes 1.22+ - -## Installing the Chart - -Full installation instructions, including details on how to configure extra -functionality in cert-manager can be found in the [installation docs](https://cert-manager.io/docs/installation/kubernetes/). - -Before installing the chart, you must first install the cert-manager CustomResourceDefinition resources. -This is performed in a separate step to allow you to easily uninstall and reinstall cert-manager without deleting your installed custom resources. - -```bash -$ kubectl apply -f https://github.com/cert-manager/cert-manager/releases/download/v1.15.1/cert-manager.crds.yaml -``` - -To install the chart with the release name `cert-manager`: - -```console -## Add the Jetstack Helm repository -$ helm repo add jetstack https://charts.jetstack.io --force-update - -## Install the cert-manager helm chart -$ helm install cert-manager --namespace cert-manager --version v1.15.1 jetstack/cert-manager -``` - -In order to begin issuing certificates, you will need to set up a ClusterIssuer -or Issuer resource (for example, by creating a 'letsencrypt-staging' issuer). - -More information on the different types of issuers and how to configure them -can be found in [our documentation](https://cert-manager.io/docs/configuration/). - -For information on how to configure cert-manager to automatically provision -Certificates for Ingress resources, take a look at the -[Securing Ingresses documentation](https://cert-manager.io/docs/usage/ingress/). - -> **Tip**: List all releases using `helm list` - -## Upgrading the Chart - -Special considerations may be required when upgrading the Helm chart, and these -are documented in our full [upgrading guide](https://cert-manager.io/docs/installation/upgrading/). - -**Please check here before performing upgrades!** - -## Uninstalling the Chart - -To uninstall/delete the `cert-manager` deployment: - -```console -$ helm delete cert-manager --namespace cert-manager -``` - -The command removes all the Kubernetes components associated with the chart and deletes the release. - -If you want to completely uninstall cert-manager from your cluster, you will also need to -delete the previously installed CustomResourceDefinition resources: - -```console -$ kubectl delete -f https://github.com/cert-manager/cert-manager/releases/download/v1.15.1/cert-manager.crds.yaml -``` - -## Configuration - - -### Global - -#### **global.imagePullSecrets** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Reference to one or more secrets to be used when pulling images. For more information, see [Pull an Image from a Private Registry](https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/). - -For example: - -```yaml -imagePullSecrets: - - name: "image-pull-secret" -``` -#### **global.commonLabels** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Labels to apply to all resources. -Please note that this does not add labels to the resources created dynamically by the controllers. For these resources, you have to add the labels in the template in the cert-manager custom resource: For example, podTemplate/ ingressTemplate in ACMEChallengeSolverHTTP01Ingress. For more information, see the [cert-manager documentation](https://cert-manager.io/docs/reference/api-docs/#acme.cert-manager.io/v1.ACMEChallengeSolverHTTP01Ingress). -For example, secretTemplate in CertificateSpec -For more information, see the [cert-manager documentation](https://cert-manager.io/docs/reference/api-docs/#cert-manager.io/v1.CertificateSpec). -#### **global.revisionHistoryLimit** ~ `number` - -The number of old ReplicaSets to retain to allow rollback (if not set, the default Kubernetes value is set to 10). - -#### **global.priorityClassName** ~ `string` -> Default value: -> ```yaml -> "" -> ``` - -The optional priority class to be used for the cert-manager pods. -#### **global.rbac.create** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -Create required ClusterRoles and ClusterRoleBindings for cert-manager. -#### **global.rbac.aggregateClusterRoles** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -Aggregate ClusterRoles to Kubernetes default user-facing roles. For more information, see [User-facing roles](https://kubernetes.io/docs/reference/access-authn-authz/rbac/#user-facing-roles) -#### **global.podSecurityPolicy.enabled** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -Create PodSecurityPolicy for cert-manager. - -Note that PodSecurityPolicy was deprecated in Kubernetes 1.21 and removed in Kubernetes 1.25. -#### **global.podSecurityPolicy.useAppArmor** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -Configure the PodSecurityPolicy to use AppArmor. -#### **global.logLevel** ~ `number` -> Default value: -> ```yaml -> 2 -> ``` - -Set the verbosity of cert-manager. A range of 0 - 6, with 6 being the most verbose. -#### **global.leaderElection.namespace** ~ `string` -> Default value: -> ```yaml -> kube-system -> ``` - -Override the namespace used for the leader election lease. -#### **global.leaderElection.leaseDuration** ~ `string` - -The duration that non-leader candidates will wait after observing a leadership renewal until attempting to acquire leadership of a led but unrenewed leader slot. This is effectively the maximum duration that a leader can be stopped before it is replaced by another candidate. - -#### **global.leaderElection.renewDeadline** ~ `string` - -The interval between attempts by the acting master to renew a leadership slot before it stops leading. This must be less than or equal to the lease duration. - -#### **global.leaderElection.retryPeriod** ~ `string` - -The duration the clients should wait between attempting acquisition and renewal of a leadership. - -#### **installCRDs** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -This option is equivalent to setting crds.enabled=true and crds.keep=true. Deprecated: use crds.enabled and crds.keep instead. -#### **crds.enabled** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -This option decides if the CRDs should be installed as part of the Helm installation. -#### **crds.keep** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -This option makes it so that the "helm.sh/resource-policy": keep annotation is added to the CRD. This will prevent Helm from uninstalling the CRD when the Helm release is uninstalled. WARNING: when the CRDs are removed, all cert-manager custom resources -(Certificates, Issuers, ...) will be removed too by the garbage collector. -### Controller - -#### **replicaCount** ~ `number` -> Default value: -> ```yaml -> 1 -> ``` - -The number of replicas of the cert-manager controller to run. - -The default is 1, but in production set this to 2 or 3 to provide high availability. - -If `replicas > 1`, consider setting `podDisruptionBudget.enabled=true`. - -Note that cert-manager uses leader election to ensure that there can only be a single instance active at a time. -#### **strategy** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Deployment update strategy for the cert-manager controller deployment. For more information, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#strategy). - -For example: - -```yaml -strategy: - type: RollingUpdate - rollingUpdate: - maxSurge: 0 - maxUnavailable: 1 -``` -#### **podDisruptionBudget.enabled** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -Enable or disable the PodDisruptionBudget resource. - -This prevents downtime during voluntary disruptions such as during a Node upgrade. For example, the PodDisruptionBudget will block `kubectl drain` if it is used on the Node where the only remaining cert-manager -Pod is currently running. -#### **podDisruptionBudget.minAvailable** ~ `number` - -This configures the minimum available pods for disruptions. It can either be set to an integer (e.g. 1) or a percentage value (e.g. 25%). -It cannot be used if `maxUnavailable` is set. - -#### **podDisruptionBudget.maxUnavailable** ~ `number` - -This configures the maximum unavailable pods for disruptions. It can either be set to an integer (e.g. 1) or a percentage value (e.g. 25%). it cannot be used if `minAvailable` is set. - -#### **featureGates** ~ `string` -> Default value: -> ```yaml -> "" -> ``` - -A comma-separated list of feature gates that should be enabled on the controller pod. -#### **maxConcurrentChallenges** ~ `number` -> Default value: -> ```yaml -> 60 -> ``` - -The maximum number of challenges that can be scheduled as 'processing' at once. -#### **image.registry** ~ `string` - -The container registry to pull the manager image from. - -#### **image.repository** ~ `string` -> Default value: -> ```yaml -> quay.io/jetstack/cert-manager-controller -> ``` - -The container image for the cert-manager controller. - -#### **image.tag** ~ `string` - -Override the image tag to deploy by setting this variable. If no value is set, the chart's appVersion is used. - -#### **image.digest** ~ `string` - -Setting a digest will override any tag. - -#### **image.pullPolicy** ~ `string` -> Default value: -> ```yaml -> IfNotPresent -> ``` - -Kubernetes imagePullPolicy on Deployment. -#### **clusterResourceNamespace** ~ `string` -> Default value: -> ```yaml -> "" -> ``` - -Override the namespace used to store DNS provider credentials etc. for ClusterIssuer resources. By default, the same namespace as cert-manager is deployed within is used. This namespace will not be automatically created by the Helm chart. -#### **namespace** ~ `string` -> Default value: -> ```yaml -> "" -> ``` - -This namespace allows you to define where the services are installed into. If not set then they use the namespace of the release. This is helpful when installing cert manager as a chart dependency (sub chart). -#### **serviceAccount.create** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -Specifies whether a service account should be created. -#### **serviceAccount.name** ~ `string` - -The name of the service account to use. -If not set and create is true, a name is generated using the fullname template. - -#### **serviceAccount.annotations** ~ `object` - -Optional additional annotations to add to the controller's Service Account. - -#### **serviceAccount.labels** ~ `object` - -Optional additional labels to add to the controller's Service Account. - -#### **serviceAccount.automountServiceAccountToken** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -Automount API credentials for a Service Account. -#### **automountServiceAccountToken** ~ `bool` - -Automounting API credentials for a particular pod. - -#### **enableCertificateOwnerRef** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -When this flag is enabled, secrets will be automatically removed when the certificate resource is deleted. -#### **config** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -This property is used to configure options for the controller pod. This allows setting options that would usually be provided using flags. An APIVersion and Kind must be specified in your values.yaml file. -Flags will override options that are set here. - -For example: - -```yaml -config: - apiVersion: controller.config.cert-manager.io/v1alpha1 - kind: ControllerConfiguration - logging: - verbosity: 2 - format: text - leaderElectionConfig: - namespace: kube-system - kubernetesAPIQPS: 9000 - kubernetesAPIBurst: 9000 - numberOfConcurrentWorkers: 200 - featureGates: - AdditionalCertificateOutputFormats: true - DisallowInsecureCSRUsageDefinition: true - ExperimentalCertificateSigningRequestControllers: true - ExperimentalGatewayAPISupport: true - LiteralCertificateSubject: true - SecretsFilteredCaching: true - ServerSideApply: true - StableCertificateRequestName: true - UseCertificateRequestBasicConstraints: true - ValidateCAA: true - metricsTLSConfig: - dynamic: - secretNamespace: "cert-manager" - secretName: "cert-manager-metrics-ca" - dnsNames: - - cert-manager-metrics - - cert-manager-metrics.cert-manager - - cert-manager-metrics.cert-manager.svc -``` -#### **dns01RecursiveNameservers** ~ `string` -> Default value: -> ```yaml -> "" -> ``` - -A comma-separated string with the host and port of the recursive nameservers cert-manager should query. -#### **dns01RecursiveNameserversOnly** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -Forces cert-manager to use only the recursive nameservers for verification. Enabling this option could cause the DNS01 self check to take longer owing to caching performed by the recursive nameservers. -#### **disableAutoApproval** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -Option to disable cert-manager's build-in auto-approver. The auto-approver approves all CertificateRequests that reference issuers matching the 'approveSignerNames' option. This 'disableAutoApproval' option is useful when you want to make all approval decisions using a different approver (like approver-policy - https://github.com/cert-manager/approver-policy). -#### **approveSignerNames** ~ `array` -> Default value: -> ```yaml -> - issuers.cert-manager.io/* -> - clusterissuers.cert-manager.io/* -> ``` - -List of signer names that cert-manager will approve by default. CertificateRequests referencing these signer names will be auto-approved by cert-manager. Defaults to just approving the cert-manager.io Issuer and ClusterIssuer issuers. When set to an empty array, ALL issuers will be auto-approved by cert-manager. To disable the auto-approval, because eg. you are using approver-policy, you can enable 'disableAutoApproval'. -ref: https://cert-manager.io/docs/concepts/certificaterequest/#approval - -#### **extraArgs** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Additional command line flags to pass to cert-manager controller binary. To see all available flags run `docker run quay.io/jetstack/cert-manager-controller: --help`. - -Use this flag to enable or disable arbitrary controllers. For example, to disable the CertificiateRequests approver. - -For example: - -```yaml -extraArgs: - - --controllers=*,-certificaterequests-approver -``` -#### **extraEnv** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Additional environment variables to pass to cert-manager controller binary. -#### **resources** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Resources to provide to the cert-manager controller pod. - -For example: - -```yaml -requests: - cpu: 10m - memory: 32Mi -``` - -For more information, see [Resource Management for Pods and Containers](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/). -#### **securityContext** ~ `object` -> Default value: -> ```yaml -> runAsNonRoot: true -> seccompProfile: -> type: RuntimeDefault -> ``` - -Pod Security Context. -For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). - -#### **containerSecurityContext** ~ `object` -> Default value: -> ```yaml -> allowPrivilegeEscalation: false -> capabilities: -> drop: -> - ALL -> readOnlyRootFilesystem: true -> ``` - -Container Security Context to be set on the controller component container. For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). - -#### **volumes** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Additional volumes to add to the cert-manager controller pod. -#### **volumeMounts** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Additional volume mounts to add to the cert-manager controller container. -#### **deploymentAnnotations** ~ `object` - -Optional additional annotations to add to the controller Deployment. - -#### **podAnnotations** ~ `object` - -Optional additional annotations to add to the controller Pods. - -#### **podLabels** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Optional additional labels to add to the controller Pods. -#### **serviceAnnotations** ~ `object` - -Optional annotations to add to the controller Service. - -#### **serviceLabels** ~ `object` - -Optional additional labels to add to the controller Service. - -#### **serviceIPFamilyPolicy** ~ `string` - -Optionally set the IP family policy for the controller Service to configure dual-stack; see [Configure dual-stack](https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services). - -#### **serviceIPFamilies** ~ `array` - -Optionally set the IP families for the controller Service that should be supported, in the order in which they should be applied to ClusterIP. Can be IPv4 and/or IPv6. - -#### **podDnsPolicy** ~ `string` - -Pod DNS policy. -For more information, see [Pod's DNS Policy](https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/#pod-s-dns-policy). - -#### **podDnsConfig** ~ `object` - -Pod DNS configuration. The podDnsConfig field is optional and can work with any podDnsPolicy settings. However, when a Pod's dnsPolicy is set to "None", the dnsConfig field has to be specified. For more information, see [Pod's DNS Config](https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/#pod-dns-config). - -#### **hostAliases** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Optional hostAliases for cert-manager-controller pods. May be useful when performing ACME DNS-01 self checks. -#### **nodeSelector** ~ `object` -> Default value: -> ```yaml -> kubernetes.io/os: linux -> ``` - -The nodeSelector on Pods tells Kubernetes to schedule Pods on the nodes with matching labels. For more information, see [Assigning Pods to Nodes](https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/). - -This default ensures that Pods are only scheduled to Linux nodes. It prevents Pods being scheduled to Windows nodes in a mixed OS cluster. - -#### **ingressShim.defaultIssuerName** ~ `string` - -Optional default issuer to use for ingress resources. - -#### **ingressShim.defaultIssuerKind** ~ `string` - -Optional default issuer kind to use for ingress resources. - -#### **ingressShim.defaultIssuerGroup** ~ `string` - -Optional default issuer group to use for ingress resources. - -#### **http_proxy** ~ `string` - -Configures the HTTP_PROXY environment variable where a HTTP proxy is required. - -#### **https_proxy** ~ `string` - -Configures the HTTPS_PROXY environment variable where a HTTP proxy is required. - -#### **no_proxy** ~ `string` - -Configures the NO_PROXY environment variable where a HTTP proxy is required, but certain domains should be excluded. - -#### **affinity** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -A Kubernetes Affinity, if required. For more information, see [Affinity v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#affinity-v1-core). - -For example: - -```yaml -affinity: - nodeAffinity: - requiredDuringSchedulingIgnoredDuringExecution: - nodeSelectorTerms: - - matchExpressions: - - key: foo.bar.com/role - operator: In - values: - - master -``` -#### **tolerations** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -A list of Kubernetes Tolerations, if required. For more information, see [Toleration v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#toleration-v1-core). - -For example: - -```yaml -tolerations: -- key: foo.bar.com/role - operator: Equal - value: master - effect: NoSchedule -``` -#### **topologySpreadConstraints** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -A list of Kubernetes TopologySpreadConstraints, if required. For more information, see [Topology spread constraint v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#topologyspreadconstraint-v1-core - -For example: - -```yaml -topologySpreadConstraints: -- maxSkew: 2 - topologyKey: topology.kubernetes.io/zone - whenUnsatisfiable: ScheduleAnyway - labelSelector: - matchLabels: - app.kubernetes.io/instance: cert-manager - app.kubernetes.io/component: controller -``` -#### **livenessProbe** ~ `object` -> Default value: -> ```yaml -> enabled: true -> failureThreshold: 8 -> initialDelaySeconds: 10 -> periodSeconds: 10 -> successThreshold: 1 -> timeoutSeconds: 15 -> ``` - -LivenessProbe settings for the controller container of the controller Pod. - -This is enabled by default, in order to enable the clock-skew liveness probe that restarts the controller in case of a skew between the system clock and the monotonic clock. LivenessProbe durations and thresholds are based on those used for the Kubernetes controller-manager. For more information see the following on the -[Kubernetes GitHub repository](https://github.com/kubernetes/kubernetes/blob/806b30170c61a38fedd54cc9ede4cd6275a1ad3b/cmd/kubeadm/app/util/staticpod/utils.go#L241-L245) - -#### **enableServiceLinks** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -enableServiceLinks indicates whether information about services should be injected into the pod's environment variables, matching the syntax of Docker links. -### Prometheus - -#### **prometheus.enabled** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -Enable Prometheus monitoring for the cert-manager controller to use with the. Prometheus Operator. If this option is enabled without enabling `prometheus.servicemonitor.enabled` or -`prometheus.podmonitor.enabled`, 'prometheus.io' annotations are added to the cert-manager Deployment -resources. Additionally, a service is created which can be used together with your own ServiceMonitor (managed outside of this Helm chart). Otherwise, a ServiceMonitor/ PodMonitor is created. -#### **prometheus.servicemonitor.enabled** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -Create a ServiceMonitor to add cert-manager to Prometheus. -#### **prometheus.servicemonitor.prometheusInstance** ~ `string` -> Default value: -> ```yaml -> default -> ``` - -Specifies the `prometheus` label on the created ServiceMonitor. This is used when different Prometheus instances have label selectors matching different ServiceMonitors. -#### **prometheus.servicemonitor.targetPort** ~ `number` -> Default value: -> ```yaml -> 9402 -> ``` - -The target port to set on the ServiceMonitor. This must match the port that the cert-manager controller is listening on for metrics. -#### **prometheus.servicemonitor.path** ~ `string` -> Default value: -> ```yaml -> /metrics -> ``` - -The path to scrape for metrics. -#### **prometheus.servicemonitor.interval** ~ `string` -> Default value: -> ```yaml -> 60s -> ``` - -The interval to scrape metrics. -#### **prometheus.servicemonitor.scrapeTimeout** ~ `string` -> Default value: -> ```yaml -> 30s -> ``` - -The timeout before a metrics scrape fails. -#### **prometheus.servicemonitor.labels** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Additional labels to add to the ServiceMonitor. -#### **prometheus.servicemonitor.annotations** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Additional annotations to add to the ServiceMonitor. -#### **prometheus.servicemonitor.honorLabels** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -Keep labels from scraped data, overriding server-side labels. -#### **prometheus.servicemonitor.endpointAdditionalProperties** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -EndpointAdditionalProperties allows setting additional properties on the endpoint such as relabelings, metricRelabelings etc. - -For example: - -```yaml -endpointAdditionalProperties: - relabelings: - - action: replace - sourceLabels: - - __meta_kubernetes_pod_node_name - targetLabel: instance -``` - - - -#### **prometheus.podmonitor.enabled** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -Create a PodMonitor to add cert-manager to Prometheus. -#### **prometheus.podmonitor.prometheusInstance** ~ `string` -> Default value: -> ```yaml -> default -> ``` - -Specifies the `prometheus` label on the created PodMonitor. This is used when different Prometheus instances have label selectors matching different PodMonitors. -#### **prometheus.podmonitor.path** ~ `string` -> Default value: -> ```yaml -> /metrics -> ``` - -The path to scrape for metrics. -#### **prometheus.podmonitor.interval** ~ `string` -> Default value: -> ```yaml -> 60s -> ``` - -The interval to scrape metrics. -#### **prometheus.podmonitor.scrapeTimeout** ~ `string` -> Default value: -> ```yaml -> 30s -> ``` - -The timeout before a metrics scrape fails. -#### **prometheus.podmonitor.labels** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Additional labels to add to the PodMonitor. -#### **prometheus.podmonitor.annotations** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Additional annotations to add to the PodMonitor. -#### **prometheus.podmonitor.honorLabels** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -Keep labels from scraped data, overriding server-side labels. -#### **prometheus.podmonitor.endpointAdditionalProperties** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -EndpointAdditionalProperties allows setting additional properties on the endpoint such as relabelings, metricRelabelings etc. - -For example: - -```yaml -endpointAdditionalProperties: - relabelings: - - action: replace - sourceLabels: - - __meta_kubernetes_pod_node_name - targetLabel: instance -``` - - - -### Webhook - -#### **webhook.replicaCount** ~ `number` -> Default value: -> ```yaml -> 1 -> ``` - -Number of replicas of the cert-manager webhook to run. - -The default is 1, but in production set this to 2 or 3 to provide high availability. - -If `replicas > 1`, consider setting `webhook.podDisruptionBudget.enabled=true`. -#### **webhook.timeoutSeconds** ~ `number` -> Default value: -> ```yaml -> 30 -> ``` - -The number of seconds the API server should wait for the webhook to respond before treating the call as a failure. The value must be between 1 and 30 seconds. For more information, see -[Validating webhook configuration v1](https://kubernetes.io/docs/reference/kubernetes-api/extend-resources/validating-webhook-configuration-v1/). - -The default is set to the maximum value of 30 seconds as users sometimes report that the connection between the K8S API server and the cert-manager webhook server times out. If *this* timeout is reached, the error message will be "context deadline exceeded", which doesn't help the user diagnose what phase of the HTTPS connection timed out. For example, it could be during DNS resolution, TCP connection, TLS negotiation, HTTP negotiation, or slow HTTP response from the webhook server. By setting this timeout to its maximum value the underlying timeout error message has more chance of being returned to the end user. -#### **webhook.config** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -This is used to configure options for the webhook pod. This allows setting options that would usually be provided using flags. An APIVersion and Kind must be specified in your values.yaml file. -Flags override options that are set here. - -For example: - -```yaml -apiVersion: webhook.config.cert-manager.io/v1alpha1 -kind: WebhookConfiguration -# The port that the webhook listens on for requests. -# In GKE private clusters, by default Kubernetes apiservers are allowed to -# talk to the cluster nodes only on 443 and 10250. Configuring -# securePort: 10250 therefore will work out-of-the-box without needing to add firewall -# rules or requiring NET_BIND_SERVICE capabilities to bind port numbers < 1000. -# This should be uncommented and set as a default by the chart once -# the apiVersion of WebhookConfiguration graduates beyond v1alpha1. -securePort: 10250 -``` -#### **webhook.strategy** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -The update strategy for the cert-manager webhook deployment. For more information, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#strategy) - -For example: - -```yaml -strategy: - type: RollingUpdate - rollingUpdate: - maxSurge: 0 - maxUnavailable: 1 -``` -#### **webhook.securityContext** ~ `object` -> Default value: -> ```yaml -> runAsNonRoot: true -> seccompProfile: -> type: RuntimeDefault -> ``` - -Pod Security Context to be set on the webhook component Pod. For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). - -#### **webhook.containerSecurityContext** ~ `object` -> Default value: -> ```yaml -> allowPrivilegeEscalation: false -> capabilities: -> drop: -> - ALL -> readOnlyRootFilesystem: true -> ``` - -Container Security Context to be set on the webhook component container. For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). - -#### **webhook.podDisruptionBudget.enabled** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -Enable or disable the PodDisruptionBudget resource. - -This prevents downtime during voluntary disruptions such as during a Node upgrade. For example, the PodDisruptionBudget will block `kubectl drain` if it is used on the Node where the only remaining cert-manager -Pod is currently running. -#### **webhook.podDisruptionBudget.minAvailable** ~ `number` - -This property configures the minimum available pods for disruptions. Can either be set to an integer (e.g. 1) or a percentage value (e.g. 25%). -It cannot be used if `maxUnavailable` is set. - -#### **webhook.podDisruptionBudget.maxUnavailable** ~ `number` - -This property configures the maximum unavailable pods for disruptions. Can either be set to an integer (e.g. 1) or a percentage value (e.g. 25%). -It cannot be used if `minAvailable` is set. - -#### **webhook.deploymentAnnotations** ~ `object` - -Optional additional annotations to add to the webhook Deployment. - -#### **webhook.podAnnotations** ~ `object` - -Optional additional annotations to add to the webhook Pods. - -#### **webhook.serviceAnnotations** ~ `object` - -Optional additional annotations to add to the webhook Service. - -#### **webhook.mutatingWebhookConfigurationAnnotations** ~ `object` - -Optional additional annotations to add to the webhook MutatingWebhookConfiguration. - -#### **webhook.validatingWebhookConfigurationAnnotations** ~ `object` - -Optional additional annotations to add to the webhook ValidatingWebhookConfiguration. - -#### **webhook.validatingWebhookConfiguration.namespaceSelector** ~ `object` -> Default value: -> ```yaml -> matchExpressions: -> - key: cert-manager.io/disable-validation -> operator: NotIn -> values: -> - "true" -> ``` - -Configure spec.namespaceSelector for validating webhooks. - -#### **webhook.mutatingWebhookConfiguration.namespaceSelector** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Configure spec.namespaceSelector for mutating webhooks. - -#### **webhook.extraArgs** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Additional command line flags to pass to cert-manager webhook binary. To see all available flags run `docker run quay.io/jetstack/cert-manager-webhook: --help`. -#### **webhook.featureGates** ~ `string` -> Default value: -> ```yaml -> "" -> ``` - -Comma separated list of feature gates that should be enabled on the webhook pod. -#### **webhook.resources** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Resources to provide to the cert-manager webhook pod. - -For example: - -```yaml -requests: - cpu: 10m - memory: 32Mi -``` - -For more information, see [Resource Management for Pods and Containers](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/). -#### **webhook.livenessProbe** ~ `object` -> Default value: -> ```yaml -> failureThreshold: 3 -> initialDelaySeconds: 60 -> periodSeconds: 10 -> successThreshold: 1 -> timeoutSeconds: 1 -> ``` - -Liveness probe values. -For more information, see [Container probes](https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#container-probes). - -#### **webhook.readinessProbe** ~ `object` -> Default value: -> ```yaml -> failureThreshold: 3 -> initialDelaySeconds: 5 -> periodSeconds: 5 -> successThreshold: 1 -> timeoutSeconds: 1 -> ``` - -Readiness probe values. -For more information, see [Container probes](https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#container-probes). - -#### **webhook.nodeSelector** ~ `object` -> Default value: -> ```yaml -> kubernetes.io/os: linux -> ``` - -The nodeSelector on Pods tells Kubernetes to schedule Pods on the nodes with matching labels. For more information, see [Assigning Pods to Nodes](https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/). - -This default ensures that Pods are only scheduled to Linux nodes. It prevents Pods being scheduled to Windows nodes in a mixed OS cluster. - -#### **webhook.affinity** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -A Kubernetes Affinity, if required. For more information, see [Affinity v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#affinity-v1-core). - -For example: - -```yaml -affinity: - nodeAffinity: - requiredDuringSchedulingIgnoredDuringExecution: - nodeSelectorTerms: - - matchExpressions: - - key: foo.bar.com/role - operator: In - values: - - master -``` -#### **webhook.tolerations** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -A list of Kubernetes Tolerations, if required. For more information, see [Toleration v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#toleration-v1-core). - -For example: - -```yaml -tolerations: -- key: foo.bar.com/role - operator: Equal - value: master - effect: NoSchedule -``` -#### **webhook.topologySpreadConstraints** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -A list of Kubernetes TopologySpreadConstraints, if required. For more information, see [Topology spread constraint v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#topologyspreadconstraint-v1-core). - -For example: - -```yaml -topologySpreadConstraints: -- maxSkew: 2 - topologyKey: topology.kubernetes.io/zone - whenUnsatisfiable: ScheduleAnyway - labelSelector: - matchLabels: - app.kubernetes.io/instance: cert-manager - app.kubernetes.io/component: controller -``` -#### **webhook.podLabels** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Optional additional labels to add to the Webhook Pods. -#### **webhook.serviceLabels** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Optional additional labels to add to the Webhook Service. -#### **webhook.serviceIPFamilyPolicy** ~ `string` -> Default value: -> ```yaml -> "" -> ``` - -Optionally set the IP family policy for the controller Service to configure dual-stack; see [Configure dual-stack](https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services). -#### **webhook.serviceIPFamilies** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Optionally set the IP families for the controller Service that should be supported, in the order in which they should be applied to ClusterIP. Can be IPv4 and/or IPv6. -#### **webhook.image.registry** ~ `string` - -The container registry to pull the webhook image from. - -#### **webhook.image.repository** ~ `string` -> Default value: -> ```yaml -> quay.io/jetstack/cert-manager-webhook -> ``` - -The container image for the cert-manager webhook - -#### **webhook.image.tag** ~ `string` - -Override the image tag to deploy by setting this variable. If no value is set, the chart's appVersion will be used. - -#### **webhook.image.digest** ~ `string` - -Setting a digest will override any tag - -#### **webhook.image.pullPolicy** ~ `string` -> Default value: -> ```yaml -> IfNotPresent -> ``` - -Kubernetes imagePullPolicy on Deployment. -#### **webhook.serviceAccount.create** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -Specifies whether a service account should be created. -#### **webhook.serviceAccount.name** ~ `string` - -The name of the service account to use. -If not set and create is true, a name is generated using the fullname template. - -#### **webhook.serviceAccount.annotations** ~ `object` - -Optional additional annotations to add to the controller's Service Account. - -#### **webhook.serviceAccount.labels** ~ `object` - -Optional additional labels to add to the webhook's Service Account. - -#### **webhook.serviceAccount.automountServiceAccountToken** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -Automount API credentials for a Service Account. -#### **webhook.automountServiceAccountToken** ~ `bool` - -Automounting API credentials for a particular pod. - -#### **webhook.securePort** ~ `number` -> Default value: -> ```yaml -> 10250 -> ``` - -The port that the webhook listens on for requests. In GKE private clusters, by default Kubernetes apiservers are allowed to talk to the cluster nodes only on 443 and 10250. Configuring securePort: 10250, therefore will work out-of-the-box without needing to add firewall rules or requiring NET_BIND_SERVICE capabilities to bind port numbers <1000. -#### **webhook.hostNetwork** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -Specifies if the webhook should be started in hostNetwork mode. - -Required for use in some managed kubernetes clusters (such as AWS EKS) with custom. CNI (such as calico), because control-plane managed by AWS cannot communicate with pods' IP CIDR and admission webhooks are not working - -Since the default port for the webhook conflicts with kubelet on the host network, `webhook.securePort` should be changed to an available port if running in hostNetwork mode. -#### **webhook.serviceType** ~ `string` -> Default value: -> ```yaml -> ClusterIP -> ``` - -Specifies how the service should be handled. Useful if you want to expose the webhook outside of the cluster. In some cases, the control plane cannot reach internal services. -#### **webhook.loadBalancerIP** ~ `string` - -Specify the load balancer IP for the created service. - -#### **webhook.url** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Overrides the mutating webhook and validating webhook so they reach the webhook service using the `url` field instead of a service. -#### **webhook.networkPolicy.enabled** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -Create network policies for the webhooks. -#### **webhook.networkPolicy.ingress** ~ `array` -> Default value: -> ```yaml -> - from: -> - ipBlock: -> cidr: 0.0.0.0/0 -> ``` - -Ingress rule for the webhook network policy. By default, it allows all inbound traffic. - -#### **webhook.networkPolicy.egress** ~ `array` -> Default value: -> ```yaml -> - ports: -> - port: 80 -> protocol: TCP -> - port: 443 -> protocol: TCP -> - port: 53 -> protocol: TCP -> - port: 53 -> protocol: UDP -> - port: 6443 -> protocol: TCP -> to: -> - ipBlock: -> cidr: 0.0.0.0/0 -> ``` - -Egress rule for the webhook network policy. By default, it allows all outbound traffic to ports 80 and 443, as well as DNS ports. - -#### **webhook.volumes** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Additional volumes to add to the cert-manager controller pod. -#### **webhook.volumeMounts** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Additional volume mounts to add to the cert-manager controller container. -#### **webhook.enableServiceLinks** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -enableServiceLinks indicates whether information about services should be injected into the pod's environment variables, matching the syntax of Docker links. -### CA Injector - -#### **cainjector.enabled** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -Create the CA Injector deployment -#### **cainjector.replicaCount** ~ `number` -> Default value: -> ```yaml -> 1 -> ``` - -The number of replicas of the cert-manager cainjector to run. - -The default is 1, but in production set this to 2 or 3 to provide high availability. - -If `replicas > 1`, consider setting `cainjector.podDisruptionBudget.enabled=true`. - -Note that cert-manager uses leader election to ensure that there can only be a single instance active at a time. -#### **cainjector.config** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -This is used to configure options for the cainjector pod. It allows setting options that are usually provided via flags. An APIVersion and Kind must be specified in your values.yaml file. -Flags override options that are set here. - -For example: - -```yaml -apiVersion: cainjector.config.cert-manager.io/v1alpha1 -kind: CAInjectorConfiguration -logging: - verbosity: 2 - format: text -leaderElectionConfig: - namespace: kube-system -``` -#### **cainjector.strategy** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Deployment update strategy for the cert-manager cainjector deployment. For more information, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#strategy). - -For example: - -```yaml -strategy: - type: RollingUpdate - rollingUpdate: - maxSurge: 0 - maxUnavailable: 1 -``` -#### **cainjector.securityContext** ~ `object` -> Default value: -> ```yaml -> runAsNonRoot: true -> seccompProfile: -> type: RuntimeDefault -> ``` - -Pod Security Context to be set on the cainjector component Pod. For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). - -#### **cainjector.containerSecurityContext** ~ `object` -> Default value: -> ```yaml -> allowPrivilegeEscalation: false -> capabilities: -> drop: -> - ALL -> readOnlyRootFilesystem: true -> ``` - -Container Security Context to be set on the cainjector component container. For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). - -#### **cainjector.podDisruptionBudget.enabled** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -Enable or disable the PodDisruptionBudget resource. - -This prevents downtime during voluntary disruptions such as during a Node upgrade. For example, the PodDisruptionBudget will block `kubectl drain` if it is used on the Node where the only remaining cert-manager -Pod is currently running. -#### **cainjector.podDisruptionBudget.minAvailable** ~ `number` - -`minAvailable` configures the minimum available pods for disruptions. It can either be set to -an integer (e.g. 1) or a percentage value (e.g. 25%). -Cannot be used if `maxUnavailable` is set. - -#### **cainjector.podDisruptionBudget.maxUnavailable** ~ `number` - -`maxUnavailable` configures the maximum unavailable pods for disruptions. It can either be set to -an integer (e.g. 1) or a percentage value (e.g. 25%). -Cannot be used if `minAvailable` is set. - -#### **cainjector.deploymentAnnotations** ~ `object` - -Optional additional annotations to add to the cainjector Deployment. - -#### **cainjector.podAnnotations** ~ `object` - -Optional additional annotations to add to the cainjector Pods. - -#### **cainjector.extraArgs** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Additional command line flags to pass to cert-manager cainjector binary. To see all available flags run `docker run quay.io/jetstack/cert-manager-cainjector: --help`. -#### **cainjector.featureGates** ~ `string` -> Default value: -> ```yaml -> "" -> ``` - -Comma separated list of feature gates that should be enabled on the cainjector pod. -#### **cainjector.resources** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Resources to provide to the cert-manager cainjector pod. - -For example: - -```yaml -requests: - cpu: 10m - memory: 32Mi -``` - -For more information, see [Resource Management for Pods and Containers](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/). -#### **cainjector.nodeSelector** ~ `object` -> Default value: -> ```yaml -> kubernetes.io/os: linux -> ``` - -The nodeSelector on Pods tells Kubernetes to schedule Pods on the nodes with matching labels. For more information, see [Assigning Pods to Nodes](https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/). - -This default ensures that Pods are only scheduled to Linux nodes. It prevents Pods being scheduled to Windows nodes in a mixed OS cluster. - -#### **cainjector.affinity** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -A Kubernetes Affinity, if required. For more information, see [Affinity v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#affinity-v1-core). - -For example: - -```yaml -affinity: - nodeAffinity: - requiredDuringSchedulingIgnoredDuringExecution: - nodeSelectorTerms: - - matchExpressions: - - key: foo.bar.com/role - operator: In - values: - - master -``` -#### **cainjector.tolerations** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -A list of Kubernetes Tolerations, if required. For more information, see [Toleration v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#toleration-v1-core). - -For example: - -```yaml -tolerations: -- key: foo.bar.com/role - operator: Equal - value: master - effect: NoSchedule -``` -#### **cainjector.topologySpreadConstraints** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -A list of Kubernetes TopologySpreadConstraints, if required. For more information, see [Topology spread constraint v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#topologyspreadconstraint-v1-core). - -For example: - -```yaml -topologySpreadConstraints: -- maxSkew: 2 - topologyKey: topology.kubernetes.io/zone - whenUnsatisfiable: ScheduleAnyway - labelSelector: - matchLabels: - app.kubernetes.io/instance: cert-manager - app.kubernetes.io/component: controller -``` -#### **cainjector.podLabels** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Optional additional labels to add to the CA Injector Pods. -#### **cainjector.image.registry** ~ `string` - -The container registry to pull the cainjector image from. - -#### **cainjector.image.repository** ~ `string` -> Default value: -> ```yaml -> quay.io/jetstack/cert-manager-cainjector -> ``` - -The container image for the cert-manager cainjector - -#### **cainjector.image.tag** ~ `string` - -Override the image tag to deploy by setting this variable. If no value is set, the chart's appVersion will be used. - -#### **cainjector.image.digest** ~ `string` - -Setting a digest will override any tag. - -#### **cainjector.image.pullPolicy** ~ `string` -> Default value: -> ```yaml -> IfNotPresent -> ``` - -Kubernetes imagePullPolicy on Deployment. -#### **cainjector.serviceAccount.create** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -Specifies whether a service account should be created. -#### **cainjector.serviceAccount.name** ~ `string` - -The name of the service account to use. -If not set and create is true, a name is generated using the fullname template - -#### **cainjector.serviceAccount.annotations** ~ `object` - -Optional additional annotations to add to the controller's Service Account. - -#### **cainjector.serviceAccount.labels** ~ `object` - -Optional additional labels to add to the cainjector's Service Account. - -#### **cainjector.serviceAccount.automountServiceAccountToken** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -Automount API credentials for a Service Account. -#### **cainjector.automountServiceAccountToken** ~ `bool` - -Automounting API credentials for a particular pod. - -#### **cainjector.volumes** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Additional volumes to add to the cert-manager controller pod. -#### **cainjector.volumeMounts** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Additional volume mounts to add to the cert-manager controller container. -#### **cainjector.enableServiceLinks** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -enableServiceLinks indicates whether information about services should be injected into the pod's environment variables, matching the syntax of Docker links. -### ACME Solver - -#### **acmesolver.image.registry** ~ `string` - -The container registry to pull the acmesolver image from. - -#### **acmesolver.image.repository** ~ `string` -> Default value: -> ```yaml -> quay.io/jetstack/cert-manager-acmesolver -> ``` - -The container image for the cert-manager acmesolver. - -#### **acmesolver.image.tag** ~ `string` - -Override the image tag to deploy by setting this variable. If no value is set, the chart's appVersion is used. - -#### **acmesolver.image.digest** ~ `string` - -Setting a digest will override any tag. - -#### **acmesolver.image.pullPolicy** ~ `string` -> Default value: -> ```yaml -> IfNotPresent -> ``` - -Kubernetes imagePullPolicy on Deployment. -### Startup API Check - - -This startupapicheck is a Helm post-install hook that waits for the webhook endpoints to become available. The check is implemented using a Kubernetes Job - if you are injecting mesh sidecar proxies into cert-manager pods, ensure that they are not injected into this Job's pod. Otherwise, the installation may time out owing to the Job never being completed because the sidecar proxy does not exit. For more information, see [this note](https://github.com/cert-manager/cert-manager/pull/4414). -#### **startupapicheck.enabled** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -Enables the startup api check. -#### **startupapicheck.securityContext** ~ `object` -> Default value: -> ```yaml -> runAsNonRoot: true -> seccompProfile: -> type: RuntimeDefault -> ``` - -Pod Security Context to be set on the startupapicheck component Pod. For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). - -#### **startupapicheck.containerSecurityContext** ~ `object` -> Default value: -> ```yaml -> allowPrivilegeEscalation: false -> capabilities: -> drop: -> - ALL -> readOnlyRootFilesystem: true -> ``` - -Container Security Context to be set on the controller component container. For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). - -#### **startupapicheck.timeout** ~ `string` -> Default value: -> ```yaml -> 1m -> ``` - -Timeout for 'kubectl check api' command. -#### **startupapicheck.backoffLimit** ~ `number` -> Default value: -> ```yaml -> 4 -> ``` - -Job backoffLimit -#### **startupapicheck.jobAnnotations** ~ `object` -> Default value: -> ```yaml -> helm.sh/hook: post-install -> helm.sh/hook-delete-policy: before-hook-creation,hook-succeeded -> helm.sh/hook-weight: "1" -> ``` - -Optional additional annotations to add to the startupapicheck Job. - -#### **startupapicheck.podAnnotations** ~ `object` - -Optional additional annotations to add to the startupapicheck Pods. - -#### **startupapicheck.extraArgs** ~ `array` -> Default value: -> ```yaml -> - -v -> ``` - -Additional command line flags to pass to startupapicheck binary. To see all available flags run `docker run quay.io/jetstack/cert-manager-startupapicheck: --help`. - -Verbose logging is enabled by default so that if startupapicheck fails, you can know what exactly caused the failure. Verbose logs include details of the webhook URL, IP address and TCP connect errors for example. - -#### **startupapicheck.resources** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Resources to provide to the cert-manager controller pod. - -For example: - -```yaml -requests: - cpu: 10m - memory: 32Mi -``` - -For more information, see [Resource Management for Pods and Containers](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/). -#### **startupapicheck.nodeSelector** ~ `object` -> Default value: -> ```yaml -> kubernetes.io/os: linux -> ``` - -The nodeSelector on Pods tells Kubernetes to schedule Pods on the nodes with matching labels. For more information, see [Assigning Pods to Nodes](https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/). - -This default ensures that Pods are only scheduled to Linux nodes. It prevents Pods being scheduled to Windows nodes in a mixed OS cluster. - -#### **startupapicheck.affinity** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -A Kubernetes Affinity, if required. For more information, see [Affinity v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#affinity-v1-core). -For example: - -```yaml -affinity: - nodeAffinity: - requiredDuringSchedulingIgnoredDuringExecution: - nodeSelectorTerms: - - matchExpressions: - - key: foo.bar.com/role - operator: In - values: - - master -``` -#### **startupapicheck.tolerations** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -A list of Kubernetes Tolerations, if required. For more information, see [Toleration v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#toleration-v1-core). - -For example: - -```yaml -tolerations: -- key: foo.bar.com/role - operator: Equal - value: master - effect: NoSchedule -``` -#### **startupapicheck.podLabels** ~ `object` -> Default value: -> ```yaml -> {} -> ``` - -Optional additional labels to add to the startupapicheck Pods. -#### **startupapicheck.image.registry** ~ `string` - -The container registry to pull the startupapicheck image from. - -#### **startupapicheck.image.repository** ~ `string` -> Default value: -> ```yaml -> quay.io/jetstack/cert-manager-startupapicheck -> ``` - -The container image for the cert-manager startupapicheck. - -#### **startupapicheck.image.tag** ~ `string` - -Override the image tag to deploy by setting this variable. If no value is set, the chart's appVersion is used. - -#### **startupapicheck.image.digest** ~ `string` - -Setting a digest will override any tag. - -#### **startupapicheck.image.pullPolicy** ~ `string` -> Default value: -> ```yaml -> IfNotPresent -> ``` - -Kubernetes imagePullPolicy on Deployment. -#### **startupapicheck.rbac.annotations** ~ `object` -> Default value: -> ```yaml -> helm.sh/hook: post-install -> helm.sh/hook-delete-policy: before-hook-creation,hook-succeeded -> helm.sh/hook-weight: "-5" -> ``` - -annotations for the startup API Check job RBAC and PSP resources. - -#### **startupapicheck.automountServiceAccountToken** ~ `bool` - -Automounting API credentials for a particular pod. - -#### **startupapicheck.serviceAccount.create** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -Specifies whether a service account should be created. -#### **startupapicheck.serviceAccount.name** ~ `string` - -The name of the service account to use. -If not set and create is true, a name is generated using the fullname template. - -#### **startupapicheck.serviceAccount.annotations** ~ `object` -> Default value: -> ```yaml -> helm.sh/hook: post-install -> helm.sh/hook-delete-policy: before-hook-creation,hook-succeeded -> helm.sh/hook-weight: "-5" -> ``` - -Optional additional annotations to add to the Job's Service Account. - -#### **startupapicheck.serviceAccount.automountServiceAccountToken** ~ `bool` -> Default value: -> ```yaml -> true -> ``` - -Automount API credentials for a Service Account. - -#### **startupapicheck.serviceAccount.labels** ~ `object` - -Optional additional labels to add to the startupapicheck's Service Account. - -#### **startupapicheck.volumes** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Additional volumes to add to the cert-manager controller pod. -#### **startupapicheck.volumeMounts** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Additional volume mounts to add to the cert-manager controller container. -#### **startupapicheck.enableServiceLinks** ~ `bool` -> Default value: -> ```yaml -> false -> ``` - -enableServiceLinks indicates whether information about services should be injected into pod's environment variables, matching the syntax of Docker links. -#### **extraObjects** ~ `array` -> Default value: -> ```yaml -> [] -> ``` - -Create dynamic manifests via values. - -For example: - -```yaml -extraObjects: - - | - apiVersion: v1 - kind: ConfigMap - metadata: - name: '{{ template "cert-manager.name" . }}-extra-configmap' -``` - - -### Default Security Contexts - -The default pod-level and container-level security contexts, below, adhere to the [restricted](https://kubernetes.io/docs/concepts/security/pod-security-standards/#restricted) Pod Security Standards policies. - -Default pod-level securityContext: -```yaml -runAsNonRoot: true -seccompProfile: - type: RuntimeDefault -``` - -Default containerSecurityContext: -```yaml -allowPrivilegeEscalation: false -capabilities: - drop: - - ALL -``` - -### Assigning Values - -Specify each parameter using the `--set key=value[,key=value]` argument to `helm install`. - -Alternatively, a YAML file that specifies the values for the above parameters can be provided while installing the chart. For example, - -```console -$ helm install my-release -f values.yaml . -``` -> **Tip**: You can use the default [values.yaml](https://github.com/cert-manager/cert-manager/blob/master/deploy/charts/cert-manager/values.yaml) - -## Contributing - -This chart is maintained at [github.com/cert-manager/cert-manager](https://github.com/cert-manager/cert-manager/tree/master/deploy/charts/cert-manager). diff --git a/charts/cert-manager/cert-manager/templates/NOTES.txt b/charts/cert-manager/cert-manager/templates/NOTES.txt deleted file mode 100644 index 341d101..0000000 --- a/charts/cert-manager/cert-manager/templates/NOTES.txt +++ /dev/null @@ -1,18 +0,0 @@ -{{- if .Values.installCRDs }} -⚠️ WARNING: `installCRDs` is deprecated, use `crds.enabled` instead. -{{- end }} -cert-manager {{ .Chart.AppVersion }} has been deployed successfully! - -In order to begin issuing certificates, you will need to set up a ClusterIssuer -or Issuer resource (for example, by creating a 'letsencrypt-staging' issuer). - -More information on the different types of issuers and how to configure them -can be found in our documentation: - -https://cert-manager.io/docs/configuration/ - -For information on how to configure cert-manager to automatically provision -Certificates for Ingress resources, take a look at the `ingress-shim` -documentation: - -https://cert-manager.io/docs/usage/ingress/ diff --git a/charts/cert-manager/cert-manager/templates/_helpers.tpl b/charts/cert-manager/cert-manager/templates/_helpers.tpl deleted file mode 100644 index 9902c08..0000000 --- a/charts/cert-manager/cert-manager/templates/_helpers.tpl +++ /dev/null @@ -1,202 +0,0 @@ -{{/* vim: set filetype=mustache: */}} -{{/* -Expand the name of the chart. -*/}} -{{- define "cert-manager.name" -}} -{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Create a default fully qualified app name. -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -*/}} -{{- define "cert-manager.fullname" -}} -{{- if .Values.fullnameOverride -}} -{{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" -}} -{{- else -}} -{{- $name := default .Chart.Name .Values.nameOverride -}} -{{- if contains $name .Release.Name -}} -{{- .Release.Name | trunc 63 | trimSuffix "-" -}} -{{- else -}} -{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} -{{- end -}} -{{- end -}} -{{- end -}} - -{{/* -Create the name of the service account to use -*/}} -{{- define "cert-manager.serviceAccountName" -}} -{{- if .Values.serviceAccount.create -}} - {{ default (include "cert-manager.fullname" .) .Values.serviceAccount.name }} -{{- else -}} - {{ default "default" .Values.serviceAccount.name }} -{{- end -}} -{{- end -}} - -{{/* -Webhook templates -*/}} - -{{/* -Expand the name of the chart. -Manually fix the 'app' and 'name' labels to 'webhook' to maintain -compatibility with the v0.9 deployment selector. -*/}} -{{- define "webhook.name" -}} -{{- printf "webhook" -}} -{{- end -}} - -{{/* -Create a default fully qualified app name. -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -If release name contains chart name it will be used as a full name. -*/}} -{{- define "webhook.fullname" -}} -{{- $trimmedName := printf "%s" (include "cert-manager.fullname" .) | trunc 55 | trimSuffix "-" -}} -{{- printf "%s-webhook" $trimmedName | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{- define "webhook.caRef" -}} -{{- template "cert-manager.namespace" }}/{{ template "webhook.fullname" . }}-ca -{{- end -}} - -{{/* -Create the name of the service account to use -*/}} -{{- define "webhook.serviceAccountName" -}} -{{- if .Values.webhook.serviceAccount.create -}} - {{ default (include "webhook.fullname" .) .Values.webhook.serviceAccount.name }} -{{- else -}} - {{ default "default" .Values.webhook.serviceAccount.name }} -{{- end -}} -{{- end -}} - -{{/* -cainjector templates -*/}} - -{{/* -Expand the name of the chart. -Manually fix the 'app' and 'name' labels to 'cainjector' to maintain -compatibility with the v0.9 deployment selector. -*/}} -{{- define "cainjector.name" -}} -{{- printf "cainjector" -}} -{{- end -}} - -{{/* -Create a default fully qualified app name. -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -If release name contains chart name it will be used as a full name. -*/}} -{{- define "cainjector.fullname" -}} -{{- $trimmedName := printf "%s" (include "cert-manager.fullname" .) | trunc 52 | trimSuffix "-" -}} -{{- printf "%s-cainjector" $trimmedName | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Create the name of the service account to use -*/}} -{{- define "cainjector.serviceAccountName" -}} -{{- if .Values.cainjector.serviceAccount.create -}} - {{ default (include "cainjector.fullname" .) .Values.cainjector.serviceAccount.name }} -{{- else -}} - {{ default "default" .Values.cainjector.serviceAccount.name }} -{{- end -}} -{{- end -}} - -{{/* -startupapicheck templates -*/}} - -{{/* -Expand the name of the chart. -Manually fix the 'app' and 'name' labels to 'startupapicheck' to maintain -compatibility with the v0.9 deployment selector. -*/}} -{{- define "startupapicheck.name" -}} -{{- printf "startupapicheck" -}} -{{- end -}} - -{{/* -Create a default fully qualified app name. -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -If release name contains chart name it will be used as a full name. -*/}} -{{- define "startupapicheck.fullname" -}} -{{- $trimmedName := printf "%s" (include "cert-manager.fullname" .) | trunc 52 | trimSuffix "-" -}} -{{- printf "%s-startupapicheck" $trimmedName | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Create the name of the service account to use -*/}} -{{- define "startupapicheck.serviceAccountName" -}} -{{- if .Values.startupapicheck.serviceAccount.create -}} - {{ default (include "startupapicheck.fullname" .) .Values.startupapicheck.serviceAccount.name }} -{{- else -}} - {{ default "default" .Values.startupapicheck.serviceAccount.name }} -{{- end -}} -{{- end -}} - -{{/* -Create chart name and version as used by the chart label. -*/}} -{{- define "chartName" -}} -{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Labels that should be added on each resource -*/}} -{{- define "labels" -}} -app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} -{{- if eq (default "helm" .Values.creator) "helm" }} -app.kubernetes.io/managed-by: {{ .Release.Service }} -helm.sh/chart: {{ include "chartName" . }} -{{- end -}} -{{- if .Values.global.commonLabels}} -{{ toYaml .Values.global.commonLabels }} -{{- end }} -{{- end -}} - -{{/* -Namespace for all resources to be installed into -If not defined in values file then the helm release namespace is used -By default this is not set so the helm release namespace will be used - -This gets around an problem within helm discussed here -https://github.com/helm/helm/issues/5358 -*/}} -{{- define "cert-manager.namespace" -}} - {{ .Values.namespace | default .Release.Namespace }} -{{- end -}} - -{{/* -Util function for generating the image URL based on the provided options. -IMPORTANT: This function is standarized across all charts in the cert-manager GH organization. -Any changes to this function should also be made in cert-manager, trust-manager, approver-policy, ... -See https://github.com/cert-manager/cert-manager/issues/6329 for a list of linked PRs. -*/}} -{{- define "image" -}} -{{- $defaultTag := index . 1 -}} -{{- with index . 0 -}} -{{- if .registry -}}{{ printf "%s/%s" .registry .repository }}{{- else -}}{{- .repository -}}{{- end -}} -{{- if .digest -}}{{ printf "@%s" .digest }}{{- else -}}{{ printf ":%s" (default $defaultTag .tag) }}{{- end -}} -{{- end }} -{{- end }} - -{{/* -Check that the user has not set both .installCRDs and .crds.enabled or -set .installCRDs and disabled .crds.keep. -.installCRDs is deprecated and users should use .crds.enabled and .crds.keep instead. -*/}} -{{- define "cert-manager.crd-check" -}} - {{- if and (.Values.installCRDs) (.Values.crds.enabled) }} - {{- fail "ERROR: the deprecated .installCRDs option cannot be enabled at the same time as its replacement .crds.enabled" }} - {{- end }} - {{- if and (.Values.installCRDs) (not .Values.crds.keep) }} - {{- fail "ERROR: .crds.keep is not compatible with .installCRDs, please use .crds.enabled and .crds.keep instead" }} - {{- end }} -{{- end -}} diff --git a/charts/cert-manager/cert-manager/templates/cainjector-config.yaml b/charts/cert-manager/cert-manager/templates/cainjector-config.yaml deleted file mode 100644 index 82399cc..0000000 --- a/charts/cert-manager/cert-manager/templates/cainjector-config.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- if .Values.cainjector.config -}} -{{- $_ := .Values.cainjector.config.apiVersion | required ".Values.cainjector.config.apiVersion must be set !" -}} -{{- $_ := .Values.cainjector.config.kind | required ".Values.cainjector.config.kind must be set !" -}} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ include "cainjector.fullname" . }} - namespace: {{ include "cert-manager.namespace" . }} - labels: - app: {{ include "cainjector.name" . }} - app.kubernetes.io/name: {{ include "cainjector.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cainjector" - {{- include "labels" . | nindent 4 }} -data: - config.yaml: | - {{- .Values.cainjector.config | toYaml | nindent 4 }} -{{- end -}} \ No newline at end of file diff --git a/charts/cert-manager/cert-manager/templates/cainjector-deployment.yaml b/charts/cert-manager/cert-manager/templates/cainjector-deployment.yaml deleted file mode 100644 index 8f9f7f3..0000000 --- a/charts/cert-manager/cert-manager/templates/cainjector-deployment.yaml +++ /dev/null @@ -1,140 +0,0 @@ -{{- if .Values.cainjector.enabled }} -apiVersion: apps/v1 -kind: Deployment -metadata: - name: {{ include "cainjector.fullname" . }} - namespace: {{ include "cert-manager.namespace" . }} - labels: - app: {{ include "cainjector.name" . }} - app.kubernetes.io/name: {{ include "cainjector.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cainjector" - {{- include "labels" . | nindent 4 }} - {{- with .Values.cainjector.deploymentAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - replicas: {{ .Values.cainjector.replicaCount }} - {{- /* The if statement below is equivalent to {{- if $value }} but will also return true for 0. */ -}} - {{- if not (has (quote .Values.global.revisionHistoryLimit) (list "" (quote ""))) }} - revisionHistoryLimit: {{ .Values.global.revisionHistoryLimit }} - {{- end }} - selector: - matchLabels: - app.kubernetes.io/name: {{ include "cainjector.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cainjector" - {{- with .Values.cainjector.strategy }} - strategy: - {{- toYaml . | nindent 4 }} - {{- end }} - template: - metadata: - labels: - app: {{ include "cainjector.name" . }} - app.kubernetes.io/name: {{ include "cainjector.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cainjector" - {{- include "labels" . | nindent 8 }} - {{- with .Values.cainjector.podLabels }} - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.cainjector.podAnnotations }} - annotations: - {{- toYaml . | nindent 8 }} - {{- end }} - spec: - serviceAccountName: {{ template "cainjector.serviceAccountName" . }} - {{- if hasKey .Values.cainjector "automountServiceAccountToken" }} - automountServiceAccountToken: {{ .Values.cainjector.automountServiceAccountToken }} - {{- end }} - enableServiceLinks: {{ .Values.cainjector.enableServiceLinks }} - {{- with .Values.global.priorityClassName }} - priorityClassName: {{ . | quote }} - {{- end }} - {{- with .Values.cainjector.securityContext }} - securityContext: - {{- toYaml . | nindent 8 }} - {{- end }} - containers: - - name: {{ .Chart.Name }}-cainjector - image: "{{ template "image" (tuple .Values.cainjector.image $.Chart.AppVersion) }}" - imagePullPolicy: {{ .Values.cainjector.image.pullPolicy }} - args: - {{- /* The if statement below is equivalent to {{- if $value }} but will also return true for 0. */ -}} - {{- if not (has (quote .Values.global.logLevel) (list "" (quote ""))) }} - - --v={{ .Values.global.logLevel }} - {{- end }} - {{- if .Values.cainjector.config }} - - --config=/var/cert-manager/config/config.yaml - {{- end }} - {{- with .Values.global.leaderElection }} - - --leader-election-namespace={{ .namespace }} - {{- if .leaseDuration }} - - --leader-election-lease-duration={{ .leaseDuration }} - {{- end }} - {{- if .renewDeadline }} - - --leader-election-renew-deadline={{ .renewDeadline }} - {{- end }} - {{- if .retryPeriod }} - - --leader-election-retry-period={{ .retryPeriod }} - {{- end }} - {{- end }} - {{- with .Values.cainjector.featureGates}} - - --feature-gates={{ . }} - {{- end}} - {{- with .Values.cainjector.extraArgs }} - {{- toYaml . | nindent 10 }} - {{- end }} - env: - - name: POD_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - {{- with .Values.cainjector.containerSecurityContext }} - securityContext: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- with .Values.cainjector.resources }} - resources: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- if or .Values.cainjector.config .Values.cainjector.volumeMounts }} - volumeMounts: - {{- if .Values.cainjector.config }} - - name: config - mountPath: /var/cert-manager/config - {{- end }} - {{- with .Values.cainjector.volumeMounts }} - {{- toYaml . | nindent 12 }} - {{- end }} - {{- end }} - {{- with .Values.cainjector.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.cainjector.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.cainjector.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.cainjector.topologySpreadConstraints }} - topologySpreadConstraints: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- if or .Values.cainjector.volumes .Values.cainjector.config }} - volumes: - {{- if .Values.cainjector.config }} - - name: config - configMap: - name: {{ include "cainjector.fullname" . }} - {{- end }} - {{ with .Values.cainjector.volumes }} - {{- toYaml . | nindent 8 }} - {{- end }} - {{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/cainjector-poddisruptionbudget.yaml b/charts/cert-manager/cert-manager/templates/cainjector-poddisruptionbudget.yaml deleted file mode 100644 index 6a7d609..0000000 --- a/charts/cert-manager/cert-manager/templates/cainjector-poddisruptionbudget.yaml +++ /dev/null @@ -1,29 +0,0 @@ -{{- if .Values.cainjector.podDisruptionBudget.enabled }} -apiVersion: policy/v1 -kind: PodDisruptionBudget -metadata: - name: {{ include "cainjector.fullname" . }} - namespace: {{ include "cert-manager.namespace" . }} - labels: - app: {{ include "cainjector.name" . }} - app.kubernetes.io/name: {{ include "cainjector.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cainjector" - {{- include "labels" . | nindent 4 }} -spec: - selector: - matchLabels: - app.kubernetes.io/name: {{ include "cainjector.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cainjector" - - {{- if not (or (hasKey .Values.cainjector.podDisruptionBudget "minAvailable") (hasKey .Values.cainjector.podDisruptionBudget "maxUnavailable")) }} - minAvailable: 1 # Default value because minAvailable and maxUnavailable are not set - {{- end }} - {{- if hasKey .Values.cainjector.podDisruptionBudget "minAvailable" }} - minAvailable: {{ .Values.cainjector.podDisruptionBudget.minAvailable }} - {{- end }} - {{- if hasKey .Values.cainjector.podDisruptionBudget "maxUnavailable" }} - maxUnavailable: {{ .Values.cainjector.podDisruptionBudget.maxUnavailable }} - {{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/cainjector-psp-clusterrole.yaml b/charts/cert-manager/cert-manager/templates/cainjector-psp-clusterrole.yaml deleted file mode 100644 index b75b9eb..0000000 --- a/charts/cert-manager/cert-manager/templates/cainjector-psp-clusterrole.yaml +++ /dev/null @@ -1,20 +0,0 @@ -{{- if .Values.cainjector.enabled }} -{{- if .Values.global.podSecurityPolicy.enabled }} -kind: ClusterRole -apiVersion: rbac.authorization.k8s.io/v1 -metadata: - name: {{ template "cainjector.fullname" . }}-psp - labels: - app: {{ include "cainjector.name" . }} - app.kubernetes.io/name: {{ include "cainjector.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cainjector" - {{- include "labels" . | nindent 4 }} -rules: -- apiGroups: ['policy'] - resources: ['podsecuritypolicies'] - verbs: ['use'] - resourceNames: - - {{ template "cainjector.fullname" . }} -{{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/cainjector-psp-clusterrolebinding.yaml b/charts/cert-manager/cert-manager/templates/cainjector-psp-clusterrolebinding.yaml deleted file mode 100644 index e2bfa26..0000000 --- a/charts/cert-manager/cert-manager/templates/cainjector-psp-clusterrolebinding.yaml +++ /dev/null @@ -1,22 +0,0 @@ -{{- if .Values.cainjector.enabled }} -{{- if .Values.global.podSecurityPolicy.enabled }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ template "cainjector.fullname" . }}-psp - labels: - app: {{ include "cainjector.name" . }} - app.kubernetes.io/name: {{ include "cainjector.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cainjector" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ template "cainjector.fullname" . }}-psp -subjects: - - kind: ServiceAccount - name: {{ template "cainjector.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} -{{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/cainjector-psp.yaml b/charts/cert-manager/cert-manager/templates/cainjector-psp.yaml deleted file mode 100644 index 24f01da..0000000 --- a/charts/cert-manager/cert-manager/templates/cainjector-psp.yaml +++ /dev/null @@ -1,51 +0,0 @@ -{{- if .Values.cainjector.enabled }} -{{- if .Values.global.podSecurityPolicy.enabled }} -apiVersion: policy/v1beta1 -kind: PodSecurityPolicy -metadata: - name: {{ template "cainjector.fullname" . }} - labels: - app: {{ include "cainjector.name" . }} - app.kubernetes.io/name: {{ include "cainjector.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cainjector" - {{- include "labels" . | nindent 4 }} - annotations: - seccomp.security.alpha.kubernetes.io/allowedProfileNames: 'docker/default' - seccomp.security.alpha.kubernetes.io/defaultProfileName: 'docker/default' - {{- if .Values.global.podSecurityPolicy.useAppArmor }} - apparmor.security.beta.kubernetes.io/allowedProfileNames: 'runtime/default' - apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' - {{- end }} -spec: - privileged: false - allowPrivilegeEscalation: false - allowedCapabilities: [] # default set of capabilities are implicitly allowed - volumes: - - 'configMap' - - 'emptyDir' - - 'projected' - - 'secret' - - 'downwardAPI' - hostNetwork: false - hostIPC: false - hostPID: false - runAsUser: - rule: 'MustRunAs' - ranges: - - min: 1000 - max: 1000 - seLinux: - rule: 'RunAsAny' - supplementalGroups: - rule: 'MustRunAs' - ranges: - - min: 1000 - max: 1000 - fsGroup: - rule: 'MustRunAs' - ranges: - - min: 1000 - max: 1000 -{{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/cainjector-rbac.yaml b/charts/cert-manager/cert-manager/templates/cainjector-rbac.yaml deleted file mode 100644 index 2aa59ee..0000000 --- a/charts/cert-manager/cert-manager/templates/cainjector-rbac.yaml +++ /dev/null @@ -1,103 +0,0 @@ -{{- if .Values.cainjector.enabled }} -{{- if .Values.global.rbac.create }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ template "cainjector.fullname" . }} - labels: - app: {{ include "cainjector.name" . }} - app.kubernetes.io/name: {{ include "cainjector.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cainjector" - {{- include "labels" . | nindent 4 }} -rules: - - apiGroups: ["cert-manager.io"] - resources: ["certificates"] - verbs: ["get", "list", "watch"] - - apiGroups: [""] - resources: ["secrets"] - verbs: ["get", "list", "watch"] - - apiGroups: [""] - resources: ["events"] - verbs: ["get", "create", "update", "patch"] - - apiGroups: ["admissionregistration.k8s.io"] - resources: ["validatingwebhookconfigurations", "mutatingwebhookconfigurations"] - verbs: ["get", "list", "watch", "update", "patch"] - - apiGroups: ["apiregistration.k8s.io"] - resources: ["apiservices"] - verbs: ["get", "list", "watch", "update", "patch"] - - apiGroups: ["apiextensions.k8s.io"] - resources: ["customresourcedefinitions"] - verbs: ["get", "list", "watch", "update", "patch"] ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ template "cainjector.fullname" . }} - labels: - app: {{ include "cainjector.name" . }} - app.kubernetes.io/name: {{ include "cainjector.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cainjector" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ template "cainjector.fullname" . }} -subjects: - - name: {{ template "cainjector.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - kind: ServiceAccount - ---- -# leader election rules -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: {{ template "cainjector.fullname" . }}:leaderelection - namespace: {{ .Values.global.leaderElection.namespace }} - labels: - app: {{ include "cainjector.name" . }} - app.kubernetes.io/name: {{ include "cainjector.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cainjector" - {{- include "labels" . | nindent 4 }} -rules: - # Used for leader election by the controller - # cert-manager-cainjector-leader-election is used by the CertificateBased injector controller - # see cmd/cainjector/start.go#L113 - # cert-manager-cainjector-leader-election-core is used by the SecretBased injector controller - # see cmd/cainjector/start.go#L137 - - apiGroups: ["coordination.k8s.io"] - resources: ["leases"] - resourceNames: ["cert-manager-cainjector-leader-election", "cert-manager-cainjector-leader-election-core"] - verbs: ["get", "update", "patch"] - - apiGroups: ["coordination.k8s.io"] - resources: ["leases"] - verbs: ["create"] - ---- - -# grant cert-manager permission to manage the leaderelection configmap in the -# leader election namespace -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: {{ include "cainjector.fullname" . }}:leaderelection - namespace: {{ .Values.global.leaderElection.namespace }} - labels: - app: {{ include "cainjector.name" . }} - app.kubernetes.io/name: {{ include "cainjector.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cainjector" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: {{ template "cainjector.fullname" . }}:leaderelection -subjects: - - kind: ServiceAccount - name: {{ template "cainjector.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} -{{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/cainjector-serviceaccount.yaml b/charts/cert-manager/cert-manager/templates/cainjector-serviceaccount.yaml deleted file mode 100644 index fedc731..0000000 --- a/charts/cert-manager/cert-manager/templates/cainjector-serviceaccount.yaml +++ /dev/null @@ -1,27 +0,0 @@ -{{- if .Values.cainjector.enabled }} -{{- if .Values.cainjector.serviceAccount.create }} -apiVersion: v1 -kind: ServiceAccount -automountServiceAccountToken: {{ .Values.cainjector.serviceAccount.automountServiceAccountToken }} -metadata: - name: {{ template "cainjector.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - {{- with .Values.cainjector.serviceAccount.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} - labels: - app: {{ include "cainjector.name" . }} - app.kubernetes.io/name: {{ include "cainjector.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cainjector" - {{- include "labels" . | nindent 4 }} - {{- with .Values.cainjector.serviceAccount.labels }} - {{ toYaml . | nindent 4 }} - {{- end }} -{{- with .Values.global.imagePullSecrets }} -imagePullSecrets: - {{- toYaml . | nindent 2 }} -{{- end }} -{{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/controller-config.yaml b/charts/cert-manager/cert-manager/templates/controller-config.yaml deleted file mode 100644 index 25f62ef..0000000 --- a/charts/cert-manager/cert-manager/templates/controller-config.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- if .Values.config -}} -{{- $_ := .Values.config.apiVersion | required ".Values.config.apiVersion must be set !" -}} -{{- $_ := .Values.config.kind | required ".Values.config.kind must be set !" -}} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ include "cert-manager.fullname" . }} - namespace: {{ include "cert-manager.namespace" . }} - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -data: - config.yaml: | - {{- .Values.config | toYaml | nindent 4 }} -{{- end -}} \ No newline at end of file diff --git a/charts/cert-manager/cert-manager/templates/crds.yaml b/charts/cert-manager/cert-manager/templates/crds.yaml deleted file mode 100644 index 2c70ca3..0000000 --- a/charts/cert-manager/cert-manager/templates/crds.yaml +++ /dev/null @@ -1,8179 +0,0 @@ -# {{- include "cert-manager.crd-check" . }} -# START crd {{- if or .Values.crds.enabled .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - name: certificaterequests.cert-manager.io - # START annotations {{- if .Values.crds.keep }} - annotations: - helm.sh/resource-policy: keep - # END annotations {{- end }} - labels: - app: '{{ template "cert-manager.name" . }}' - app.kubernetes.io/name: '{{ template "cert-manager.name" . }}' - app.kubernetes.io/instance: '{{ .Release.Name }}' - # Generated labels {{- include "labels" . | nindent 4 }} -spec: - group: cert-manager.io - names: - kind: CertificateRequest - listKind: CertificateRequestList - plural: certificaterequests - shortNames: - - cr - - crs - singular: certificaterequest - categories: - - cert-manager - scope: Namespaced - versions: - - name: v1 - subresources: - status: {} - additionalPrinterColumns: - - jsonPath: .status.conditions[?(@.type=="Approved")].status - name: Approved - type: string - - jsonPath: .status.conditions[?(@.type=="Denied")].status - name: Denied - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .spec.issuerRef.name - name: Issuer - type: string - - jsonPath: .spec.username - name: Requestor - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - priority: 1 - type: string - - jsonPath: .metadata.creationTimestamp - description: CreationTimestamp is a timestamp representing the server time when this object was created. It is not guaranteed to be set in happens-before order across separate operations. Clients may not set this value. It is represented in RFC3339 form and is in UTC. - name: Age - type: date - schema: - openAPIV3Schema: - description: |- - A CertificateRequest is used to request a signed certificate from one of the - configured issuers. - - - All fields within the CertificateRequest's `spec` are immutable after creation. - A CertificateRequest will either succeed or fail, as denoted by its `Ready` status - condition and its `status.failureTime` field. - - - A CertificateRequest is a one-shot resource, meaning it represents a single - point in time request for a certificate and cannot be re-used. - type: object - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - Specification of the desired state of the CertificateRequest resource. - https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#spec-and-status - type: object - required: - - issuerRef - - request - properties: - duration: - description: |- - Requested 'duration' (i.e. lifetime) of the Certificate. Note that the - issuer may choose to ignore the requested duration, just like any other - requested attribute. - type: string - extra: - description: |- - Extra contains extra attributes of the user that created the CertificateRequest. - Populated by the cert-manager webhook on creation and immutable. - type: object - additionalProperties: - type: array - items: - type: string - groups: - description: |- - Groups contains group membership of the user that created the CertificateRequest. - Populated by the cert-manager webhook on creation and immutable. - type: array - items: - type: string - x-kubernetes-list-type: atomic - isCA: - description: |- - Requested basic constraints isCA value. Note that the issuer may choose - to ignore the requested isCA value, just like any other requested attribute. - - - NOTE: If the CSR in the `Request` field has a BasicConstraints extension, - it must have the same isCA value as specified here. - - - If true, this will automatically add the `cert sign` usage to the list - of requested `usages`. - type: boolean - issuerRef: - description: |- - Reference to the issuer responsible for issuing the certificate. - If the issuer is namespace-scoped, it must be in the same namespace - as the Certificate. If the issuer is cluster-scoped, it can be used - from any namespace. - - - The `name` field of the reference must always be specified. - type: object - required: - - name - properties: - group: - description: Group of the resource being referred to. - type: string - kind: - description: Kind of the resource being referred to. - type: string - name: - description: Name of the resource being referred to. - type: string - request: - description: |- - The PEM-encoded X.509 certificate signing request to be submitted to the - issuer for signing. - - - If the CSR has a BasicConstraints extension, its isCA attribute must - match the `isCA` value of this CertificateRequest. - If the CSR has a KeyUsage extension, its key usages must match the - key usages in the `usages` field of this CertificateRequest. - If the CSR has a ExtKeyUsage extension, its extended key usages - must match the extended key usages in the `usages` field of this - CertificateRequest. - type: string - format: byte - uid: - description: |- - UID contains the uid of the user that created the CertificateRequest. - Populated by the cert-manager webhook on creation and immutable. - type: string - usages: - description: |- - Requested key usages and extended key usages. - - - NOTE: If the CSR in the `Request` field has uses the KeyUsage or - ExtKeyUsage extension, these extensions must have the same values - as specified here without any additional values. - - - If unset, defaults to `digital signature` and `key encipherment`. - type: array - items: - description: |- - KeyUsage specifies valid usage contexts for keys. - See: - https://tools.ietf.org/html/rfc5280#section-4.2.1.3 - https://tools.ietf.org/html/rfc5280#section-4.2.1.12 - - - Valid KeyUsage values are as follows: - "signing", - "digital signature", - "content commitment", - "key encipherment", - "key agreement", - "data encipherment", - "cert sign", - "crl sign", - "encipher only", - "decipher only", - "any", - "server auth", - "client auth", - "code signing", - "email protection", - "s/mime", - "ipsec end system", - "ipsec tunnel", - "ipsec user", - "timestamping", - "ocsp signing", - "microsoft sgc", - "netscape sgc" - type: string - enum: - - signing - - digital signature - - content commitment - - key encipherment - - key agreement - - data encipherment - - cert sign - - crl sign - - encipher only - - decipher only - - any - - server auth - - client auth - - code signing - - email protection - - s/mime - - ipsec end system - - ipsec tunnel - - ipsec user - - timestamping - - ocsp signing - - microsoft sgc - - netscape sgc - username: - description: |- - Username contains the name of the user that created the CertificateRequest. - Populated by the cert-manager webhook on creation and immutable. - type: string - status: - description: |- - Status of the CertificateRequest. - This is set and managed automatically. - Read-only. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#spec-and-status - type: object - properties: - ca: - description: |- - The PEM encoded X.509 certificate of the signer, also known as the CA - (Certificate Authority). - This is set on a best-effort basis by different issuers. - If not set, the CA is assumed to be unknown/not available. - type: string - format: byte - certificate: - description: |- - The PEM encoded X.509 certificate resulting from the certificate - signing request. - If not set, the CertificateRequest has either not been completed or has - failed. More information on failure can be found by checking the - `conditions` field. - type: string - format: byte - conditions: - description: |- - List of status conditions to indicate the status of a CertificateRequest. - Known condition types are `Ready`, `InvalidRequest`, `Approved` and `Denied`. - type: array - items: - description: CertificateRequestCondition contains condition information for a CertificateRequest. - type: object - required: - - status - - type - properties: - lastTransitionTime: - description: |- - LastTransitionTime is the timestamp corresponding to the last status - change of this condition. - type: string - format: date-time - message: - description: |- - Message is a human readable description of the details of the last - transition, complementing reason. - type: string - reason: - description: |- - Reason is a brief machine readable explanation for the condition's last - transition. - type: string - status: - description: Status of the condition, one of (`True`, `False`, `Unknown`). - type: string - enum: - - "True" - - "False" - - Unknown - type: - description: |- - Type of the condition, known values are (`Ready`, `InvalidRequest`, - `Approved`, `Denied`). - type: string - x-kubernetes-list-map-keys: - - type - x-kubernetes-list-type: map - failureTime: - description: |- - FailureTime stores the time that this CertificateRequest failed. This is - used to influence garbage collection and back-off. - type: string - format: date-time - served: true - storage: true - -# END crd {{- end }} - ---- -# START crd {{- if or .Values.crds.enabled .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - name: certificates.cert-manager.io - # START annotations {{- if .Values.crds.keep }} - annotations: - helm.sh/resource-policy: keep - # END annotations {{- end }} - labels: - app: '{{ template "cert-manager.name" . }}' - app.kubernetes.io/name: '{{ template "cert-manager.name" . }}' - app.kubernetes.io/instance: '{{ .Release.Name }}' - # Generated labels {{- include "labels" . | nindent 4 }} -spec: - group: cert-manager.io - names: - kind: Certificate - listKind: CertificateList - plural: certificates - shortNames: - - cert - - certs - singular: certificate - categories: - - cert-manager - scope: Namespaced - versions: - - name: v1 - subresources: - status: {} - additionalPrinterColumns: - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .spec.secretName - name: Secret - type: string - - jsonPath: .spec.issuerRef.name - name: Issuer - priority: 1 - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - priority: 1 - type: string - - jsonPath: .metadata.creationTimestamp - description: CreationTimestamp is a timestamp representing the server time when this object was created. It is not guaranteed to be set in happens-before order across separate operations. Clients may not set this value. It is represented in RFC3339 form and is in UTC. - name: Age - type: date - schema: - openAPIV3Schema: - description: |- - A Certificate resource should be created to ensure an up to date and signed - X.509 certificate is stored in the Kubernetes Secret resource named in `spec.secretName`. - - - The stored certificate will be renewed before it expires (as configured by `spec.renewBefore`). - type: object - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - Specification of the desired state of the Certificate resource. - https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#spec-and-status - type: object - required: - - issuerRef - - secretName - properties: - additionalOutputFormats: - description: |- - Defines extra output formats of the private key and signed certificate chain - to be written to this Certificate's target Secret. - - - This is a Beta Feature enabled by default. It can be disabled with the - `--feature-gates=AdditionalCertificateOutputFormats=false` option set on both - the controller and webhook components. - type: array - items: - description: |- - CertificateAdditionalOutputFormat defines an additional output format of a - Certificate resource. These contain supplementary data formats of the signed - certificate chain and paired private key. - type: object - required: - - type - properties: - type: - description: |- - Type is the name of the format type that should be written to the - Certificate's target Secret. - type: string - enum: - - DER - - CombinedPEM - commonName: - description: |- - Requested common name X509 certificate subject attribute. - More info: https://datatracker.ietf.org/doc/html/rfc5280#section-4.1.2.6 - NOTE: TLS clients will ignore this value when any subject alternative name is - set (see https://tools.ietf.org/html/rfc6125#section-6.4.4). - - - Should have a length of 64 characters or fewer to avoid generating invalid CSRs. - Cannot be set if the `literalSubject` field is set. - type: string - dnsNames: - description: Requested DNS subject alternative names. - type: array - items: - type: string - duration: - description: |- - Requested 'duration' (i.e. lifetime) of the Certificate. Note that the - issuer may choose to ignore the requested duration, just like any other - requested attribute. - - - If unset, this defaults to 90 days. - Minimum accepted duration is 1 hour. - Value must be in units accepted by Go time.ParseDuration https://golang.org/pkg/time/#ParseDuration. - type: string - emailAddresses: - description: Requested email subject alternative names. - type: array - items: - type: string - encodeUsagesInRequest: - description: |- - Whether the KeyUsage and ExtKeyUsage extensions should be set in the encoded CSR. - - - This option defaults to true, and should only be disabled if the target - issuer does not support CSRs with these X509 KeyUsage/ ExtKeyUsage extensions. - type: boolean - ipAddresses: - description: Requested IP address subject alternative names. - type: array - items: - type: string - isCA: - description: |- - Requested basic constraints isCA value. - The isCA value is used to set the `isCA` field on the created CertificateRequest - resources. Note that the issuer may choose to ignore the requested isCA value, just - like any other requested attribute. - - - If true, this will automatically add the `cert sign` usage to the list - of requested `usages`. - type: boolean - issuerRef: - description: |- - Reference to the issuer responsible for issuing the certificate. - If the issuer is namespace-scoped, it must be in the same namespace - as the Certificate. If the issuer is cluster-scoped, it can be used - from any namespace. - - - The `name` field of the reference must always be specified. - type: object - required: - - name - properties: - group: - description: Group of the resource being referred to. - type: string - kind: - description: Kind of the resource being referred to. - type: string - name: - description: Name of the resource being referred to. - type: string - keystores: - description: Additional keystore output formats to be stored in the Certificate's Secret. - type: object - properties: - jks: - description: |- - JKS configures options for storing a JKS keystore in the - `spec.secretName` Secret resource. - type: object - required: - - create - - passwordSecretRef - properties: - alias: - description: |- - Alias specifies the alias of the key in the keystore, required by the JKS format. - If not provided, the default alias `certificate` will be used. - type: string - create: - description: |- - Create enables JKS keystore creation for the Certificate. - If true, a file named `keystore.jks` will be created in the target - Secret resource, encrypted using the password stored in - `passwordSecretRef`. - The keystore file will be updated immediately. - If the issuer provided a CA certificate, a file named `truststore.jks` - will also be created in the target Secret resource, encrypted using the - password stored in `passwordSecretRef` - containing the issuing Certificate Authority - type: boolean - passwordSecretRef: - description: |- - PasswordSecretRef is a reference to a key in a Secret resource - containing the password used to encrypt the JKS keystore. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - pkcs12: - description: |- - PKCS12 configures options for storing a PKCS12 keystore in the - `spec.secretName` Secret resource. - type: object - required: - - create - - passwordSecretRef - properties: - create: - description: |- - Create enables PKCS12 keystore creation for the Certificate. - If true, a file named `keystore.p12` will be created in the target - Secret resource, encrypted using the password stored in - `passwordSecretRef`. - The keystore file will be updated immediately. - If the issuer provided a CA certificate, a file named `truststore.p12` will - also be created in the target Secret resource, encrypted using the - password stored in `passwordSecretRef` containing the issuing Certificate - Authority - type: boolean - passwordSecretRef: - description: |- - PasswordSecretRef is a reference to a key in a Secret resource - containing the password used to encrypt the PKCS12 keystore. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - profile: - description: |- - Profile specifies the key and certificate encryption algorithms and the HMAC algorithm - used to create the PKCS12 keystore. Default value is `LegacyRC2` for backward compatibility. - - - If provided, allowed values are: - `LegacyRC2`: Deprecated. Not supported by default in OpenSSL 3 or Java 20. - `LegacyDES`: Less secure algorithm. Use this option for maximal compatibility. - `Modern2023`: Secure algorithm. Use this option in case you have to always use secure algorithms - (eg. because of company policy). Please note that the security of the algorithm is not that important - in reality, because the unencrypted certificate and private key are also stored in the Secret. - type: string - enum: - - LegacyRC2 - - LegacyDES - - Modern2023 - literalSubject: - description: |- - Requested X.509 certificate subject, represented using the LDAP "String - Representation of a Distinguished Name" [1]. - Important: the LDAP string format also specifies the order of the attributes - in the subject, this is important when issuing certs for LDAP authentication. - Example: `CN=foo,DC=corp,DC=example,DC=com` - More info [1]: https://datatracker.ietf.org/doc/html/rfc4514 - More info: https://github.com/cert-manager/cert-manager/issues/3203 - More info: https://github.com/cert-manager/cert-manager/issues/4424 - - - Cannot be set if the `subject` or `commonName` field is set. - type: string - nameConstraints: - description: |- - x.509 certificate NameConstraint extension which MUST NOT be used in a non-CA certificate. - More Info: https://datatracker.ietf.org/doc/html/rfc5280#section-4.2.1.10 - - - This is an Alpha Feature and is only enabled with the - `--feature-gates=NameConstraints=true` option set on both - the controller and webhook components. - type: object - properties: - critical: - description: if true then the name constraints are marked critical. - type: boolean - excluded: - description: |- - Excluded contains the constraints which must be disallowed. Any name matching a - restriction in the excluded field is invalid regardless - of information appearing in the permitted - type: object - properties: - dnsDomains: - description: DNSDomains is a list of DNS domains that are permitted or excluded. - type: array - items: - type: string - emailAddresses: - description: EmailAddresses is a list of Email Addresses that are permitted or excluded. - type: array - items: - type: string - ipRanges: - description: |- - IPRanges is a list of IP Ranges that are permitted or excluded. - This should be a valid CIDR notation. - type: array - items: - type: string - uriDomains: - description: URIDomains is a list of URI domains that are permitted or excluded. - type: array - items: - type: string - permitted: - description: Permitted contains the constraints in which the names must be located. - type: object - properties: - dnsDomains: - description: DNSDomains is a list of DNS domains that are permitted or excluded. - type: array - items: - type: string - emailAddresses: - description: EmailAddresses is a list of Email Addresses that are permitted or excluded. - type: array - items: - type: string - ipRanges: - description: |- - IPRanges is a list of IP Ranges that are permitted or excluded. - This should be a valid CIDR notation. - type: array - items: - type: string - uriDomains: - description: URIDomains is a list of URI domains that are permitted or excluded. - type: array - items: - type: string - otherNames: - description: |- - `otherNames` is an escape hatch for SAN that allows any type. We currently restrict the support to string like otherNames, cf RFC 5280 p 37 - Any UTF8 String valued otherName can be passed with by setting the keys oid: x.x.x.x and UTF8Value: somevalue for `otherName`. - Most commonly this would be UPN set with oid: 1.3.6.1.4.1.311.20.2.3 - You should ensure that any OID passed is valid for the UTF8String type as we do not explicitly validate this. - type: array - items: - type: object - properties: - oid: - description: |- - OID is the object identifier for the otherName SAN. - The object identifier must be expressed as a dotted string, for - example, "1.2.840.113556.1.4.221". - type: string - utf8Value: - description: |- - utf8Value is the string value of the otherName SAN. - The utf8Value accepts any valid UTF8 string to set as value for the otherName SAN. - type: string - privateKey: - description: |- - Private key options. These include the key algorithm and size, the used - encoding and the rotation policy. - type: object - properties: - algorithm: - description: |- - Algorithm is the private key algorithm of the corresponding private key - for this certificate. - - - If provided, allowed values are either `RSA`, `ECDSA` or `Ed25519`. - If `algorithm` is specified and `size` is not provided, - key size of 2048 will be used for `RSA` key algorithm and - key size of 256 will be used for `ECDSA` key algorithm. - key size is ignored when using the `Ed25519` key algorithm. - type: string - enum: - - RSA - - ECDSA - - Ed25519 - encoding: - description: |- - The private key cryptography standards (PKCS) encoding for this - certificate's private key to be encoded in. - - - If provided, allowed values are `PKCS1` and `PKCS8` standing for PKCS#1 - and PKCS#8, respectively. - Defaults to `PKCS1` if not specified. - type: string - enum: - - PKCS1 - - PKCS8 - rotationPolicy: - description: |- - RotationPolicy controls how private keys should be regenerated when a - re-issuance is being processed. - - - If set to `Never`, a private key will only be generated if one does not - already exist in the target `spec.secretName`. If one does exists but it - does not have the correct algorithm or size, a warning will be raised - to await user intervention. - If set to `Always`, a private key matching the specified requirements - will be generated whenever a re-issuance occurs. - Default is `Never` for backward compatibility. - type: string - enum: - - Never - - Always - size: - description: |- - Size is the key bit size of the corresponding private key for this certificate. - - - If `algorithm` is set to `RSA`, valid values are `2048`, `4096` or `8192`, - and will default to `2048` if not specified. - If `algorithm` is set to `ECDSA`, valid values are `256`, `384` or `521`, - and will default to `256` if not specified. - If `algorithm` is set to `Ed25519`, Size is ignored. - No other values are allowed. - type: integer - renewBefore: - description: |- - How long before the currently issued certificate's expiry cert-manager should - renew the certificate. For example, if a certificate is valid for 60 minutes, - and `renewBefore=10m`, cert-manager will begin to attempt to renew the certificate - 50 minutes after it was issued (i.e. when there are 10 minutes remaining until - the certificate is no longer valid). - - - NOTE: The actual lifetime of the issued certificate is used to determine the - renewal time. If an issuer returns a certificate with a different lifetime than - the one requested, cert-manager will use the lifetime of the issued certificate. - - - If unset, this defaults to 1/3 of the issued certificate's lifetime. - Minimum accepted value is 5 minutes. - Value must be in units accepted by Go time.ParseDuration https://golang.org/pkg/time/#ParseDuration. - type: string - revisionHistoryLimit: - description: |- - The maximum number of CertificateRequest revisions that are maintained in - the Certificate's history. Each revision represents a single `CertificateRequest` - created by this Certificate, either when it was created, renewed, or Spec - was changed. Revisions will be removed by oldest first if the number of - revisions exceeds this number. - - - If set, revisionHistoryLimit must be a value of `1` or greater. - If unset (`nil`), revisions will not be garbage collected. - Default value is `nil`. - type: integer - format: int32 - secretName: - description: |- - Name of the Secret resource that will be automatically created and - managed by this Certificate resource. It will be populated with a - private key and certificate, signed by the denoted issuer. The Secret - resource lives in the same namespace as the Certificate resource. - type: string - secretTemplate: - description: |- - Defines annotations and labels to be copied to the Certificate's Secret. - Labels and annotations on the Secret will be changed as they appear on the - SecretTemplate when added or removed. SecretTemplate annotations are added - in conjunction with, and cannot overwrite, the base set of annotations - cert-manager sets on the Certificate's Secret. - type: object - properties: - annotations: - description: Annotations is a key value map to be copied to the target Kubernetes Secret. - type: object - additionalProperties: - type: string - labels: - description: Labels is a key value map to be copied to the target Kubernetes Secret. - type: object - additionalProperties: - type: string - subject: - description: |- - Requested set of X509 certificate subject attributes. - More info: https://datatracker.ietf.org/doc/html/rfc5280#section-4.1.2.6 - - - The common name attribute is specified separately in the `commonName` field. - Cannot be set if the `literalSubject` field is set. - type: object - properties: - countries: - description: Countries to be used on the Certificate. - type: array - items: - type: string - localities: - description: Cities to be used on the Certificate. - type: array - items: - type: string - organizationalUnits: - description: Organizational Units to be used on the Certificate. - type: array - items: - type: string - organizations: - description: Organizations to be used on the Certificate. - type: array - items: - type: string - postalCodes: - description: Postal codes to be used on the Certificate. - type: array - items: - type: string - provinces: - description: State/Provinces to be used on the Certificate. - type: array - items: - type: string - serialNumber: - description: Serial number to be used on the Certificate. - type: string - streetAddresses: - description: Street addresses to be used on the Certificate. - type: array - items: - type: string - uris: - description: Requested URI subject alternative names. - type: array - items: - type: string - usages: - description: |- - Requested key usages and extended key usages. - These usages are used to set the `usages` field on the created CertificateRequest - resources. If `encodeUsagesInRequest` is unset or set to `true`, the usages - will additionally be encoded in the `request` field which contains the CSR blob. - - - If unset, defaults to `digital signature` and `key encipherment`. - type: array - items: - description: |- - KeyUsage specifies valid usage contexts for keys. - See: - https://tools.ietf.org/html/rfc5280#section-4.2.1.3 - https://tools.ietf.org/html/rfc5280#section-4.2.1.12 - - - Valid KeyUsage values are as follows: - "signing", - "digital signature", - "content commitment", - "key encipherment", - "key agreement", - "data encipherment", - "cert sign", - "crl sign", - "encipher only", - "decipher only", - "any", - "server auth", - "client auth", - "code signing", - "email protection", - "s/mime", - "ipsec end system", - "ipsec tunnel", - "ipsec user", - "timestamping", - "ocsp signing", - "microsoft sgc", - "netscape sgc" - type: string - enum: - - signing - - digital signature - - content commitment - - key encipherment - - key agreement - - data encipherment - - cert sign - - crl sign - - encipher only - - decipher only - - any - - server auth - - client auth - - code signing - - email protection - - s/mime - - ipsec end system - - ipsec tunnel - - ipsec user - - timestamping - - ocsp signing - - microsoft sgc - - netscape sgc - status: - description: |- - Status of the Certificate. - This is set and managed automatically. - Read-only. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#spec-and-status - type: object - properties: - conditions: - description: |- - List of status conditions to indicate the status of certificates. - Known condition types are `Ready` and `Issuing`. - type: array - items: - description: CertificateCondition contains condition information for an Certificate. - type: object - required: - - status - - type - properties: - lastTransitionTime: - description: |- - LastTransitionTime is the timestamp corresponding to the last status - change of this condition. - type: string - format: date-time - message: - description: |- - Message is a human readable description of the details of the last - transition, complementing reason. - type: string - observedGeneration: - description: |- - If set, this represents the .metadata.generation that the condition was - set based upon. - For instance, if .metadata.generation is currently 12, but the - .status.condition[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the Certificate. - type: integer - format: int64 - reason: - description: |- - Reason is a brief machine readable explanation for the condition's last - transition. - type: string - status: - description: Status of the condition, one of (`True`, `False`, `Unknown`). - type: string - enum: - - "True" - - "False" - - Unknown - type: - description: Type of the condition, known values are (`Ready`, `Issuing`). - type: string - x-kubernetes-list-map-keys: - - type - x-kubernetes-list-type: map - failedIssuanceAttempts: - description: |- - The number of continuous failed issuance attempts up till now. This - field gets removed (if set) on a successful issuance and gets set to - 1 if unset and an issuance has failed. If an issuance has failed, the - delay till the next issuance will be calculated using formula - time.Hour * 2 ^ (failedIssuanceAttempts - 1). - type: integer - lastFailureTime: - description: |- - LastFailureTime is set only if the lastest issuance for this - Certificate failed and contains the time of the failure. If an - issuance has failed, the delay till the next issuance will be - calculated using formula time.Hour * 2 ^ (failedIssuanceAttempts - - 1). If the latest issuance has succeeded this field will be unset. - type: string - format: date-time - nextPrivateKeySecretName: - description: |- - The name of the Secret resource containing the private key to be used - for the next certificate iteration. - The keymanager controller will automatically set this field if the - `Issuing` condition is set to `True`. - It will automatically unset this field when the Issuing condition is - not set or False. - type: string - notAfter: - description: |- - The expiration time of the certificate stored in the secret named - by this resource in `spec.secretName`. - type: string - format: date-time - notBefore: - description: |- - The time after which the certificate stored in the secret named - by this resource in `spec.secretName` is valid. - type: string - format: date-time - renewalTime: - description: |- - RenewalTime is the time at which the certificate will be next - renewed. - If not set, no upcoming renewal is scheduled. - type: string - format: date-time - revision: - description: |- - The current 'revision' of the certificate as issued. - - - When a CertificateRequest resource is created, it will have the - `cert-manager.io/certificate-revision` set to one greater than the - current value of this field. - - - Upon issuance, this field will be set to the value of the annotation - on the CertificateRequest resource used to issue the certificate. - - - Persisting the value on the CertificateRequest resource allows the - certificates controller to know whether a request is part of an old - issuance or if it is part of the ongoing revision's issuance by - checking if the revision value in the annotation is greater than this - field. - type: integer - served: true - storage: true - -# END crd {{- end }} - ---- -# START crd {{- if or .Values.crds.enabled .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - name: challenges.acme.cert-manager.io - # START annotations {{- if .Values.crds.keep }} - annotations: - helm.sh/resource-policy: keep - # END annotations {{- end }} - labels: - app: '{{ template "cert-manager.name" . }}' - app.kubernetes.io/name: '{{ template "cert-manager.name" . }}' - app.kubernetes.io/instance: '{{ .Release.Name }}' - # Generated labels {{- include "labels" . | nindent 4 }} -spec: - group: acme.cert-manager.io - names: - kind: Challenge - listKind: ChallengeList - plural: challenges - singular: challenge - categories: - - cert-manager - - cert-manager-acme - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .status.state - name: State - type: string - - jsonPath: .spec.dnsName - name: Domain - type: string - - jsonPath: .status.reason - name: Reason - priority: 1 - type: string - - description: CreationTimestamp is a timestamp representing the server time when this object was created. It is not guaranteed to be set in happens-before order across separate operations. Clients may not set this value. It is represented in RFC3339 form and is in UTC. - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1 - schema: - openAPIV3Schema: - description: Challenge is a type to represent a Challenge request with an ACME server - type: object - required: - - metadata - - spec - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - type: object - required: - - authorizationURL - - dnsName - - issuerRef - - key - - solver - - token - - type - - url - properties: - authorizationURL: - description: |- - The URL to the ACME Authorization resource that this - challenge is a part of. - type: string - dnsName: - description: |- - dnsName is the identifier that this challenge is for, e.g. example.com. - If the requested DNSName is a 'wildcard', this field MUST be set to the - non-wildcard domain, e.g. for `*.example.com`, it must be `example.com`. - type: string - issuerRef: - description: |- - References a properly configured ACME-type Issuer which should - be used to create this Challenge. - If the Issuer does not exist, processing will be retried. - If the Issuer is not an 'ACME' Issuer, an error will be returned and the - Challenge will be marked as failed. - type: object - required: - - name - properties: - group: - description: Group of the resource being referred to. - type: string - kind: - description: Kind of the resource being referred to. - type: string - name: - description: Name of the resource being referred to. - type: string - key: - description: |- - The ACME challenge key for this challenge - For HTTP01 challenges, this is the value that must be responded with to - complete the HTTP01 challenge in the format: - `.`. - For DNS01 challenges, this is the base64 encoded SHA256 sum of the - `.` - text that must be set as the TXT record content. - type: string - solver: - description: |- - Contains the domain solving configuration that should be used to - solve this challenge resource. - type: object - properties: - dns01: - description: |- - Configures cert-manager to attempt to complete authorizations by - performing the DNS01 challenge flow. - type: object - properties: - acmeDNS: - description: |- - Use the 'ACME DNS' (https://github.com/joohoi/acme-dns) API to manage - DNS01 challenge records. - type: object - required: - - accountSecretRef - - host - properties: - accountSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - host: - type: string - akamai: - description: Use the Akamai DNS zone management API to manage DNS01 challenge records. - type: object - required: - - accessTokenSecretRef - - clientSecretSecretRef - - clientTokenSecretRef - - serviceConsumerDomain - properties: - accessTokenSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - clientSecretSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - clientTokenSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - serviceConsumerDomain: - type: string - azureDNS: - description: Use the Microsoft Azure DNS API to manage DNS01 challenge records. - type: object - required: - - resourceGroupName - - subscriptionID - properties: - clientID: - description: |- - Auth: Azure Service Principal: - The ClientID of the Azure Service Principal used to authenticate with Azure DNS. - If set, ClientSecret and TenantID must also be set. - type: string - clientSecretSecretRef: - description: |- - Auth: Azure Service Principal: - A reference to a Secret containing the password associated with the Service Principal. - If set, ClientID and TenantID must also be set. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - environment: - description: name of the Azure environment (default AzurePublicCloud) - type: string - enum: - - AzurePublicCloud - - AzureChinaCloud - - AzureGermanCloud - - AzureUSGovernmentCloud - hostedZoneName: - description: name of the DNS zone that should be used - type: string - managedIdentity: - description: |- - Auth: Azure Workload Identity or Azure Managed Service Identity: - Settings to enable Azure Workload Identity or Azure Managed Service Identity - If set, ClientID, ClientSecret and TenantID must not be set. - type: object - properties: - clientID: - description: client ID of the managed identity, can not be used at the same time as resourceID - type: string - resourceID: - description: |- - resource ID of the managed identity, can not be used at the same time as clientID - Cannot be used for Azure Managed Service Identity - type: string - resourceGroupName: - description: resource group the DNS zone is located in - type: string - subscriptionID: - description: ID of the Azure subscription - type: string - tenantID: - description: |- - Auth: Azure Service Principal: - The TenantID of the Azure Service Principal used to authenticate with Azure DNS. - If set, ClientID and ClientSecret must also be set. - type: string - cloudDNS: - description: Use the Google Cloud DNS API to manage DNS01 challenge records. - type: object - required: - - project - properties: - hostedZoneName: - description: |- - HostedZoneName is an optional field that tells cert-manager in which - Cloud DNS zone the challenge record has to be created. - If left empty cert-manager will automatically choose a zone. - type: string - project: - type: string - serviceAccountSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - cloudflare: - description: Use the Cloudflare API to manage DNS01 challenge records. - type: object - properties: - apiKeySecretRef: - description: |- - API key to use to authenticate with Cloudflare. - Note: using an API token to authenticate is now the recommended method - as it allows greater control of permissions. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - apiTokenSecretRef: - description: API token used to authenticate with Cloudflare. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - email: - description: Email of the account, only required when using API key based authentication. - type: string - cnameStrategy: - description: |- - CNAMEStrategy configures how the DNS01 provider should handle CNAME - records when found in DNS zones. - type: string - enum: - - None - - Follow - digitalocean: - description: Use the DigitalOcean DNS API to manage DNS01 challenge records. - type: object - required: - - tokenSecretRef - properties: - tokenSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - rfc2136: - description: |- - Use RFC2136 ("Dynamic Updates in the Domain Name System") (https://datatracker.ietf.org/doc/rfc2136/) - to manage DNS01 challenge records. - type: object - required: - - nameserver - properties: - nameserver: - description: |- - The IP address or hostname of an authoritative DNS server supporting - RFC2136 in the form host:port. If the host is an IPv6 address it must be - enclosed in square brackets (e.g [2001:db8::1]) ; port is optional. - This field is required. - type: string - tsigAlgorithm: - description: |- - The TSIG Algorithm configured in the DNS supporting RFC2136. Used only - when ``tsigSecretSecretRef`` and ``tsigKeyName`` are defined. - Supported values are (case-insensitive): ``HMACMD5`` (default), - ``HMACSHA1``, ``HMACSHA256`` or ``HMACSHA512``. - type: string - tsigKeyName: - description: |- - The TSIG Key name configured in the DNS. - If ``tsigSecretSecretRef`` is defined, this field is required. - type: string - tsigSecretSecretRef: - description: |- - The name of the secret containing the TSIG value. - If ``tsigKeyName`` is defined, this field is required. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - route53: - description: Use the AWS Route53 API to manage DNS01 challenge records. - type: object - required: - - region - properties: - accessKeyID: - description: |- - The AccessKeyID is used for authentication. - Cannot be set when SecretAccessKeyID is set. - If neither the Access Key nor Key ID are set, we fall-back to using env - vars, shared credentials file or AWS Instance metadata, - see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials - type: string - accessKeyIDSecretRef: - description: |- - The SecretAccessKey is used for authentication. If set, pull the AWS - access key ID from a key within a Kubernetes Secret. - Cannot be set when AccessKeyID is set. - If neither the Access Key nor Key ID are set, we fall-back to using env - vars, shared credentials file or AWS Instance metadata, - see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - auth: - description: Auth configures how cert-manager authenticates. - type: object - required: - - kubernetes - properties: - kubernetes: - description: |- - Kubernetes authenticates with Route53 using AssumeRoleWithWebIdentity - by passing a bound ServiceAccount token. - type: object - required: - - serviceAccountRef - properties: - serviceAccountRef: - description: |- - A reference to a service account that will be used to request a bound - token (also known as "projected token"). To use this field, you must - configure an RBAC rule to let cert-manager request a token. - type: object - required: - - name - properties: - audiences: - description: |- - TokenAudiences is an optional list of audiences to include in the - token passed to AWS. The default token consisting of the issuer's namespace - and name is always included. - If unset the audience defaults to `sts.amazonaws.com`. - type: array - items: - type: string - name: - description: Name of the ServiceAccount used to request a token. - type: string - hostedZoneID: - description: If set, the provider will manage only this zone in Route53 and will not do an lookup using the route53:ListHostedZonesByName api call. - type: string - region: - description: Always set the region when using AccessKeyID and SecretAccessKey - type: string - role: - description: |- - Role is a Role ARN which the Route53 provider will assume using either the explicit credentials AccessKeyID/SecretAccessKey - or the inferred credentials from environment variables, shared credentials file or AWS Instance metadata - type: string - secretAccessKeySecretRef: - description: |- - The SecretAccessKey is used for authentication. - If neither the Access Key nor Key ID are set, we fall-back to using env - vars, shared credentials file or AWS Instance metadata, - see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - webhook: - description: |- - Configure an external webhook based DNS01 challenge solver to manage - DNS01 challenge records. - type: object - required: - - groupName - - solverName - properties: - config: - description: |- - Additional configuration that should be passed to the webhook apiserver - when challenges are processed. - This can contain arbitrary JSON data. - Secret values should not be specified in this stanza. - If secret values are needed (e.g. credentials for a DNS service), you - should use a SecretKeySelector to reference a Secret resource. - For details on the schema of this field, consult the webhook provider - implementation's documentation. - x-kubernetes-preserve-unknown-fields: true - groupName: - description: |- - The API group name that should be used when POSTing ChallengePayload - resources to the webhook apiserver. - This should be the same as the GroupName specified in the webhook - provider implementation. - type: string - solverName: - description: |- - The name of the solver to use, as defined in the webhook provider - implementation. - This will typically be the name of the provider, e.g. 'cloudflare'. - type: string - http01: - description: |- - Configures cert-manager to attempt to complete authorizations by - performing the HTTP01 challenge flow. - It is not possible to obtain certificates for wildcard domain names - (e.g. `*.example.com`) using the HTTP01 challenge mechanism. - type: object - properties: - gatewayHTTPRoute: - description: |- - The Gateway API is a sig-network community API that models service networking - in Kubernetes (https://gateway-api.sigs.k8s.io/). The Gateway solver will - create HTTPRoutes with the specified labels in the same namespace as the challenge. - This solver is experimental, and fields / behaviour may change in the future. - type: object - properties: - labels: - description: |- - Custom labels that will be applied to HTTPRoutes created by cert-manager - while solving HTTP-01 challenges. - type: object - additionalProperties: - type: string - parentRefs: - description: |- - When solving an HTTP-01 challenge, cert-manager creates an HTTPRoute. - cert-manager needs to know which parentRefs should be used when creating - the HTTPRoute. Usually, the parentRef references a Gateway. See: - https://gateway-api.sigs.k8s.io/api-types/httproute/#attaching-to-gateways - type: array - items: - description: |- - ParentReference identifies an API object (usually a Gateway) that can be considered - a parent of this resource (usually a route). There are two kinds of parent resources - with "Core" support: - - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - - This API may be extended in the future to support additional kinds of parent - resources. - - - The API object must be valid in the cluster; the Group and Kind must - be registered in the cluster for this reference to be valid. - type: object - required: - - name - properties: - group: - description: |- - Group is the group of the referent. - When unspecified, "gateway.networking.k8s.io" is inferred. - To set the core API group (such as for a "Service" kind referent), - Group must be explicitly set to "" (empty string). - - - Support: Core - type: string - default: gateway.networking.k8s.io - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - kind: - description: |- - Kind is kind of the referent. - - - There are two kinds of parent resources with "Core" support: - - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - - Support for other resources is Implementation-Specific. - type: string - default: Gateway - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - name: - description: |- - Name is the name of the referent. - - - Support: Core - type: string - maxLength: 253 - minLength: 1 - namespace: - description: |- - Namespace is the namespace of the referent. When unspecified, this refers - to the local namespace of the Route. - - - Note that there are specific rules for ParentRefs which cross namespace - boundaries. Cross-namespace references are only valid if they are explicitly - allowed by something in the namespace they are referring to. For example: - Gateway has the AllowedRoutes field, and ReferenceGrant provides a - generic way to enable any other kind of cross-namespace reference. - - - - ParentRefs from a Route to a Service in the same namespace are "producer" - routes, which apply default routing rules to inbound connections from - any namespace to the Service. - - - ParentRefs from a Route to a Service in a different namespace are - "consumer" routes, and these routing rules are only applied to outbound - connections originating from the same namespace as the Route, for which - the intended destination of the connections are a Service targeted as a - ParentRef of the Route. - - - - Support: Core - type: string - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - port: - description: |- - Port is the network port this Route targets. It can be interpreted - differently based on the type of parent resource. - - - When the parent resource is a Gateway, this targets all listeners - listening on the specified port that also support this kind of Route(and - select this Route). It's not recommended to set `Port` unless the - networking behaviors specified in a Route must apply to a specific port - as opposed to a listener(s) whose port(s) may be changed. When both Port - and SectionName are specified, the name and port of the selected listener - must match both specified values. - - - - When the parent resource is a Service, this targets a specific port in the - Service spec. When both Port (experimental) and SectionName are specified, - the name and port of the selected port must match both specified values. - - - - Implementations MAY choose to support other parent resources. - Implementations supporting other types of parent resources MUST clearly - document how/if Port is interpreted. - - - For the purpose of status, an attachment is considered successful as - long as the parent resource accepts it partially. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment - from the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, - the Route MUST be considered detached from the Gateway. - - - Support: Extended - type: integer - format: int32 - maximum: 65535 - minimum: 1 - sectionName: - description: |- - SectionName is the name of a section within the target resource. In the - following resources, SectionName is interpreted as the following: - - - * Gateway: Listener name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - * Service: Port name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - - - Implementations MAY choose to support attaching Routes to other resources. - If that is the case, they MUST clearly document how SectionName is - interpreted. - - - When unspecified (empty string), this will reference the entire resource. - For the purpose of status, an attachment is considered successful if at - least one section in the parent resource accepts it. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from - the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, the - Route MUST be considered detached from the Gateway. - - - Support: Core - type: string - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - serviceType: - description: |- - Optional service type for Kubernetes solver service. Supported values - are NodePort or ClusterIP. If unset, defaults to NodePort. - type: string - ingress: - description: |- - The ingress based HTTP01 challenge solver will solve challenges by - creating or modifying Ingress resources in order to route requests for - '/.well-known/acme-challenge/XYZ' to 'challenge solver' pods that are - provisioned by cert-manager for each Challenge to be completed. - type: object - properties: - class: - description: |- - This field configures the annotation `kubernetes.io/ingress.class` when - creating Ingress resources to solve ACME challenges that use this - challenge solver. Only one of `class`, `name` or `ingressClassName` may - be specified. - type: string - ingressClassName: - description: |- - This field configures the field `ingressClassName` on the created Ingress - resources used to solve ACME challenges that use this challenge solver. - This is the recommended way of configuring the ingress class. Only one of - `class`, `name` or `ingressClassName` may be specified. - type: string - ingressTemplate: - description: |- - Optional ingress template used to configure the ACME challenge solver - ingress used for HTTP01 challenges. - type: object - properties: - metadata: - description: |- - ObjectMeta overrides for the ingress used to solve HTTP01 challenges. - Only the 'labels' and 'annotations' fields may be set. - If labels or annotations overlap with in-built values, the values here - will override the in-built values. - type: object - properties: - annotations: - description: Annotations that should be added to the created ACME HTTP01 solver ingress. - type: object - additionalProperties: - type: string - labels: - description: Labels that should be added to the created ACME HTTP01 solver ingress. - type: object - additionalProperties: - type: string - name: - description: |- - The name of the ingress resource that should have ACME challenge solving - routes inserted into it in order to solve HTTP01 challenges. - This is typically used in conjunction with ingress controllers like - ingress-gce, which maintains a 1:1 mapping between external IPs and - ingress resources. Only one of `class`, `name` or `ingressClassName` may - be specified. - type: string - podTemplate: - description: |- - Optional pod template used to configure the ACME challenge solver pods - used for HTTP01 challenges. - type: object - properties: - metadata: - description: |- - ObjectMeta overrides for the pod used to solve HTTP01 challenges. - Only the 'labels' and 'annotations' fields may be set. - If labels or annotations overlap with in-built values, the values here - will override the in-built values. - type: object - properties: - annotations: - description: Annotations that should be added to the create ACME HTTP01 solver pods. - type: object - additionalProperties: - type: string - labels: - description: Labels that should be added to the created ACME HTTP01 solver pods. - type: object - additionalProperties: - type: string - spec: - description: |- - PodSpec defines overrides for the HTTP01 challenge solver pod. - Check ACMEChallengeSolverHTTP01IngressPodSpec to find out currently supported fields. - All other fields will be ignored. - type: object - properties: - affinity: - description: If specified, the pod's scheduling constraints - type: object - properties: - nodeAffinity: - description: Describes node affinity scheduling rules for the pod. - type: object - properties: - preferredDuringSchedulingIgnoredDuringExecution: - description: |- - The scheduler will prefer to schedule pods to nodes that satisfy - the affinity expressions specified by this field, but it may choose - a node that violates one or more of the expressions. The node that is - most preferred is the one with the greatest sum of weights, i.e. - for each node that meets all of the scheduling requirements (resource - request, requiredDuringScheduling affinity expressions, etc.), - compute a sum by iterating through the elements of this field and adding - "weight" to the sum if the node matches the corresponding matchExpressions; the - node(s) with the highest sum are the most preferred. - type: array - items: - description: |- - An empty preferred scheduling term matches all objects with implicit weight 0 - (i.e. it's a no-op). A null preferred scheduling term matches no objects (i.e. is also a no-op). - type: object - required: - - preference - - weight - properties: - preference: - description: A node selector term, associated with the corresponding weight. - type: object - properties: - matchExpressions: - description: A list of node selector requirements by node's labels. - type: array - items: - description: |- - A node selector requirement is a selector that contains values, a key, and an operator - that relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: The label key that the selector applies to. - type: string - operator: - description: |- - Represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. - type: string - values: - description: |- - An array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. If the operator is Gt or Lt, the values - array must have a single element, which will be interpreted as an integer. - This array is replaced during a strategic merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchFields: - description: A list of node selector requirements by node's fields. - type: array - items: - description: |- - A node selector requirement is a selector that contains values, a key, and an operator - that relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: The label key that the selector applies to. - type: string - operator: - description: |- - Represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. - type: string - values: - description: |- - An array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. If the operator is Gt or Lt, the values - array must have a single element, which will be interpreted as an integer. - This array is replaced during a strategic merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - x-kubernetes-map-type: atomic - weight: - description: Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100. - type: integer - format: int32 - x-kubernetes-list-type: atomic - requiredDuringSchedulingIgnoredDuringExecution: - description: |- - If the affinity requirements specified by this field are not met at - scheduling time, the pod will not be scheduled onto the node. - If the affinity requirements specified by this field cease to be met - at some point during pod execution (e.g. due to an update), the system - may or may not try to eventually evict the pod from its node. - type: object - required: - - nodeSelectorTerms - properties: - nodeSelectorTerms: - description: Required. A list of node selector terms. The terms are ORed. - type: array - items: - description: |- - A null or empty node selector term matches no objects. The requirements of - them are ANDed. - The TopologySelectorTerm type implements a subset of the NodeSelectorTerm. - type: object - properties: - matchExpressions: - description: A list of node selector requirements by node's labels. - type: array - items: - description: |- - A node selector requirement is a selector that contains values, a key, and an operator - that relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: The label key that the selector applies to. - type: string - operator: - description: |- - Represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. - type: string - values: - description: |- - An array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. If the operator is Gt or Lt, the values - array must have a single element, which will be interpreted as an integer. - This array is replaced during a strategic merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchFields: - description: A list of node selector requirements by node's fields. - type: array - items: - description: |- - A node selector requirement is a selector that contains values, a key, and an operator - that relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: The label key that the selector applies to. - type: string - operator: - description: |- - Represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. - type: string - values: - description: |- - An array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. If the operator is Gt or Lt, the values - array must have a single element, which will be interpreted as an integer. - This array is replaced during a strategic merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - x-kubernetes-map-type: atomic - x-kubernetes-list-type: atomic - x-kubernetes-map-type: atomic - podAffinity: - description: Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)). - type: object - properties: - preferredDuringSchedulingIgnoredDuringExecution: - description: |- - The scheduler will prefer to schedule pods to nodes that satisfy - the affinity expressions specified by this field, but it may choose - a node that violates one or more of the expressions. The node that is - most preferred is the one with the greatest sum of weights, i.e. - for each node that meets all of the scheduling requirements (resource - request, requiredDuringScheduling affinity expressions, etc.), - compute a sum by iterating through the elements of this field and adding - "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the - node(s) with the highest sum are the most preferred. - type: array - items: - description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) - type: object - required: - - podAffinityTerm - - weight - properties: - podAffinityTerm: - description: Required. A pod affinity term, associated with the corresponding weight. - type: object - required: - - topologyKey - properties: - labelSelector: - description: |- - A label query over a set of resources, in this case pods. - If it's null, this PodAffinityTerm matches with no Pods. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - matchLabelKeys: - description: |- - MatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both matchLabelKeys and labelSelector. - Also, matchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - mismatchLabelKeys: - description: |- - MismatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. - Also, mismatchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - namespaceSelector: - description: |- - A label query over the set of namespaces that the term applies to. - The term is applied to the union of the namespaces selected by this field - and the ones listed in the namespaces field. - null selector and null or empty namespaces list means "this pod's namespace". - An empty selector ({}) matches all namespaces. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - namespaces: - description: |- - namespaces specifies a static list of namespace names that the term applies to. - The term is applied to the union of the namespaces listed in this field - and the ones selected by namespaceSelector. - null or empty namespaces list and null namespaceSelector means "this pod's namespace". - type: array - items: - type: string - x-kubernetes-list-type: atomic - topologyKey: - description: |- - This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching - the labelSelector in the specified namespaces, where co-located is defined as running on a node - whose value of the label with key topologyKey matches that of any node on which any of the - selected pods is running. - Empty topologyKey is not allowed. - type: string - weight: - description: |- - weight associated with matching the corresponding podAffinityTerm, - in the range 1-100. - type: integer - format: int32 - x-kubernetes-list-type: atomic - requiredDuringSchedulingIgnoredDuringExecution: - description: |- - If the affinity requirements specified by this field are not met at - scheduling time, the pod will not be scheduled onto the node. - If the affinity requirements specified by this field cease to be met - at some point during pod execution (e.g. due to a pod label update), the - system may or may not try to eventually evict the pod from its node. - When there are multiple elements, the lists of nodes corresponding to each - podAffinityTerm are intersected, i.e. all terms must be satisfied. - type: array - items: - description: |- - Defines a set of pods (namely those matching the labelSelector - relative to the given namespace(s)) that this pod should be - co-located (affinity) or not co-located (anti-affinity) with, - where co-located is defined as running on a node whose value of - the label with key matches that of any node on which - a pod of the set of pods is running - type: object - required: - - topologyKey - properties: - labelSelector: - description: |- - A label query over a set of resources, in this case pods. - If it's null, this PodAffinityTerm matches with no Pods. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - matchLabelKeys: - description: |- - MatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both matchLabelKeys and labelSelector. - Also, matchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - mismatchLabelKeys: - description: |- - MismatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. - Also, mismatchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - namespaceSelector: - description: |- - A label query over the set of namespaces that the term applies to. - The term is applied to the union of the namespaces selected by this field - and the ones listed in the namespaces field. - null selector and null or empty namespaces list means "this pod's namespace". - An empty selector ({}) matches all namespaces. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - namespaces: - description: |- - namespaces specifies a static list of namespace names that the term applies to. - The term is applied to the union of the namespaces listed in this field - and the ones selected by namespaceSelector. - null or empty namespaces list and null namespaceSelector means "this pod's namespace". - type: array - items: - type: string - x-kubernetes-list-type: atomic - topologyKey: - description: |- - This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching - the labelSelector in the specified namespaces, where co-located is defined as running on a node - whose value of the label with key topologyKey matches that of any node on which any of the - selected pods is running. - Empty topologyKey is not allowed. - type: string - x-kubernetes-list-type: atomic - podAntiAffinity: - description: Describes pod anti-affinity scheduling rules (e.g. avoid putting this pod in the same node, zone, etc. as some other pod(s)). - type: object - properties: - preferredDuringSchedulingIgnoredDuringExecution: - description: |- - The scheduler will prefer to schedule pods to nodes that satisfy - the anti-affinity expressions specified by this field, but it may choose - a node that violates one or more of the expressions. The node that is - most preferred is the one with the greatest sum of weights, i.e. - for each node that meets all of the scheduling requirements (resource - request, requiredDuringScheduling anti-affinity expressions, etc.), - compute a sum by iterating through the elements of this field and adding - "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the - node(s) with the highest sum are the most preferred. - type: array - items: - description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) - type: object - required: - - podAffinityTerm - - weight - properties: - podAffinityTerm: - description: Required. A pod affinity term, associated with the corresponding weight. - type: object - required: - - topologyKey - properties: - labelSelector: - description: |- - A label query over a set of resources, in this case pods. - If it's null, this PodAffinityTerm matches with no Pods. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - matchLabelKeys: - description: |- - MatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both matchLabelKeys and labelSelector. - Also, matchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - mismatchLabelKeys: - description: |- - MismatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. - Also, mismatchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - namespaceSelector: - description: |- - A label query over the set of namespaces that the term applies to. - The term is applied to the union of the namespaces selected by this field - and the ones listed in the namespaces field. - null selector and null or empty namespaces list means "this pod's namespace". - An empty selector ({}) matches all namespaces. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - namespaces: - description: |- - namespaces specifies a static list of namespace names that the term applies to. - The term is applied to the union of the namespaces listed in this field - and the ones selected by namespaceSelector. - null or empty namespaces list and null namespaceSelector means "this pod's namespace". - type: array - items: - type: string - x-kubernetes-list-type: atomic - topologyKey: - description: |- - This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching - the labelSelector in the specified namespaces, where co-located is defined as running on a node - whose value of the label with key topologyKey matches that of any node on which any of the - selected pods is running. - Empty topologyKey is not allowed. - type: string - weight: - description: |- - weight associated with matching the corresponding podAffinityTerm, - in the range 1-100. - type: integer - format: int32 - x-kubernetes-list-type: atomic - requiredDuringSchedulingIgnoredDuringExecution: - description: |- - If the anti-affinity requirements specified by this field are not met at - scheduling time, the pod will not be scheduled onto the node. - If the anti-affinity requirements specified by this field cease to be met - at some point during pod execution (e.g. due to a pod label update), the - system may or may not try to eventually evict the pod from its node. - When there are multiple elements, the lists of nodes corresponding to each - podAffinityTerm are intersected, i.e. all terms must be satisfied. - type: array - items: - description: |- - Defines a set of pods (namely those matching the labelSelector - relative to the given namespace(s)) that this pod should be - co-located (affinity) or not co-located (anti-affinity) with, - where co-located is defined as running on a node whose value of - the label with key matches that of any node on which - a pod of the set of pods is running - type: object - required: - - topologyKey - properties: - labelSelector: - description: |- - A label query over a set of resources, in this case pods. - If it's null, this PodAffinityTerm matches with no Pods. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - matchLabelKeys: - description: |- - MatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both matchLabelKeys and labelSelector. - Also, matchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - mismatchLabelKeys: - description: |- - MismatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. - Also, mismatchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - namespaceSelector: - description: |- - A label query over the set of namespaces that the term applies to. - The term is applied to the union of the namespaces selected by this field - and the ones listed in the namespaces field. - null selector and null or empty namespaces list means "this pod's namespace". - An empty selector ({}) matches all namespaces. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - namespaces: - description: |- - namespaces specifies a static list of namespace names that the term applies to. - The term is applied to the union of the namespaces listed in this field - and the ones selected by namespaceSelector. - null or empty namespaces list and null namespaceSelector means "this pod's namespace". - type: array - items: - type: string - x-kubernetes-list-type: atomic - topologyKey: - description: |- - This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching - the labelSelector in the specified namespaces, where co-located is defined as running on a node - whose value of the label with key topologyKey matches that of any node on which any of the - selected pods is running. - Empty topologyKey is not allowed. - type: string - x-kubernetes-list-type: atomic - imagePullSecrets: - description: If specified, the pod's imagePullSecrets - type: array - items: - description: |- - LocalObjectReference contains enough information to let you locate the - referenced object inside the same namespace. - type: object - properties: - name: - description: |- - Name of the referent. - This field is effectively required, but due to backwards compatibility is - allowed to be empty. Instances of this type with an empty value here are - almost certainly wrong. - TODO: Add other useful fields. apiVersion, kind, uid? - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - TODO: Drop `kubebuilder:default` when controller-gen doesn't need it https://github.com/kubernetes-sigs/kubebuilder/issues/3896. - type: string - default: "" - x-kubernetes-map-type: atomic - nodeSelector: - description: |- - NodeSelector is a selector which must be true for the pod to fit on a node. - Selector which must match a node's labels for the pod to be scheduled on that node. - More info: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/ - type: object - additionalProperties: - type: string - priorityClassName: - description: If specified, the pod's priorityClassName. - type: string - serviceAccountName: - description: If specified, the pod's service account - type: string - tolerations: - description: If specified, the pod's tolerations. - type: array - items: - description: |- - The pod this Toleration is attached to tolerates any taint that matches - the triple using the matching operator . - type: object - properties: - effect: - description: |- - Effect indicates the taint effect to match. Empty means match all taint effects. - When specified, allowed values are NoSchedule, PreferNoSchedule and NoExecute. - type: string - key: - description: |- - Key is the taint key that the toleration applies to. Empty means match all taint keys. - If the key is empty, operator must be Exists; this combination means to match all values and all keys. - type: string - operator: - description: |- - Operator represents a key's relationship to the value. - Valid operators are Exists and Equal. Defaults to Equal. - Exists is equivalent to wildcard for value, so that a pod can - tolerate all taints of a particular category. - type: string - tolerationSeconds: - description: |- - TolerationSeconds represents the period of time the toleration (which must be - of effect NoExecute, otherwise this field is ignored) tolerates the taint. By default, - it is not set, which means tolerate the taint forever (do not evict). Zero and - negative values will be treated as 0 (evict immediately) by the system. - type: integer - format: int64 - value: - description: |- - Value is the taint value the toleration matches to. - If the operator is Exists, the value should be empty, otherwise just a regular string. - type: string - serviceType: - description: |- - Optional service type for Kubernetes solver service. Supported values - are NodePort or ClusterIP. If unset, defaults to NodePort. - type: string - selector: - description: |- - Selector selects a set of DNSNames on the Certificate resource that - should be solved using this challenge solver. - If not specified, the solver will be treated as the 'default' solver - with the lowest priority, i.e. if any other solver has a more specific - match, it will be used instead. - type: object - properties: - dnsNames: - description: |- - List of DNSNames that this solver will be used to solve. - If specified and a match is found, a dnsNames selector will take - precedence over a dnsZones selector. - If multiple solvers match with the same dnsNames value, the solver - with the most matching labels in matchLabels will be selected. - If neither has more matches, the solver defined earlier in the list - will be selected. - type: array - items: - type: string - dnsZones: - description: |- - List of DNSZones that this solver will be used to solve. - The most specific DNS zone match specified here will take precedence - over other DNS zone matches, so a solver specifying sys.example.com - will be selected over one specifying example.com for the domain - www.sys.example.com. - If multiple solvers match with the same dnsZones value, the solver - with the most matching labels in matchLabels will be selected. - If neither has more matches, the solver defined earlier in the list - will be selected. - type: array - items: - type: string - matchLabels: - description: |- - A label selector that is used to refine the set of certificate's that - this challenge solver will apply to. - type: object - additionalProperties: - type: string - token: - description: |- - The ACME challenge token for this challenge. - This is the raw value returned from the ACME server. - type: string - type: - description: |- - The type of ACME challenge this resource represents. - One of "HTTP-01" or "DNS-01". - type: string - enum: - - HTTP-01 - - DNS-01 - url: - description: |- - The URL of the ACME Challenge resource for this challenge. - This can be used to lookup details about the status of this challenge. - type: string - wildcard: - description: |- - wildcard will be true if this challenge is for a wildcard identifier, - for example '*.example.com'. - type: boolean - status: - type: object - properties: - presented: - description: |- - presented will be set to true if the challenge values for this challenge - are currently 'presented'. - This *does not* imply the self check is passing. Only that the values - have been 'submitted' for the appropriate challenge mechanism (i.e. the - DNS01 TXT record has been presented, or the HTTP01 configuration has been - configured). - type: boolean - processing: - description: |- - Used to denote whether this challenge should be processed or not. - This field will only be set to true by the 'scheduling' component. - It will only be set to false by the 'challenges' controller, after the - challenge has reached a final state or timed out. - If this field is set to false, the challenge controller will not take - any more action. - type: boolean - reason: - description: |- - Contains human readable information on why the Challenge is in the - current state. - type: string - state: - description: |- - Contains the current 'state' of the challenge. - If not set, the state of the challenge is unknown. - type: string - enum: - - valid - - ready - - pending - - processing - - invalid - - expired - - errored - served: true - storage: true - subresources: - status: {} - -# END crd {{- end }} - ---- -# START crd {{- if or .Values.crds.enabled .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - name: clusterissuers.cert-manager.io - # START annotations {{- if .Values.crds.keep }} - annotations: - helm.sh/resource-policy: keep - # END annotations {{- end }} - labels: - app: '{{ template "cert-manager.name" . }}' - app.kubernetes.io/name: '{{ template "cert-manager.name" . }}' - app.kubernetes.io/instance: '{{ .Release.Name }}' - # Generated labels {{- include "labels" . | nindent 4 }} -spec: - group: cert-manager.io - names: - kind: ClusterIssuer - listKind: ClusterIssuerList - plural: clusterissuers - singular: clusterissuer - categories: - - cert-manager - scope: Cluster - versions: - - name: v1 - subresources: - status: {} - additionalPrinterColumns: - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - priority: 1 - type: string - - jsonPath: .metadata.creationTimestamp - description: CreationTimestamp is a timestamp representing the server time when this object was created. It is not guaranteed to be set in happens-before order across separate operations. Clients may not set this value. It is represented in RFC3339 form and is in UTC. - name: Age - type: date - schema: - openAPIV3Schema: - description: |- - A ClusterIssuer represents a certificate issuing authority which can be - referenced as part of `issuerRef` fields. - It is similar to an Issuer, however it is cluster-scoped and therefore can - be referenced by resources that exist in *any* namespace, not just the same - namespace as the referent. - type: object - required: - - spec - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: Desired state of the ClusterIssuer resource. - type: object - properties: - acme: - description: |- - ACME configures this issuer to communicate with a RFC8555 (ACME) server - to obtain signed x509 certificates. - type: object - required: - - privateKeySecretRef - - server - properties: - caBundle: - description: |- - Base64-encoded bundle of PEM CAs which can be used to validate the certificate - chain presented by the ACME server. - Mutually exclusive with SkipTLSVerify; prefer using CABundle to prevent various - kinds of security vulnerabilities. - If CABundle and SkipTLSVerify are unset, the system certificate bundle inside - the container is used to validate the TLS connection. - type: string - format: byte - disableAccountKeyGeneration: - description: |- - Enables or disables generating a new ACME account key. - If true, the Issuer resource will *not* request a new account but will expect - the account key to be supplied via an existing secret. - If false, the cert-manager system will generate a new ACME account key - for the Issuer. - Defaults to false. - type: boolean - email: - description: |- - Email is the email address to be associated with the ACME account. - This field is optional, but it is strongly recommended to be set. - It will be used to contact you in case of issues with your account or - certificates, including expiry notification emails. - This field may be updated after the account is initially registered. - type: string - enableDurationFeature: - description: |- - Enables requesting a Not After date on certificates that matches the - duration of the certificate. This is not supported by all ACME servers - like Let's Encrypt. If set to true when the ACME server does not support - it, it will create an error on the Order. - Defaults to false. - type: boolean - externalAccountBinding: - description: |- - ExternalAccountBinding is a reference to a CA external account of the ACME - server. - If set, upon registration cert-manager will attempt to associate the given - external account credentials with the registered ACME account. - type: object - required: - - keyID - - keySecretRef - properties: - keyAlgorithm: - description: |- - Deprecated: keyAlgorithm field exists for historical compatibility - reasons and should not be used. The algorithm is now hardcoded to HS256 - in golang/x/crypto/acme. - type: string - enum: - - HS256 - - HS384 - - HS512 - keyID: - description: keyID is the ID of the CA key that the External Account is bound to. - type: string - keySecretRef: - description: |- - keySecretRef is a Secret Key Selector referencing a data item in a Kubernetes - Secret which holds the symmetric MAC key of the External Account Binding. - The `key` is the index string that is paired with the key data in the - Secret and should not be confused with the key data itself, or indeed with - the External Account Binding keyID above. - The secret key stored in the Secret **must** be un-padded, base64 URL - encoded data. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - preferredChain: - description: |- - PreferredChain is the chain to use if the ACME server outputs multiple. - PreferredChain is no guarantee that this one gets delivered by the ACME - endpoint. - For example, for Let's Encrypt's DST crosssign you would use: - "DST Root CA X3" or "ISRG Root X1" for the newer Let's Encrypt root CA. - This value picks the first certificate bundle in the combined set of - ACME default and alternative chains that has a root-most certificate with - this value as its issuer's commonname. - type: string - maxLength: 64 - privateKeySecretRef: - description: |- - PrivateKey is the name of a Kubernetes Secret resource that will be used to - store the automatically generated ACME account private key. - Optionally, a `key` may be specified to select a specific entry within - the named Secret resource. - If `key` is not specified, a default of `tls.key` will be used. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - server: - description: |- - Server is the URL used to access the ACME server's 'directory' endpoint. - For example, for Let's Encrypt's staging endpoint, you would use: - "https://acme-staging-v02.api.letsencrypt.org/directory". - Only ACME v2 endpoints (i.e. RFC 8555) are supported. - type: string - skipTLSVerify: - description: |- - INSECURE: Enables or disables validation of the ACME server TLS certificate. - If true, requests to the ACME server will not have the TLS certificate chain - validated. - Mutually exclusive with CABundle; prefer using CABundle to prevent various - kinds of security vulnerabilities. - Only enable this option in development environments. - If CABundle and SkipTLSVerify are unset, the system certificate bundle inside - the container is used to validate the TLS connection. - Defaults to false. - type: boolean - solvers: - description: |- - Solvers is a list of challenge solvers that will be used to solve - ACME challenges for the matching domains. - Solver configurations must be provided in order to obtain certificates - from an ACME server. - For more information, see: https://cert-manager.io/docs/configuration/acme/ - type: array - items: - description: |- - An ACMEChallengeSolver describes how to solve ACME challenges for the issuer it is part of. - A selector may be provided to use different solving strategies for different DNS names. - Only one of HTTP01 or DNS01 must be provided. - type: object - properties: - dns01: - description: |- - Configures cert-manager to attempt to complete authorizations by - performing the DNS01 challenge flow. - type: object - properties: - acmeDNS: - description: |- - Use the 'ACME DNS' (https://github.com/joohoi/acme-dns) API to manage - DNS01 challenge records. - type: object - required: - - accountSecretRef - - host - properties: - accountSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - host: - type: string - akamai: - description: Use the Akamai DNS zone management API to manage DNS01 challenge records. - type: object - required: - - accessTokenSecretRef - - clientSecretSecretRef - - clientTokenSecretRef - - serviceConsumerDomain - properties: - accessTokenSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - clientSecretSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - clientTokenSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - serviceConsumerDomain: - type: string - azureDNS: - description: Use the Microsoft Azure DNS API to manage DNS01 challenge records. - type: object - required: - - resourceGroupName - - subscriptionID - properties: - clientID: - description: |- - Auth: Azure Service Principal: - The ClientID of the Azure Service Principal used to authenticate with Azure DNS. - If set, ClientSecret and TenantID must also be set. - type: string - clientSecretSecretRef: - description: |- - Auth: Azure Service Principal: - A reference to a Secret containing the password associated with the Service Principal. - If set, ClientID and TenantID must also be set. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - environment: - description: name of the Azure environment (default AzurePublicCloud) - type: string - enum: - - AzurePublicCloud - - AzureChinaCloud - - AzureGermanCloud - - AzureUSGovernmentCloud - hostedZoneName: - description: name of the DNS zone that should be used - type: string - managedIdentity: - description: |- - Auth: Azure Workload Identity or Azure Managed Service Identity: - Settings to enable Azure Workload Identity or Azure Managed Service Identity - If set, ClientID, ClientSecret and TenantID must not be set. - type: object - properties: - clientID: - description: client ID of the managed identity, can not be used at the same time as resourceID - type: string - resourceID: - description: |- - resource ID of the managed identity, can not be used at the same time as clientID - Cannot be used for Azure Managed Service Identity - type: string - resourceGroupName: - description: resource group the DNS zone is located in - type: string - subscriptionID: - description: ID of the Azure subscription - type: string - tenantID: - description: |- - Auth: Azure Service Principal: - The TenantID of the Azure Service Principal used to authenticate with Azure DNS. - If set, ClientID and ClientSecret must also be set. - type: string - cloudDNS: - description: Use the Google Cloud DNS API to manage DNS01 challenge records. - type: object - required: - - project - properties: - hostedZoneName: - description: |- - HostedZoneName is an optional field that tells cert-manager in which - Cloud DNS zone the challenge record has to be created. - If left empty cert-manager will automatically choose a zone. - type: string - project: - type: string - serviceAccountSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - cloudflare: - description: Use the Cloudflare API to manage DNS01 challenge records. - type: object - properties: - apiKeySecretRef: - description: |- - API key to use to authenticate with Cloudflare. - Note: using an API token to authenticate is now the recommended method - as it allows greater control of permissions. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - apiTokenSecretRef: - description: API token used to authenticate with Cloudflare. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - email: - description: Email of the account, only required when using API key based authentication. - type: string - cnameStrategy: - description: |- - CNAMEStrategy configures how the DNS01 provider should handle CNAME - records when found in DNS zones. - type: string - enum: - - None - - Follow - digitalocean: - description: Use the DigitalOcean DNS API to manage DNS01 challenge records. - type: object - required: - - tokenSecretRef - properties: - tokenSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - rfc2136: - description: |- - Use RFC2136 ("Dynamic Updates in the Domain Name System") (https://datatracker.ietf.org/doc/rfc2136/) - to manage DNS01 challenge records. - type: object - required: - - nameserver - properties: - nameserver: - description: |- - The IP address or hostname of an authoritative DNS server supporting - RFC2136 in the form host:port. If the host is an IPv6 address it must be - enclosed in square brackets (e.g [2001:db8::1]) ; port is optional. - This field is required. - type: string - tsigAlgorithm: - description: |- - The TSIG Algorithm configured in the DNS supporting RFC2136. Used only - when ``tsigSecretSecretRef`` and ``tsigKeyName`` are defined. - Supported values are (case-insensitive): ``HMACMD5`` (default), - ``HMACSHA1``, ``HMACSHA256`` or ``HMACSHA512``. - type: string - tsigKeyName: - description: |- - The TSIG Key name configured in the DNS. - If ``tsigSecretSecretRef`` is defined, this field is required. - type: string - tsigSecretSecretRef: - description: |- - The name of the secret containing the TSIG value. - If ``tsigKeyName`` is defined, this field is required. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - route53: - description: Use the AWS Route53 API to manage DNS01 challenge records. - type: object - required: - - region - properties: - accessKeyID: - description: |- - The AccessKeyID is used for authentication. - Cannot be set when SecretAccessKeyID is set. - If neither the Access Key nor Key ID are set, we fall-back to using env - vars, shared credentials file or AWS Instance metadata, - see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials - type: string - accessKeyIDSecretRef: - description: |- - The SecretAccessKey is used for authentication. If set, pull the AWS - access key ID from a key within a Kubernetes Secret. - Cannot be set when AccessKeyID is set. - If neither the Access Key nor Key ID are set, we fall-back to using env - vars, shared credentials file or AWS Instance metadata, - see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - auth: - description: Auth configures how cert-manager authenticates. - type: object - required: - - kubernetes - properties: - kubernetes: - description: |- - Kubernetes authenticates with Route53 using AssumeRoleWithWebIdentity - by passing a bound ServiceAccount token. - type: object - required: - - serviceAccountRef - properties: - serviceAccountRef: - description: |- - A reference to a service account that will be used to request a bound - token (also known as "projected token"). To use this field, you must - configure an RBAC rule to let cert-manager request a token. - type: object - required: - - name - properties: - audiences: - description: |- - TokenAudiences is an optional list of audiences to include in the - token passed to AWS. The default token consisting of the issuer's namespace - and name is always included. - If unset the audience defaults to `sts.amazonaws.com`. - type: array - items: - type: string - name: - description: Name of the ServiceAccount used to request a token. - type: string - hostedZoneID: - description: If set, the provider will manage only this zone in Route53 and will not do an lookup using the route53:ListHostedZonesByName api call. - type: string - region: - description: Always set the region when using AccessKeyID and SecretAccessKey - type: string - role: - description: |- - Role is a Role ARN which the Route53 provider will assume using either the explicit credentials AccessKeyID/SecretAccessKey - or the inferred credentials from environment variables, shared credentials file or AWS Instance metadata - type: string - secretAccessKeySecretRef: - description: |- - The SecretAccessKey is used for authentication. - If neither the Access Key nor Key ID are set, we fall-back to using env - vars, shared credentials file or AWS Instance metadata, - see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - webhook: - description: |- - Configure an external webhook based DNS01 challenge solver to manage - DNS01 challenge records. - type: object - required: - - groupName - - solverName - properties: - config: - description: |- - Additional configuration that should be passed to the webhook apiserver - when challenges are processed. - This can contain arbitrary JSON data. - Secret values should not be specified in this stanza. - If secret values are needed (e.g. credentials for a DNS service), you - should use a SecretKeySelector to reference a Secret resource. - For details on the schema of this field, consult the webhook provider - implementation's documentation. - x-kubernetes-preserve-unknown-fields: true - groupName: - description: |- - The API group name that should be used when POSTing ChallengePayload - resources to the webhook apiserver. - This should be the same as the GroupName specified in the webhook - provider implementation. - type: string - solverName: - description: |- - The name of the solver to use, as defined in the webhook provider - implementation. - This will typically be the name of the provider, e.g. 'cloudflare'. - type: string - http01: - description: |- - Configures cert-manager to attempt to complete authorizations by - performing the HTTP01 challenge flow. - It is not possible to obtain certificates for wildcard domain names - (e.g. `*.example.com`) using the HTTP01 challenge mechanism. - type: object - properties: - gatewayHTTPRoute: - description: |- - The Gateway API is a sig-network community API that models service networking - in Kubernetes (https://gateway-api.sigs.k8s.io/). The Gateway solver will - create HTTPRoutes with the specified labels in the same namespace as the challenge. - This solver is experimental, and fields / behaviour may change in the future. - type: object - properties: - labels: - description: |- - Custom labels that will be applied to HTTPRoutes created by cert-manager - while solving HTTP-01 challenges. - type: object - additionalProperties: - type: string - parentRefs: - description: |- - When solving an HTTP-01 challenge, cert-manager creates an HTTPRoute. - cert-manager needs to know which parentRefs should be used when creating - the HTTPRoute. Usually, the parentRef references a Gateway. See: - https://gateway-api.sigs.k8s.io/api-types/httproute/#attaching-to-gateways - type: array - items: - description: |- - ParentReference identifies an API object (usually a Gateway) that can be considered - a parent of this resource (usually a route). There are two kinds of parent resources - with "Core" support: - - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - - This API may be extended in the future to support additional kinds of parent - resources. - - - The API object must be valid in the cluster; the Group and Kind must - be registered in the cluster for this reference to be valid. - type: object - required: - - name - properties: - group: - description: |- - Group is the group of the referent. - When unspecified, "gateway.networking.k8s.io" is inferred. - To set the core API group (such as for a "Service" kind referent), - Group must be explicitly set to "" (empty string). - - - Support: Core - type: string - default: gateway.networking.k8s.io - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - kind: - description: |- - Kind is kind of the referent. - - - There are two kinds of parent resources with "Core" support: - - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - - Support for other resources is Implementation-Specific. - type: string - default: Gateway - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - name: - description: |- - Name is the name of the referent. - - - Support: Core - type: string - maxLength: 253 - minLength: 1 - namespace: - description: |- - Namespace is the namespace of the referent. When unspecified, this refers - to the local namespace of the Route. - - - Note that there are specific rules for ParentRefs which cross namespace - boundaries. Cross-namespace references are only valid if they are explicitly - allowed by something in the namespace they are referring to. For example: - Gateway has the AllowedRoutes field, and ReferenceGrant provides a - generic way to enable any other kind of cross-namespace reference. - - - - ParentRefs from a Route to a Service in the same namespace are "producer" - routes, which apply default routing rules to inbound connections from - any namespace to the Service. - - - ParentRefs from a Route to a Service in a different namespace are - "consumer" routes, and these routing rules are only applied to outbound - connections originating from the same namespace as the Route, for which - the intended destination of the connections are a Service targeted as a - ParentRef of the Route. - - - - Support: Core - type: string - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - port: - description: |- - Port is the network port this Route targets. It can be interpreted - differently based on the type of parent resource. - - - When the parent resource is a Gateway, this targets all listeners - listening on the specified port that also support this kind of Route(and - select this Route). It's not recommended to set `Port` unless the - networking behaviors specified in a Route must apply to a specific port - as opposed to a listener(s) whose port(s) may be changed. When both Port - and SectionName are specified, the name and port of the selected listener - must match both specified values. - - - - When the parent resource is a Service, this targets a specific port in the - Service spec. When both Port (experimental) and SectionName are specified, - the name and port of the selected port must match both specified values. - - - - Implementations MAY choose to support other parent resources. - Implementations supporting other types of parent resources MUST clearly - document how/if Port is interpreted. - - - For the purpose of status, an attachment is considered successful as - long as the parent resource accepts it partially. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment - from the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, - the Route MUST be considered detached from the Gateway. - - - Support: Extended - type: integer - format: int32 - maximum: 65535 - minimum: 1 - sectionName: - description: |- - SectionName is the name of a section within the target resource. In the - following resources, SectionName is interpreted as the following: - - - * Gateway: Listener name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - * Service: Port name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - - - Implementations MAY choose to support attaching Routes to other resources. - If that is the case, they MUST clearly document how SectionName is - interpreted. - - - When unspecified (empty string), this will reference the entire resource. - For the purpose of status, an attachment is considered successful if at - least one section in the parent resource accepts it. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from - the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, the - Route MUST be considered detached from the Gateway. - - - Support: Core - type: string - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - serviceType: - description: |- - Optional service type for Kubernetes solver service. Supported values - are NodePort or ClusterIP. If unset, defaults to NodePort. - type: string - ingress: - description: |- - The ingress based HTTP01 challenge solver will solve challenges by - creating or modifying Ingress resources in order to route requests for - '/.well-known/acme-challenge/XYZ' to 'challenge solver' pods that are - provisioned by cert-manager for each Challenge to be completed. - type: object - properties: - class: - description: |- - This field configures the annotation `kubernetes.io/ingress.class` when - creating Ingress resources to solve ACME challenges that use this - challenge solver. Only one of `class`, `name` or `ingressClassName` may - be specified. - type: string - ingressClassName: - description: |- - This field configures the field `ingressClassName` on the created Ingress - resources used to solve ACME challenges that use this challenge solver. - This is the recommended way of configuring the ingress class. Only one of - `class`, `name` or `ingressClassName` may be specified. - type: string - ingressTemplate: - description: |- - Optional ingress template used to configure the ACME challenge solver - ingress used for HTTP01 challenges. - type: object - properties: - metadata: - description: |- - ObjectMeta overrides for the ingress used to solve HTTP01 challenges. - Only the 'labels' and 'annotations' fields may be set. - If labels or annotations overlap with in-built values, the values here - will override the in-built values. - type: object - properties: - annotations: - description: Annotations that should be added to the created ACME HTTP01 solver ingress. - type: object - additionalProperties: - type: string - labels: - description: Labels that should be added to the created ACME HTTP01 solver ingress. - type: object - additionalProperties: - type: string - name: - description: |- - The name of the ingress resource that should have ACME challenge solving - routes inserted into it in order to solve HTTP01 challenges. - This is typically used in conjunction with ingress controllers like - ingress-gce, which maintains a 1:1 mapping between external IPs and - ingress resources. Only one of `class`, `name` or `ingressClassName` may - be specified. - type: string - podTemplate: - description: |- - Optional pod template used to configure the ACME challenge solver pods - used for HTTP01 challenges. - type: object - properties: - metadata: - description: |- - ObjectMeta overrides for the pod used to solve HTTP01 challenges. - Only the 'labels' and 'annotations' fields may be set. - If labels or annotations overlap with in-built values, the values here - will override the in-built values. - type: object - properties: - annotations: - description: Annotations that should be added to the create ACME HTTP01 solver pods. - type: object - additionalProperties: - type: string - labels: - description: Labels that should be added to the created ACME HTTP01 solver pods. - type: object - additionalProperties: - type: string - spec: - description: |- - PodSpec defines overrides for the HTTP01 challenge solver pod. - Check ACMEChallengeSolverHTTP01IngressPodSpec to find out currently supported fields. - All other fields will be ignored. - type: object - properties: - affinity: - description: If specified, the pod's scheduling constraints - type: object - properties: - nodeAffinity: - description: Describes node affinity scheduling rules for the pod. - type: object - properties: - preferredDuringSchedulingIgnoredDuringExecution: - description: |- - The scheduler will prefer to schedule pods to nodes that satisfy - the affinity expressions specified by this field, but it may choose - a node that violates one or more of the expressions. The node that is - most preferred is the one with the greatest sum of weights, i.e. - for each node that meets all of the scheduling requirements (resource - request, requiredDuringScheduling affinity expressions, etc.), - compute a sum by iterating through the elements of this field and adding - "weight" to the sum if the node matches the corresponding matchExpressions; the - node(s) with the highest sum are the most preferred. - type: array - items: - description: |- - An empty preferred scheduling term matches all objects with implicit weight 0 - (i.e. it's a no-op). A null preferred scheduling term matches no objects (i.e. is also a no-op). - type: object - required: - - preference - - weight - properties: - preference: - description: A node selector term, associated with the corresponding weight. - type: object - properties: - matchExpressions: - description: A list of node selector requirements by node's labels. - type: array - items: - description: |- - A node selector requirement is a selector that contains values, a key, and an operator - that relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: The label key that the selector applies to. - type: string - operator: - description: |- - Represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. - type: string - values: - description: |- - An array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. If the operator is Gt or Lt, the values - array must have a single element, which will be interpreted as an integer. - This array is replaced during a strategic merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchFields: - description: A list of node selector requirements by node's fields. - type: array - items: - description: |- - A node selector requirement is a selector that contains values, a key, and an operator - that relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: The label key that the selector applies to. - type: string - operator: - description: |- - Represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. - type: string - values: - description: |- - An array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. If the operator is Gt or Lt, the values - array must have a single element, which will be interpreted as an integer. - This array is replaced during a strategic merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - x-kubernetes-map-type: atomic - weight: - description: Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100. - type: integer - format: int32 - x-kubernetes-list-type: atomic - requiredDuringSchedulingIgnoredDuringExecution: - description: |- - If the affinity requirements specified by this field are not met at - scheduling time, the pod will not be scheduled onto the node. - If the affinity requirements specified by this field cease to be met - at some point during pod execution (e.g. due to an update), the system - may or may not try to eventually evict the pod from its node. - type: object - required: - - nodeSelectorTerms - properties: - nodeSelectorTerms: - description: Required. A list of node selector terms. The terms are ORed. - type: array - items: - description: |- - A null or empty node selector term matches no objects. The requirements of - them are ANDed. - The TopologySelectorTerm type implements a subset of the NodeSelectorTerm. - type: object - properties: - matchExpressions: - description: A list of node selector requirements by node's labels. - type: array - items: - description: |- - A node selector requirement is a selector that contains values, a key, and an operator - that relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: The label key that the selector applies to. - type: string - operator: - description: |- - Represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. - type: string - values: - description: |- - An array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. If the operator is Gt or Lt, the values - array must have a single element, which will be interpreted as an integer. - This array is replaced during a strategic merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchFields: - description: A list of node selector requirements by node's fields. - type: array - items: - description: |- - A node selector requirement is a selector that contains values, a key, and an operator - that relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: The label key that the selector applies to. - type: string - operator: - description: |- - Represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. - type: string - values: - description: |- - An array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. If the operator is Gt or Lt, the values - array must have a single element, which will be interpreted as an integer. - This array is replaced during a strategic merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - x-kubernetes-map-type: atomic - x-kubernetes-list-type: atomic - x-kubernetes-map-type: atomic - podAffinity: - description: Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)). - type: object - properties: - preferredDuringSchedulingIgnoredDuringExecution: - description: |- - The scheduler will prefer to schedule pods to nodes that satisfy - the affinity expressions specified by this field, but it may choose - a node that violates one or more of the expressions. The node that is - most preferred is the one with the greatest sum of weights, i.e. - for each node that meets all of the scheduling requirements (resource - request, requiredDuringScheduling affinity expressions, etc.), - compute a sum by iterating through the elements of this field and adding - "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the - node(s) with the highest sum are the most preferred. - type: array - items: - description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) - type: object - required: - - podAffinityTerm - - weight - properties: - podAffinityTerm: - description: Required. A pod affinity term, associated with the corresponding weight. - type: object - required: - - topologyKey - properties: - labelSelector: - description: |- - A label query over a set of resources, in this case pods. - If it's null, this PodAffinityTerm matches with no Pods. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - matchLabelKeys: - description: |- - MatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both matchLabelKeys and labelSelector. - Also, matchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - mismatchLabelKeys: - description: |- - MismatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. - Also, mismatchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - namespaceSelector: - description: |- - A label query over the set of namespaces that the term applies to. - The term is applied to the union of the namespaces selected by this field - and the ones listed in the namespaces field. - null selector and null or empty namespaces list means "this pod's namespace". - An empty selector ({}) matches all namespaces. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - namespaces: - description: |- - namespaces specifies a static list of namespace names that the term applies to. - The term is applied to the union of the namespaces listed in this field - and the ones selected by namespaceSelector. - null or empty namespaces list and null namespaceSelector means "this pod's namespace". - type: array - items: - type: string - x-kubernetes-list-type: atomic - topologyKey: - description: |- - This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching - the labelSelector in the specified namespaces, where co-located is defined as running on a node - whose value of the label with key topologyKey matches that of any node on which any of the - selected pods is running. - Empty topologyKey is not allowed. - type: string - weight: - description: |- - weight associated with matching the corresponding podAffinityTerm, - in the range 1-100. - type: integer - format: int32 - x-kubernetes-list-type: atomic - requiredDuringSchedulingIgnoredDuringExecution: - description: |- - If the affinity requirements specified by this field are not met at - scheduling time, the pod will not be scheduled onto the node. - If the affinity requirements specified by this field cease to be met - at some point during pod execution (e.g. due to a pod label update), the - system may or may not try to eventually evict the pod from its node. - When there are multiple elements, the lists of nodes corresponding to each - podAffinityTerm are intersected, i.e. all terms must be satisfied. - type: array - items: - description: |- - Defines a set of pods (namely those matching the labelSelector - relative to the given namespace(s)) that this pod should be - co-located (affinity) or not co-located (anti-affinity) with, - where co-located is defined as running on a node whose value of - the label with key matches that of any node on which - a pod of the set of pods is running - type: object - required: - - topologyKey - properties: - labelSelector: - description: |- - A label query over a set of resources, in this case pods. - If it's null, this PodAffinityTerm matches with no Pods. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - matchLabelKeys: - description: |- - MatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both matchLabelKeys and labelSelector. - Also, matchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - mismatchLabelKeys: - description: |- - MismatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. - Also, mismatchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - namespaceSelector: - description: |- - A label query over the set of namespaces that the term applies to. - The term is applied to the union of the namespaces selected by this field - and the ones listed in the namespaces field. - null selector and null or empty namespaces list means "this pod's namespace". - An empty selector ({}) matches all namespaces. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - namespaces: - description: |- - namespaces specifies a static list of namespace names that the term applies to. - The term is applied to the union of the namespaces listed in this field - and the ones selected by namespaceSelector. - null or empty namespaces list and null namespaceSelector means "this pod's namespace". - type: array - items: - type: string - x-kubernetes-list-type: atomic - topologyKey: - description: |- - This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching - the labelSelector in the specified namespaces, where co-located is defined as running on a node - whose value of the label with key topologyKey matches that of any node on which any of the - selected pods is running. - Empty topologyKey is not allowed. - type: string - x-kubernetes-list-type: atomic - podAntiAffinity: - description: Describes pod anti-affinity scheduling rules (e.g. avoid putting this pod in the same node, zone, etc. as some other pod(s)). - type: object - properties: - preferredDuringSchedulingIgnoredDuringExecution: - description: |- - The scheduler will prefer to schedule pods to nodes that satisfy - the anti-affinity expressions specified by this field, but it may choose - a node that violates one or more of the expressions. The node that is - most preferred is the one with the greatest sum of weights, i.e. - for each node that meets all of the scheduling requirements (resource - request, requiredDuringScheduling anti-affinity expressions, etc.), - compute a sum by iterating through the elements of this field and adding - "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the - node(s) with the highest sum are the most preferred. - type: array - items: - description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) - type: object - required: - - podAffinityTerm - - weight - properties: - podAffinityTerm: - description: Required. A pod affinity term, associated with the corresponding weight. - type: object - required: - - topologyKey - properties: - labelSelector: - description: |- - A label query over a set of resources, in this case pods. - If it's null, this PodAffinityTerm matches with no Pods. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - matchLabelKeys: - description: |- - MatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both matchLabelKeys and labelSelector. - Also, matchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - mismatchLabelKeys: - description: |- - MismatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. - Also, mismatchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - namespaceSelector: - description: |- - A label query over the set of namespaces that the term applies to. - The term is applied to the union of the namespaces selected by this field - and the ones listed in the namespaces field. - null selector and null or empty namespaces list means "this pod's namespace". - An empty selector ({}) matches all namespaces. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - namespaces: - description: |- - namespaces specifies a static list of namespace names that the term applies to. - The term is applied to the union of the namespaces listed in this field - and the ones selected by namespaceSelector. - null or empty namespaces list and null namespaceSelector means "this pod's namespace". - type: array - items: - type: string - x-kubernetes-list-type: atomic - topologyKey: - description: |- - This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching - the labelSelector in the specified namespaces, where co-located is defined as running on a node - whose value of the label with key topologyKey matches that of any node on which any of the - selected pods is running. - Empty topologyKey is not allowed. - type: string - weight: - description: |- - weight associated with matching the corresponding podAffinityTerm, - in the range 1-100. - type: integer - format: int32 - x-kubernetes-list-type: atomic - requiredDuringSchedulingIgnoredDuringExecution: - description: |- - If the anti-affinity requirements specified by this field are not met at - scheduling time, the pod will not be scheduled onto the node. - If the anti-affinity requirements specified by this field cease to be met - at some point during pod execution (e.g. due to a pod label update), the - system may or may not try to eventually evict the pod from its node. - When there are multiple elements, the lists of nodes corresponding to each - podAffinityTerm are intersected, i.e. all terms must be satisfied. - type: array - items: - description: |- - Defines a set of pods (namely those matching the labelSelector - relative to the given namespace(s)) that this pod should be - co-located (affinity) or not co-located (anti-affinity) with, - where co-located is defined as running on a node whose value of - the label with key matches that of any node on which - a pod of the set of pods is running - type: object - required: - - topologyKey - properties: - labelSelector: - description: |- - A label query over a set of resources, in this case pods. - If it's null, this PodAffinityTerm matches with no Pods. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - matchLabelKeys: - description: |- - MatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both matchLabelKeys and labelSelector. - Also, matchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - mismatchLabelKeys: - description: |- - MismatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. - Also, mismatchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - namespaceSelector: - description: |- - A label query over the set of namespaces that the term applies to. - The term is applied to the union of the namespaces selected by this field - and the ones listed in the namespaces field. - null selector and null or empty namespaces list means "this pod's namespace". - An empty selector ({}) matches all namespaces. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - namespaces: - description: |- - namespaces specifies a static list of namespace names that the term applies to. - The term is applied to the union of the namespaces listed in this field - and the ones selected by namespaceSelector. - null or empty namespaces list and null namespaceSelector means "this pod's namespace". - type: array - items: - type: string - x-kubernetes-list-type: atomic - topologyKey: - description: |- - This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching - the labelSelector in the specified namespaces, where co-located is defined as running on a node - whose value of the label with key topologyKey matches that of any node on which any of the - selected pods is running. - Empty topologyKey is not allowed. - type: string - x-kubernetes-list-type: atomic - imagePullSecrets: - description: If specified, the pod's imagePullSecrets - type: array - items: - description: |- - LocalObjectReference contains enough information to let you locate the - referenced object inside the same namespace. - type: object - properties: - name: - description: |- - Name of the referent. - This field is effectively required, but due to backwards compatibility is - allowed to be empty. Instances of this type with an empty value here are - almost certainly wrong. - TODO: Add other useful fields. apiVersion, kind, uid? - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - TODO: Drop `kubebuilder:default` when controller-gen doesn't need it https://github.com/kubernetes-sigs/kubebuilder/issues/3896. - type: string - default: "" - x-kubernetes-map-type: atomic - nodeSelector: - description: |- - NodeSelector is a selector which must be true for the pod to fit on a node. - Selector which must match a node's labels for the pod to be scheduled on that node. - More info: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/ - type: object - additionalProperties: - type: string - priorityClassName: - description: If specified, the pod's priorityClassName. - type: string - serviceAccountName: - description: If specified, the pod's service account - type: string - tolerations: - description: If specified, the pod's tolerations. - type: array - items: - description: |- - The pod this Toleration is attached to tolerates any taint that matches - the triple using the matching operator . - type: object - properties: - effect: - description: |- - Effect indicates the taint effect to match. Empty means match all taint effects. - When specified, allowed values are NoSchedule, PreferNoSchedule and NoExecute. - type: string - key: - description: |- - Key is the taint key that the toleration applies to. Empty means match all taint keys. - If the key is empty, operator must be Exists; this combination means to match all values and all keys. - type: string - operator: - description: |- - Operator represents a key's relationship to the value. - Valid operators are Exists and Equal. Defaults to Equal. - Exists is equivalent to wildcard for value, so that a pod can - tolerate all taints of a particular category. - type: string - tolerationSeconds: - description: |- - TolerationSeconds represents the period of time the toleration (which must be - of effect NoExecute, otherwise this field is ignored) tolerates the taint. By default, - it is not set, which means tolerate the taint forever (do not evict). Zero and - negative values will be treated as 0 (evict immediately) by the system. - type: integer - format: int64 - value: - description: |- - Value is the taint value the toleration matches to. - If the operator is Exists, the value should be empty, otherwise just a regular string. - type: string - serviceType: - description: |- - Optional service type for Kubernetes solver service. Supported values - are NodePort or ClusterIP. If unset, defaults to NodePort. - type: string - selector: - description: |- - Selector selects a set of DNSNames on the Certificate resource that - should be solved using this challenge solver. - If not specified, the solver will be treated as the 'default' solver - with the lowest priority, i.e. if any other solver has a more specific - match, it will be used instead. - type: object - properties: - dnsNames: - description: |- - List of DNSNames that this solver will be used to solve. - If specified and a match is found, a dnsNames selector will take - precedence over a dnsZones selector. - If multiple solvers match with the same dnsNames value, the solver - with the most matching labels in matchLabels will be selected. - If neither has more matches, the solver defined earlier in the list - will be selected. - type: array - items: - type: string - dnsZones: - description: |- - List of DNSZones that this solver will be used to solve. - The most specific DNS zone match specified here will take precedence - over other DNS zone matches, so a solver specifying sys.example.com - will be selected over one specifying example.com for the domain - www.sys.example.com. - If multiple solvers match with the same dnsZones value, the solver - with the most matching labels in matchLabels will be selected. - If neither has more matches, the solver defined earlier in the list - will be selected. - type: array - items: - type: string - matchLabels: - description: |- - A label selector that is used to refine the set of certificate's that - this challenge solver will apply to. - type: object - additionalProperties: - type: string - ca: - description: |- - CA configures this issuer to sign certificates using a signing CA keypair - stored in a Secret resource. - This is used to build internal PKIs that are managed by cert-manager. - type: object - required: - - secretName - properties: - crlDistributionPoints: - description: |- - The CRL distribution points is an X.509 v3 certificate extension which identifies - the location of the CRL from which the revocation of this certificate can be checked. - If not set, certificates will be issued without distribution points set. - type: array - items: - type: string - issuingCertificateURLs: - description: |- - IssuingCertificateURLs is a list of URLs which this issuer should embed into certificates - it creates. See https://www.rfc-editor.org/rfc/rfc5280#section-4.2.2.1 for more details. - As an example, such a URL might be "http://ca.domain.com/ca.crt". - type: array - items: - type: string - ocspServers: - description: |- - The OCSP server list is an X.509 v3 extension that defines a list of - URLs of OCSP responders. The OCSP responders can be queried for the - revocation status of an issued certificate. If not set, the - certificate will be issued with no OCSP servers set. For example, an - OCSP server URL could be "http://ocsp.int-x3.letsencrypt.org". - type: array - items: - type: string - secretName: - description: |- - SecretName is the name of the secret used to sign Certificates issued - by this Issuer. - type: string - selfSigned: - description: |- - SelfSigned configures this issuer to 'self sign' certificates using the - private key used to create the CertificateRequest object. - type: object - properties: - crlDistributionPoints: - description: |- - The CRL distribution points is an X.509 v3 certificate extension which identifies - the location of the CRL from which the revocation of this certificate can be checked. - If not set certificate will be issued without CDP. Values are strings. - type: array - items: - type: string - vault: - description: |- - Vault configures this issuer to sign certificates using a HashiCorp Vault - PKI backend. - type: object - required: - - auth - - path - - server - properties: - auth: - description: Auth configures how cert-manager authenticates with the Vault server. - type: object - properties: - appRole: - description: |- - AppRole authenticates with Vault using the App Role auth mechanism, - with the role and secret stored in a Kubernetes Secret resource. - type: object - required: - - path - - roleId - - secretRef - properties: - path: - description: |- - Path where the App Role authentication backend is mounted in Vault, e.g: - "approle" - type: string - roleId: - description: |- - RoleID configured in the App Role authentication backend when setting - up the authentication backend in Vault. - type: string - secretRef: - description: |- - Reference to a key in a Secret that contains the App Role secret used - to authenticate with Vault. - The `key` field must be specified and denotes which entry within the Secret - resource is used as the app role secret. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - kubernetes: - description: |- - Kubernetes authenticates with Vault by passing the ServiceAccount - token stored in the named Secret resource to the Vault server. - type: object - required: - - role - properties: - mountPath: - description: |- - The Vault mountPath here is the mount path to use when authenticating with - Vault. For example, setting a value to `/v1/auth/foo`, will use the path - `/v1/auth/foo/login` to authenticate with Vault. If unspecified, the - default value "/v1/auth/kubernetes" will be used. - type: string - role: - description: |- - A required field containing the Vault Role to assume. A Role binds a - Kubernetes ServiceAccount with a set of Vault policies. - type: string - secretRef: - description: |- - The required Secret field containing a Kubernetes ServiceAccount JWT used - for authenticating with Vault. Use of 'ambient credentials' is not - supported. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - serviceAccountRef: - description: |- - A reference to a service account that will be used to request a bound - token (also known as "projected token"). Compared to using "secretRef", - using this field means that you don't rely on statically bound tokens. To - use this field, you must configure an RBAC rule to let cert-manager - request a token. - type: object - required: - - name - properties: - audiences: - description: |- - TokenAudiences is an optional list of extra audiences to include in the token passed to Vault. The default token - consisting of the issuer's namespace and name is always included. - type: array - items: - type: string - name: - description: Name of the ServiceAccount used to request a token. - type: string - tokenSecretRef: - description: TokenSecretRef authenticates with Vault by presenting a token. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - caBundle: - description: |- - Base64-encoded bundle of PEM CAs which will be used to validate the certificate - chain presented by Vault. Only used if using HTTPS to connect to Vault and - ignored for HTTP connections. - Mutually exclusive with CABundleSecretRef. - If neither CABundle nor CABundleSecretRef are defined, the certificate bundle in - the cert-manager controller container is used to validate the TLS connection. - type: string - format: byte - caBundleSecretRef: - description: |- - Reference to a Secret containing a bundle of PEM-encoded CAs to use when - verifying the certificate chain presented by Vault when using HTTPS. - Mutually exclusive with CABundle. - If neither CABundle nor CABundleSecretRef are defined, the certificate bundle in - the cert-manager controller container is used to validate the TLS connection. - If no key for the Secret is specified, cert-manager will default to 'ca.crt'. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - clientCertSecretRef: - description: |- - Reference to a Secret containing a PEM-encoded Client Certificate to use when the - Vault server requires mTLS. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - clientKeySecretRef: - description: |- - Reference to a Secret containing a PEM-encoded Client Private Key to use when the - Vault server requires mTLS. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - namespace: - description: |- - Name of the vault namespace. Namespaces is a set of features within Vault Enterprise that allows Vault environments to support Secure Multi-tenancy. e.g: "ns1" - More about namespaces can be found here https://www.vaultproject.io/docs/enterprise/namespaces - type: string - path: - description: |- - Path is the mount path of the Vault PKI backend's `sign` endpoint, e.g: - "my_pki_mount/sign/my-role-name". - type: string - server: - description: 'Server is the connection address for the Vault server, e.g: "https://vault.example.com:8200".' - type: string - venafi: - description: |- - Venafi configures this issuer to sign certificates using a Venafi TPP - or Venafi Cloud policy zone. - type: object - required: - - zone - properties: - cloud: - description: |- - Cloud specifies the Venafi cloud configuration settings. - Only one of TPP or Cloud may be specified. - type: object - required: - - apiTokenSecretRef - properties: - apiTokenSecretRef: - description: APITokenSecretRef is a secret key selector for the Venafi Cloud API token. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - url: - description: |- - URL is the base URL for Venafi Cloud. - Defaults to "https://api.venafi.cloud/v1". - type: string - tpp: - description: |- - TPP specifies Trust Protection Platform configuration settings. - Only one of TPP or Cloud may be specified. - type: object - required: - - credentialsRef - - url - properties: - caBundle: - description: |- - Base64-encoded bundle of PEM CAs which will be used to validate the certificate - chain presented by the TPP server. Only used if using HTTPS; ignored for HTTP. - If undefined, the certificate bundle in the cert-manager controller container - is used to validate the chain. - type: string - format: byte - credentialsRef: - description: |- - CredentialsRef is a reference to a Secret containing the username and - password for the TPP server. - The secret must contain two keys, 'username' and 'password'. - type: object - required: - - name - properties: - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - url: - description: |- - URL is the base URL for the vedsdk endpoint of the Venafi TPP instance, - for example: "https://tpp.example.com/vedsdk". - type: string - zone: - description: |- - Zone is the Venafi Policy Zone to use for this issuer. - All requests made to the Venafi platform will be restricted by the named - zone policy. - This field is required. - type: string - status: - description: Status of the ClusterIssuer. This is set and managed automatically. - type: object - properties: - acme: - description: |- - ACME specific status options. - This field should only be set if the Issuer is configured to use an ACME - server to issue certificates. - type: object - properties: - lastPrivateKeyHash: - description: |- - LastPrivateKeyHash is a hash of the private key associated with the latest - registered ACME account, in order to track changes made to registered account - associated with the Issuer - type: string - lastRegisteredEmail: - description: |- - LastRegisteredEmail is the email associated with the latest registered - ACME account, in order to track changes made to registered account - associated with the Issuer - type: string - uri: - description: |- - URI is the unique account identifier, which can also be used to retrieve - account details from the CA - type: string - conditions: - description: |- - List of status conditions to indicate the status of a CertificateRequest. - Known condition types are `Ready`. - type: array - items: - description: IssuerCondition contains condition information for an Issuer. - type: object - required: - - status - - type - properties: - lastTransitionTime: - description: |- - LastTransitionTime is the timestamp corresponding to the last status - change of this condition. - type: string - format: date-time - message: - description: |- - Message is a human readable description of the details of the last - transition, complementing reason. - type: string - observedGeneration: - description: |- - If set, this represents the .metadata.generation that the condition was - set based upon. - For instance, if .metadata.generation is currently 12, but the - .status.condition[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the Issuer. - type: integer - format: int64 - reason: - description: |- - Reason is a brief machine readable explanation for the condition's last - transition. - type: string - status: - description: Status of the condition, one of (`True`, `False`, `Unknown`). - type: string - enum: - - "True" - - "False" - - Unknown - type: - description: Type of the condition, known values are (`Ready`). - type: string - x-kubernetes-list-map-keys: - - type - x-kubernetes-list-type: map - served: true - storage: true - -# END crd {{- end }} - ---- -# START crd {{- if or .Values.crds.enabled .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - name: issuers.cert-manager.io - # START annotations {{- if .Values.crds.keep }} - annotations: - helm.sh/resource-policy: keep - # END annotations {{- end }} - labels: - app: '{{ template "cert-manager.name" . }}' - app.kubernetes.io/name: '{{ template "cert-manager.name" . }}' - app.kubernetes.io/instance: '{{ .Release.Name }}' - app.kubernetes.io/component: "crds" - # Generated labels {{- include "labels" . | nindent 4 }} -spec: - group: cert-manager.io - names: - kind: Issuer - listKind: IssuerList - plural: issuers - singular: issuer - categories: - - cert-manager - scope: Namespaced - versions: - - name: v1 - subresources: - status: {} - additionalPrinterColumns: - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - priority: 1 - type: string - - jsonPath: .metadata.creationTimestamp - description: CreationTimestamp is a timestamp representing the server time when this object was created. It is not guaranteed to be set in happens-before order across separate operations. Clients may not set this value. It is represented in RFC3339 form and is in UTC. - name: Age - type: date - schema: - openAPIV3Schema: - description: |- - An Issuer represents a certificate issuing authority which can be - referenced as part of `issuerRef` fields. - It is scoped to a single namespace and can therefore only be referenced by - resources within the same namespace. - type: object - required: - - spec - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: Desired state of the Issuer resource. - type: object - properties: - acme: - description: |- - ACME configures this issuer to communicate with a RFC8555 (ACME) server - to obtain signed x509 certificates. - type: object - required: - - privateKeySecretRef - - server - properties: - caBundle: - description: |- - Base64-encoded bundle of PEM CAs which can be used to validate the certificate - chain presented by the ACME server. - Mutually exclusive with SkipTLSVerify; prefer using CABundle to prevent various - kinds of security vulnerabilities. - If CABundle and SkipTLSVerify are unset, the system certificate bundle inside - the container is used to validate the TLS connection. - type: string - format: byte - disableAccountKeyGeneration: - description: |- - Enables or disables generating a new ACME account key. - If true, the Issuer resource will *not* request a new account but will expect - the account key to be supplied via an existing secret. - If false, the cert-manager system will generate a new ACME account key - for the Issuer. - Defaults to false. - type: boolean - email: - description: |- - Email is the email address to be associated with the ACME account. - This field is optional, but it is strongly recommended to be set. - It will be used to contact you in case of issues with your account or - certificates, including expiry notification emails. - This field may be updated after the account is initially registered. - type: string - enableDurationFeature: - description: |- - Enables requesting a Not After date on certificates that matches the - duration of the certificate. This is not supported by all ACME servers - like Let's Encrypt. If set to true when the ACME server does not support - it, it will create an error on the Order. - Defaults to false. - type: boolean - externalAccountBinding: - description: |- - ExternalAccountBinding is a reference to a CA external account of the ACME - server. - If set, upon registration cert-manager will attempt to associate the given - external account credentials with the registered ACME account. - type: object - required: - - keyID - - keySecretRef - properties: - keyAlgorithm: - description: |- - Deprecated: keyAlgorithm field exists for historical compatibility - reasons and should not be used. The algorithm is now hardcoded to HS256 - in golang/x/crypto/acme. - type: string - enum: - - HS256 - - HS384 - - HS512 - keyID: - description: keyID is the ID of the CA key that the External Account is bound to. - type: string - keySecretRef: - description: |- - keySecretRef is a Secret Key Selector referencing a data item in a Kubernetes - Secret which holds the symmetric MAC key of the External Account Binding. - The `key` is the index string that is paired with the key data in the - Secret and should not be confused with the key data itself, or indeed with - the External Account Binding keyID above. - The secret key stored in the Secret **must** be un-padded, base64 URL - encoded data. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - preferredChain: - description: |- - PreferredChain is the chain to use if the ACME server outputs multiple. - PreferredChain is no guarantee that this one gets delivered by the ACME - endpoint. - For example, for Let's Encrypt's DST crosssign you would use: - "DST Root CA X3" or "ISRG Root X1" for the newer Let's Encrypt root CA. - This value picks the first certificate bundle in the combined set of - ACME default and alternative chains that has a root-most certificate with - this value as its issuer's commonname. - type: string - maxLength: 64 - privateKeySecretRef: - description: |- - PrivateKey is the name of a Kubernetes Secret resource that will be used to - store the automatically generated ACME account private key. - Optionally, a `key` may be specified to select a specific entry within - the named Secret resource. - If `key` is not specified, a default of `tls.key` will be used. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - server: - description: |- - Server is the URL used to access the ACME server's 'directory' endpoint. - For example, for Let's Encrypt's staging endpoint, you would use: - "https://acme-staging-v02.api.letsencrypt.org/directory". - Only ACME v2 endpoints (i.e. RFC 8555) are supported. - type: string - skipTLSVerify: - description: |- - INSECURE: Enables or disables validation of the ACME server TLS certificate. - If true, requests to the ACME server will not have the TLS certificate chain - validated. - Mutually exclusive with CABundle; prefer using CABundle to prevent various - kinds of security vulnerabilities. - Only enable this option in development environments. - If CABundle and SkipTLSVerify are unset, the system certificate bundle inside - the container is used to validate the TLS connection. - Defaults to false. - type: boolean - solvers: - description: |- - Solvers is a list of challenge solvers that will be used to solve - ACME challenges for the matching domains. - Solver configurations must be provided in order to obtain certificates - from an ACME server. - For more information, see: https://cert-manager.io/docs/configuration/acme/ - type: array - items: - description: |- - An ACMEChallengeSolver describes how to solve ACME challenges for the issuer it is part of. - A selector may be provided to use different solving strategies for different DNS names. - Only one of HTTP01 or DNS01 must be provided. - type: object - properties: - dns01: - description: |- - Configures cert-manager to attempt to complete authorizations by - performing the DNS01 challenge flow. - type: object - properties: - acmeDNS: - description: |- - Use the 'ACME DNS' (https://github.com/joohoi/acme-dns) API to manage - DNS01 challenge records. - type: object - required: - - accountSecretRef - - host - properties: - accountSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - host: - type: string - akamai: - description: Use the Akamai DNS zone management API to manage DNS01 challenge records. - type: object - required: - - accessTokenSecretRef - - clientSecretSecretRef - - clientTokenSecretRef - - serviceConsumerDomain - properties: - accessTokenSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - clientSecretSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - clientTokenSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - serviceConsumerDomain: - type: string - azureDNS: - description: Use the Microsoft Azure DNS API to manage DNS01 challenge records. - type: object - required: - - resourceGroupName - - subscriptionID - properties: - clientID: - description: |- - Auth: Azure Service Principal: - The ClientID of the Azure Service Principal used to authenticate with Azure DNS. - If set, ClientSecret and TenantID must also be set. - type: string - clientSecretSecretRef: - description: |- - Auth: Azure Service Principal: - A reference to a Secret containing the password associated with the Service Principal. - If set, ClientID and TenantID must also be set. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - environment: - description: name of the Azure environment (default AzurePublicCloud) - type: string - enum: - - AzurePublicCloud - - AzureChinaCloud - - AzureGermanCloud - - AzureUSGovernmentCloud - hostedZoneName: - description: name of the DNS zone that should be used - type: string - managedIdentity: - description: |- - Auth: Azure Workload Identity or Azure Managed Service Identity: - Settings to enable Azure Workload Identity or Azure Managed Service Identity - If set, ClientID, ClientSecret and TenantID must not be set. - type: object - properties: - clientID: - description: client ID of the managed identity, can not be used at the same time as resourceID - type: string - resourceID: - description: |- - resource ID of the managed identity, can not be used at the same time as clientID - Cannot be used for Azure Managed Service Identity - type: string - resourceGroupName: - description: resource group the DNS zone is located in - type: string - subscriptionID: - description: ID of the Azure subscription - type: string - tenantID: - description: |- - Auth: Azure Service Principal: - The TenantID of the Azure Service Principal used to authenticate with Azure DNS. - If set, ClientID and ClientSecret must also be set. - type: string - cloudDNS: - description: Use the Google Cloud DNS API to manage DNS01 challenge records. - type: object - required: - - project - properties: - hostedZoneName: - description: |- - HostedZoneName is an optional field that tells cert-manager in which - Cloud DNS zone the challenge record has to be created. - If left empty cert-manager will automatically choose a zone. - type: string - project: - type: string - serviceAccountSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - cloudflare: - description: Use the Cloudflare API to manage DNS01 challenge records. - type: object - properties: - apiKeySecretRef: - description: |- - API key to use to authenticate with Cloudflare. - Note: using an API token to authenticate is now the recommended method - as it allows greater control of permissions. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - apiTokenSecretRef: - description: API token used to authenticate with Cloudflare. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - email: - description: Email of the account, only required when using API key based authentication. - type: string - cnameStrategy: - description: |- - CNAMEStrategy configures how the DNS01 provider should handle CNAME - records when found in DNS zones. - type: string - enum: - - None - - Follow - digitalocean: - description: Use the DigitalOcean DNS API to manage DNS01 challenge records. - type: object - required: - - tokenSecretRef - properties: - tokenSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource. - In some instances, `key` is a required field. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - rfc2136: - description: |- - Use RFC2136 ("Dynamic Updates in the Domain Name System") (https://datatracker.ietf.org/doc/rfc2136/) - to manage DNS01 challenge records. - type: object - required: - - nameserver - properties: - nameserver: - description: |- - The IP address or hostname of an authoritative DNS server supporting - RFC2136 in the form host:port. If the host is an IPv6 address it must be - enclosed in square brackets (e.g [2001:db8::1]) ; port is optional. - This field is required. - type: string - tsigAlgorithm: - description: |- - The TSIG Algorithm configured in the DNS supporting RFC2136. Used only - when ``tsigSecretSecretRef`` and ``tsigKeyName`` are defined. - Supported values are (case-insensitive): ``HMACMD5`` (default), - ``HMACSHA1``, ``HMACSHA256`` or ``HMACSHA512``. - type: string - tsigKeyName: - description: |- - The TSIG Key name configured in the DNS. - If ``tsigSecretSecretRef`` is defined, this field is required. - type: string - tsigSecretSecretRef: - description: |- - The name of the secret containing the TSIG value. - If ``tsigKeyName`` is defined, this field is required. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - route53: - description: Use the AWS Route53 API to manage DNS01 challenge records. - type: object - required: - - region - properties: - accessKeyID: - description: |- - The AccessKeyID is used for authentication. - Cannot be set when SecretAccessKeyID is set. - If neither the Access Key nor Key ID are set, we fall-back to using env - vars, shared credentials file or AWS Instance metadata, - see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials - type: string - accessKeyIDSecretRef: - description: |- - The SecretAccessKey is used for authentication. If set, pull the AWS - access key ID from a key within a Kubernetes Secret. - Cannot be set when AccessKeyID is set. - If neither the Access Key nor Key ID are set, we fall-back to using env - vars, shared credentials file or AWS Instance metadata, - see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - auth: - description: Auth configures how cert-manager authenticates. - type: object - required: - - kubernetes - properties: - kubernetes: - description: |- - Kubernetes authenticates with Route53 using AssumeRoleWithWebIdentity - by passing a bound ServiceAccount token. - type: object - required: - - serviceAccountRef - properties: - serviceAccountRef: - description: |- - A reference to a service account that will be used to request a bound - token (also known as "projected token"). To use this field, you must - configure an RBAC rule to let cert-manager request a token. - type: object - required: - - name - properties: - audiences: - description: |- - TokenAudiences is an optional list of audiences to include in the - token passed to AWS. The default token consisting of the issuer's namespace - and name is always included. - If unset the audience defaults to `sts.amazonaws.com`. - type: array - items: - type: string - name: - description: Name of the ServiceAccount used to request a token. - type: string - hostedZoneID: - description: If set, the provider will manage only this zone in Route53 and will not do an lookup using the route53:ListHostedZonesByName api call. - type: string - region: - description: Always set the region when using AccessKeyID and SecretAccessKey - type: string - role: - description: |- - Role is a Role ARN which the Route53 provider will assume using either the explicit credentials AccessKeyID/SecretAccessKey - or the inferred credentials from environment variables, shared credentials file or AWS Instance metadata - type: string - secretAccessKeySecretRef: - description: |- - The SecretAccessKey is used for authentication. - If neither the Access Key nor Key ID are set, we fall-back to using env - vars, shared credentials file or AWS Instance metadata, - see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - webhook: - description: |- - Configure an external webhook based DNS01 challenge solver to manage - DNS01 challenge records. - type: object - required: - - groupName - - solverName - properties: - config: - description: |- - Additional configuration that should be passed to the webhook apiserver - when challenges are processed. - This can contain arbitrary JSON data. - Secret values should not be specified in this stanza. - If secret values are needed (e.g. credentials for a DNS service), you - should use a SecretKeySelector to reference a Secret resource. - For details on the schema of this field, consult the webhook provider - implementation's documentation. - x-kubernetes-preserve-unknown-fields: true - groupName: - description: |- - The API group name that should be used when POSTing ChallengePayload - resources to the webhook apiserver. - This should be the same as the GroupName specified in the webhook - provider implementation. - type: string - solverName: - description: |- - The name of the solver to use, as defined in the webhook provider - implementation. - This will typically be the name of the provider, e.g. 'cloudflare'. - type: string - http01: - description: |- - Configures cert-manager to attempt to complete authorizations by - performing the HTTP01 challenge flow. - It is not possible to obtain certificates for wildcard domain names - (e.g. `*.example.com`) using the HTTP01 challenge mechanism. - type: object - properties: - gatewayHTTPRoute: - description: |- - The Gateway API is a sig-network community API that models service networking - in Kubernetes (https://gateway-api.sigs.k8s.io/). The Gateway solver will - create HTTPRoutes with the specified labels in the same namespace as the challenge. - This solver is experimental, and fields / behaviour may change in the future. - type: object - properties: - labels: - description: |- - Custom labels that will be applied to HTTPRoutes created by cert-manager - while solving HTTP-01 challenges. - type: object - additionalProperties: - type: string - parentRefs: - description: |- - When solving an HTTP-01 challenge, cert-manager creates an HTTPRoute. - cert-manager needs to know which parentRefs should be used when creating - the HTTPRoute. Usually, the parentRef references a Gateway. See: - https://gateway-api.sigs.k8s.io/api-types/httproute/#attaching-to-gateways - type: array - items: - description: |- - ParentReference identifies an API object (usually a Gateway) that can be considered - a parent of this resource (usually a route). There are two kinds of parent resources - with "Core" support: - - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - - This API may be extended in the future to support additional kinds of parent - resources. - - - The API object must be valid in the cluster; the Group and Kind must - be registered in the cluster for this reference to be valid. - type: object - required: - - name - properties: - group: - description: |- - Group is the group of the referent. - When unspecified, "gateway.networking.k8s.io" is inferred. - To set the core API group (such as for a "Service" kind referent), - Group must be explicitly set to "" (empty string). - - - Support: Core - type: string - default: gateway.networking.k8s.io - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - kind: - description: |- - Kind is kind of the referent. - - - There are two kinds of parent resources with "Core" support: - - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - - Support for other resources is Implementation-Specific. - type: string - default: Gateway - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - name: - description: |- - Name is the name of the referent. - - - Support: Core - type: string - maxLength: 253 - minLength: 1 - namespace: - description: |- - Namespace is the namespace of the referent. When unspecified, this refers - to the local namespace of the Route. - - - Note that there are specific rules for ParentRefs which cross namespace - boundaries. Cross-namespace references are only valid if they are explicitly - allowed by something in the namespace they are referring to. For example: - Gateway has the AllowedRoutes field, and ReferenceGrant provides a - generic way to enable any other kind of cross-namespace reference. - - - - ParentRefs from a Route to a Service in the same namespace are "producer" - routes, which apply default routing rules to inbound connections from - any namespace to the Service. - - - ParentRefs from a Route to a Service in a different namespace are - "consumer" routes, and these routing rules are only applied to outbound - connections originating from the same namespace as the Route, for which - the intended destination of the connections are a Service targeted as a - ParentRef of the Route. - - - - Support: Core - type: string - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - port: - description: |- - Port is the network port this Route targets. It can be interpreted - differently based on the type of parent resource. - - - When the parent resource is a Gateway, this targets all listeners - listening on the specified port that also support this kind of Route(and - select this Route). It's not recommended to set `Port` unless the - networking behaviors specified in a Route must apply to a specific port - as opposed to a listener(s) whose port(s) may be changed. When both Port - and SectionName are specified, the name and port of the selected listener - must match both specified values. - - - - When the parent resource is a Service, this targets a specific port in the - Service spec. When both Port (experimental) and SectionName are specified, - the name and port of the selected port must match both specified values. - - - - Implementations MAY choose to support other parent resources. - Implementations supporting other types of parent resources MUST clearly - document how/if Port is interpreted. - - - For the purpose of status, an attachment is considered successful as - long as the parent resource accepts it partially. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment - from the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, - the Route MUST be considered detached from the Gateway. - - - Support: Extended - type: integer - format: int32 - maximum: 65535 - minimum: 1 - sectionName: - description: |- - SectionName is the name of a section within the target resource. In the - following resources, SectionName is interpreted as the following: - - - * Gateway: Listener name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - * Service: Port name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - - - Implementations MAY choose to support attaching Routes to other resources. - If that is the case, they MUST clearly document how SectionName is - interpreted. - - - When unspecified (empty string), this will reference the entire resource. - For the purpose of status, an attachment is considered successful if at - least one section in the parent resource accepts it. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from - the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, the - Route MUST be considered detached from the Gateway. - - - Support: Core - type: string - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - serviceType: - description: |- - Optional service type for Kubernetes solver service. Supported values - are NodePort or ClusterIP. If unset, defaults to NodePort. - type: string - ingress: - description: |- - The ingress based HTTP01 challenge solver will solve challenges by - creating or modifying Ingress resources in order to route requests for - '/.well-known/acme-challenge/XYZ' to 'challenge solver' pods that are - provisioned by cert-manager for each Challenge to be completed. - type: object - properties: - class: - description: |- - This field configures the annotation `kubernetes.io/ingress.class` when - creating Ingress resources to solve ACME challenges that use this - challenge solver. Only one of `class`, `name` or `ingressClassName` may - be specified. - type: string - ingressClassName: - description: |- - This field configures the field `ingressClassName` on the created Ingress - resources used to solve ACME challenges that use this challenge solver. - This is the recommended way of configuring the ingress class. Only one of - `class`, `name` or `ingressClassName` may be specified. - type: string - ingressTemplate: - description: |- - Optional ingress template used to configure the ACME challenge solver - ingress used for HTTP01 challenges. - type: object - properties: - metadata: - description: |- - ObjectMeta overrides for the ingress used to solve HTTP01 challenges. - Only the 'labels' and 'annotations' fields may be set. - If labels or annotations overlap with in-built values, the values here - will override the in-built values. - type: object - properties: - annotations: - description: Annotations that should be added to the created ACME HTTP01 solver ingress. - type: object - additionalProperties: - type: string - labels: - description: Labels that should be added to the created ACME HTTP01 solver ingress. - type: object - additionalProperties: - type: string - name: - description: |- - The name of the ingress resource that should have ACME challenge solving - routes inserted into it in order to solve HTTP01 challenges. - This is typically used in conjunction with ingress controllers like - ingress-gce, which maintains a 1:1 mapping between external IPs and - ingress resources. Only one of `class`, `name` or `ingressClassName` may - be specified. - type: string - podTemplate: - description: |- - Optional pod template used to configure the ACME challenge solver pods - used for HTTP01 challenges. - type: object - properties: - metadata: - description: |- - ObjectMeta overrides for the pod used to solve HTTP01 challenges. - Only the 'labels' and 'annotations' fields may be set. - If labels or annotations overlap with in-built values, the values here - will override the in-built values. - type: object - properties: - annotations: - description: Annotations that should be added to the create ACME HTTP01 solver pods. - type: object - additionalProperties: - type: string - labels: - description: Labels that should be added to the created ACME HTTP01 solver pods. - type: object - additionalProperties: - type: string - spec: - description: |- - PodSpec defines overrides for the HTTP01 challenge solver pod. - Check ACMEChallengeSolverHTTP01IngressPodSpec to find out currently supported fields. - All other fields will be ignored. - type: object - properties: - affinity: - description: If specified, the pod's scheduling constraints - type: object - properties: - nodeAffinity: - description: Describes node affinity scheduling rules for the pod. - type: object - properties: - preferredDuringSchedulingIgnoredDuringExecution: - description: |- - The scheduler will prefer to schedule pods to nodes that satisfy - the affinity expressions specified by this field, but it may choose - a node that violates one or more of the expressions. The node that is - most preferred is the one with the greatest sum of weights, i.e. - for each node that meets all of the scheduling requirements (resource - request, requiredDuringScheduling affinity expressions, etc.), - compute a sum by iterating through the elements of this field and adding - "weight" to the sum if the node matches the corresponding matchExpressions; the - node(s) with the highest sum are the most preferred. - type: array - items: - description: |- - An empty preferred scheduling term matches all objects with implicit weight 0 - (i.e. it's a no-op). A null preferred scheduling term matches no objects (i.e. is also a no-op). - type: object - required: - - preference - - weight - properties: - preference: - description: A node selector term, associated with the corresponding weight. - type: object - properties: - matchExpressions: - description: A list of node selector requirements by node's labels. - type: array - items: - description: |- - A node selector requirement is a selector that contains values, a key, and an operator - that relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: The label key that the selector applies to. - type: string - operator: - description: |- - Represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. - type: string - values: - description: |- - An array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. If the operator is Gt or Lt, the values - array must have a single element, which will be interpreted as an integer. - This array is replaced during a strategic merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchFields: - description: A list of node selector requirements by node's fields. - type: array - items: - description: |- - A node selector requirement is a selector that contains values, a key, and an operator - that relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: The label key that the selector applies to. - type: string - operator: - description: |- - Represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. - type: string - values: - description: |- - An array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. If the operator is Gt or Lt, the values - array must have a single element, which will be interpreted as an integer. - This array is replaced during a strategic merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - x-kubernetes-map-type: atomic - weight: - description: Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100. - type: integer - format: int32 - x-kubernetes-list-type: atomic - requiredDuringSchedulingIgnoredDuringExecution: - description: |- - If the affinity requirements specified by this field are not met at - scheduling time, the pod will not be scheduled onto the node. - If the affinity requirements specified by this field cease to be met - at some point during pod execution (e.g. due to an update), the system - may or may not try to eventually evict the pod from its node. - type: object - required: - - nodeSelectorTerms - properties: - nodeSelectorTerms: - description: Required. A list of node selector terms. The terms are ORed. - type: array - items: - description: |- - A null or empty node selector term matches no objects. The requirements of - them are ANDed. - The TopologySelectorTerm type implements a subset of the NodeSelectorTerm. - type: object - properties: - matchExpressions: - description: A list of node selector requirements by node's labels. - type: array - items: - description: |- - A node selector requirement is a selector that contains values, a key, and an operator - that relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: The label key that the selector applies to. - type: string - operator: - description: |- - Represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. - type: string - values: - description: |- - An array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. If the operator is Gt or Lt, the values - array must have a single element, which will be interpreted as an integer. - This array is replaced during a strategic merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchFields: - description: A list of node selector requirements by node's fields. - type: array - items: - description: |- - A node selector requirement is a selector that contains values, a key, and an operator - that relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: The label key that the selector applies to. - type: string - operator: - description: |- - Represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. - type: string - values: - description: |- - An array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. If the operator is Gt or Lt, the values - array must have a single element, which will be interpreted as an integer. - This array is replaced during a strategic merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - x-kubernetes-map-type: atomic - x-kubernetes-list-type: atomic - x-kubernetes-map-type: atomic - podAffinity: - description: Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)). - type: object - properties: - preferredDuringSchedulingIgnoredDuringExecution: - description: |- - The scheduler will prefer to schedule pods to nodes that satisfy - the affinity expressions specified by this field, but it may choose - a node that violates one or more of the expressions. The node that is - most preferred is the one with the greatest sum of weights, i.e. - for each node that meets all of the scheduling requirements (resource - request, requiredDuringScheduling affinity expressions, etc.), - compute a sum by iterating through the elements of this field and adding - "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the - node(s) with the highest sum are the most preferred. - type: array - items: - description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) - type: object - required: - - podAffinityTerm - - weight - properties: - podAffinityTerm: - description: Required. A pod affinity term, associated with the corresponding weight. - type: object - required: - - topologyKey - properties: - labelSelector: - description: |- - A label query over a set of resources, in this case pods. - If it's null, this PodAffinityTerm matches with no Pods. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - matchLabelKeys: - description: |- - MatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both matchLabelKeys and labelSelector. - Also, matchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - mismatchLabelKeys: - description: |- - MismatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. - Also, mismatchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - namespaceSelector: - description: |- - A label query over the set of namespaces that the term applies to. - The term is applied to the union of the namespaces selected by this field - and the ones listed in the namespaces field. - null selector and null or empty namespaces list means "this pod's namespace". - An empty selector ({}) matches all namespaces. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - namespaces: - description: |- - namespaces specifies a static list of namespace names that the term applies to. - The term is applied to the union of the namespaces listed in this field - and the ones selected by namespaceSelector. - null or empty namespaces list and null namespaceSelector means "this pod's namespace". - type: array - items: - type: string - x-kubernetes-list-type: atomic - topologyKey: - description: |- - This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching - the labelSelector in the specified namespaces, where co-located is defined as running on a node - whose value of the label with key topologyKey matches that of any node on which any of the - selected pods is running. - Empty topologyKey is not allowed. - type: string - weight: - description: |- - weight associated with matching the corresponding podAffinityTerm, - in the range 1-100. - type: integer - format: int32 - x-kubernetes-list-type: atomic - requiredDuringSchedulingIgnoredDuringExecution: - description: |- - If the affinity requirements specified by this field are not met at - scheduling time, the pod will not be scheduled onto the node. - If the affinity requirements specified by this field cease to be met - at some point during pod execution (e.g. due to a pod label update), the - system may or may not try to eventually evict the pod from its node. - When there are multiple elements, the lists of nodes corresponding to each - podAffinityTerm are intersected, i.e. all terms must be satisfied. - type: array - items: - description: |- - Defines a set of pods (namely those matching the labelSelector - relative to the given namespace(s)) that this pod should be - co-located (affinity) or not co-located (anti-affinity) with, - where co-located is defined as running on a node whose value of - the label with key matches that of any node on which - a pod of the set of pods is running - type: object - required: - - topologyKey - properties: - labelSelector: - description: |- - A label query over a set of resources, in this case pods. - If it's null, this PodAffinityTerm matches with no Pods. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - matchLabelKeys: - description: |- - MatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both matchLabelKeys and labelSelector. - Also, matchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - mismatchLabelKeys: - description: |- - MismatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. - Also, mismatchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - namespaceSelector: - description: |- - A label query over the set of namespaces that the term applies to. - The term is applied to the union of the namespaces selected by this field - and the ones listed in the namespaces field. - null selector and null or empty namespaces list means "this pod's namespace". - An empty selector ({}) matches all namespaces. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - namespaces: - description: |- - namespaces specifies a static list of namespace names that the term applies to. - The term is applied to the union of the namespaces listed in this field - and the ones selected by namespaceSelector. - null or empty namespaces list and null namespaceSelector means "this pod's namespace". - type: array - items: - type: string - x-kubernetes-list-type: atomic - topologyKey: - description: |- - This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching - the labelSelector in the specified namespaces, where co-located is defined as running on a node - whose value of the label with key topologyKey matches that of any node on which any of the - selected pods is running. - Empty topologyKey is not allowed. - type: string - x-kubernetes-list-type: atomic - podAntiAffinity: - description: Describes pod anti-affinity scheduling rules (e.g. avoid putting this pod in the same node, zone, etc. as some other pod(s)). - type: object - properties: - preferredDuringSchedulingIgnoredDuringExecution: - description: |- - The scheduler will prefer to schedule pods to nodes that satisfy - the anti-affinity expressions specified by this field, but it may choose - a node that violates one or more of the expressions. The node that is - most preferred is the one with the greatest sum of weights, i.e. - for each node that meets all of the scheduling requirements (resource - request, requiredDuringScheduling anti-affinity expressions, etc.), - compute a sum by iterating through the elements of this field and adding - "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the - node(s) with the highest sum are the most preferred. - type: array - items: - description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) - type: object - required: - - podAffinityTerm - - weight - properties: - podAffinityTerm: - description: Required. A pod affinity term, associated with the corresponding weight. - type: object - required: - - topologyKey - properties: - labelSelector: - description: |- - A label query over a set of resources, in this case pods. - If it's null, this PodAffinityTerm matches with no Pods. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - matchLabelKeys: - description: |- - MatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both matchLabelKeys and labelSelector. - Also, matchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - mismatchLabelKeys: - description: |- - MismatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. - Also, mismatchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - namespaceSelector: - description: |- - A label query over the set of namespaces that the term applies to. - The term is applied to the union of the namespaces selected by this field - and the ones listed in the namespaces field. - null selector and null or empty namespaces list means "this pod's namespace". - An empty selector ({}) matches all namespaces. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - namespaces: - description: |- - namespaces specifies a static list of namespace names that the term applies to. - The term is applied to the union of the namespaces listed in this field - and the ones selected by namespaceSelector. - null or empty namespaces list and null namespaceSelector means "this pod's namespace". - type: array - items: - type: string - x-kubernetes-list-type: atomic - topologyKey: - description: |- - This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching - the labelSelector in the specified namespaces, where co-located is defined as running on a node - whose value of the label with key topologyKey matches that of any node on which any of the - selected pods is running. - Empty topologyKey is not allowed. - type: string - weight: - description: |- - weight associated with matching the corresponding podAffinityTerm, - in the range 1-100. - type: integer - format: int32 - x-kubernetes-list-type: atomic - requiredDuringSchedulingIgnoredDuringExecution: - description: |- - If the anti-affinity requirements specified by this field are not met at - scheduling time, the pod will not be scheduled onto the node. - If the anti-affinity requirements specified by this field cease to be met - at some point during pod execution (e.g. due to a pod label update), the - system may or may not try to eventually evict the pod from its node. - When there are multiple elements, the lists of nodes corresponding to each - podAffinityTerm are intersected, i.e. all terms must be satisfied. - type: array - items: - description: |- - Defines a set of pods (namely those matching the labelSelector - relative to the given namespace(s)) that this pod should be - co-located (affinity) or not co-located (anti-affinity) with, - where co-located is defined as running on a node whose value of - the label with key matches that of any node on which - a pod of the set of pods is running - type: object - required: - - topologyKey - properties: - labelSelector: - description: |- - A label query over a set of resources, in this case pods. - If it's null, this PodAffinityTerm matches with no Pods. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - matchLabelKeys: - description: |- - MatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both matchLabelKeys and labelSelector. - Also, matchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - mismatchLabelKeys: - description: |- - MismatchLabelKeys is a set of pod label keys to select which pods will - be taken into consideration. The keys are used to lookup values from the - incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` - to select the group of existing pods which pods will be taken into consideration - for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming - pod labels will be ignored. The default value is empty. - The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. - Also, mismatchLabelKeys cannot be set when labelSelector isn't set. - This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. - type: array - items: - type: string - x-kubernetes-list-type: atomic - namespaceSelector: - description: |- - A label query over the set of namespaces that the term applies to. - The term is applied to the union of the namespaces selected by this field - and the ones listed in the namespaces field. - null selector and null or empty namespaces list means "this pod's namespace". - An empty selector ({}) matches all namespaces. - type: object - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - type: array - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - type: object - required: - - key - - operator - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - type: array - items: - type: string - x-kubernetes-list-type: atomic - x-kubernetes-list-type: atomic - matchLabels: - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - additionalProperties: - type: string - x-kubernetes-map-type: atomic - namespaces: - description: |- - namespaces specifies a static list of namespace names that the term applies to. - The term is applied to the union of the namespaces listed in this field - and the ones selected by namespaceSelector. - null or empty namespaces list and null namespaceSelector means "this pod's namespace". - type: array - items: - type: string - x-kubernetes-list-type: atomic - topologyKey: - description: |- - This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching - the labelSelector in the specified namespaces, where co-located is defined as running on a node - whose value of the label with key topologyKey matches that of any node on which any of the - selected pods is running. - Empty topologyKey is not allowed. - type: string - x-kubernetes-list-type: atomic - imagePullSecrets: - description: If specified, the pod's imagePullSecrets - type: array - items: - description: |- - LocalObjectReference contains enough information to let you locate the - referenced object inside the same namespace. - type: object - properties: - name: - description: |- - Name of the referent. - This field is effectively required, but due to backwards compatibility is - allowed to be empty. Instances of this type with an empty value here are - almost certainly wrong. - TODO: Add other useful fields. apiVersion, kind, uid? - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - TODO: Drop `kubebuilder:default` when controller-gen doesn't need it https://github.com/kubernetes-sigs/kubebuilder/issues/3896. - type: string - default: "" - x-kubernetes-map-type: atomic - nodeSelector: - description: |- - NodeSelector is a selector which must be true for the pod to fit on a node. - Selector which must match a node's labels for the pod to be scheduled on that node. - More info: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/ - type: object - additionalProperties: - type: string - priorityClassName: - description: If specified, the pod's priorityClassName. - type: string - serviceAccountName: - description: If specified, the pod's service account - type: string - tolerations: - description: If specified, the pod's tolerations. - type: array - items: - description: |- - The pod this Toleration is attached to tolerates any taint that matches - the triple using the matching operator . - type: object - properties: - effect: - description: |- - Effect indicates the taint effect to match. Empty means match all taint effects. - When specified, allowed values are NoSchedule, PreferNoSchedule and NoExecute. - type: string - key: - description: |- - Key is the taint key that the toleration applies to. Empty means match all taint keys. - If the key is empty, operator must be Exists; this combination means to match all values and all keys. - type: string - operator: - description: |- - Operator represents a key's relationship to the value. - Valid operators are Exists and Equal. Defaults to Equal. - Exists is equivalent to wildcard for value, so that a pod can - tolerate all taints of a particular category. - type: string - tolerationSeconds: - description: |- - TolerationSeconds represents the period of time the toleration (which must be - of effect NoExecute, otherwise this field is ignored) tolerates the taint. By default, - it is not set, which means tolerate the taint forever (do not evict). Zero and - negative values will be treated as 0 (evict immediately) by the system. - type: integer - format: int64 - value: - description: |- - Value is the taint value the toleration matches to. - If the operator is Exists, the value should be empty, otherwise just a regular string. - type: string - serviceType: - description: |- - Optional service type for Kubernetes solver service. Supported values - are NodePort or ClusterIP. If unset, defaults to NodePort. - type: string - selector: - description: |- - Selector selects a set of DNSNames on the Certificate resource that - should be solved using this challenge solver. - If not specified, the solver will be treated as the 'default' solver - with the lowest priority, i.e. if any other solver has a more specific - match, it will be used instead. - type: object - properties: - dnsNames: - description: |- - List of DNSNames that this solver will be used to solve. - If specified and a match is found, a dnsNames selector will take - precedence over a dnsZones selector. - If multiple solvers match with the same dnsNames value, the solver - with the most matching labels in matchLabels will be selected. - If neither has more matches, the solver defined earlier in the list - will be selected. - type: array - items: - type: string - dnsZones: - description: |- - List of DNSZones that this solver will be used to solve. - The most specific DNS zone match specified here will take precedence - over other DNS zone matches, so a solver specifying sys.example.com - will be selected over one specifying example.com for the domain - www.sys.example.com. - If multiple solvers match with the same dnsZones value, the solver - with the most matching labels in matchLabels will be selected. - If neither has more matches, the solver defined earlier in the list - will be selected. - type: array - items: - type: string - matchLabels: - description: |- - A label selector that is used to refine the set of certificate's that - this challenge solver will apply to. - type: object - additionalProperties: - type: string - ca: - description: |- - CA configures this issuer to sign certificates using a signing CA keypair - stored in a Secret resource. - This is used to build internal PKIs that are managed by cert-manager. - type: object - required: - - secretName - properties: - crlDistributionPoints: - description: |- - The CRL distribution points is an X.509 v3 certificate extension which identifies - the location of the CRL from which the revocation of this certificate can be checked. - If not set, certificates will be issued without distribution points set. - type: array - items: - type: string - issuingCertificateURLs: - description: |- - IssuingCertificateURLs is a list of URLs which this issuer should embed into certificates - it creates. See https://www.rfc-editor.org/rfc/rfc5280#section-4.2.2.1 for more details. - As an example, such a URL might be "http://ca.domain.com/ca.crt". - type: array - items: - type: string - ocspServers: - description: |- - The OCSP server list is an X.509 v3 extension that defines a list of - URLs of OCSP responders. The OCSP responders can be queried for the - revocation status of an issued certificate. If not set, the - certificate will be issued with no OCSP servers set. For example, an - OCSP server URL could be "http://ocsp.int-x3.letsencrypt.org". - type: array - items: - type: string - secretName: - description: |- - SecretName is the name of the secret used to sign Certificates issued - by this Issuer. - type: string - selfSigned: - description: |- - SelfSigned configures this issuer to 'self sign' certificates using the - private key used to create the CertificateRequest object. - type: object - properties: - crlDistributionPoints: - description: |- - The CRL distribution points is an X.509 v3 certificate extension which identifies - the location of the CRL from which the revocation of this certificate can be checked. - If not set certificate will be issued without CDP. Values are strings. - type: array - items: - type: string - vault: - description: |- - Vault configures this issuer to sign certificates using a HashiCorp Vault - PKI backend. - type: object - required: - - auth - - path - - server - properties: - auth: - description: Auth configures how cert-manager authenticates with the Vault server. - type: object - properties: - appRole: - description: |- - AppRole authenticates with Vault using the App Role auth mechanism, - with the role and secret stored in a Kubernetes Secret resource. - type: object - required: - - path - - roleId - - secretRef - properties: - path: - description: |- - Path where the App Role authentication backend is mounted in Vault, e.g: - "approle" - type: string - roleId: - description: |- - RoleID configured in the App Role authentication backend when setting - up the authentication backend in Vault. - type: string - secretRef: - description: |- - Reference to a key in a Secret that contains the App Role secret used - to authenticate with Vault. - The `key` field must be specified and denotes which entry within the Secret - resource is used as the app role secret. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - kubernetes: - description: |- - Kubernetes authenticates with Vault by passing the ServiceAccount - token stored in the named Secret resource to the Vault server. - type: object - required: - - role - properties: - mountPath: - description: |- - The Vault mountPath here is the mount path to use when authenticating with - Vault. For example, setting a value to `/v1/auth/foo`, will use the path - `/v1/auth/foo/login` to authenticate with Vault. If unspecified, the - default value "/v1/auth/kubernetes" will be used. - type: string - role: - description: |- - A required field containing the Vault Role to assume. A Role binds a - Kubernetes ServiceAccount with a set of Vault policies. - type: string - secretRef: - description: |- - The required Secret field containing a Kubernetes ServiceAccount JWT used - for authenticating with Vault. Use of 'ambient credentials' is not - supported. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - serviceAccountRef: - description: |- - A reference to a service account that will be used to request a bound - token (also known as "projected token"). Compared to using "secretRef", - using this field means that you don't rely on statically bound tokens. To - use this field, you must configure an RBAC rule to let cert-manager - request a token. - type: object - required: - - name - properties: - audiences: - description: |- - TokenAudiences is an optional list of extra audiences to include in the token passed to Vault. The default token - consisting of the issuer's namespace and name is always included. - type: array - items: - type: string - name: - description: Name of the ServiceAccount used to request a token. - type: string - tokenSecretRef: - description: TokenSecretRef authenticates with Vault by presenting a token. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - caBundle: - description: |- - Base64-encoded bundle of PEM CAs which will be used to validate the certificate - chain presented by Vault. Only used if using HTTPS to connect to Vault and - ignored for HTTP connections. - Mutually exclusive with CABundleSecretRef. - If neither CABundle nor CABundleSecretRef are defined, the certificate bundle in - the cert-manager controller container is used to validate the TLS connection. - type: string - format: byte - caBundleSecretRef: - description: |- - Reference to a Secret containing a bundle of PEM-encoded CAs to use when - verifying the certificate chain presented by Vault when using HTTPS. - Mutually exclusive with CABundle. - If neither CABundle nor CABundleSecretRef are defined, the certificate bundle in - the cert-manager controller container is used to validate the TLS connection. - If no key for the Secret is specified, cert-manager will default to 'ca.crt'. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - clientCertSecretRef: - description: |- - Reference to a Secret containing a PEM-encoded Client Certificate to use when the - Vault server requires mTLS. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - clientKeySecretRef: - description: |- - Reference to a Secret containing a PEM-encoded Client Private Key to use when the - Vault server requires mTLS. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - namespace: - description: |- - Name of the vault namespace. Namespaces is a set of features within Vault Enterprise that allows Vault environments to support Secure Multi-tenancy. e.g: "ns1" - More about namespaces can be found here https://www.vaultproject.io/docs/enterprise/namespaces - type: string - path: - description: |- - Path is the mount path of the Vault PKI backend's `sign` endpoint, e.g: - "my_pki_mount/sign/my-role-name". - type: string - server: - description: 'Server is the connection address for the Vault server, e.g: "https://vault.example.com:8200".' - type: string - venafi: - description: |- - Venafi configures this issuer to sign certificates using a Venafi TPP - or Venafi Cloud policy zone. - type: object - required: - - zone - properties: - cloud: - description: |- - Cloud specifies the Venafi cloud configuration settings. - Only one of TPP or Cloud may be specified. - type: object - required: - - apiTokenSecretRef - properties: - apiTokenSecretRef: - description: APITokenSecretRef is a secret key selector for the Venafi Cloud API token. - type: object - required: - - name - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. - Some instances of this field may be defaulted, in others it may be - required. - type: string - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - url: - description: |- - URL is the base URL for Venafi Cloud. - Defaults to "https://api.venafi.cloud/v1". - type: string - tpp: - description: |- - TPP specifies Trust Protection Platform configuration settings. - Only one of TPP or Cloud may be specified. - type: object - required: - - credentialsRef - - url - properties: - caBundle: - description: |- - Base64-encoded bundle of PEM CAs which will be used to validate the certificate - chain presented by the TPP server. Only used if using HTTPS; ignored for HTTP. - If undefined, the certificate bundle in the cert-manager controller container - is used to validate the chain. - type: string - format: byte - credentialsRef: - description: |- - CredentialsRef is a reference to a Secret containing the username and - password for the TPP server. - The secret must contain two keys, 'username' and 'password'. - type: object - required: - - name - properties: - name: - description: |- - Name of the resource being referred to. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - url: - description: |- - URL is the base URL for the vedsdk endpoint of the Venafi TPP instance, - for example: "https://tpp.example.com/vedsdk". - type: string - zone: - description: |- - Zone is the Venafi Policy Zone to use for this issuer. - All requests made to the Venafi platform will be restricted by the named - zone policy. - This field is required. - type: string - status: - description: Status of the Issuer. This is set and managed automatically. - type: object - properties: - acme: - description: |- - ACME specific status options. - This field should only be set if the Issuer is configured to use an ACME - server to issue certificates. - type: object - properties: - lastPrivateKeyHash: - description: |- - LastPrivateKeyHash is a hash of the private key associated with the latest - registered ACME account, in order to track changes made to registered account - associated with the Issuer - type: string - lastRegisteredEmail: - description: |- - LastRegisteredEmail is the email associated with the latest registered - ACME account, in order to track changes made to registered account - associated with the Issuer - type: string - uri: - description: |- - URI is the unique account identifier, which can also be used to retrieve - account details from the CA - type: string - conditions: - description: |- - List of status conditions to indicate the status of a CertificateRequest. - Known condition types are `Ready`. - type: array - items: - description: IssuerCondition contains condition information for an Issuer. - type: object - required: - - status - - type - properties: - lastTransitionTime: - description: |- - LastTransitionTime is the timestamp corresponding to the last status - change of this condition. - type: string - format: date-time - message: - description: |- - Message is a human readable description of the details of the last - transition, complementing reason. - type: string - observedGeneration: - description: |- - If set, this represents the .metadata.generation that the condition was - set based upon. - For instance, if .metadata.generation is currently 12, but the - .status.condition[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the Issuer. - type: integer - format: int64 - reason: - description: |- - Reason is a brief machine readable explanation for the condition's last - transition. - type: string - status: - description: Status of the condition, one of (`True`, `False`, `Unknown`). - type: string - enum: - - "True" - - "False" - - Unknown - type: - description: Type of the condition, known values are (`Ready`). - type: string - x-kubernetes-list-map-keys: - - type - x-kubernetes-list-type: map - served: true - storage: true - -# END crd {{- end }} - ---- -# START crd {{- if or .Values.crds.enabled .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - name: orders.acme.cert-manager.io - # START annotations {{- if .Values.crds.keep }} - annotations: - helm.sh/resource-policy: keep - # END annotations {{- end }} - labels: - app: '{{ template "cert-manager.name" . }}' - app.kubernetes.io/name: '{{ template "cert-manager.name" . }}' - app.kubernetes.io/instance: '{{ .Release.Name }}' - app.kubernetes.io/component: "crds" - # Generated labels {{- include "labels" . | nindent 4 }} -spec: - group: acme.cert-manager.io - names: - kind: Order - listKind: OrderList - plural: orders - singular: order - categories: - - cert-manager - - cert-manager-acme - scope: Namespaced - versions: - - name: v1 - subresources: - status: {} - additionalPrinterColumns: - - jsonPath: .status.state - name: State - type: string - - jsonPath: .spec.issuerRef.name - name: Issuer - priority: 1 - type: string - - jsonPath: .status.reason - name: Reason - priority: 1 - type: string - - jsonPath: .metadata.creationTimestamp - description: CreationTimestamp is a timestamp representing the server time when this object was created. It is not guaranteed to be set in happens-before order across separate operations. Clients may not set this value. It is represented in RFC3339 form and is in UTC. - name: Age - type: date - schema: - openAPIV3Schema: - description: Order is a type to represent an Order with an ACME server - type: object - required: - - metadata - - spec - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - type: object - required: - - issuerRef - - request - properties: - commonName: - description: |- - CommonName is the common name as specified on the DER encoded CSR. - If specified, this value must also be present in `dnsNames` or `ipAddresses`. - This field must match the corresponding field on the DER encoded CSR. - type: string - dnsNames: - description: |- - DNSNames is a list of DNS names that should be included as part of the Order - validation process. - This field must match the corresponding field on the DER encoded CSR. - type: array - items: - type: string - duration: - description: |- - Duration is the duration for the not after date for the requested certificate. - this is set on order creation as pe the ACME spec. - type: string - ipAddresses: - description: |- - IPAddresses is a list of IP addresses that should be included as part of the Order - validation process. - This field must match the corresponding field on the DER encoded CSR. - type: array - items: - type: string - issuerRef: - description: |- - IssuerRef references a properly configured ACME-type Issuer which should - be used to create this Order. - If the Issuer does not exist, processing will be retried. - If the Issuer is not an 'ACME' Issuer, an error will be returned and the - Order will be marked as failed. - type: object - required: - - name - properties: - group: - description: Group of the resource being referred to. - type: string - kind: - description: Kind of the resource being referred to. - type: string - name: - description: Name of the resource being referred to. - type: string - request: - description: |- - Certificate signing request bytes in DER encoding. - This will be used when finalizing the order. - This field must be set on the order. - type: string - format: byte - status: - type: object - properties: - authorizations: - description: |- - Authorizations contains data returned from the ACME server on what - authorizations must be completed in order to validate the DNS names - specified on the Order. - type: array - items: - description: |- - ACMEAuthorization contains data returned from the ACME server on an - authorization that must be completed in order validate a DNS name on an ACME - Order resource. - type: object - required: - - url - properties: - challenges: - description: |- - Challenges specifies the challenge types offered by the ACME server. - One of these challenge types will be selected when validating the DNS - name and an appropriate Challenge resource will be created to perform - the ACME challenge process. - type: array - items: - description: |- - Challenge specifies a challenge offered by the ACME server for an Order. - An appropriate Challenge resource can be created to perform the ACME - challenge process. - type: object - required: - - token - - type - - url - properties: - token: - description: |- - Token is the token that must be presented for this challenge. - This is used to compute the 'key' that must also be presented. - type: string - type: - description: |- - Type is the type of challenge being offered, e.g. 'http-01', 'dns-01', - 'tls-sni-01', etc. - This is the raw value retrieved from the ACME server. - Only 'http-01' and 'dns-01' are supported by cert-manager, other values - will be ignored. - type: string - url: - description: |- - URL is the URL of this challenge. It can be used to retrieve additional - metadata about the Challenge from the ACME server. - type: string - identifier: - description: Identifier is the DNS name to be validated as part of this authorization - type: string - initialState: - description: |- - InitialState is the initial state of the ACME authorization when first - fetched from the ACME server. - If an Authorization is already 'valid', the Order controller will not - create a Challenge resource for the authorization. This will occur when - working with an ACME server that enables 'authz reuse' (such as Let's - Encrypt's production endpoint). - If not set and 'identifier' is set, the state is assumed to be pending - and a Challenge will be created. - type: string - enum: - - valid - - ready - - pending - - processing - - invalid - - expired - - errored - url: - description: URL is the URL of the Authorization that must be completed - type: string - wildcard: - description: |- - Wildcard will be true if this authorization is for a wildcard DNS name. - If this is true, the identifier will be the *non-wildcard* version of - the DNS name. - For example, if '*.example.com' is the DNS name being validated, this - field will be 'true' and the 'identifier' field will be 'example.com'. - type: boolean - certificate: - description: |- - Certificate is a copy of the PEM encoded certificate for this Order. - This field will be populated after the order has been successfully - finalized with the ACME server, and the order has transitioned to the - 'valid' state. - type: string - format: byte - failureTime: - description: |- - FailureTime stores the time that this order failed. - This is used to influence garbage collection and back-off. - type: string - format: date-time - finalizeURL: - description: |- - FinalizeURL of the Order. - This is used to obtain certificates for this order once it has been completed. - type: string - reason: - description: |- - Reason optionally provides more information about a why the order is in - the current state. - type: string - state: - description: |- - State contains the current state of this Order resource. - States 'success' and 'expired' are 'final' - type: string - enum: - - valid - - ready - - pending - - processing - - invalid - - expired - - errored - url: - description: |- - URL of the Order. - This will initially be empty when the resource is first created. - The Order controller will populate this field when the Order is first processed. - This field will be immutable after it is initially set. - type: string - served: true - storage: true - -# END crd {{- end }} diff --git a/charts/cert-manager/cert-manager/templates/deployment.yaml b/charts/cert-manager/cert-manager/templates/deployment.yaml deleted file mode 100644 index e6f3f68..0000000 --- a/charts/cert-manager/cert-manager/templates/deployment.yaml +++ /dev/null @@ -1,231 +0,0 @@ -apiVersion: apps/v1 -kind: Deployment -metadata: - name: {{ template "cert-manager.fullname" . }} - namespace: {{ include "cert-manager.namespace" . }} - labels: - app: {{ template "cert-manager.name" . }} - app.kubernetes.io/name: {{ template "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} - {{- with .Values.deploymentAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - replicas: {{ .Values.replicaCount }} - {{- /* The if statement below is equivalent to {{- if $value }} but will also return true for 0. */ -}} - {{- if not (has (quote .Values.global.revisionHistoryLimit) (list "" (quote ""))) }} - revisionHistoryLimit: {{ .Values.global.revisionHistoryLimit }} - {{- end }} - selector: - matchLabels: - app.kubernetes.io/name: {{ template "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- with .Values.strategy }} - strategy: - {{- toYaml . | nindent 4 }} - {{- end }} - template: - metadata: - labels: - app: {{ template "cert-manager.name" . }} - app.kubernetes.io/name: {{ template "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 8 }} - {{- with .Values.podLabels }} - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.podAnnotations }} - annotations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- if and .Values.prometheus.enabled (not (or .Values.prometheus.servicemonitor.enabled .Values.prometheus.podmonitor.enabled)) }} - {{- if not .Values.podAnnotations }} - annotations: - {{- end }} - prometheus.io/path: "/metrics" - prometheus.io/scrape: 'true' - prometheus.io/port: '9402' - {{- end }} - spec: - serviceAccountName: {{ template "cert-manager.serviceAccountName" . }} - {{- if hasKey .Values "automountServiceAccountToken" }} - automountServiceAccountToken: {{ .Values.automountServiceAccountToken }} - {{- end }} - enableServiceLinks: {{ .Values.enableServiceLinks }} - {{- with .Values.global.priorityClassName }} - priorityClassName: {{ . | quote }} - {{- end }} - {{- with .Values.securityContext }} - securityContext: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- if or .Values.volumes .Values.config}} - volumes: - {{- if .Values.config }} - - name: config - configMap: - name: {{ include "cert-manager.fullname" . }} - {{- end }} - {{ with .Values.volumes }} - {{- toYaml . | nindent 8 }} - {{- end }} - {{- end }} - containers: - - name: {{ .Chart.Name }}-controller - image: "{{ template "image" (tuple .Values.image $.Chart.AppVersion) }}" - imagePullPolicy: {{ .Values.image.pullPolicy }} - args: - {{- /* The if statement below is equivalent to {{- if $value }} but will also return true for 0. */ -}} - {{- if not (has (quote .Values.global.logLevel) (list "" (quote ""))) }} - - --v={{ .Values.global.logLevel }} - {{- end }} - {{- if .Values.config }} - - --config=/var/cert-manager/config/config.yaml - {{- end }} - {{- $config := default .Values.config "" }} - {{- if .Values.clusterResourceNamespace }} - - --cluster-resource-namespace={{ .Values.clusterResourceNamespace }} - {{- else }} - - --cluster-resource-namespace=$(POD_NAMESPACE) - {{- end }} - {{- with .Values.global.leaderElection }} - - --leader-election-namespace={{ .namespace }} - {{- if .leaseDuration }} - - --leader-election-lease-duration={{ .leaseDuration }} - {{- end }} - {{- if .renewDeadline }} - - --leader-election-renew-deadline={{ .renewDeadline }} - {{- end }} - {{- if .retryPeriod }} - - --leader-election-retry-period={{ .retryPeriod }} - {{- end }} - {{- end }} - {{- with .Values.acmesolver.image }} - - --acme-http01-solver-image={{- if .registry -}}{{ .registry }}/{{- end -}}{{ .repository }}{{- if (.digest) -}} @{{ .digest }}{{- else -}}:{{ default $.Chart.AppVersion .tag }} {{- end -}} - {{- end }} - {{- with .Values.extraArgs }} - {{- toYaml . | nindent 10 }} - {{- end }} - {{- with .Values.ingressShim }} - {{- if .defaultIssuerName }} - - --default-issuer-name={{ .defaultIssuerName }} - {{- end }} - {{- if .defaultIssuerKind }} - - --default-issuer-kind={{ .defaultIssuerKind }} - {{- end }} - {{- if .defaultIssuerGroup }} - - --default-issuer-group={{ .defaultIssuerGroup }} - {{- end }} - {{- end }} - {{- if .Values.featureGates }} - - --feature-gates={{ .Values.featureGates }} - {{- end }} - {{- if .Values.maxConcurrentChallenges }} - - --max-concurrent-challenges={{ .Values.maxConcurrentChallenges }} - {{- end }} - {{- if .Values.enableCertificateOwnerRef }} - - --enable-certificate-owner-ref=true - {{- end }} - {{- if .Values.dns01RecursiveNameserversOnly }} - - --dns01-recursive-nameservers-only=true - {{- end }} - {{- with .Values.dns01RecursiveNameservers }} - - --dns01-recursive-nameservers={{ . }} - {{- end }} - {{- if .Values.disableAutoApproval }} - - --controllers=-certificaterequests-approver - {{- end }} - ports: - - containerPort: 9402 - name: http-metrics - protocol: TCP - - containerPort: 9403 - name: http-healthz - protocol: TCP - {{- with .Values.containerSecurityContext }} - securityContext: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- if or .Values.config .Values.volumeMounts }} - volumeMounts: - {{- if .Values.config }} - - name: config - mountPath: /var/cert-manager/config - {{- end }} - {{- with .Values.volumeMounts }} - {{- toYaml . | nindent 12 }} - {{- end }} - {{- end }} - env: - - name: POD_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - {{- with .Values.extraEnv }} - {{- toYaml . | nindent 10 }} - {{- end }} - {{- with .Values.http_proxy }} - - name: HTTP_PROXY - value: {{ . }} - {{- end }} - {{- with .Values.https_proxy }} - - name: HTTPS_PROXY - value: {{ . }} - {{- end }} - {{- with .Values.no_proxy }} - - name: NO_PROXY - value: {{ . }} - {{- end }} - {{- with .Values.resources }} - resources: - {{- toYaml . | nindent 12 }} - {{- end }} - - {{- with .Values.livenessProbe }} - {{- if .enabled }} - # LivenessProbe settings are based on those used for the Kubernetes - # controller-manager. See: - # https://github.com/kubernetes/kubernetes/blob/806b30170c61a38fedd54cc9ede4cd6275a1ad3b/cmd/kubeadm/app/util/staticpod/utils.go#L241-L245 - livenessProbe: - httpGet: - port: http-healthz - path: /livez - scheme: HTTP - initialDelaySeconds: {{ .initialDelaySeconds }} - periodSeconds: {{ .periodSeconds }} - timeoutSeconds: {{ .timeoutSeconds }} - successThreshold: {{ .successThreshold }} - failureThreshold: {{ .failureThreshold }} - {{- end }} - {{- end }} - {{- with .Values.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.topologySpreadConstraints }} - topologySpreadConstraints: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.podDnsPolicy }} - dnsPolicy: {{ . }} - {{- end }} - {{- with .Values.podDnsConfig }} - dnsConfig: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.hostAliases }} - hostAliases: {{ toYaml . | nindent 8 }} - {{- end }} \ No newline at end of file diff --git a/charts/cert-manager/cert-manager/templates/extras-objects.yaml b/charts/cert-manager/cert-manager/templates/extras-objects.yaml deleted file mode 100644 index 9ec3a7e..0000000 --- a/charts/cert-manager/cert-manager/templates/extras-objects.yaml +++ /dev/null @@ -1,4 +0,0 @@ -{{ range .Values.extraObjects }} ---- -{{ tpl . $ }} -{{ end }} diff --git a/charts/cert-manager/cert-manager/templates/networkpolicy-egress.yaml b/charts/cert-manager/cert-manager/templates/networkpolicy-egress.yaml deleted file mode 100644 index 37f90bd..0000000 --- a/charts/cert-manager/cert-manager/templates/networkpolicy-egress.yaml +++ /dev/null @@ -1,19 +0,0 @@ -{{- if .Values.webhook.networkPolicy.enabled }} -apiVersion: networking.k8s.io/v1 -kind: NetworkPolicy -metadata: - name: {{ template "webhook.fullname" . }}-allow-egress - namespace: {{ include "cert-manager.namespace" . }} -spec: - egress: - {{- with .Values.webhook.networkPolicy.egress }} - {{- toYaml . | nindent 2 }} - {{- end }} - podSelector: - matchLabels: - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - policyTypes: - - Egress -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/networkpolicy-webhooks.yaml b/charts/cert-manager/cert-manager/templates/networkpolicy-webhooks.yaml deleted file mode 100644 index 3a0ed7a..0000000 --- a/charts/cert-manager/cert-manager/templates/networkpolicy-webhooks.yaml +++ /dev/null @@ -1,21 +0,0 @@ -{{- if .Values.webhook.networkPolicy.enabled }} - -apiVersion: networking.k8s.io/v1 -kind: NetworkPolicy -metadata: - name: {{ template "webhook.fullname" . }}-allow-ingress - namespace: {{ include "cert-manager.namespace" . }} -spec: - ingress: - {{- with .Values.webhook.networkPolicy.ingress }} - {{- toYaml . | nindent 2 }} - {{- end }} - podSelector: - matchLabels: - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - policyTypes: - - Ingress - -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/poddisruptionbudget.yaml b/charts/cert-manager/cert-manager/templates/poddisruptionbudget.yaml deleted file mode 100644 index ae71eed..0000000 --- a/charts/cert-manager/cert-manager/templates/poddisruptionbudget.yaml +++ /dev/null @@ -1,29 +0,0 @@ -{{- if .Values.podDisruptionBudget.enabled }} -apiVersion: policy/v1 -kind: PodDisruptionBudget -metadata: - name: {{ include "cert-manager.fullname" . }} - namespace: {{ include "cert-manager.namespace" . }} - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -spec: - selector: - matchLabels: - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - - {{- if not (or (hasKey .Values.podDisruptionBudget "minAvailable") (hasKey .Values.podDisruptionBudget "maxUnavailable")) }} - minAvailable: 1 # Default value because minAvailable and maxUnavailable are not set - {{- end }} - {{- if hasKey .Values.podDisruptionBudget "minAvailable" }} - minAvailable: {{ .Values.podDisruptionBudget.minAvailable }} - {{- end }} - {{- if hasKey .Values.podDisruptionBudget "maxUnavailable" }} - maxUnavailable: {{ .Values.podDisruptionBudget.maxUnavailable }} - {{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/podmonitor.yaml b/charts/cert-manager/cert-manager/templates/podmonitor.yaml deleted file mode 100644 index 1adc060..0000000 --- a/charts/cert-manager/cert-manager/templates/podmonitor.yaml +++ /dev/null @@ -1,50 +0,0 @@ -{{- if and .Values.prometheus.enabled (and .Values.prometheus.podmonitor.enabled .Values.prometheus.servicemonitor.enabled) }} -{{- fail "Either .Values.prometheus.podmonitor.enabled or .Values.prometheus.servicemonitor.enabled can be enabled at a time, but not both." }} -{{- else if and .Values.prometheus.enabled .Values.prometheus.podmonitor.enabled }} -apiVersion: monitoring.coreos.com/v1 -kind: PodMonitor -metadata: - name: {{ template "cert-manager.fullname" . }} -{{- if .Values.prometheus.podmonitor.namespace }} - namespace: {{ .Values.prometheus.podmonitor.namespace }} -{{- else }} - namespace: {{ include "cert-manager.namespace" . }} -{{- end }} - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} - prometheus: {{ .Values.prometheus.podmonitor.prometheusInstance }} - {{- with .Values.prometheus.podmonitor.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} -{{- if .Values.prometheus.podmonitor.annotations }} - annotations: - {{- with .Values.prometheus.podmonitor.annotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end }} -spec: - jobLabel: {{ template "cert-manager.fullname" . }} - selector: - matchLabels: - app.kubernetes.io/name: {{ template "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" -{{- if .Values.prometheus.podmonitor.namespace }} - namespaceSelector: - matchNames: - - {{ include "cert-manager.namespace" . }} -{{- end }} - podMetricsEndpoints: - - port: http-metrics - path: {{ .Values.prometheus.podmonitor.path }} - interval: {{ .Values.prometheus.podmonitor.interval }} - scrapeTimeout: {{ .Values.prometheus.podmonitor.scrapeTimeout }} - honorLabels: {{ .Values.prometheus.podmonitor.honorLabels }} - {{- with .Values.prometheus.servicemonitor.endpointAdditionalProperties }} - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/psp-clusterrole.yaml b/charts/cert-manager/cert-manager/templates/psp-clusterrole.yaml deleted file mode 100644 index 1d40a02..0000000 --- a/charts/cert-manager/cert-manager/templates/psp-clusterrole.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- if .Values.global.podSecurityPolicy.enabled }} -kind: ClusterRole -apiVersion: rbac.authorization.k8s.io/v1 -metadata: - name: {{ template "cert-manager.fullname" . }}-psp - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -rules: -- apiGroups: ['policy'] - resources: ['podsecuritypolicies'] - verbs: ['use'] - resourceNames: - - {{ template "cert-manager.fullname" . }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/psp-clusterrolebinding.yaml b/charts/cert-manager/cert-manager/templates/psp-clusterrolebinding.yaml deleted file mode 100644 index 4f09b6b..0000000 --- a/charts/cert-manager/cert-manager/templates/psp-clusterrolebinding.yaml +++ /dev/null @@ -1,20 +0,0 @@ -{{- if .Values.global.podSecurityPolicy.enabled }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ template "cert-manager.fullname" . }}-psp - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ template "cert-manager.fullname" . }}-psp -subjects: - - kind: ServiceAccount - name: {{ template "cert-manager.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/psp.yaml b/charts/cert-manager/cert-manager/templates/psp.yaml deleted file mode 100644 index 9e99f5c..0000000 --- a/charts/cert-manager/cert-manager/templates/psp.yaml +++ /dev/null @@ -1,49 +0,0 @@ -{{- if .Values.global.podSecurityPolicy.enabled }} -apiVersion: policy/v1beta1 -kind: PodSecurityPolicy -metadata: - name: {{ template "cert-manager.fullname" . }} - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} - annotations: - seccomp.security.alpha.kubernetes.io/allowedProfileNames: 'docker/default' - seccomp.security.alpha.kubernetes.io/defaultProfileName: 'docker/default' - {{- if .Values.global.podSecurityPolicy.useAppArmor }} - apparmor.security.beta.kubernetes.io/allowedProfileNames: 'runtime/default' - apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' - {{- end }} -spec: - privileged: false - allowPrivilegeEscalation: false - allowedCapabilities: [] # default set of capabilities are implicitly allowed - volumes: - - 'configMap' - - 'emptyDir' - - 'projected' - - 'secret' - - 'downwardAPI' - hostNetwork: false - hostIPC: false - hostPID: false - runAsUser: - rule: 'MustRunAs' - ranges: - - min: 1000 - max: 1000 - seLinux: - rule: 'RunAsAny' - supplementalGroups: - rule: 'MustRunAs' - ranges: - - min: 1000 - max: 1000 - fsGroup: - rule: 'MustRunAs' - ranges: - - min: 1000 - max: 1000 -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/rbac.yaml b/charts/cert-manager/cert-manager/templates/rbac.yaml deleted file mode 100644 index 7a27d4f..0000000 --- a/charts/cert-manager/cert-manager/templates/rbac.yaml +++ /dev/null @@ -1,574 +0,0 @@ -{{- if .Values.global.rbac.create }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: {{ template "cert-manager.fullname" . }}:leaderelection - namespace: {{ .Values.global.leaderElection.namespace }} - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -rules: - - apiGroups: ["coordination.k8s.io"] - resources: ["leases"] - resourceNames: ["cert-manager-controller"] - verbs: ["get", "update", "patch"] - - apiGroups: ["coordination.k8s.io"] - resources: ["leases"] - verbs: ["create"] - ---- - -# grant cert-manager permission to manage the leaderelection configmap in the -# leader election namespace -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: {{ include "cert-manager.fullname" . }}:leaderelection - namespace: {{ .Values.global.leaderElection.namespace }} - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: {{ template "cert-manager.fullname" . }}:leaderelection -subjects: - - apiGroup: "" - kind: ServiceAccount - name: {{ template "cert-manager.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - ---- - -# Issuer controller role -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-issuers - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -rules: - - apiGroups: ["cert-manager.io"] - resources: ["issuers", "issuers/status"] - verbs: ["update", "patch"] - - apiGroups: ["cert-manager.io"] - resources: ["issuers"] - verbs: ["get", "list", "watch"] - - apiGroups: [""] - resources: ["secrets"] - verbs: ["get", "list", "watch", "create", "update", "delete"] - - apiGroups: [""] - resources: ["events"] - verbs: ["create", "patch"] ---- - -# ClusterIssuer controller role -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-clusterissuers - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -rules: - - apiGroups: ["cert-manager.io"] - resources: ["clusterissuers", "clusterissuers/status"] - verbs: ["update", "patch"] - - apiGroups: ["cert-manager.io"] - resources: ["clusterissuers"] - verbs: ["get", "list", "watch"] - - apiGroups: [""] - resources: ["secrets"] - verbs: ["get", "list", "watch", "create", "update", "delete"] - - apiGroups: [""] - resources: ["events"] - verbs: ["create", "patch"] - ---- - -# Certificates controller role -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-certificates - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -rules: - - apiGroups: ["cert-manager.io"] - resources: ["certificates", "certificates/status", "certificaterequests", "certificaterequests/status"] - verbs: ["update", "patch"] - - apiGroups: ["cert-manager.io"] - resources: ["certificates", "certificaterequests", "clusterissuers", "issuers"] - verbs: ["get", "list", "watch"] - # We require these rules to support users with the OwnerReferencesPermissionEnforcement - # admission controller enabled: - # https://kubernetes.io/docs/reference/access-authn-authz/admission-controllers/#ownerreferencespermissionenforcement - - apiGroups: ["cert-manager.io"] - resources: ["certificates/finalizers", "certificaterequests/finalizers"] - verbs: ["update"] - - apiGroups: ["acme.cert-manager.io"] - resources: ["orders"] - verbs: ["create", "delete", "get", "list", "watch"] - - apiGroups: [""] - resources: ["secrets"] - verbs: ["get", "list", "watch", "create", "update", "delete", "patch"] - - apiGroups: [""] - resources: ["events"] - verbs: ["create", "patch"] - ---- - -# Orders controller role -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-orders - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -rules: - - apiGroups: ["acme.cert-manager.io"] - resources: ["orders", "orders/status"] - verbs: ["update", "patch"] - - apiGroups: ["acme.cert-manager.io"] - resources: ["orders", "challenges"] - verbs: ["get", "list", "watch"] - - apiGroups: ["cert-manager.io"] - resources: ["clusterissuers", "issuers"] - verbs: ["get", "list", "watch"] - - apiGroups: ["acme.cert-manager.io"] - resources: ["challenges"] - verbs: ["create", "delete"] - # We require these rules to support users with the OwnerReferencesPermissionEnforcement - # admission controller enabled: - # https://kubernetes.io/docs/reference/access-authn-authz/admission-controllers/#ownerreferencespermissionenforcement - - apiGroups: ["acme.cert-manager.io"] - resources: ["orders/finalizers"] - verbs: ["update"] - - apiGroups: [""] - resources: ["secrets"] - verbs: ["get", "list", "watch"] - - apiGroups: [""] - resources: ["events"] - verbs: ["create", "patch"] - ---- - -# Challenges controller role -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-challenges - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -rules: - # Use to update challenge resource status - - apiGroups: ["acme.cert-manager.io"] - resources: ["challenges", "challenges/status"] - verbs: ["update", "patch"] - # Used to watch challenge resources - - apiGroups: ["acme.cert-manager.io"] - resources: ["challenges"] - verbs: ["get", "list", "watch"] - # Used to watch challenges, issuer and clusterissuer resources - - apiGroups: ["cert-manager.io"] - resources: ["issuers", "clusterissuers"] - verbs: ["get", "list", "watch"] - # Need to be able to retrieve ACME account private key to complete challenges - - apiGroups: [""] - resources: ["secrets"] - verbs: ["get", "list", "watch"] - # Used to create events - - apiGroups: [""] - resources: ["events"] - verbs: ["create", "patch"] - # HTTP01 rules - - apiGroups: [""] - resources: ["pods", "services"] - verbs: ["get", "list", "watch", "create", "delete"] - - apiGroups: ["networking.k8s.io"] - resources: ["ingresses"] - verbs: ["get", "list", "watch", "create", "delete", "update"] - - apiGroups: [ "gateway.networking.k8s.io" ] - resources: [ "httproutes" ] - verbs: ["get", "list", "watch", "create", "delete", "update"] - # We require the ability to specify a custom hostname when we are creating - # new ingress resources. - # See: https://github.com/openshift/origin/blob/21f191775636f9acadb44fa42beeb4f75b255532/pkg/route/apiserver/admission/ingress_admission.go#L84-L148 - - apiGroups: ["route.openshift.io"] - resources: ["routes/custom-host"] - verbs: ["create"] - # We require these rules to support users with the OwnerReferencesPermissionEnforcement - # admission controller enabled: - # https://kubernetes.io/docs/reference/access-authn-authz/admission-controllers/#ownerreferencespermissionenforcement - - apiGroups: ["acme.cert-manager.io"] - resources: ["challenges/finalizers"] - verbs: ["update"] - # DNS01 rules (duplicated above) - - apiGroups: [""] - resources: ["secrets"] - verbs: ["get", "list", "watch"] - ---- - -# ingress-shim controller role -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-ingress-shim - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -rules: - - apiGroups: ["cert-manager.io"] - resources: ["certificates", "certificaterequests"] - verbs: ["create", "update", "delete"] - - apiGroups: ["cert-manager.io"] - resources: ["certificates", "certificaterequests", "issuers", "clusterissuers"] - verbs: ["get", "list", "watch"] - - apiGroups: ["networking.k8s.io"] - resources: ["ingresses"] - verbs: ["get", "list", "watch"] - # We require these rules to support users with the OwnerReferencesPermissionEnforcement - # admission controller enabled: - # https://kubernetes.io/docs/reference/access-authn-authz/admission-controllers/#ownerreferencespermissionenforcement - - apiGroups: ["networking.k8s.io"] - resources: ["ingresses/finalizers"] - verbs: ["update"] - - apiGroups: ["gateway.networking.k8s.io"] - resources: ["gateways", "httproutes"] - verbs: ["get", "list", "watch"] - - apiGroups: ["gateway.networking.k8s.io"] - resources: ["gateways/finalizers", "httproutes/finalizers"] - verbs: ["update"] - - apiGroups: [""] - resources: ["events"] - verbs: ["create", "patch"] - ---- - -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-issuers - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ template "cert-manager.fullname" . }}-controller-issuers -subjects: - - name: {{ template "cert-manager.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - kind: ServiceAccount - ---- - -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-clusterissuers - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ template "cert-manager.fullname" . }}-controller-clusterissuers -subjects: - - name: {{ template "cert-manager.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - kind: ServiceAccount - ---- - -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-certificates - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ template "cert-manager.fullname" . }}-controller-certificates -subjects: - - name: {{ template "cert-manager.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - kind: ServiceAccount - ---- - -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-orders - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ template "cert-manager.fullname" . }}-controller-orders -subjects: - - name: {{ template "cert-manager.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - kind: ServiceAccount - ---- - -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-challenges - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ template "cert-manager.fullname" . }}-controller-challenges -subjects: - - name: {{ template "cert-manager.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - kind: ServiceAccount - ---- - -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-ingress-shim - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ template "cert-manager.fullname" . }}-controller-ingress-shim -subjects: - - name: {{ template "cert-manager.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - kind: ServiceAccount - -{{- if .Values.global.rbac.aggregateClusterRoles }} ---- - -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ template "cert-manager.fullname" . }}-cluster-view - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} - rbac.authorization.k8s.io/aggregate-to-cluster-reader: "true" -rules: - - apiGroups: ["cert-manager.io"] - resources: ["clusterissuers"] - verbs: ["get", "list", "watch"] - -{{- end }} ---- - -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ template "cert-manager.fullname" . }}-view - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} - {{- if .Values.global.rbac.aggregateClusterRoles }} - rbac.authorization.k8s.io/aggregate-to-view: "true" - rbac.authorization.k8s.io/aggregate-to-edit: "true" - rbac.authorization.k8s.io/aggregate-to-admin: "true" - rbac.authorization.k8s.io/aggregate-to-cluster-reader: "true" - {{- end }} -rules: - - apiGroups: ["cert-manager.io"] - resources: ["certificates", "certificaterequests", "issuers"] - verbs: ["get", "list", "watch"] - - apiGroups: ["acme.cert-manager.io"] - resources: ["challenges", "orders"] - verbs: ["get", "list", "watch"] - - ---- - -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ template "cert-manager.fullname" . }}-edit - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} - {{- if .Values.global.rbac.aggregateClusterRoles }} - rbac.authorization.k8s.io/aggregate-to-edit: "true" - rbac.authorization.k8s.io/aggregate-to-admin: "true" - {{- end }} -rules: - - apiGroups: ["cert-manager.io"] - resources: ["certificates", "certificaterequests", "issuers"] - verbs: ["create", "delete", "deletecollection", "patch", "update"] - - apiGroups: ["cert-manager.io"] - resources: ["certificates/status"] - verbs: ["update"] - - apiGroups: ["acme.cert-manager.io"] - resources: ["challenges", "orders"] - verbs: ["create", "delete", "deletecollection", "patch", "update"] - ---- - -{{- if not .Values.disableAutoApproval -}} - -# Permission to approve CertificateRequests referencing cert-manager.io Issuers and ClusterIssuers -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-approve:cert-manager-io - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cert-manager" - {{- include "labels" . | nindent 4 }} -rules: - - apiGroups: ["cert-manager.io"] - resources: ["signers"] - verbs: ["approve"] - {{- with .Values.approveSignerNames }} - resourceNames: - {{- range . }} - - {{ . | quote }} - {{- end }} - {{- end }} - ---- - -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-approve:cert-manager-io - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cert-manager" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ template "cert-manager.fullname" . }}-controller-approve:cert-manager-io -subjects: - - name: {{ template "cert-manager.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - kind: ServiceAccount - ---- - -{{- end -}} - -# Permission to: -# - Update and sign CertificatSigningeRequests referencing cert-manager.io Issuers and ClusterIssuers -# - Perform SubjectAccessReviews to test whether users are able to reference Namespaced Issuers -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-certificatesigningrequests - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cert-manager" - {{- include "labels" . | nindent 4 }} -rules: - - apiGroups: ["certificates.k8s.io"] - resources: ["certificatesigningrequests"] - verbs: ["get", "list", "watch", "update"] - - apiGroups: ["certificates.k8s.io"] - resources: ["certificatesigningrequests/status"] - verbs: ["update", "patch"] - - apiGroups: ["certificates.k8s.io"] - resources: ["signers"] - resourceNames: ["issuers.cert-manager.io/*", "clusterissuers.cert-manager.io/*"] - verbs: ["sign"] - - apiGroups: ["authorization.k8s.io"] - resources: ["subjectaccessreviews"] - verbs: ["create"] - ---- - -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ template "cert-manager.fullname" . }}-controller-certificatesigningrequests - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "cert-manager" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ template "cert-manager.fullname" . }}-controller-certificatesigningrequests -subjects: - - name: {{ template "cert-manager.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - kind: ServiceAccount -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/service.yaml b/charts/cert-manager/cert-manager/templates/service.yaml deleted file mode 100644 index 360ec64..0000000 --- a/charts/cert-manager/cert-manager/templates/service.yaml +++ /dev/null @@ -1,37 +0,0 @@ -{{- if and .Values.prometheus.enabled (not .Values.prometheus.podmonitor.enabled) }} -apiVersion: v1 -kind: Service -metadata: - name: {{ template "cert-manager.fullname" . }} - namespace: {{ include "cert-manager.namespace" . }} -{{- with .Values.serviceAnnotations }} - annotations: -{{ toYaml . | indent 4 }} -{{- end }} - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} - {{- with .Values.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - type: ClusterIP - {{- if .Values.serviceIPFamilyPolicy }} - ipFamilyPolicy: {{ .Values.serviceIPFamilyPolicy }} - {{- end }} - {{- if .Values.serviceIPFamilies }} - ipFamilies: {{ .Values.serviceIPFamilies | toYaml | nindent 2 }} - {{- end }} - ports: - - protocol: TCP - port: 9402 - name: tcp-prometheus-servicemonitor - targetPort: {{ .Values.prometheus.servicemonitor.targetPort }} - selector: - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/serviceaccount.yaml b/charts/cert-manager/cert-manager/templates/serviceaccount.yaml deleted file mode 100644 index 87fc00e..0000000 --- a/charts/cert-manager/cert-manager/templates/serviceaccount.yaml +++ /dev/null @@ -1,25 +0,0 @@ -{{- if .Values.serviceAccount.create }} -apiVersion: v1 -kind: ServiceAccount -{{- with .Values.global.imagePullSecrets }} -imagePullSecrets: - {{- toYaml . | nindent 2 }} -{{- end }} -automountServiceAccountToken: {{ .Values.serviceAccount.automountServiceAccountToken }} -metadata: - name: {{ template "cert-manager.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - {{- with .Values.serviceAccount.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} - {{- with .Values.serviceAccount.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/servicemonitor.yaml b/charts/cert-manager/cert-manager/templates/servicemonitor.yaml deleted file mode 100644 index b638860..0000000 --- a/charts/cert-manager/cert-manager/templates/servicemonitor.yaml +++ /dev/null @@ -1,50 +0,0 @@ -{{- if and .Values.prometheus.enabled (and .Values.prometheus.podmonitor.enabled .Values.prometheus.servicemonitor.enabled) }} -{{- fail "Either .Values.prometheus.podmonitor.enabled or .Values.prometheus.servicemonitor.enabled can be enabled at a time, but not both." }} -{{- else if and .Values.prometheus.enabled .Values.prometheus.servicemonitor.enabled }} -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ template "cert-manager.fullname" . }} -{{- if .Values.prometheus.servicemonitor.namespace }} - namespace: {{ .Values.prometheus.servicemonitor.namespace }} -{{- else }} - namespace: {{ include "cert-manager.namespace" . }} -{{- end }} - labels: - app: {{ include "cert-manager.name" . }} - app.kubernetes.io/name: {{ include "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" - {{- include "labels" . | nindent 4 }} - prometheus: {{ .Values.prometheus.servicemonitor.prometheusInstance }} - {{- with .Values.prometheus.servicemonitor.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} -{{- if .Values.prometheus.servicemonitor.annotations }} - annotations: - {{- with .Values.prometheus.servicemonitor.annotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end }} -spec: - jobLabel: {{ template "cert-manager.fullname" . }} - selector: - matchLabels: - app.kubernetes.io/name: {{ template "cert-manager.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "controller" -{{- if .Values.prometheus.servicemonitor.namespace }} - namespaceSelector: - matchNames: - - {{ include "cert-manager.namespace" . }} -{{- end }} - endpoints: - - targetPort: {{ .Values.prometheus.servicemonitor.targetPort }} - path: {{ .Values.prometheus.servicemonitor.path }} - interval: {{ .Values.prometheus.servicemonitor.interval }} - scrapeTimeout: {{ .Values.prometheus.servicemonitor.scrapeTimeout }} - honorLabels: {{ .Values.prometheus.servicemonitor.honorLabels }} - {{- with .Values.prometheus.servicemonitor.endpointAdditionalProperties }} - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/startupapicheck-job.yaml b/charts/cert-manager/cert-manager/templates/startupapicheck-job.yaml deleted file mode 100644 index 311b4c4..0000000 --- a/charts/cert-manager/cert-manager/templates/startupapicheck-job.yaml +++ /dev/null @@ -1,87 +0,0 @@ -{{- if .Values.startupapicheck.enabled }} -apiVersion: batch/v1 -kind: Job -metadata: - name: {{ include "startupapicheck.fullname" . }} - namespace: {{ include "cert-manager.namespace" . }} - labels: - app: {{ include "startupapicheck.name" . }} - app.kubernetes.io/name: {{ include "startupapicheck.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "startupapicheck" - {{- include "labels" . | nindent 4 }} - {{- with .Values.startupapicheck.jobAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - backoffLimit: {{ .Values.startupapicheck.backoffLimit }} - template: - metadata: - labels: - app: {{ include "startupapicheck.name" . }} - app.kubernetes.io/name: {{ include "startupapicheck.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "startupapicheck" - {{- include "labels" . | nindent 8 }} - {{- with .Values.startupapicheck.podLabels }} - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.startupapicheck.podAnnotations }} - annotations: - {{- toYaml . | nindent 8 }} - {{- end }} - spec: - restartPolicy: OnFailure - serviceAccountName: {{ template "startupapicheck.serviceAccountName" . }} - {{- if hasKey .Values.startupapicheck "automountServiceAccountToken" }} - automountServiceAccountToken: {{ .Values.startupapicheck.automountServiceAccountToken }} - {{- end }} - enableServiceLinks: {{ .Values.startupapicheck.enableServiceLinks }} - {{- with .Values.global.priorityClassName }} - priorityClassName: {{ . | quote }} - {{- end }} - {{- with .Values.startupapicheck.securityContext }} - securityContext: - {{- toYaml . | nindent 8 }} - {{- end }} - containers: - - name: {{ .Chart.Name }}-startupapicheck - image: "{{ template "image" (tuple .Values.startupapicheck.image $.Chart.AppVersion) }}" - imagePullPolicy: {{ .Values.startupapicheck.image.pullPolicy }} - args: - - check - - api - - --wait={{ .Values.startupapicheck.timeout }} - {{- with .Values.startupapicheck.extraArgs }} - {{- toYaml . | nindent 10 }} - {{- end }} - {{- with .Values.startupapicheck.containerSecurityContext }} - securityContext: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- with .Values.startupapicheck.resources }} - resources: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- with .Values.startupapicheck.volumeMounts }} - volumeMounts: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- with .Values.startupapicheck.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.startupapicheck.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.startupapicheck.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.startupapicheck.volumes }} - volumes: - {{- toYaml . | nindent 8 }} - {{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/startupapicheck-psp-clusterrole.yaml b/charts/cert-manager/cert-manager/templates/startupapicheck-psp-clusterrole.yaml deleted file mode 100644 index dacd4be..0000000 --- a/charts/cert-manager/cert-manager/templates/startupapicheck-psp-clusterrole.yaml +++ /dev/null @@ -1,24 +0,0 @@ -{{- if .Values.startupapicheck.enabled }} -{{- if .Values.global.podSecurityPolicy.enabled }} -kind: ClusterRole -apiVersion: rbac.authorization.k8s.io/v1 -metadata: - name: {{ template "startupapicheck.fullname" . }}-psp - labels: - app: {{ include "startupapicheck.name" . }} - app.kubernetes.io/name: {{ include "startupapicheck.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "startupapicheck" - {{- include "labels" . | nindent 4 }} - {{- with .Values.startupapicheck.rbac.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -rules: -- apiGroups: ['policy'] - resources: ['podsecuritypolicies'] - verbs: ['use'] - resourceNames: - - {{ template "startupapicheck.fullname" . }} -{{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/startupapicheck-psp-clusterrolebinding.yaml b/charts/cert-manager/cert-manager/templates/startupapicheck-psp-clusterrolebinding.yaml deleted file mode 100644 index 54d5a42..0000000 --- a/charts/cert-manager/cert-manager/templates/startupapicheck-psp-clusterrolebinding.yaml +++ /dev/null @@ -1,26 +0,0 @@ -{{- if .Values.startupapicheck.enabled }} -{{- if .Values.global.podSecurityPolicy.enabled }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ template "startupapicheck.fullname" . }}-psp - labels: - app: {{ include "startupapicheck.name" . }} - app.kubernetes.io/name: {{ include "startupapicheck.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "startupapicheck" - {{- include "labels" . | nindent 4 }} - {{- with .Values.startupapicheck.rbac.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ template "startupapicheck.fullname" . }}-psp -subjects: - - kind: ServiceAccount - name: {{ template "startupapicheck.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} -{{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/startupapicheck-psp.yaml b/charts/cert-manager/cert-manager/templates/startupapicheck-psp.yaml deleted file mode 100644 index f09d60d..0000000 --- a/charts/cert-manager/cert-manager/templates/startupapicheck-psp.yaml +++ /dev/null @@ -1,51 +0,0 @@ -{{- if .Values.startupapicheck.enabled }} -{{- if .Values.global.podSecurityPolicy.enabled }} -apiVersion: policy/v1beta1 -kind: PodSecurityPolicy -metadata: - name: {{ template "startupapicheck.fullname" . }} - labels: - app: {{ include "startupapicheck.name" . }} - app.kubernetes.io/name: {{ include "startupapicheck.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "startupapicheck" - {{- include "labels" . | nindent 4 }} - annotations: - seccomp.security.alpha.kubernetes.io/allowedProfileNames: 'docker/default' - seccomp.security.alpha.kubernetes.io/defaultProfileName: 'docker/default' - {{- if .Values.global.podSecurityPolicy.useAppArmor }} - apparmor.security.beta.kubernetes.io/allowedProfileNames: 'runtime/default' - apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' - {{- end }} - {{- with .Values.startupapicheck.rbac.annotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - privileged: false - allowPrivilegeEscalation: false - allowedCapabilities: [] # default set of capabilities are implicitly allowed - volumes: - - 'projected' - - 'secret' - hostNetwork: false - hostIPC: false - hostPID: false - runAsUser: - rule: 'MustRunAs' - ranges: - - min: 1000 - max: 1000 - seLinux: - rule: 'RunAsAny' - supplementalGroups: - rule: 'MustRunAs' - ranges: - - min: 1000 - max: 1000 - fsGroup: - rule: 'MustRunAs' - ranges: - - min: 1000 - max: 1000 -{{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/startupapicheck-rbac.yaml b/charts/cert-manager/cert-manager/templates/startupapicheck-rbac.yaml deleted file mode 100644 index 606e725..0000000 --- a/charts/cert-manager/cert-manager/templates/startupapicheck-rbac.yaml +++ /dev/null @@ -1,48 +0,0 @@ -{{- if .Values.startupapicheck.enabled }} -{{- if .Values.global.rbac.create }} -# create certificate role -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: {{ template "startupapicheck.fullname" . }}:create-cert - namespace: {{ include "cert-manager.namespace" . }} - labels: - app: {{ include "startupapicheck.name" . }} - app.kubernetes.io/name: {{ include "startupapicheck.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "startupapicheck" - {{- include "labels" . | nindent 4 }} - {{- with .Values.startupapicheck.rbac.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -rules: - - apiGroups: ["cert-manager.io"] - resources: ["certificates"] - verbs: ["create"] ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: {{ include "startupapicheck.fullname" . }}:create-cert - namespace: {{ include "cert-manager.namespace" . }} - labels: - app: {{ include "startupapicheck.name" . }} - app.kubernetes.io/name: {{ include "startupapicheck.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "startupapicheck" - {{- include "labels" . | nindent 4 }} - {{- with .Values.startupapicheck.rbac.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: {{ template "startupapicheck.fullname" . }}:create-cert -subjects: - - kind: ServiceAccount - name: {{ template "startupapicheck.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} -{{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/startupapicheck-serviceaccount.yaml b/charts/cert-manager/cert-manager/templates/startupapicheck-serviceaccount.yaml deleted file mode 100644 index 8c41760..0000000 --- a/charts/cert-manager/cert-manager/templates/startupapicheck-serviceaccount.yaml +++ /dev/null @@ -1,27 +0,0 @@ -{{- if .Values.startupapicheck.enabled }} -{{- if .Values.startupapicheck.serviceAccount.create }} -apiVersion: v1 -kind: ServiceAccount -automountServiceAccountToken: {{ .Values.startupapicheck.serviceAccount.automountServiceAccountToken }} -metadata: - name: {{ template "startupapicheck.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - {{- with .Values.startupapicheck.serviceAccount.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} - labels: - app: {{ include "startupapicheck.name" . }} - app.kubernetes.io/name: {{ include "startupapicheck.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "startupapicheck" - {{- include "labels" . | nindent 4 }} - {{- with .Values.startupapicheck.serviceAccount.labels }} - {{ toYaml . | nindent 4 }} - {{- end }} -{{- with .Values.global.imagePullSecrets }} -imagePullSecrets: - {{- toYaml . | nindent 2 }} -{{- end }} -{{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/webhook-config.yaml b/charts/cert-manager/cert-manager/templates/webhook-config.yaml deleted file mode 100644 index 8f3ce20..0000000 --- a/charts/cert-manager/cert-manager/templates/webhook-config.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- if .Values.webhook.config -}} -{{- $_ := .Values.webhook.config.apiVersion | required ".Values.webhook.config.apiVersion must be set !" -}} -{{- $_ := .Values.webhook.config.kind | required ".Values.webhook.config.kind must be set !" -}} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ include "webhook.fullname" . }} - namespace: {{ include "cert-manager.namespace" . }} - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 4 }} -data: - config.yaml: | - {{- .Values.webhook.config | toYaml | nindent 4 }} -{{- end -}} \ No newline at end of file diff --git a/charts/cert-manager/cert-manager/templates/webhook-deployment.yaml b/charts/cert-manager/cert-manager/templates/webhook-deployment.yaml deleted file mode 100644 index ae5399e..0000000 --- a/charts/cert-manager/cert-manager/templates/webhook-deployment.yaml +++ /dev/null @@ -1,192 +0,0 @@ -apiVersion: apps/v1 -kind: Deployment -metadata: - name: {{ include "webhook.fullname" . }} - namespace: {{ include "cert-manager.namespace" . }} - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 4 }} - {{- with .Values.webhook.deploymentAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - replicas: {{ .Values.webhook.replicaCount }} - {{- /* The if statement below is equivalent to {{- if $value }} but will also return true for 0. */ -}} - {{- if not (has (quote .Values.global.revisionHistoryLimit) (list "" (quote ""))) }} - revisionHistoryLimit: {{ .Values.global.revisionHistoryLimit }} - {{- end }} - selector: - matchLabels: - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- with .Values.webhook.strategy }} - strategy: - {{- toYaml . | nindent 4 }} - {{- end }} - template: - metadata: - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 8 }} - {{- with .Values.webhook.podLabels }} - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.webhook.podAnnotations }} - annotations: - {{- toYaml . | nindent 8 }} - {{- end }} - spec: - serviceAccountName: {{ template "webhook.serviceAccountName" . }} - {{- if hasKey .Values.webhook "automountServiceAccountToken" }} - automountServiceAccountToken: {{ .Values.webhook.automountServiceAccountToken }} - {{- end }} - enableServiceLinks: {{ .Values.webhook.enableServiceLinks }} - {{- with .Values.global.priorityClassName }} - priorityClassName: {{ . | quote }} - {{- end }} - {{- with .Values.webhook.securityContext }} - securityContext: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- if .Values.webhook.hostNetwork }} - hostNetwork: true - {{- end }} - {{- if .Values.webhook.hostNetwork }} - dnsPolicy: ClusterFirstWithHostNet - {{- end }} - containers: - - name: {{ .Chart.Name }}-webhook - image: "{{ template "image" (tuple .Values.webhook.image $.Chart.AppVersion) }}" - imagePullPolicy: {{ .Values.webhook.image.pullPolicy }} - args: - {{- /* The if statement below is equivalent to {{- if $value }} but will also return true for 0. */ -}} - {{- if not (has (quote .Values.global.logLevel) (list "" (quote ""))) }} - - --v={{ .Values.global.logLevel }} - {{- end }} - {{- if .Values.webhook.config }} - - --config=/var/cert-manager/config/config.yaml - {{- end }} - {{- $config := default .Values.webhook.config "" }} - {{ if not $config.securePort -}} - - --secure-port={{ .Values.webhook.securePort }} - {{- end }} - {{- if .Values.webhook.featureGates }} - - --feature-gates={{ .Values.webhook.featureGates }} - {{- end }} - {{- $tlsConfig := default $config.tlsConfig "" }} - {{ if or (not $config.tlsConfig) (and (not $tlsConfig.dynamic) (not $tlsConfig.filesystem) ) -}} - - --dynamic-serving-ca-secret-namespace=$(POD_NAMESPACE) - - --dynamic-serving-ca-secret-name={{ template "webhook.fullname" . }}-ca - - --dynamic-serving-dns-names={{ template "webhook.fullname" . }} - - --dynamic-serving-dns-names={{ template "webhook.fullname" . }}.$(POD_NAMESPACE) - - --dynamic-serving-dns-names={{ template "webhook.fullname" . }}.$(POD_NAMESPACE).svc - {{ if .Values.webhook.url.host }} - - --dynamic-serving-dns-names={{ .Values.webhook.url.host }} - {{- end }} - {{- end }} - {{- with .Values.webhook.extraArgs }} - {{- toYaml . | nindent 10 }} - {{- end }} - ports: - - name: https - protocol: TCP - {{- if $config.securePort }} - containerPort: {{ $config.securePort }} - {{- else if .Values.webhook.securePort }} - containerPort: {{ .Values.webhook.securePort }} - {{- else }} - containerPort: 6443 - {{- end }} - - name: healthcheck - protocol: TCP - {{- if $config.healthzPort }} - containerPort: {{ $config.healthzPort }} - {{- else }} - containerPort: 6080 - {{- end }} - livenessProbe: - httpGet: - path: /livez - {{- if $config.healthzPort }} - port: {{ $config.healthzPort }} - {{- else }} - port: 6080 - {{- end }} - scheme: HTTP - initialDelaySeconds: {{ .Values.webhook.livenessProbe.initialDelaySeconds }} - periodSeconds: {{ .Values.webhook.livenessProbe.periodSeconds }} - timeoutSeconds: {{ .Values.webhook.livenessProbe.timeoutSeconds }} - successThreshold: {{ .Values.webhook.livenessProbe.successThreshold }} - failureThreshold: {{ .Values.webhook.livenessProbe.failureThreshold }} - readinessProbe: - httpGet: - path: /healthz - {{- if $config.healthzPort }} - port: {{ $config.healthzPort }} - {{- else }} - port: 6080 - {{- end }} - scheme: HTTP - initialDelaySeconds: {{ .Values.webhook.readinessProbe.initialDelaySeconds }} - periodSeconds: {{ .Values.webhook.readinessProbe.periodSeconds }} - timeoutSeconds: {{ .Values.webhook.readinessProbe.timeoutSeconds }} - successThreshold: {{ .Values.webhook.readinessProbe.successThreshold }} - failureThreshold: {{ .Values.webhook.readinessProbe.failureThreshold }} - {{- with .Values.webhook.containerSecurityContext }} - securityContext: - {{- toYaml . | nindent 12 }} - {{- end }} - env: - - name: POD_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - {{- with .Values.webhook.resources }} - resources: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- if or .Values.webhook.config .Values.webhook.volumeMounts }} - volumeMounts: - {{- if .Values.webhook.config }} - - name: config - mountPath: /var/cert-manager/config - {{- end }} - {{- with .Values.webhook.volumeMounts }} - {{- toYaml . | nindent 12 }} - {{- end }} - {{- end }} - {{- with .Values.webhook.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.webhook.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.webhook.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.webhook.topologySpreadConstraints }} - topologySpreadConstraints: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- if or .Values.webhook.config .Values.webhook.volumes }} - volumes: - {{- if .Values.webhook.config }} - - name: config - configMap: - name: {{ include "webhook.fullname" . }} - {{- end }} - {{- with .Values.webhook.volumes }} - {{- toYaml . | nindent 8 }} - {{- end }} - {{- end }} diff --git a/charts/cert-manager/cert-manager/templates/webhook-mutating-webhook.yaml b/charts/cert-manager/cert-manager/templates/webhook-mutating-webhook.yaml deleted file mode 100644 index 9ea2977..0000000 --- a/charts/cert-manager/cert-manager/templates/webhook-mutating-webhook.yaml +++ /dev/null @@ -1,48 +0,0 @@ -apiVersion: admissionregistration.k8s.io/v1 -kind: MutatingWebhookConfiguration -metadata: - name: {{ include "webhook.fullname" . }} - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 4 }} - annotations: - cert-manager.io/inject-ca-from-secret: {{ printf "%s/%s-ca" (include "cert-manager.namespace" .) (include "webhook.fullname" .) | quote }} - {{- with .Values.webhook.mutatingWebhookConfigurationAnnotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -webhooks: - - name: webhook.cert-manager.io - {{- with .Values.webhook.mutatingWebhookConfiguration.namespaceSelector }} - namespaceSelector: - {{- toYaml . | nindent 6 }} - {{- end }} - rules: - - apiGroups: - - "cert-manager.io" - apiVersions: - - "v1" - operations: - - CREATE - resources: - - "certificaterequests" - admissionReviewVersions: ["v1"] - # This webhook only accepts v1 cert-manager resources. - # Equivalent matchPolicy ensures that non-v1 resource requests are sent to - # this webhook (after the resources have been converted to v1). - matchPolicy: Equivalent - timeoutSeconds: {{ .Values.webhook.timeoutSeconds }} - failurePolicy: Fail - # Only include 'sideEffects' field in Kubernetes 1.12+ - sideEffects: None - clientConfig: - {{- if .Values.webhook.url.host }} - url: https://{{ .Values.webhook.url.host }}/mutate - {{- else }} - service: - name: {{ template "webhook.fullname" . }} - namespace: {{ include "cert-manager.namespace" . }} - path: /mutate - {{- end }} \ No newline at end of file diff --git a/charts/cert-manager/cert-manager/templates/webhook-poddisruptionbudget.yaml b/charts/cert-manager/cert-manager/templates/webhook-poddisruptionbudget.yaml deleted file mode 100644 index ab2a481..0000000 --- a/charts/cert-manager/cert-manager/templates/webhook-poddisruptionbudget.yaml +++ /dev/null @@ -1,29 +0,0 @@ -{{- if .Values.webhook.podDisruptionBudget.enabled }} -apiVersion: policy/v1 -kind: PodDisruptionBudget -metadata: - name: {{ include "webhook.fullname" . }} - namespace: {{ include "cert-manager.namespace" . }} - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 4 }} -spec: - selector: - matchLabels: - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - - {{- if not (or (hasKey .Values.webhook.podDisruptionBudget "minAvailable") (hasKey .Values.webhook.podDisruptionBudget "maxUnavailable")) }} - minAvailable: 1 # Default value because minAvailable and maxUnavailable are not set - {{- end }} - {{- if hasKey .Values.webhook.podDisruptionBudget "minAvailable" }} - minAvailable: {{ .Values.webhook.podDisruptionBudget.minAvailable }} - {{- end }} - {{- if hasKey .Values.webhook.podDisruptionBudget "maxUnavailable" }} - maxUnavailable: {{ .Values.webhook.podDisruptionBudget.maxUnavailable }} - {{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/webhook-psp-clusterrole.yaml b/charts/cert-manager/cert-manager/templates/webhook-psp-clusterrole.yaml deleted file mode 100644 index f6fa4c5..0000000 --- a/charts/cert-manager/cert-manager/templates/webhook-psp-clusterrole.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- if .Values.global.podSecurityPolicy.enabled }} -kind: ClusterRole -apiVersion: rbac.authorization.k8s.io/v1 -metadata: - name: {{ template "webhook.fullname" . }}-psp - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 4 }} -rules: -- apiGroups: ['policy'] - resources: ['podsecuritypolicies'] - verbs: ['use'] - resourceNames: - - {{ template "webhook.fullname" . }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/webhook-psp-clusterrolebinding.yaml b/charts/cert-manager/cert-manager/templates/webhook-psp-clusterrolebinding.yaml deleted file mode 100644 index 858df8f..0000000 --- a/charts/cert-manager/cert-manager/templates/webhook-psp-clusterrolebinding.yaml +++ /dev/null @@ -1,20 +0,0 @@ -{{- if .Values.global.podSecurityPolicy.enabled }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ template "webhook.fullname" . }}-psp - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ template "webhook.fullname" . }}-psp -subjects: - - kind: ServiceAccount - name: {{ template "webhook.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/webhook-psp.yaml b/charts/cert-manager/cert-manager/templates/webhook-psp.yaml deleted file mode 100644 index 4d5d959..0000000 --- a/charts/cert-manager/cert-manager/templates/webhook-psp.yaml +++ /dev/null @@ -1,54 +0,0 @@ -{{- if .Values.global.podSecurityPolicy.enabled }} -apiVersion: policy/v1beta1 -kind: PodSecurityPolicy -metadata: - name: {{ template "webhook.fullname" . }} - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 4 }} - annotations: - seccomp.security.alpha.kubernetes.io/allowedProfileNames: 'docker/default' - seccomp.security.alpha.kubernetes.io/defaultProfileName: 'docker/default' - {{- if .Values.global.podSecurityPolicy.useAppArmor }} - apparmor.security.beta.kubernetes.io/allowedProfileNames: 'runtime/default' - apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' - {{- end }} -spec: - privileged: false - allowPrivilegeEscalation: false - allowedCapabilities: [] # default set of capabilities are implicitly allowed - volumes: - - 'configMap' - - 'emptyDir' - - 'projected' - - 'secret' - - 'downwardAPI' - hostNetwork: {{ .Values.webhook.hostNetwork }} - {{- if .Values.webhook.hostNetwork }} - hostPorts: - - max: {{ .Values.webhook.securePort }} - min: {{ .Values.webhook.securePort }} - {{- end }} - hostIPC: false - hostPID: false - runAsUser: - rule: 'MustRunAs' - ranges: - - min: 1000 - max: 1000 - seLinux: - rule: 'RunAsAny' - supplementalGroups: - rule: 'MustRunAs' - ranges: - - min: 1000 - max: 1000 - fsGroup: - rule: 'MustRunAs' - ranges: - - min: 1000 - max: 1000 -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/webhook-rbac.yaml b/charts/cert-manager/cert-manager/templates/webhook-rbac.yaml deleted file mode 100644 index b075ffd..0000000 --- a/charts/cert-manager/cert-manager/templates/webhook-rbac.yaml +++ /dev/null @@ -1,83 +0,0 @@ -{{- if .Values.global.rbac.create }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: {{ template "webhook.fullname" . }}:dynamic-serving - namespace: {{ include "cert-manager.namespace" . }} - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 4 }} -rules: -- apiGroups: [""] - resources: ["secrets"] - resourceNames: - - '{{ template "webhook.fullname" . }}-ca' - verbs: ["get", "list", "watch", "update"] -# It's not possible to grant CREATE permission on a single resourceName. -- apiGroups: [""] - resources: ["secrets"] - verbs: ["create"] ---- - -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: {{ template "webhook.fullname" . }}:dynamic-serving - namespace: {{ include "cert-manager.namespace" . }} - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: {{ template "webhook.fullname" . }}:dynamic-serving -subjects: -- apiGroup: "" - kind: ServiceAccount - name: {{ template "webhook.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - ---- - -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ template "webhook.fullname" . }}:subjectaccessreviews - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 4 }} -rules: -- apiGroups: ["authorization.k8s.io"] - resources: ["subjectaccessreviews"] - verbs: ["create"] ---- - -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ template "webhook.fullname" . }}:subjectaccessreviews - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ template "webhook.fullname" . }}:subjectaccessreviews -subjects: -- apiGroup: "" - kind: ServiceAccount - name: {{ template "webhook.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/webhook-service.yaml b/charts/cert-manager/cert-manager/templates/webhook-service.yaml deleted file mode 100644 index 86d47f1..0000000 --- a/charts/cert-manager/cert-manager/templates/webhook-service.yaml +++ /dev/null @@ -1,38 +0,0 @@ -apiVersion: v1 -kind: Service -metadata: - name: {{ template "webhook.fullname" . }} - namespace: {{ include "cert-manager.namespace" . }} -{{- with .Values.webhook.serviceAnnotations }} - annotations: -{{ toYaml . | indent 4 }} -{{- end }} - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 4 }} - {{- with .Values.webhook.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - type: {{ .Values.webhook.serviceType }} - {{- if .Values.webhook.serviceIPFamilyPolicy }} - ipFamilyPolicy: {{ .Values.webhook.serviceIPFamilyPolicy }} - {{- end }} - {{- if .Values.webhook.serviceIPFamilies }} - ipFamilies: {{ .Values.webhook.serviceIPFamilies | toYaml | nindent 2 }} - {{- end }} - {{- with .Values.webhook.loadBalancerIP }} - loadBalancerIP: {{ . }} - {{- end }} - ports: - - name: https - port: 443 - protocol: TCP - targetPort: "https" - selector: - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" diff --git a/charts/cert-manager/cert-manager/templates/webhook-serviceaccount.yaml b/charts/cert-manager/cert-manager/templates/webhook-serviceaccount.yaml deleted file mode 100644 index dff5c06..0000000 --- a/charts/cert-manager/cert-manager/templates/webhook-serviceaccount.yaml +++ /dev/null @@ -1,25 +0,0 @@ -{{- if .Values.webhook.serviceAccount.create }} -apiVersion: v1 -kind: ServiceAccount -automountServiceAccountToken: {{ .Values.webhook.serviceAccount.automountServiceAccountToken }} -metadata: - name: {{ template "webhook.serviceAccountName" . }} - namespace: {{ include "cert-manager.namespace" . }} - {{- with .Values.webhook.serviceAccount.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 4 }} - {{- with .Values.webhook.serviceAccount.labels }} - {{ toYaml . | nindent 4 }} - {{- end }} -{{- with .Values.global.imagePullSecrets }} -imagePullSecrets: - {{- toYaml . | nindent 2 }} -{{- end }} -{{- end }} diff --git a/charts/cert-manager/cert-manager/templates/webhook-validating-webhook.yaml b/charts/cert-manager/cert-manager/templates/webhook-validating-webhook.yaml deleted file mode 100644 index 76235fd..0000000 --- a/charts/cert-manager/cert-manager/templates/webhook-validating-webhook.yaml +++ /dev/null @@ -1,49 +0,0 @@ -apiVersion: admissionregistration.k8s.io/v1 -kind: ValidatingWebhookConfiguration -metadata: - name: {{ include "webhook.fullname" . }} - labels: - app: {{ include "webhook.name" . }} - app.kubernetes.io/name: {{ include "webhook.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/component: "webhook" - {{- include "labels" . | nindent 4 }} - annotations: - cert-manager.io/inject-ca-from-secret: {{ printf "%s/%s-ca" (include "cert-manager.namespace" .) (include "webhook.fullname" .) | quote}} - {{- with .Values.webhook.validatingWebhookConfigurationAnnotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -webhooks: - - name: webhook.cert-manager.io - {{- with .Values.webhook.validatingWebhookConfiguration.namespaceSelector }} - namespaceSelector: - {{- toYaml . | nindent 6 }} - {{- end }} - rules: - - apiGroups: - - "cert-manager.io" - - "acme.cert-manager.io" - apiVersions: - - "v1" - operations: - - CREATE - - UPDATE - resources: - - "*/*" - admissionReviewVersions: ["v1"] - # This webhook only accepts v1 cert-manager resources. - # Equivalent matchPolicy ensures that non-v1 resource requests are sent to - # this webhook (after the resources have been converted to v1). - matchPolicy: Equivalent - timeoutSeconds: {{ .Values.webhook.timeoutSeconds }} - failurePolicy: Fail - sideEffects: None - clientConfig: - {{- if .Values.webhook.url.host }} - url: https://{{ .Values.webhook.url.host }}/validate - {{- else }} - service: - name: {{ template "webhook.fullname" . }} - namespace: {{ include "cert-manager.namespace" . }} - path: /validate - {{- end }} diff --git a/charts/cert-manager/cert-manager/values.yaml b/charts/cert-manager/cert-manager/values.yaml deleted file mode 100644 index 7630c04..0000000 --- a/charts/cert-manager/cert-manager/values.yaml +++ /dev/null @@ -1,1349 +0,0 @@ -# +docs:section=Global - -# Default values for cert-manager. -# This is a YAML-formatted file. -# Declare variables to be passed into your templates. -global: - # Reference to one or more secrets to be used when pulling images. - # For more information, see [Pull an Image from a Private Registry](https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/). - # - # For example: - # imagePullSecrets: - # - name: "image-pull-secret" - imagePullSecrets: [] - - # Labels to apply to all resources. - # Please note that this does not add labels to the resources created dynamically by the controllers. - # For these resources, you have to add the labels in the template in the cert-manager custom resource: - # For example, podTemplate/ ingressTemplate in ACMEChallengeSolverHTTP01Ingress - # For more information, see the [cert-manager documentation](https://cert-manager.io/docs/reference/api-docs/#acme.cert-manager.io/v1.ACMEChallengeSolverHTTP01Ingress). - # For example, secretTemplate in CertificateSpec - # For more information, see the [cert-manager documentation](https://cert-manager.io/docs/reference/api-docs/#cert-manager.io/v1.CertificateSpec). - commonLabels: {} - - # The number of old ReplicaSets to retain to allow rollback (if not set, the default Kubernetes value is set to 10). - # +docs:property - # revisionHistoryLimit: 1 - - # The optional priority class to be used for the cert-manager pods. - priorityClassName: "" - - rbac: - # Create required ClusterRoles and ClusterRoleBindings for cert-manager. - create: true - # Aggregate ClusterRoles to Kubernetes default user-facing roles. For more information, see [User-facing roles](https://kubernetes.io/docs/reference/access-authn-authz/rbac/#user-facing-roles) - aggregateClusterRoles: true - - podSecurityPolicy: - # Create PodSecurityPolicy for cert-manager. - # - # Note that PodSecurityPolicy was deprecated in Kubernetes 1.21 and removed in Kubernetes 1.25. - enabled: false - # Configure the PodSecurityPolicy to use AppArmor. - useAppArmor: true - - # Set the verbosity of cert-manager. A range of 0 - 6, with 6 being the most verbose. - logLevel: 2 - - leaderElection: - # Override the namespace used for the leader election lease. - namespace: "kube-system" - - # The duration that non-leader candidates will wait after observing a - # leadership renewal until attempting to acquire leadership of a led but - # unrenewed leader slot. This is effectively the maximum duration that a - # leader can be stopped before it is replaced by another candidate. - # +docs:property - # leaseDuration: 60s - - # The interval between attempts by the acting master to renew a leadership - # slot before it stops leading. This must be less than or equal to the - # lease duration. - # +docs:property - # renewDeadline: 40s - - # The duration the clients should wait between attempting acquisition and - # renewal of a leadership. - # +docs:property - # retryPeriod: 15s - -# This option is equivalent to setting crds.enabled=true and crds.keep=true. -# Deprecated: use crds.enabled and crds.keep instead. -installCRDs: false - -crds: - # This option decides if the CRDs should be installed - # as part of the Helm installation. - enabled: false - - # This option makes it so that the "helm.sh/resource-policy": keep - # annotation is added to the CRD. This will prevent Helm from uninstalling - # the CRD when the Helm release is uninstalled. - # WARNING: when the CRDs are removed, all cert-manager custom resources - # (Certificates, Issuers, ...) will be removed too by the garbage collector. - keep: true - -# +docs:section=Controller - -# The number of replicas of the cert-manager controller to run. -# -# The default is 1, but in production set this to 2 or 3 to provide high -# availability. -# -# If `replicas > 1`, consider setting `podDisruptionBudget.enabled=true`. -# -# Note that cert-manager uses leader election to ensure that there can -# only be a single instance active at a time. -replicaCount: 1 - -# Deployment update strategy for the cert-manager controller deployment. -# For more information, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#strategy). -# -# For example: -# strategy: -# type: RollingUpdate -# rollingUpdate: -# maxSurge: 0 -# maxUnavailable: 1 -strategy: {} - -podDisruptionBudget: - # Enable or disable the PodDisruptionBudget resource. - # - # This prevents downtime during voluntary disruptions such as during a Node upgrade. - # For example, the PodDisruptionBudget will block `kubectl drain` - # if it is used on the Node where the only remaining cert-manager - # Pod is currently running. - enabled: false - - # This configures the minimum available pods for disruptions. It can either be set to - # an integer (e.g. 1) or a percentage value (e.g. 25%). - # It cannot be used if `maxUnavailable` is set. - # +docs:property - # minAvailable: 1 - - # This configures the maximum unavailable pods for disruptions. It can either be set to - # an integer (e.g. 1) or a percentage value (e.g. 25%). - # it cannot be used if `minAvailable` is set. - # +docs:property - # maxUnavailable: 1 - -# A comma-separated list of feature gates that should be enabled on the -# controller pod. -featureGates: "" - -# The maximum number of challenges that can be scheduled as 'processing' at once. -maxConcurrentChallenges: 60 - -image: - # The container registry to pull the manager image from. - # +docs:property - # registry: quay.io - - # The container image for the cert-manager controller. - # +docs:property - repository: quay.io/jetstack/cert-manager-controller - - # Override the image tag to deploy by setting this variable. - # If no value is set, the chart's appVersion is used. - # +docs:property - # tag: vX.Y.Z - - # Setting a digest will override any tag. - # +docs:property - # digest: sha256:0e072dddd1f7f8fc8909a2ca6f65e76c5f0d2fcfb8be47935ae3457e8bbceb20 - - # Kubernetes imagePullPolicy on Deployment. - pullPolicy: IfNotPresent - -# Override the namespace used to store DNS provider credentials etc. for ClusterIssuer -# resources. By default, the same namespace as cert-manager is deployed within is -# used. This namespace will not be automatically created by the Helm chart. -clusterResourceNamespace: "" - -# This namespace allows you to define where the services are installed into. -# If not set then they use the namespace of the release. -# This is helpful when installing cert manager as a chart dependency (sub chart). -namespace: "" - -serviceAccount: - # Specifies whether a service account should be created. - create: true - - # The name of the service account to use. - # If not set and create is true, a name is generated using the fullname template. - # +docs:property - # name: "" - - # Optional additional annotations to add to the controller's Service Account. - # +docs:property - # annotations: {} - - # Optional additional labels to add to the controller's Service Account. - # +docs:property - # labels: {} - - # Automount API credentials for a Service Account. - automountServiceAccountToken: true - -# Automounting API credentials for a particular pod. -# +docs:property -# automountServiceAccountToken: true - -# When this flag is enabled, secrets will be automatically removed when the certificate resource is deleted. -enableCertificateOwnerRef: false - -# This property is used to configure options for the controller pod. -# This allows setting options that would usually be provided using flags. -# An APIVersion and Kind must be specified in your values.yaml file. -# Flags will override options that are set here. -# -# For example: -# config: -# apiVersion: controller.config.cert-manager.io/v1alpha1 -# kind: ControllerConfiguration -# logging: -# verbosity: 2 -# format: text -# leaderElectionConfig: -# namespace: kube-system -# kubernetesAPIQPS: 9000 -# kubernetesAPIBurst: 9000 -# numberOfConcurrentWorkers: 200 -# featureGates: -# AdditionalCertificateOutputFormats: true -# DisallowInsecureCSRUsageDefinition: true -# ExperimentalCertificateSigningRequestControllers: true -# ExperimentalGatewayAPISupport: true -# LiteralCertificateSubject: true -# SecretsFilteredCaching: true -# ServerSideApply: true -# StableCertificateRequestName: true -# UseCertificateRequestBasicConstraints: true -# ValidateCAA: true -# metricsTLSConfig: -# dynamic: -# secretNamespace: "cert-manager" -# secretName: "cert-manager-metrics-ca" -# dnsNames: -# - cert-manager-metrics -# - cert-manager-metrics.cert-manager -# - cert-manager-metrics.cert-manager.svc -config: {} - -# Setting Nameservers for DNS01 Self Check. -# For more information, see the [cert-manager documentation](https://cert-manager.io/docs/configuration/acme/dns01/#setting-nameservers-for-dns01-self-check). - -# A comma-separated string with the host and port of the recursive nameservers cert-manager should query. -dns01RecursiveNameservers: "" - -# Forces cert-manager to use only the recursive nameservers for verification. -# Enabling this option could cause the DNS01 self check to take longer owing to caching performed by the recursive nameservers. -dns01RecursiveNameserversOnly: false - -# Option to disable cert-manager's build-in auto-approver. The auto-approver -# approves all CertificateRequests that reference issuers matching the 'approveSignerNames' -# option. This 'disableAutoApproval' option is useful when you want to make all approval decisions -# using a different approver (like approver-policy - https://github.com/cert-manager/approver-policy). -disableAutoApproval: false - -# List of signer names that cert-manager will approve by default. CertificateRequests -# referencing these signer names will be auto-approved by cert-manager. Defaults to just -# approving the cert-manager.io Issuer and ClusterIssuer issuers. When set to an empty -# array, ALL issuers will be auto-approved by cert-manager. To disable the auto-approval, -# because eg. you are using approver-policy, you can enable 'disableAutoApproval'. -# ref: https://cert-manager.io/docs/concepts/certificaterequest/#approval -# +docs:property -approveSignerNames: -- issuers.cert-manager.io/* -- clusterissuers.cert-manager.io/* - -# Additional command line flags to pass to cert-manager controller binary. -# To see all available flags run `docker run quay.io/jetstack/cert-manager-controller: --help`. -# -# Use this flag to enable or disable arbitrary controllers. For example, to disable the CertificiateRequests approver. -# -# For example: -# extraArgs: -# - --controllers=*,-certificaterequests-approver -extraArgs: [] - -# Additional environment variables to pass to cert-manager controller binary. -extraEnv: [] -# - name: SOME_VAR -# value: 'some value' - -# Resources to provide to the cert-manager controller pod. -# -# For example: -# requests: -# cpu: 10m -# memory: 32Mi -# -# For more information, see [Resource Management for Pods and Containers](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/). -resources: {} - -# Pod Security Context. -# For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). -# +docs:property -securityContext: - runAsNonRoot: true - seccompProfile: - type: RuntimeDefault - -# Container Security Context to be set on the controller component container. -# For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). -# +docs:property -containerSecurityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - readOnlyRootFilesystem: true - -# Additional volumes to add to the cert-manager controller pod. -volumes: [] - -# Additional volume mounts to add to the cert-manager controller container. -volumeMounts: [] - -# Optional additional annotations to add to the controller Deployment. -# +docs:property -# deploymentAnnotations: {} - -# Optional additional annotations to add to the controller Pods. -# +docs:property -# podAnnotations: {} - -# Optional additional labels to add to the controller Pods. -podLabels: {} - -# Optional annotations to add to the controller Service. -# +docs:property -# serviceAnnotations: {} - -# Optional additional labels to add to the controller Service. -# +docs:property -# serviceLabels: {} - -# Optionally set the IP family policy for the controller Service to configure dual-stack; see [Configure dual-stack](https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services). -# +docs:property -# serviceIPFamilyPolicy: "" - -# Optionally set the IP families for the controller Service that should be supported, in the order in which they should be applied to ClusterIP. Can be IPv4 and/or IPv6. -# +docs:property -# serviceIPFamilies: [] - -# Optional DNS settings. These are useful if you have a public and private DNS zone for -# the same domain on Route 53. The following is an example of ensuring -# cert-manager can access an ingress or DNS TXT records at all times. -# Note that this requires Kubernetes 1.10 or `CustomPodDNS` feature gate enabled for -# the cluster to work. - -# Pod DNS policy. -# For more information, see [Pod's DNS Policy](https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/#pod-s-dns-policy). -# +docs:property -# podDnsPolicy: "None" - -# Pod DNS configuration. The podDnsConfig field is optional and can work with any podDnsPolicy -# settings. However, when a Pod's dnsPolicy is set to "None", the dnsConfig field has to be specified. -# For more information, see [Pod's DNS Config](https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/#pod-dns-config). -# +docs:property -# podDnsConfig: -# nameservers: -# - "1.1.1.1" -# - "8.8.8.8" - -# Optional hostAliases for cert-manager-controller pods. May be useful when performing ACME DNS-01 self checks. -hostAliases: [] -# - ip: 127.0.0.1 -# hostnames: -# - foo.local -# - bar.local -# - ip: 10.1.2.3 -# hostnames: -# - foo.remote -# - bar.remote - -# The nodeSelector on Pods tells Kubernetes to schedule Pods on the nodes with -# matching labels. -# For more information, see [Assigning Pods to Nodes](https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/). -# -# This default ensures that Pods are only scheduled to Linux nodes. -# It prevents Pods being scheduled to Windows nodes in a mixed OS cluster. -# +docs:property -nodeSelector: - kubernetes.io/os: linux - -# +docs:ignore -ingressShim: {} - - # Optional default issuer to use for ingress resources. - # +docs:property=ingressShim.defaultIssuerName - # defaultIssuerName: "" - - # Optional default issuer kind to use for ingress resources. - # +docs:property=ingressShim.defaultIssuerKind - # defaultIssuerKind: "" - - # Optional default issuer group to use for ingress resources. - # +docs:property=ingressShim.defaultIssuerGroup - # defaultIssuerGroup: "" - -# Use these variables to configure the HTTP_PROXY environment variables. - -# Configures the HTTP_PROXY environment variable where a HTTP proxy is required. -# +docs:property -# http_proxy: "http://proxy:8080" - -# Configures the HTTPS_PROXY environment variable where a HTTP proxy is required. -# +docs:property -# https_proxy: "https://proxy:8080" - -# Configures the NO_PROXY environment variable where a HTTP proxy is required, -# but certain domains should be excluded. -# +docs:property -# no_proxy: 127.0.0.1,localhost - - -# A Kubernetes Affinity, if required. For more information, see [Affinity v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#affinity-v1-core). -# -# For example: -# affinity: -# nodeAffinity: -# requiredDuringSchedulingIgnoredDuringExecution: -# nodeSelectorTerms: -# - matchExpressions: -# - key: foo.bar.com/role -# operator: In -# values: -# - master -affinity: {} - -# A list of Kubernetes Tolerations, if required. For more information, see [Toleration v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#toleration-v1-core). -# -# For example: -# tolerations: -# - key: foo.bar.com/role -# operator: Equal -# value: master -# effect: NoSchedule -tolerations: [] - -# A list of Kubernetes TopologySpreadConstraints, if required. For more information, see [Topology spread constraint v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#topologyspreadconstraint-v1-core -# -# For example: -# topologySpreadConstraints: -# - maxSkew: 2 -# topologyKey: topology.kubernetes.io/zone -# whenUnsatisfiable: ScheduleAnyway -# labelSelector: -# matchLabels: -# app.kubernetes.io/instance: cert-manager -# app.kubernetes.io/component: controller -topologySpreadConstraints: [] - -# LivenessProbe settings for the controller container of the controller Pod. -# -# This is enabled by default, in order to enable the clock-skew liveness probe that -# restarts the controller in case of a skew between the system clock and the monotonic clock. -# LivenessProbe durations and thresholds are based on those used for the Kubernetes -# controller-manager. For more information see the following on the -# [Kubernetes GitHub repository](https://github.com/kubernetes/kubernetes/blob/806b30170c61a38fedd54cc9ede4cd6275a1ad3b/cmd/kubeadm/app/util/staticpod/utils.go#L241-L245) -# +docs:property -livenessProbe: - enabled: true - initialDelaySeconds: 10 - periodSeconds: 10 - timeoutSeconds: 15 - successThreshold: 1 - failureThreshold: 8 - -# enableServiceLinks indicates whether information about services should be -# injected into the pod's environment variables, matching the syntax of Docker -# links. -enableServiceLinks: false - -# +docs:section=Prometheus - -prometheus: - # Enable Prometheus monitoring for the cert-manager controller to use with the - # Prometheus Operator. If this option is enabled without enabling `prometheus.servicemonitor.enabled` or - # `prometheus.podmonitor.enabled`, 'prometheus.io' annotations are added to the cert-manager Deployment - # resources. Additionally, a service is created which can be used together - # with your own ServiceMonitor (managed outside of this Helm chart). - # Otherwise, a ServiceMonitor/ PodMonitor is created. - enabled: true - - servicemonitor: - # Create a ServiceMonitor to add cert-manager to Prometheus. - enabled: false - - # Specifies the `prometheus` label on the created ServiceMonitor. This is - # used when different Prometheus instances have label selectors matching - # different ServiceMonitors. - prometheusInstance: default - - # The target port to set on the ServiceMonitor. This must match the port that the - # cert-manager controller is listening on for metrics. - targetPort: 9402 - - # The path to scrape for metrics. - path: /metrics - - # The interval to scrape metrics. - interval: 60s - - # The timeout before a metrics scrape fails. - scrapeTimeout: 30s - - # Additional labels to add to the ServiceMonitor. - labels: {} - - # Additional annotations to add to the ServiceMonitor. - annotations: {} - - # Keep labels from scraped data, overriding server-side labels. - honorLabels: false - - # EndpointAdditionalProperties allows setting additional properties on the - # endpoint such as relabelings, metricRelabelings etc. - # - # For example: - # endpointAdditionalProperties: - # relabelings: - # - action: replace - # sourceLabels: - # - __meta_kubernetes_pod_node_name - # targetLabel: instance - # - # +docs:property - endpointAdditionalProperties: {} - - # Note that you can not enable both PodMonitor and ServiceMonitor as they are mutually exclusive. Enabling both will result in a error. - podmonitor: - # Create a PodMonitor to add cert-manager to Prometheus. - enabled: false - - # Specifies the `prometheus` label on the created PodMonitor. This is - # used when different Prometheus instances have label selectors matching - # different PodMonitors. - prometheusInstance: default - - # The path to scrape for metrics. - path: /metrics - - # The interval to scrape metrics. - interval: 60s - - # The timeout before a metrics scrape fails. - scrapeTimeout: 30s - - # Additional labels to add to the PodMonitor. - labels: {} - - # Additional annotations to add to the PodMonitor. - annotations: {} - - # Keep labels from scraped data, overriding server-side labels. - honorLabels: false - - # EndpointAdditionalProperties allows setting additional properties on the - # endpoint such as relabelings, metricRelabelings etc. - # - # For example: - # endpointAdditionalProperties: - # relabelings: - # - action: replace - # sourceLabels: - # - __meta_kubernetes_pod_node_name - # targetLabel: instance - # - # +docs:property - endpointAdditionalProperties: {} - -# +docs:section=Webhook - -webhook: - # Number of replicas of the cert-manager webhook to run. - # - # The default is 1, but in production set this to 2 or 3 to provide high - # availability. - # - # If `replicas > 1`, consider setting `webhook.podDisruptionBudget.enabled=true`. - replicaCount: 1 - - # The number of seconds the API server should wait for the webhook to respond before treating the call as a failure. - # The value must be between 1 and 30 seconds. For more information, see - # [Validating webhook configuration v1](https://kubernetes.io/docs/reference/kubernetes-api/extend-resources/validating-webhook-configuration-v1/). - # - # The default is set to the maximum value of 30 seconds as - # users sometimes report that the connection between the K8S API server and - # the cert-manager webhook server times out. - # If *this* timeout is reached, the error message will be "context deadline exceeded", - # which doesn't help the user diagnose what phase of the HTTPS connection timed out. - # For example, it could be during DNS resolution, TCP connection, TLS - # negotiation, HTTP negotiation, or slow HTTP response from the webhook - # server. - # By setting this timeout to its maximum value the underlying timeout error - # message has more chance of being returned to the end user. - timeoutSeconds: 30 - - # This is used to configure options for the webhook pod. - # This allows setting options that would usually be provided using flags. - # An APIVersion and Kind must be specified in your values.yaml file. - # Flags override options that are set here. - # - # For example: - # apiVersion: webhook.config.cert-manager.io/v1alpha1 - # kind: WebhookConfiguration - # # The port that the webhook listens on for requests. - # # In GKE private clusters, by default Kubernetes apiservers are allowed to - # # talk to the cluster nodes only on 443 and 10250. Configuring - # # securePort: 10250 therefore will work out-of-the-box without needing to add firewall - # # rules or requiring NET_BIND_SERVICE capabilities to bind port numbers < 1000. - # # This should be uncommented and set as a default by the chart once - # # the apiVersion of WebhookConfiguration graduates beyond v1alpha1. - # securePort: 10250 - config: {} - - # The update strategy for the cert-manager webhook deployment. - # For more information, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#strategy) - # - # For example: - # strategy: - # type: RollingUpdate - # rollingUpdate: - # maxSurge: 0 - # maxUnavailable: 1 - strategy: {} - - # Pod Security Context to be set on the webhook component Pod. - # For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). - # +docs:property - securityContext: - runAsNonRoot: true - seccompProfile: - type: RuntimeDefault - - # Container Security Context to be set on the webhook component container. - # For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). - # +docs:property - containerSecurityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - readOnlyRootFilesystem: true - - podDisruptionBudget: - # Enable or disable the PodDisruptionBudget resource. - # - # This prevents downtime during voluntary disruptions such as during a Node upgrade. - # For example, the PodDisruptionBudget will block `kubectl drain` - # if it is used on the Node where the only remaining cert-manager - # Pod is currently running. - enabled: false - - # This property configures the minimum available pods for disruptions. Can either be set to - # an integer (e.g. 1) or a percentage value (e.g. 25%). - # It cannot be used if `maxUnavailable` is set. - # +docs:property - # minAvailable: 1 - - # This property configures the maximum unavailable pods for disruptions. Can either be set to - # an integer (e.g. 1) or a percentage value (e.g. 25%). - # It cannot be used if `minAvailable` is set. - # +docs:property - # maxUnavailable: 1 - - # Optional additional annotations to add to the webhook Deployment. - # +docs:property - # deploymentAnnotations: {} - - # Optional additional annotations to add to the webhook Pods. - # +docs:property - # podAnnotations: {} - - # Optional additional annotations to add to the webhook Service. - # +docs:property - # serviceAnnotations: {} - - # Optional additional annotations to add to the webhook MutatingWebhookConfiguration. - # +docs:property - # mutatingWebhookConfigurationAnnotations: {} - - # Optional additional annotations to add to the webhook ValidatingWebhookConfiguration. - # +docs:property - # validatingWebhookConfigurationAnnotations: {} - - validatingWebhookConfiguration: - # Configure spec.namespaceSelector for validating webhooks. - # +docs:property - namespaceSelector: - matchExpressions: - - key: "cert-manager.io/disable-validation" - operator: "NotIn" - values: - - "true" - - mutatingWebhookConfiguration: - # Configure spec.namespaceSelector for mutating webhooks. - # +docs:property - namespaceSelector: {} - # matchLabels: - # key: value - # matchExpressions: - # - key: kubernetes.io/metadata.name - # operator: NotIn - # values: - # - kube-system - - - # Additional command line flags to pass to cert-manager webhook binary. - # To see all available flags run `docker run quay.io/jetstack/cert-manager-webhook: --help`. - extraArgs: [] - # Path to a file containing a WebhookConfiguration object used to configure the webhook. - # - --config= - - # Comma separated list of feature gates that should be enabled on the - # webhook pod. - featureGates: "" - - # Resources to provide to the cert-manager webhook pod. - # - # For example: - # requests: - # cpu: 10m - # memory: 32Mi - # - # For more information, see [Resource Management for Pods and Containers](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/). - resources: {} - - # Liveness probe values. - # For more information, see [Container probes](https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#container-probes). - # - # +docs:property - livenessProbe: - failureThreshold: 3 - initialDelaySeconds: 60 - periodSeconds: 10 - successThreshold: 1 - timeoutSeconds: 1 - - # Readiness probe values. - # For more information, see [Container probes](https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#container-probes). - # - # +docs:property - readinessProbe: - failureThreshold: 3 - initialDelaySeconds: 5 - periodSeconds: 5 - successThreshold: 1 - timeoutSeconds: 1 - - # The nodeSelector on Pods tells Kubernetes to schedule Pods on the nodes with - # matching labels. - # For more information, see [Assigning Pods to Nodes](https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/). - # - # This default ensures that Pods are only scheduled to Linux nodes. - # It prevents Pods being scheduled to Windows nodes in a mixed OS cluster. - # +docs:property - nodeSelector: - kubernetes.io/os: linux - - # A Kubernetes Affinity, if required. For more information, see [Affinity v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#affinity-v1-core). - # - # For example: - # affinity: - # nodeAffinity: - # requiredDuringSchedulingIgnoredDuringExecution: - # nodeSelectorTerms: - # - matchExpressions: - # - key: foo.bar.com/role - # operator: In - # values: - # - master - affinity: {} - - # A list of Kubernetes Tolerations, if required. For more information, see [Toleration v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#toleration-v1-core). - # - # For example: - # tolerations: - # - key: foo.bar.com/role - # operator: Equal - # value: master - # effect: NoSchedule - tolerations: [] - - # A list of Kubernetes TopologySpreadConstraints, if required. For more information, see [Topology spread constraint v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#topologyspreadconstraint-v1-core). - # - # For example: - # topologySpreadConstraints: - # - maxSkew: 2 - # topologyKey: topology.kubernetes.io/zone - # whenUnsatisfiable: ScheduleAnyway - # labelSelector: - # matchLabels: - # app.kubernetes.io/instance: cert-manager - # app.kubernetes.io/component: controller - topologySpreadConstraints: [] - - # Optional additional labels to add to the Webhook Pods. - podLabels: {} - - # Optional additional labels to add to the Webhook Service. - serviceLabels: {} - - # Optionally set the IP family policy for the controller Service to configure dual-stack; see [Configure dual-stack](https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services). - serviceIPFamilyPolicy: "" - - # Optionally set the IP families for the controller Service that should be supported, in the order in which they should be applied to ClusterIP. Can be IPv4 and/or IPv6. - serviceIPFamilies: [] - - image: - # The container registry to pull the webhook image from. - # +docs:property - # registry: quay.io - - # The container image for the cert-manager webhook - # +docs:property - repository: quay.io/jetstack/cert-manager-webhook - - # Override the image tag to deploy by setting this variable. - # If no value is set, the chart's appVersion will be used. - # +docs:property - # tag: vX.Y.Z - - # Setting a digest will override any tag - # +docs:property - # digest: sha256:0e072dddd1f7f8fc8909a2ca6f65e76c5f0d2fcfb8be47935ae3457e8bbceb20 - - # Kubernetes imagePullPolicy on Deployment. - pullPolicy: IfNotPresent - - serviceAccount: - # Specifies whether a service account should be created. - create: true - - # The name of the service account to use. - # If not set and create is true, a name is generated using the fullname template. - # +docs:property - # name: "" - - # Optional additional annotations to add to the controller's Service Account. - # +docs:property - # annotations: {} - - # Optional additional labels to add to the webhook's Service Account. - # +docs:property - # labels: {} - - # Automount API credentials for a Service Account. - automountServiceAccountToken: true - - # Automounting API credentials for a particular pod. - # +docs:property - # automountServiceAccountToken: true - - # The port that the webhook listens on for requests. - # In GKE private clusters, by default Kubernetes apiservers are allowed to - # talk to the cluster nodes only on 443 and 10250. Configuring - # securePort: 10250, therefore will work out-of-the-box without needing to add firewall - # rules or requiring NET_BIND_SERVICE capabilities to bind port numbers <1000. - securePort: 10250 - - # Specifies if the webhook should be started in hostNetwork mode. - # - # Required for use in some managed kubernetes clusters (such as AWS EKS) with custom - # CNI (such as calico), because control-plane managed by AWS cannot communicate - # with pods' IP CIDR and admission webhooks are not working - # - # Since the default port for the webhook conflicts with kubelet on the host - # network, `webhook.securePort` should be changed to an available port if - # running in hostNetwork mode. - hostNetwork: false - - # Specifies how the service should be handled. Useful if you want to expose the - # webhook outside of the cluster. In some cases, the control plane cannot - # reach internal services. - serviceType: ClusterIP - - # Specify the load balancer IP for the created service. - # +docs:property - # loadBalancerIP: "10.10.10.10" - - # Overrides the mutating webhook and validating webhook so they reach the webhook - # service using the `url` field instead of a service. - url: {} - # host: - - # Enables default network policies for webhooks. - networkPolicy: - # Create network policies for the webhooks. - enabled: false - - # Ingress rule for the webhook network policy. By default, it allows all - # inbound traffic. - # +docs:property - ingress: - - from: - - ipBlock: - cidr: 0.0.0.0/0 - - # Egress rule for the webhook network policy. By default, it allows all - # outbound traffic to ports 80 and 443, as well as DNS ports. - # +docs:property - egress: - - ports: - - port: 80 - protocol: TCP - - port: 443 - protocol: TCP - - port: 53 - protocol: TCP - - port: 53 - protocol: UDP - # On OpenShift and OKD, the Kubernetes API server listens on. - # port 6443. - - port: 6443 - protocol: TCP - to: - - ipBlock: - cidr: 0.0.0.0/0 - - # Additional volumes to add to the cert-manager controller pod. - volumes: [] - - # Additional volume mounts to add to the cert-manager controller container. - volumeMounts: [] - - # enableServiceLinks indicates whether information about services should be - # injected into the pod's environment variables, matching the syntax of Docker - # links. - enableServiceLinks: false - -# +docs:section=CA Injector - -cainjector: - # Create the CA Injector deployment - enabled: true - - # The number of replicas of the cert-manager cainjector to run. - # - # The default is 1, but in production set this to 2 or 3 to provide high - # availability. - # - # If `replicas > 1`, consider setting `cainjector.podDisruptionBudget.enabled=true`. - # - # Note that cert-manager uses leader election to ensure that there can - # only be a single instance active at a time. - replicaCount: 1 - - # This is used to configure options for the cainjector pod. - # It allows setting options that are usually provided via flags. - # An APIVersion and Kind must be specified in your values.yaml file. - # Flags override options that are set here. - # - # For example: - # apiVersion: cainjector.config.cert-manager.io/v1alpha1 - # kind: CAInjectorConfiguration - # logging: - # verbosity: 2 - # format: text - # leaderElectionConfig: - # namespace: kube-system - config: {} - - # Deployment update strategy for the cert-manager cainjector deployment. - # For more information, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#strategy). - # - # For example: - # strategy: - # type: RollingUpdate - # rollingUpdate: - # maxSurge: 0 - # maxUnavailable: 1 - strategy: {} - - # Pod Security Context to be set on the cainjector component Pod - # For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). - # +docs:property - securityContext: - runAsNonRoot: true - seccompProfile: - type: RuntimeDefault - - # Container Security Context to be set on the cainjector component container - # For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). - # +docs:property - containerSecurityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - readOnlyRootFilesystem: true - - podDisruptionBudget: - # Enable or disable the PodDisruptionBudget resource. - # - # This prevents downtime during voluntary disruptions such as during a Node upgrade. - # For example, the PodDisruptionBudget will block `kubectl drain` - # if it is used on the Node where the only remaining cert-manager - # Pod is currently running. - enabled: false - - # `minAvailable` configures the minimum available pods for disruptions. It can either be set to - # an integer (e.g. 1) or a percentage value (e.g. 25%). - # Cannot be used if `maxUnavailable` is set. - # +docs:property - # minAvailable: 1 - - # `maxUnavailable` configures the maximum unavailable pods for disruptions. It can either be set to - # an integer (e.g. 1) or a percentage value (e.g. 25%). - # Cannot be used if `minAvailable` is set. - # +docs:property - # maxUnavailable: 1 - - # Optional additional annotations to add to the cainjector Deployment. - # +docs:property - # deploymentAnnotations: {} - - # Optional additional annotations to add to the cainjector Pods. - # +docs:property - # podAnnotations: {} - - # Additional command line flags to pass to cert-manager cainjector binary. - # To see all available flags run `docker run quay.io/jetstack/cert-manager-cainjector: --help`. - extraArgs: [] - # Enable profiling for cainjector. - # - --enable-profiling=true - - # Comma separated list of feature gates that should be enabled on the - # cainjector pod. - featureGates: "" - - # Resources to provide to the cert-manager cainjector pod. - # - # For example: - # requests: - # cpu: 10m - # memory: 32Mi - # - # For more information, see [Resource Management for Pods and Containers](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/). - resources: {} - - - # The nodeSelector on Pods tells Kubernetes to schedule Pods on the nodes with - # matching labels. - # For more information, see [Assigning Pods to Nodes](https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/). - # - # This default ensures that Pods are only scheduled to Linux nodes. - # It prevents Pods being scheduled to Windows nodes in a mixed OS cluster. - # +docs:property - nodeSelector: - kubernetes.io/os: linux - - # A Kubernetes Affinity, if required. For more information, see [Affinity v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#affinity-v1-core). - # - # For example: - # affinity: - # nodeAffinity: - # requiredDuringSchedulingIgnoredDuringExecution: - # nodeSelectorTerms: - # - matchExpressions: - # - key: foo.bar.com/role - # operator: In - # values: - # - master - affinity: {} - - # A list of Kubernetes Tolerations, if required. For more information, see [Toleration v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#toleration-v1-core). - # - # For example: - # tolerations: - # - key: foo.bar.com/role - # operator: Equal - # value: master - # effect: NoSchedule - tolerations: [] - - # A list of Kubernetes TopologySpreadConstraints, if required. For more information, see [Topology spread constraint v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#topologyspreadconstraint-v1-core). - # - # For example: - # topologySpreadConstraints: - # - maxSkew: 2 - # topologyKey: topology.kubernetes.io/zone - # whenUnsatisfiable: ScheduleAnyway - # labelSelector: - # matchLabels: - # app.kubernetes.io/instance: cert-manager - # app.kubernetes.io/component: controller - topologySpreadConstraints: [] - - # Optional additional labels to add to the CA Injector Pods. - podLabels: {} - - image: - # The container registry to pull the cainjector image from. - # +docs:property - # registry: quay.io - - # The container image for the cert-manager cainjector - # +docs:property - repository: quay.io/jetstack/cert-manager-cainjector - - # Override the image tag to deploy by setting this variable. - # If no value is set, the chart's appVersion will be used. - # +docs:property - # tag: vX.Y.Z - - # Setting a digest will override any tag. - # +docs:property - # digest: sha256:0e072dddd1f7f8fc8909a2ca6f65e76c5f0d2fcfb8be47935ae3457e8bbceb20 - - # Kubernetes imagePullPolicy on Deployment. - pullPolicy: IfNotPresent - - serviceAccount: - # Specifies whether a service account should be created. - create: true - - # The name of the service account to use. - # If not set and create is true, a name is generated using the fullname template - # +docs:property - # name: "" - - # Optional additional annotations to add to the controller's Service Account. - # +docs:property - # annotations: {} - - # Optional additional labels to add to the cainjector's Service Account. - # +docs:property - # labels: {} - - # Automount API credentials for a Service Account. - automountServiceAccountToken: true - - # Automounting API credentials for a particular pod. - # +docs:property - # automountServiceAccountToken: true - - # Additional volumes to add to the cert-manager controller pod. - volumes: [] - - # Additional volume mounts to add to the cert-manager controller container. - volumeMounts: [] - - # enableServiceLinks indicates whether information about services should be - # injected into the pod's environment variables, matching the syntax of Docker - # links. - enableServiceLinks: false - -# +docs:section=ACME Solver - -acmesolver: - image: - # The container registry to pull the acmesolver image from. - # +docs:property - # registry: quay.io - - # The container image for the cert-manager acmesolver. - # +docs:property - repository: quay.io/jetstack/cert-manager-acmesolver - - # Override the image tag to deploy by setting this variable. - # If no value is set, the chart's appVersion is used. - # +docs:property - # tag: vX.Y.Z - - # Setting a digest will override any tag. - # +docs:property - # digest: sha256:0e072dddd1f7f8fc8909a2ca6f65e76c5f0d2fcfb8be47935ae3457e8bbceb20 - - # Kubernetes imagePullPolicy on Deployment. - pullPolicy: IfNotPresent - -# +docs:section=Startup API Check -# This startupapicheck is a Helm post-install hook that waits for the webhook -# endpoints to become available. -# The check is implemented using a Kubernetes Job - if you are injecting mesh -# sidecar proxies into cert-manager pods, ensure that they -# are not injected into this Job's pod. Otherwise, the installation may time out -# owing to the Job never being completed because the sidecar proxy does not exit. -# For more information, see [this note](https://github.com/cert-manager/cert-manager/pull/4414). - -startupapicheck: - # Enables the startup api check. - enabled: true - - # Pod Security Context to be set on the startupapicheck component Pod. - # For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). - # +docs:property - securityContext: - runAsNonRoot: true - seccompProfile: - type: RuntimeDefault - - # Container Security Context to be set on the controller component container. - # For more information, see [Configure a Security Context for a Pod or Container](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/). - # +docs:property - containerSecurityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - readOnlyRootFilesystem: true - - # Timeout for 'kubectl check api' command. - timeout: 1m - - # Job backoffLimit - backoffLimit: 4 - - # Optional additional annotations to add to the startupapicheck Job. - # +docs:property - jobAnnotations: - helm.sh/hook: post-install - helm.sh/hook-weight: "1" - helm.sh/hook-delete-policy: before-hook-creation,hook-succeeded - - # Optional additional annotations to add to the startupapicheck Pods. - # +docs:property - # podAnnotations: {} - - # Additional command line flags to pass to startupapicheck binary. - # To see all available flags run `docker run quay.io/jetstack/cert-manager-startupapicheck: --help`. - # - # Verbose logging is enabled by default so that if startupapicheck fails, you - # can know what exactly caused the failure. Verbose logs include details of - # the webhook URL, IP address and TCP connect errors for example. - # +docs:property - extraArgs: - - -v - - # Resources to provide to the cert-manager controller pod. - # - # For example: - # requests: - # cpu: 10m - # memory: 32Mi - # - # For more information, see [Resource Management for Pods and Containers](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/). - resources: {} - - - # The nodeSelector on Pods tells Kubernetes to schedule Pods on the nodes with - # matching labels. - # For more information, see [Assigning Pods to Nodes](https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/). - # - # This default ensures that Pods are only scheduled to Linux nodes. - # It prevents Pods being scheduled to Windows nodes in a mixed OS cluster. - # +docs:property - nodeSelector: - kubernetes.io/os: linux - - # A Kubernetes Affinity, if required. For more information, see [Affinity v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#affinity-v1-core). - # For example: - # affinity: - # nodeAffinity: - # requiredDuringSchedulingIgnoredDuringExecution: - # nodeSelectorTerms: - # - matchExpressions: - # - key: foo.bar.com/role - # operator: In - # values: - # - master - affinity: {} - - # A list of Kubernetes Tolerations, if required. For more information, see [Toleration v1 core](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#toleration-v1-core). - # - # For example: - # tolerations: - # - key: foo.bar.com/role - # operator: Equal - # value: master - # effect: NoSchedule - tolerations: [] - - # Optional additional labels to add to the startupapicheck Pods. - podLabels: {} - - image: - # The container registry to pull the startupapicheck image from. - # +docs:property - # registry: quay.io - - # The container image for the cert-manager startupapicheck. - # +docs:property - repository: quay.io/jetstack/cert-manager-startupapicheck - - # Override the image tag to deploy by setting this variable. - # If no value is set, the chart's appVersion is used. - # +docs:property - # tag: vX.Y.Z - - # Setting a digest will override any tag. - # +docs:property - # digest: sha256:0e072dddd1f7f8fc8909a2ca6f65e76c5f0d2fcfb8be47935ae3457e8bbceb20 - - # Kubernetes imagePullPolicy on Deployment. - pullPolicy: IfNotPresent - - rbac: - # annotations for the startup API Check job RBAC and PSP resources. - # +docs:property - annotations: - helm.sh/hook: post-install - helm.sh/hook-weight: "-5" - helm.sh/hook-delete-policy: before-hook-creation,hook-succeeded - - # Automounting API credentials for a particular pod. - # +docs:property - # automountServiceAccountToken: true - - serviceAccount: - # Specifies whether a service account should be created. - create: true - - # The name of the service account to use. - # If not set and create is true, a name is generated using the fullname template. - # +docs:property - # name: "" - - # Optional additional annotations to add to the Job's Service Account. - # +docs:property - annotations: - helm.sh/hook: post-install - helm.sh/hook-weight: "-5" - helm.sh/hook-delete-policy: before-hook-creation,hook-succeeded - - # Automount API credentials for a Service Account. - # +docs:property - automountServiceAccountToken: true - - # Optional additional labels to add to the startupapicheck's Service Account. - # +docs:property - # labels: {} - - # Additional volumes to add to the cert-manager controller pod. - volumes: [] - - # Additional volume mounts to add to the cert-manager controller container. - volumeMounts: [] - - # enableServiceLinks indicates whether information about services should be - # injected into pod's environment variables, matching the syntax of Docker - # links. - enableServiceLinks: false - -# Create dynamic manifests via values. -# -# For example: -# extraObjects: -# - | -# apiVersion: v1 -# kind: ConfigMap -# metadata: -# name: '{{ template "cert-manager.name" . }}-extra-configmap' -extraObjects: [] diff --git a/charts/cert-manager/values-overrides.yaml b/charts/cert-manager/values-overrides.yaml deleted file mode 100644 index 6dc95ab..0000000 --- a/charts/cert-manager/values-overrides.yaml +++ /dev/null @@ -1,3 +0,0 @@ -fullnameOverride: cert-manager -crds: - enabled: true \ No newline at end of file diff --git a/charts/chisel-operator/chisel-operator/.helmignore b/charts/chisel-operator/chisel-operator/.helmignore deleted file mode 100644 index 0e8a0eb..0000000 --- a/charts/chisel-operator/chisel-operator/.helmignore +++ /dev/null @@ -1,23 +0,0 @@ -# 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 -*.orig -*~ -# Various IDEs -.project -.idea/ -*.tmproj -.vscode/ diff --git a/charts/chisel-operator/chisel-operator/Chart.yaml b/charts/chisel-operator/chisel-operator/Chart.yaml deleted file mode 100644 index 5c12ad5..0000000 --- a/charts/chisel-operator/chisel-operator/Chart.yaml +++ /dev/null @@ -1,6 +0,0 @@ -apiVersion: v2 -appVersion: v0.4.1 -description: Kubernetes Operator for deploying Chisel reverse proxies -name: chisel-operator -type: application -version: 0.1.0 diff --git a/charts/chisel-operator/chisel-operator/templates/_helpers.tpl b/charts/chisel-operator/chisel-operator/templates/_helpers.tpl deleted file mode 100644 index a13353f..0000000 --- a/charts/chisel-operator/chisel-operator/templates/_helpers.tpl +++ /dev/null @@ -1,62 +0,0 @@ -{{/* -Expand the name of the chart. -*/}} -{{- define "chisel-operator.name" -}} -{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }} -{{- end }} - -{{/* -Create a default fully qualified app name. -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -If release name contains chart name it will be used as a full name. -*/}} -{{- define "chisel-operator.fullname" -}} -{{- if .Values.fullnameOverride }} -{{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" }} -{{- else }} -{{- $name := default .Chart.Name .Values.nameOverride }} -{{- if contains $name .Release.Name }} -{{- .Release.Name | trunc 63 | trimSuffix "-" }} -{{- else }} -{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" }} -{{- end }} -{{- end }} -{{- end }} - -{{/* -Create chart name and version as used by the chart label. -*/}} -{{- define "chisel-operator.chart" -}} -{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} -{{- end }} - -{{/* -Common labels -*/}} -{{- define "chisel-operator.labels" -}} -helm.sh/chart: {{ include "chisel-operator.chart" . }} -{{ include "chisel-operator.selectorLabels" . }} -{{- if .Chart.AppVersion }} -app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} -{{- end }} -app.kubernetes.io/managed-by: {{ .Release.Service }} -{{- end }} - -{{/* -Selector labels -*/}} -{{- define "chisel-operator.selectorLabels" -}} -app.kubernetes.io/name: {{ include "chisel-operator.name" . }} -app.kubernetes.io/instance: {{ .Release.Name }} -{{- end }} - -{{/* -Create the name of the service account to use -*/}} -{{- define "chisel-operator.serviceAccountName" -}} -{{- if .Values.serviceAccount.create }} -{{- default (include "chisel-operator.fullname" .) .Values.serviceAccount.name }} -{{- else }} -{{- default "default" .Values.serviceAccount.name }} -{{- end }} -{{- end }} diff --git a/charts/chisel-operator/chisel-operator/templates/crds/exit-node-provisioner.yaml b/charts/chisel-operator/chisel-operator/templates/crds/exit-node-provisioner.yaml deleted file mode 100644 index 64ff698..0000000 --- a/charts/chisel-operator/chisel-operator/templates/crds/exit-node-provisioner.yaml +++ /dev/null @@ -1,101 +0,0 @@ - - -{{- if .Values.createCrds }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - name: exitnodeprovisioners.chisel-operator.io -spec: - group: chisel-operator.io - names: - categories: [] - kind: ExitNodeProvisioner - plural: exitnodeprovisioners - shortNames: [] - singular: exitnodeprovisioner - scope: Namespaced - versions: - - additionalPrinterColumns: [] - name: v1 - schema: - openAPIV3Schema: - description: Auto-generated derived type for ExitNodeProvisionerSpec via `CustomResource` - properties: - spec: - description: ExitNodeProvisioner is a custom resource that represents a Chisel exit node provisioner on a cloud provider. - oneOf: - - required: - - DigitalOcean - - required: - - Linode - - required: - - AWS - properties: - AWS: - properties: - auth: - description: Reference to a secret containing the AWS access key ID and secret access key, under the `access_key_id` and `secret_access_key` secret keys - type: string - region: - description: Region ID for the AWS region to provision the exit node in See https://docs.aws.amazon.com/AWSEC2/latest/UserGuide/using-regions-availability-zones.html - type: string - security_group: - description: Security group name to use for the exit node, uses the default security group if not specified - nullable: true - type: string - size: - default: t2.micro - description: Size for the EC2 instance See https://aws.amazon.com/ec2/instance-types/ - type: string - required: - - auth - - region - type: object - DigitalOcean: - properties: - auth: - description: Reference to a secret containing the DigitalOcean API token, under the `DIGITALOCEAN_TOKEN` secret key - type: string - region: - default: '' - description: Region ID of the DigitalOcean datacenter to provision the exit node in If empty, DigitalOcean will randomly select a region for you, which might not be what you want See https://slugs.do-api.dev/ - type: string - size: - default: s-1vcpu-1gb - description: Size for the DigitalOcean droplet See https://slugs.do-api.dev/ - type: string - ssh_fingerprints: - default: [] - description: SSH key fingerprints to add to the exit node - items: - type: string - type: array - required: - - auth - type: object - Linode: - properties: - auth: - description: Name of the secret containing the Linode API token, under the `LINODE_TOKEN` secret key - type: string - region: - description: Region ID of the Linode datacenter to provision the exit node in See https://api.linode.com/v4/regions - type: string - size: - default: g6-nanode-1 - description: Size for the Linode instance See https://api.linode.com/v4/linode/ - type: string - required: - - auth - - region - type: object - type: object - required: - - spec - title: ExitNodeProvisioner - type: object - served: true - storage: true - subresources: {} - -{{- end }} \ No newline at end of file diff --git a/charts/chisel-operator/chisel-operator/templates/crds/exit-node.yaml b/charts/chisel-operator/chisel-operator/templates/crds/exit-node.yaml deleted file mode 100644 index 7d679f2..0000000 --- a/charts/chisel-operator/chisel-operator/templates/crds/exit-node.yaml +++ /dev/null @@ -1,84 +0,0 @@ -{{- if .Values.createCrds -}} - -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - name: exitnodes.chisel-operator.io -spec: - group: chisel-operator.io - names: - categories: [] - kind: ExitNode - plural: exitnodes - shortNames: [] - singular: exitnode - scope: Namespaced - versions: - - additionalPrinterColumns: [] - name: v1 - schema: - openAPIV3Schema: - description: Auto-generated derived type for ExitNodeSpec via `CustomResource` - properties: - spec: - description: ExitNode is a custom resource that represents a Chisel exit node. It will be used as the reverse proxy for all services in the cluster. - properties: - auth: - description: Optional authentication secret name to connect to the control plane - nullable: true - type: string - chisel_image: - description: Optional value for the chisel client image used to connect to the chisel server If not provided, jpillora/chisel:latest is used - nullable: true - type: string - default_route: - default: false - description: Optional boolean value for whether to make the exit node the default route for the cluster If true, the exit node will be the default route for the cluster default value is false - type: boolean - external_host: - description: Optional real external hostname/IP of exit node If not provided, the host field will be used - nullable: true - type: string - fingerprint: - description: Optional but highly recommended fingerprint to perform host-key validation against the server's public key - nullable: true - type: string - host: - description: Hostname or IP address of the chisel server - type: string - port: - description: Control plane port of the chisel server - format: uint16 - minimum: 0.0 - type: integer - required: - - host - - port - type: object - status: - nullable: true - properties: - id: - nullable: true - type: string - ip: - type: string - name: - type: string - provider: - type: string - required: - - ip - - name - - provider - type: object - required: - - spec - title: ExitNode - type: object - served: true - storage: true - subresources: - status: {} - -{{- end -}} \ No newline at end of file diff --git a/charts/chisel-operator/chisel-operator/templates/deployment.yaml b/charts/chisel-operator/chisel-operator/templates/deployment.yaml deleted file mode 100644 index af36049..0000000 --- a/charts/chisel-operator/chisel-operator/templates/deployment.yaml +++ /dev/null @@ -1,49 +0,0 @@ -apiVersion: apps/v1 -kind: Deployment -metadata: - name: {{ include "chisel-operator.fullname" . }} - labels: - name: {{- include "chisel-operator.labels" . | nindent 4 }} -spec: - replicas: {{ .Values.replicaCount}} - selector: - matchLabels: - {{- include "chisel-operator.selectorLabels" . | nindent 6 }} - template: - metadata: - {{- with .Values.podAnnotations }} - annotations: - {{- toYaml . | nindent 8 }} - {{- end }} - labels: - {{- include "chisel-operator.selectorLabels" . | nindent 8 }} - spec: - {{- if .Values.serviceAccount.create }} - serviceAccountName: {{ include "chisel-operator.serviceAccountName" . }} - automountServiceAccountToken: true # This is required - {{- end }} - {{- with .Values.imagePullSecrets }} - imagePullSecrets: - {{- toYaml . | nindent 8 }} - {{- end }} - serviceAccountName: {{ include "chisel-operator.serviceAccountName" . }} - containers: - - name: {{ .Chart.Name }} - image: "{{ .Values.image.repository }}:{{ .Values.image.tag | default .Chart.AppVersion }}" - imagePullPolicy: {{ .Values.image.pullPolicy }} - resources: - {{- toYaml .Values.resources | nindent 12 }} - - {{- with .Values.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - diff --git a/charts/chisel-operator/chisel-operator/templates/serviceaccount.yaml b/charts/chisel-operator/chisel-operator/templates/serviceaccount.yaml deleted file mode 100644 index beac4f6..0000000 --- a/charts/chisel-operator/chisel-operator/templates/serviceaccount.yaml +++ /dev/null @@ -1,58 +0,0 @@ -{{- if .Values.serviceAccount.create -}} -apiVersion: v1 -kind: ServiceAccount -metadata: - name: {{ include "chisel-operator.serviceAccountName" . }} - labels: - {{- include "chisel-operator.labels" . | nindent 4 }} - {{- with .Values.serviceAccount.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ include "chisel-operator.serviceAccountName" . }} - labels: - {{- include "chisel-operator.labels" . | nindent 4 }} - {{- with .Values.serviceAccount.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -rules: - - apiGroups: ["apps"] - resources: ["deployments", "deployments/*"] - verbs: ["*"] - - apiGroups: [""] - resources: ["services", "services/status", "services/finalizers"] - verbs: ["get", "list", "watch", "update", "patch"] - - apiGroups: ["chisel-operator.io"] - resources: ["*"] - verbs: ["*"] - - apiGroups: [""] - resources: ["secrets"] - verbs: ["get", "list", "create", "update", "patch", "delete"] - ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ include "chisel-operator.serviceAccountName" . }} - labels: - {{- include "chisel-operator.labels" . | nindent 4 }} - {{- with .Values.serviceAccount.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -subjects: - - kind: ServiceAccount - name: {{ include "chisel-operator.serviceAccountName" . }} - namespace: {{ .Release.Namespace }} -roleRef: - kind: ClusterRole - name: {{ include "chisel-operator.serviceAccountName" . }} - apiGroup: rbac.authorization.k8s.io - - -{{- end }} diff --git a/charts/chisel-operator/chisel-operator/values.yaml b/charts/chisel-operator/chisel-operator/values.yaml deleted file mode 100644 index 0bab700..0000000 --- a/charts/chisel-operator/chisel-operator/values.yaml +++ /dev/null @@ -1,51 +0,0 @@ -# Default values for chisel-operator. -# This is a YAML-formatted file. -# Declare variables to be passed into your templates. - -replicaCount: 1 # Right now only 1 replica is supported -# LeaderElection and multiple replicas may be supported in the future. -# For now, we recommend running only 1 replica else Chisel Operator may constantly -# recreate resources, wasting your API resources and costing you money. - -image: - repository: ghcr.io/fyralabs/chisel-operator - pullPolicy: IfNotPresent - # Overrides the image tag whose default is the chart appVersion. - # tag: - -imagePullSecrets: [] -nameOverride: "" -fullnameOverride: "" - -serviceAccount: - # Specifies whether a service account should be created - create: true - # Annotations to add to the service account - annotations: {} - # The name of the service account to use. - # If not set and create is true, a name is generated using the fullname template - name: "" - -podAnnotations: {} - - -resources: {} - # We usually recommend not to specify default resources and to leave this as a conscious - # choice for the user. This also increases chances charts run on environments with little - # resources, such as Minikube. If you do want to specify resources, uncomment the following - # lines, adjust them as necessary, and remove the curly braces after 'resources:'. - # limits: - # cpu: 100m - # memory: 128Mi - # requests: - # cpu: 100m - # memory: 128Mi - -nodeSelector: {} - -tolerations: [] - -affinity: {} - -# Create CRDs for Chisel Operator -createCrds: true diff --git a/charts/chisel-operator/values-overrides.yaml b/charts/chisel-operator/values-overrides.yaml deleted file mode 100644 index 1ef6ac0..0000000 --- a/charts/chisel-operator/values-overrides.yaml +++ /dev/null @@ -1 +0,0 @@ -fullnameOverride: "chisel-operator" \ No newline at end of file diff --git a/charts/drupal/README.md b/charts/drupal/README.md deleted file mode 100644 index b07acbd..0000000 --- a/charts/drupal/README.md +++ /dev/null @@ -1,3 +0,0 @@ -## drupal notes - -To initialize drupal with all the modules we need, use the "Initialize Drupal" button in Tilt's UI. Or see the script it executes in ../../scripts/drupal-init.sh \ No newline at end of file diff --git a/charts/drupal/drupal/.helmignore b/charts/drupal/drupal/.helmignore deleted file mode 100644 index 207983f..0000000 --- a/charts/drupal/drupal/.helmignore +++ /dev/null @@ -1,25 +0,0 @@ -# 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 -# img folder -img/ -# Changelog -CHANGELOG.md diff --git a/charts/drupal/drupal/Chart.lock b/charts/drupal/drupal/Chart.lock deleted file mode 100644 index 9e7f677..0000000 --- a/charts/drupal/drupal/Chart.lock +++ /dev/null @@ -1,9 +0,0 @@ -dependencies: -- name: mariadb - repository: oci://registry-1.docker.io/bitnamicharts - version: 19.0.7 -- name: common - repository: oci://registry-1.docker.io/bitnamicharts - version: 2.24.0 -digest: sha256:1a41985df21415bc1a7fdd62b58daa1b60b31f2c2ebda4b9506e099a372b338c -generated: "2024-10-03T11:38:53.545931255Z" diff --git a/charts/drupal/drupal/Chart.yaml b/charts/drupal/drupal/Chart.yaml deleted file mode 100644 index 6826be6..0000000 --- a/charts/drupal/drupal/Chart.yaml +++ /dev/null @@ -1,42 +0,0 @@ -annotations: - category: CMS - images: | - - name: apache-exporter - image: docker.io/bitnami/apache-exporter:1.0.9-debian-12-r1 - - name: drupal - image: docker.io/bitnami/drupal:11.0.5-debian-12-r1 - - name: os-shell - image: docker.io/bitnami/os-shell:12-debian-12-r31 - licenses: Apache-2.0 -apiVersion: v2 -appVersion: 11.0.5 -dependencies: -- condition: mariadb.enabled - name: mariadb - repository: oci://registry-1.docker.io/bitnamicharts - version: 19.x.x -- name: common - repository: oci://registry-1.docker.io/bitnamicharts - tags: - - bitnami-common - version: 2.x.x -description: Drupal is one of the most versatile open source content management systems - in the world. It is pre-configured with the Ctools and Views modules, Drush and - Let's Encrypt auto-configuration support. -home: https://bitnami.com -icon: https://bitnami.com/assets/stacks/drupal/img/drupal-stack-220x234.png -keywords: -- drupal -- cms -- blog -- http -- web -- application -- php -maintainers: -- name: Broadcom, Inc. All Rights Reserved. - url: https://github.com/bitnami/charts -name: drupal -sources: -- https://github.com/bitnami/charts/tree/main/bitnami/drupal -version: 20.0.10 diff --git a/charts/drupal/drupal/README.md b/charts/drupal/drupal/README.md deleted file mode 100644 index 41108fc..0000000 --- a/charts/drupal/drupal/README.md +++ /dev/null @@ -1,586 +0,0 @@ - - -# Bitnami package for Drupal - -Drupal is one of the most versatile open source content management systems in the world. It is pre-configured with the Ctools and Views modules, Drush and Let's Encrypt auto-configuration support. - -[Overview of Drupal](http://drupal.org) - -Trademarks: This software listing is packaged by Bitnami. The respective trademarks mentioned in the offering are owned by the respective companies, and use of them does not imply any affiliation or endorsement. - -## TL;DR - -```console -helm install my-release oci://registry-1.docker.io/bitnamicharts/drupal -``` - -Looking to use Drupal in production? Try [VMware Tanzu Application Catalog](https://bitnami.com/enterprise), the commercial edition of the Bitnami catalog. - -## Introduction - -This chart bootstraps a [Drupal](https://github.com/bitnami/containers/tree/main/bitnami/drupal) deployment on a [Kubernetes](https://kubernetes.io) cluster using the [Helm](https://helm.sh) package manager. - -It also packages the [Bitnami MariaDB chart](https://github.com/bitnami/charts/tree/main/bitnami/mariadb) which is required for bootstrapping a MariaDB deployment as a database for the Drupal application. - -Bitnami charts can be used with [Kubeapps](https://kubeapps.dev/) for deployment and management of Helm Charts in clusters. - -## Prerequisites - -- Kubernetes 1.23+ -- Helm 3.8.0+ -- PV provisioner support in the underlying infrastructure -- ReadWriteMany volumes for deployment scaling - -## Installing the Chart - -To install the chart with the release name `my-release`: - -```console -helm install my-release oci://REGISTRY_NAME/REPOSITORY_NAME/drupal -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -The command deploys Drupal on the Kubernetes cluster in the default configuration. The [Parameters](#parameters) section lists the parameters that can be configured during installation. - -> **Tip**: List all releases using `helm list` - -## Configuration and installation details - -### Resource requests and limits - -Bitnami charts allow setting resource requests and limits for all containers inside the chart deployment. These are inside the `resources` value (check parameter table). Setting requests is essential for production workloads and these should be adapted to your specific use case. - -To make this process easier, the chart contains the `resourcesPreset` values, which automatically sets the `resources` section according to different presets. Check these presets in [the bitnami/common chart](https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15). However, in production workloads using `resourcePreset` is discouraged as it may not fully adapt to your specific needs. Find more information on container resource management in the [official Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/). - -### [Rolling VS Immutable tags](https://docs.vmware.com/en/VMware-Tanzu-Application-Catalog/services/tutorials/GUID-understand-rolling-tags-containers-index.html) - -It is strongly recommended to use immutable tags in a production environment. This ensures your deployment does not change automatically if the same tag is updated with a different image. - -Bitnami will release a new chart updating its containers if a new version of the main container, significant changes, or critical vulnerabilities exist. - -### Image - -The `image` parameter allows specifying which image will be pulled for the chart. - -#### Private registry - -If you configure the `image` value to one in a private registry, you will need to [specify an image pull secret](https://kubernetes.io/docs/concepts/containers/images/#specifying-imagepullsecrets-on-a-pod). - -1. Manually create image pull secret(s) in the namespace. See [this YAML example reference](https://kubernetes.io/docs/concepts/containers/images/#creating-a-secret-with-a-docker-config). Consult your image registry's documentation about getting the appropriate secret. -2. Note that the `imagePullSecrets` configuration value cannot currently be passed to helm using the `--set` parameter, so you must supply these using a `values.yaml` file, such as: - - ```yaml - imagePullSecrets: - - name: SECRET_NAME - ``` - -3. Install the chart - -### Setting Pod's affinity - -This chart allows you to set your custom affinity using the `affinity` parameter. Find more information about Pod's affinity in the [kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity). - -As an alternative, you can use of the preset configurations for pod affinity, pod anti-affinity, and node affinity available at the [bitnami/common](https://github.com/bitnami/charts/tree/main/bitnami/common#affinities) chart. To do so, set the `podAffinityPreset`, `podAntiAffinityPreset`, or `nodeAffinityPreset` parameters. - -## Persistence - -The [Bitnami Drupal](https://github.com/bitnami/containers/tree/main/bitnami/drupal) image stores the Drupal data and configurations at the `/bitnami/drupal` path of the container. - -Persistent Volume Claims are used to keep the data across deployments. This is known to work in GCE, AWS, and minikube. -See the [Parameters](#parameters) section to configure the PVC or to disable persistence. - -### Existing PersistentVolumeClaim - -1. Create the PersistentVolume -2. Create the PersistentVolumeClaim -3. Install the chart - -```console -helm install my-release --set persistence.existingClaim=PVC_NAME oci://REGISTRY_NAME/REPOSITORY_NAME/drupal -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -### Host path - -#### System compatibility - -- The local filesystem accessibility to a container in a pod with `hostPath` has been tested on OSX/MacOS with xhyve, and Linux with VirtualBox. -- Windows has not been tested with the supported VM drivers. Minikube does however officially support [Mounting Host Folders](https://minikube.sigs.k8s.io/docs/handbook/mount/) per pod. Or you may manually sync your container whenever host files are changed with tools like [docker-sync](https://github.com/EugenMayer/docker-sync) or [docker-bg-sync](https://github.com/cweagans/docker-bg-sync). - -#### Mounting steps - -1. The specified `hostPath` directory must already exist (create one if it does not). -2. Install the chart - - ```console - helm install my-release --set persistence.hostPath=/PATH/TO/HOST/MOUNT oci://REGISTRY_NAME/REPOSITORY_NAME/drupal - ``` - - > Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - - This will mount the `drupal-data` volume into the `hostPath` directory. The site data will be persisted if the mount path contains valid data, else the site data will be initialized at first launch. -3. Because the container cannot control the host machine's directory permissions, you must set the Drupal file directory permissions yourself and disable or clear Drupal cache. See Drupal Core's [INSTALL.txt](https://cgit.drupalcode.org/drupal/tree/core/INSTALL.txt?h=8.3.x#n152) for setting file permissions, and see [Drupal handbook page](https://www.drupal.org/node/2598914) to disable the cache, or [Drush handbook](https://drushcommands.com/drush-8x/cache/cache-rebuild/) to clear cache. - -## Parameters - -### Global parameters - -| Name | Description | Value | -| ----------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------ | -| `global.imageRegistry` | Global Docker image registry | `""` | -| `global.imagePullSecrets` | Global Docker registry secret names as an array | `[]` | -| `global.defaultStorageClass` | Global default StorageClass for Persistent Volume(s) | `""` | -| `global.storageClass` | DEPRECATED: use global.defaultStorageClass instead | `""` | -| `global.compatibility.openshift.adaptSecurityContext` | Adapt the securityContext sections of the deployment to make them compatible with Openshift restricted-v2 SCC: remove runAsUser, runAsGroup and fsGroup and let the platform use their allowed default IDs. Possible values: auto (apply if the detected running cluster is Openshift), force (perform the adaptation always), disabled (do not perform adaptation) | `auto` | - -### Common parameters - -| Name | Description | Value | -| ------------------- | ---------------------------------------------------------------------------------------------------------- | ----- | -| `kubeVersion` | Force target Kubernetes version (using Helm capabilities if not set) | `""` | -| `nameOverride` | String to partially override drupal.fullname template (will maintain the release name) | `""` | -| `fullnameOverride` | String to fully override drupal.fullname template | `""` | -| `namespaceOverride` | String to fully override common.names.namespace | `""` | -| `commonAnnotations` | Common annotations to add to all Drupal resources (sub-charts are not considered). Evaluated as a template | `{}` | -| `commonLabels` | Common labels to add to all Drupal resources (sub-charts are not considered). Evaluated as a template | `{}` | -| `extraDeploy` | Array of extra objects to deploy with the release (evaluated as a template). | `[]` | - -### Drupal parameters - -| Name | Description | Value | -| --------------------------------------------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------------ | -| `image.registry` | Drupal image registry | `REGISTRY_NAME` | -| `image.repository` | Drupal Image name | `REPOSITORY_NAME/drupal` | -| `image.digest` | Drupal image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `image.pullPolicy` | Drupal image pull policy | `IfNotPresent` | -| `image.pullSecrets` | Specify docker-registry secret names as an array | `[]` | -| `image.debug` | Specify if debug logs should be enabled | `false` | -| `replicaCount` | Number of Drupal Pods to run (requires ReadWriteMany PVC support) | `1` | -| `drupalProfile` | Drupal installation profile | `standard` | -| `drupalSkipInstall` | Skip Drupal installation wizard. Useful for migrations and restoring from SQL dump | `false` | -| `drupalUsername` | User of the application | `user` | -| `drupalPassword` | Application password | `""` | -| `drupalEmail` | Admin email | `user@example.com` | -| `allowEmptyPassword` | Allow DB blank passwords | `true` | -| `command` | Override default container command (useful when using custom images) | `[]` | -| `args` | Override default container args (useful when using custom images) | `[]` | -| `updateStrategy.type` | Update strategy - only really applicable for deployments with RWO PVs attached | `RollingUpdate` | -| `priorityClassName` | Drupal pods' priorityClassName | `""` | -| `schedulerName` | Name of the k8s scheduler (other than default) | `""` | -| `topologySpreadConstraints` | Topology Spread Constraints for pod assignment | `[]` | -| `automountServiceAccountToken` | Mount Service Account token in pod | `false` | -| `hostAliases` | Add deployment host aliases | `[]` | -| `extraEnvVars` | Extra environment variables | `[]` | -| `extraEnvVarsCM` | ConfigMap containing extra env vars | `""` | -| `extraEnvVarsSecret` | Secret containing extra env vars (in case of sensitive data) | `""` | -| `extraVolumes` | Array of extra volumes to be added to the deployment (evaluated as template). Requires setting `extraVolumeMounts` | `[]` | -| `extraVolumeMounts` | Array of extra volume mounts to be added to the container (evaluated as template). Normally used with `extraVolumes`. | `[]` | -| `initContainers` | Add additional init containers to the pod (evaluated as a template) | `[]` | -| `pdb.create` | Enable/disable a Pod Disruption Budget creation | `true` | -| `pdb.minAvailable` | Minimum number/percentage of pods that should remain scheduled | `""` | -| `pdb.maxUnavailable` | Maximum number/percentage of pods that may be made unavailable. Defaults to `1` if both `pdb.minAvailable` and `pdb.maxUnavailable` are empty. | `""` | -| `sidecars` | Attach additional containers to the pod (evaluated as a template) | `[]` | -| `tolerations` | Tolerations for pod assignment | `[]` | -| `serviceAccount.create` | Specifies whether a service account should be created | `true` | -| `serviceAccount.name` | The name of the service account to use. If not set and create is true, a name is generated using the fullname template | `""` | -| `serviceAccount.annotations` | Add annotations | `{}` | -| `serviceAccount.automountServiceAccountToken` | Automount API credentials for a service account. | `false` | -| `existingSecret` | Name of a secret with the application password | `""` | -| `smtpHost` | SMTP host | `""` | -| `smtpPort` | SMTP port | `""` | -| `smtpUser` | SMTP user | `""` | -| `smtpPassword` | SMTP password | `""` | -| `smtpProtocol` | SMTP Protocol (options: ssl,tls, nil) | `""` | -| `containerPorts` | Container ports | `{}` | -| `extraContainerPorts` | Optionally specify extra list of additional ports for Drupal container(s) | `[]` | -| `sessionAffinity` | Control where client requests go, to the same pod or round-robin. Values: ClientIP or None | `None` | -| `persistence.enabled` | Enable persistence using PVC | `true` | -| `persistence.storageClass` | PVC Storage Class for Drupal volume | `""` | -| `persistence.accessModes` | PVC Access Mode for Drupal volume | `["ReadWriteOnce"]` | -| `persistence.size` | PVC Storage Request for Drupal volume | `8Gi` | -| `persistence.existingClaim` | A manually managed Persistent Volume Claim | `""` | -| `persistence.hostPath` | If defined, the drupal-data volume will mount to the specified hostPath. | `""` | -| `persistence.annotations` | Persistent Volume Claim annotations | `{}` | -| `podAffinityPreset` | Pod affinity preset. Ignored if `affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `podAntiAffinityPreset` | Pod anti-affinity preset. Ignored if `affinity` is set. Allowed values: `soft` or `hard` | `soft` | -| `nodeAffinityPreset.type` | Node affinity preset type. Ignored if `affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `nodeAffinityPreset.key` | Node label key to match Ignored if `affinity` is set. | `""` | -| `nodeAffinityPreset.values` | Node label values to match. Ignored if `affinity` is set. | `[]` | -| `affinity` | Affinity for pod assignment | `{}` | -| `nodeSelector` | Node labels for pod assignment. Evaluated as a template. | `{}` | -| `resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if resources is set (resources is recommended for production). | `micro` | -| `resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `podSecurityContext.enabled` | Enable Drupal pods' Security Context | `true` | -| `podSecurityContext.fsGroupChangePolicy` | Set filesystem group change policy | `Always` | -| `podSecurityContext.sysctls` | Set kernel settings using the sysctl interface | `[]` | -| `podSecurityContext.supplementalGroups` | Set filesystem extra groups | `[]` | -| `podSecurityContext.fsGroup` | Drupal pods' group ID | `1001` | -| `containerSecurityContext.enabled` | Enabled Drupal containers' Security Context | `true` | -| `containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `containerSecurityContext.runAsUser` | Set Drupal containers' Security Context runAsUser | `1001` | -| `containerSecurityContext.runAsGroup` | Set Drupal containers' Security Context runAsGroup | `1001` | -| `containerSecurityContext.runAsNonRoot` | Set Controller container's Security Context runAsNonRoot | `true` | -| `containerSecurityContext.privileged` | Set Drupal container's Security Context privileged | `false` | -| `containerSecurityContext.allowPrivilegeEscalation` | Set Drupal container's Security Context allowPrivilegeEscalation | `false` | -| `containerSecurityContext.capabilities.drop` | List of capabilities to be dropped | `["ALL"]` | -| `containerSecurityContext.seccompProfile.type` | Set container's Security Context seccomp profile | `RuntimeDefault` | -| `containerSecurityContext.readOnlyRootFilesystem` | Set Drupal container's Security Context readOnlyRootFilesystem | `true` | -| `startupProbe.enabled` | Enable startupProbe | `false` | -| `startupProbe.path` | Request path for startupProbe | `/user/login` | -| `startupProbe.initialDelaySeconds` | Initial delay seconds for startupProbe | `600` | -| `startupProbe.periodSeconds` | Period seconds for startupProbe | `10` | -| `startupProbe.timeoutSeconds` | Timeout seconds for startupProbe | `5` | -| `startupProbe.failureThreshold` | Failure threshold for startupProbe | `5` | -| `startupProbe.successThreshold` | Success threshold for startupProbe | `1` | -| `livenessProbe.enabled` | Enable livenessProbe | `true` | -| `livenessProbe.initialDelaySeconds` | Initial delay seconds for livenessProbe | `600` | -| `livenessProbe.periodSeconds` | Period seconds for livenessProbe | `10` | -| `livenessProbe.timeoutSeconds` | Timeout seconds for livenessProbe | `5` | -| `livenessProbe.failureThreshold` | Failure threshold for livenessProbe | `5` | -| `livenessProbe.successThreshold` | Success threshold for livenessProbe | `1` | -| `readinessProbe.enabled` | Enable readinessProbe | `true` | -| `readinessProbe.path` | Request path for readinessProbe | `/user/login` | -| `readinessProbe.initialDelaySeconds` | Initial delay seconds for readinessProbe | `30` | -| `readinessProbe.periodSeconds` | Period seconds for readinessProbe | `5` | -| `readinessProbe.timeoutSeconds` | Timeout seconds for readinessProbe | `1` | -| `readinessProbe.failureThreshold` | Failure threshold for readinessProbe | `5` | -| `readinessProbe.successThreshold` | Success threshold for readinessProbe | `1` | -| `customStartupProbe` | Override default startup probe | `{}` | -| `customLivenessProbe` | Override default liveness probe | `{}` | -| `customReadinessProbe` | Override default readiness probe | `{}` | -| `lifecycleHooks` | LifecycleHook to set additional configuration at startup Evaluated as a template | `{}` | -| `podAnnotations` | Pod annotations | `{}` | -| `podLabels` | Add additional labels to the pod (evaluated as a template) | `{}` | - -### Traffic Exposure Parameters - -| Name | Description | Value | -| ---------------------------------- | -------------------------------------------------------------------------------------------------------------------------------- | ------------------------ | -| `service.type` | Kubernetes Service type | `LoadBalancer` | -| `service.ports.http` | Service HTTP port | `80` | -| `service.ports.https` | Service HTTPS port | `443` | -| `service.loadBalancerSourceRanges` | Restricts access for LoadBalancer (only with `service.type: LoadBalancer`) | `[]` | -| `service.loadBalancerIP` | loadBalancerIP for the Drupal Service (optional, cloud specific) | `""` | -| `service.nodePorts` | Kubernetes node port | `{}` | -| `service.externalTrafficPolicy` | Enable client source IP preservation | `Cluster` | -| `service.clusterIP` | %%MAIN_CONTAINER_NAME%% service Cluster IP | `""` | -| `service.extraPorts` | Extra ports to expose (normally used with the `sidecar` value) | `[]` | -| `service.annotations` | Additional custom annotations for %%MAIN_CONTAINER_NAME%% service | `{}` | -| `service.sessionAffinity` | Session Affinity for Kubernetes service, can be "None" or "ClientIP" | `None` | -| `service.sessionAffinityConfig` | Additional settings for the sessionAffinity | `{}` | -| `ingress.enabled` | Enable ingress controller resource | `false` | -| `ingress.pathType` | Ingress Path type | `ImplementationSpecific` | -| `ingress.apiVersion` | Override API Version (automatically detected if not set) | `""` | -| `ingress.ingressClassName` | IngressClass that will be be used to implement the Ingress (Kubernetes 1.18+) | `""` | -| `ingress.hostname` | Default host for the ingress resource | `drupal.local` | -| `ingress.path` | The Path to Drupal. You may need to set this to '/*' in order to use this | `/` | -| `ingress.annotations` | Additional annotations for the Ingress resource. To enable certificate autogeneration, place here your cert-manager annotations. | `{}` | -| `ingress.tls` | Enable TLS configuration for the hostname defined at ingress.hostname parameter | `false` | -| `ingress.tlsWwwPrefix` | Adds www subdomain to default cert | `false` | -| `ingress.extraHosts` | The list of additional hostnames to be covered with this ingress record. | `[]` | -| `ingress.extraPaths` | Any additional arbitrary paths that may need to be added to the ingress under the main host. | `[]` | -| `ingress.extraTls` | The tls configuration for additional hostnames to be covered with this ingress record. | `[]` | -| `ingress.secrets` | If you're providing your own certificates, please use this to add the certificates as secrets | `[]` | -| `ingress.extraRules` | Additional rules to be covered with this ingress record | `[]` | - -### Database parameters - -| Name | Description | Value | -| ------------------------------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------- | -| `mariadb.enabled` | Whether to deploy a mariadb server to satisfy the applications database requirements | `true` | -| `mariadb.architecture` | MariaDB architecture (`standalone` or `replication`) | `standalone` | -| `mariadb.auth.rootPassword` | Password for the MariaDB `root` user | `""` | -| `mariadb.auth.database` | Database name to create | `bitnami_drupal` | -| `mariadb.auth.username` | Database user to create | `bn_drupal` | -| `mariadb.auth.password` | Password for the database | `""` | -| `mariadb.primary.persistence.enabled` | Enable database persistence using PVC | `true` | -| `mariadb.primary.persistence.storageClass` | MariaDB primary persistent volume storage Class | `""` | -| `mariadb.primary.persistence.accessModes` | Database Persistent Volume Access Modes | `["ReadWriteOnce"]` | -| `mariadb.primary.persistence.size` | Database Persistent Volume Size | `8Gi` | -| `mariadb.primary.persistence.hostPath` | Set path in case you want to use local host path volumes (not recommended in production) | `""` | -| `mariadb.primary.persistence.existingClaim` | Name of an existing `PersistentVolumeClaim` for MariaDB primary replicas | `""` | -| `externalDatabase.host` | Host of the existing database | `""` | -| `externalDatabase.port` | Port of the existing database | `3306` | -| `externalDatabase.user` | Existing username in the external db | `bn_drupal` | -| `externalDatabase.password` | Password for the above username. Ignored if existing secret is provided | `""` | -| `externalDatabase.database` | Name of the existing database | `bitnami_drupal` | -| `externalDatabase.existingSecret` | Name of a secret with the database password. (externalDatabase.password will be ignored and picked up from this secret). The secret has to contain the key db-password | `""` | - -### Volume Permissions parameters - -| Name | Description | Value | -| ------------------------------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | -------------------------- | -| `volumePermissions.enabled` | Enable init container that changes volume permissions in the data directory (for cases where the default k8s `runAsUser` and `fsUser` values do not work) | `false` | -| `volumePermissions.image.registry` | Init container volume-permissions image registry | `REGISTRY_NAME` | -| `volumePermissions.image.repository` | Init container volume-permissions image name | `REPOSITORY_NAME/os-shell` | -| `volumePermissions.image.digest` | Init container volume-permissions image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `volumePermissions.image.pullPolicy` | Init container volume-permissions image pull policy | `IfNotPresent` | -| `volumePermissions.image.pullSecrets` | Specify docker-registry secret names as an array | `[]` | -| `volumePermissions.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if volumePermissions.resources is set (volumePermissions.resources is recommended for production). | `none` | -| `volumePermissions.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | - -### Metrics parameters - -| Name | Description | Value | -| ------------------------------------------ | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | --------------------------------- | -| `metrics.enabled` | Start a exporter side-car | `false` | -| `metrics.image.registry` | Apache exporter image registry | `REGISTRY_NAME` | -| `metrics.image.repository` | Apache exporter image repository | `REPOSITORY_NAME/apache-exporter` | -| `metrics.image.digest` | Apache exporter image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `metrics.image.pullPolicy` | Image pull policy | `IfNotPresent` | -| `metrics.image.pullSecrets` | Specify docker-registry secret names as an array | `[]` | -| `metrics.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if metrics.resources is set (metrics.resources is recommended for production). | `none` | -| `metrics.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `metrics.podAnnotations` | Additional annotations for Metrics exporter pod | `{}` | -| `metrics.service.type` | Drupal exporter service type | `ClusterIP` | -| `metrics.service.ports.metrics` | Drupal exporter service port | `9117` | -| `metrics.service.externalTrafficPolicy` | Drupal exporter service external traffic policy | `Cluster` | -| `metrics.service.extraPorts` | Extra ports to expose (normally used with the `sidecar` value) | `[]` | -| `metrics.service.loadBalancerIP` | Drupal exporter service Load Balancer IP | `""` | -| `metrics.service.loadBalancerSourceRanges` | Drupal exporter service Load Balancer sources | `[]` | -| `metrics.service.annotations` | Additional custom annotations for Drupal exporter service | `{}` | -| `metrics.serviceMonitor.enabled` | Create ServiceMonitor resource(s) for scraping metrics using PrometheusOperator | `false` | -| `metrics.serviceMonitor.namespace` | The namespace in which the ServiceMonitor will be created | `""` | -| `metrics.serviceMonitor.interval` | The interval at which metrics should be scraped | `30s` | -| `metrics.serviceMonitor.scrapeTimeout` | The timeout after which the scrape is ended | `""` | -| `metrics.serviceMonitor.relabellings` | Metrics RelabelConfigs to apply to samples before scraping. | `[]` | -| `metrics.serviceMonitor.metricRelabelings` | Metrics RelabelConfigs to apply to samples before ingestion. | `[]` | -| `metrics.serviceMonitor.honorLabels` | Specify honorLabels parameter to add the scrape endpoint | `false` | -| `metrics.serviceMonitor.additionalLabels` | Additional labels that can be used so ServiceMonitor resource(s) can be discovered by Prometheus | `{}` | -| `metrics.prometheusRule.enabled` | Create a custom prometheusRule Resource for scraping metrics using PrometheusOperator | `false` | -| `metrics.prometheusRule.namespace` | The namespace in which the prometheusRule will be created | `""` | -| `metrics.prometheusRule.additionalLabels` | Additional labels for the prometheusRule | `{}` | -| `metrics.prometheusRule.rules` | Custom Prometheus rules | `[]` | - -### Certificate injection parameters - -| Name | Description | Value | -| ---------------------------------------------------- | ----------------------------------------------------------------------------------------------------------------- | ---------------------------------------- | -| `certificates.customCertificate.certificateSecret` | Secret containing the certificate and key to add | `""` | -| `certificates.customCertificate.chainSecret.name` | Name of the secret containing the certificate chain | `secret-name` | -| `certificates.customCertificate.chainSecret.key` | Key of the certificate chain file inside the secret | `secret-key` | -| `certificates.customCertificate.certificateLocation` | Location in the container to store the certificate | `/etc/ssl/certs/ssl-cert-snakeoil.pem` | -| `certificates.customCertificate.keyLocation` | Location in the container to store the private key | `/etc/ssl/private/ssl-cert-snakeoil.key` | -| `certificates.customCertificate.chainLocation` | Location in the container to store the certificate chain | `/etc/ssl/certs/mychain.pem` | -| `certificates.customCAs` | Defines a list of secrets to import into the container trust store | `[]` | -| `certificates.command` | Override default container command (useful when using custom images) | `[]` | -| `certificates.args` | Override default container args (useful when using custom images) | `[]` | -| `certificates.extraEnvVars` | Container sidecar extra environment variables (eg proxy) | `[]` | -| `certificates.extraEnvVarsCM` | ConfigMap containing extra env vars | `""` | -| `certificates.extraEnvVarsSecret` | Secret containing extra env vars (in case of sensitive data) | `""` | -| `certificates.image.registry` | Container sidecar registry | `REGISTRY_NAME` | -| `certificates.image.repository` | Container sidecar image | `REPOSITORY_NAME/os-shell` | -| `certificates.image.digest` | Container sidecar image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `certificates.image.pullPolicy` | Container sidecar image pull policy | `IfNotPresent` | -| `certificates.image.pullSecrets` | Container sidecar image pull secrets | `[]` | - -### NetworkPolicy parameters - -| Name | Description | Value | -| --------------------------------------- | --------------------------------------------------------------- | ------ | -| `networkPolicy.enabled` | Specifies whether a NetworkPolicy should be created | `true` | -| `networkPolicy.allowExternal` | Don't require server label for connections | `true` | -| `networkPolicy.allowExternalEgress` | Allow the pod to access any range of port and all destinations. | `true` | -| `networkPolicy.extraIngress` | Add extra ingress rules to the NetworkPolicy | `[]` | -| `networkPolicy.extraEgress` | Add extra ingress rules to the NetworkPolicy | `[]` | -| `networkPolicy.ingressNSMatchLabels` | Labels to match to allow traffic from other namespaces | `{}` | -| `networkPolicy.ingressNSPodMatchLabels` | Pod labels to match to allow traffic from other namespaces | `{}` | - -The above parameters map to the env variables defined in [bitnami/drupal](https://github.com/bitnami/containers/tree/main/bitnami/drupal). For more information please refer to the [bitnami/drupal](https://github.com/bitnami/containers/tree/main/bitnami/drupal) image documentation. - -Specify each parameter using the `--set key=value[,key=value]` argument to `helm install`. For example, - -```console -helm install my-release \ - --set drupalUsername=admin,drupalPassword=password,mariadb.auth.rootPassword=secretpassword \ - oci://REGISTRY_NAME/REPOSITORY_NAME/drupal -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -The above command sets the Drupal administrator account username and password to `admin` and `password` respectively. Additionally, it sets the MariaDB `root` user password to `secretpassword`. - -> NOTE: Once this chart is deployed, it is not possible to change the application's access credentials, such as usernames or passwords, using Helm. To change these application credentials after deployment, delete any persistent volumes (PVs) used by the chart and re-deploy it, or use the application's built-in administrative tools if available. - -Alternatively, a YAML file that specifies the values for the above parameters can be provided while installing the chart. For example, - -```console -helm install my-release -f values.yaml oci://REGISTRY_NAME/REPOSITORY_NAME/drupal -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. -> **Tip**: You can use the default [values.yaml](https://github.com/bitnami/charts/tree/main/bitnami/drupal/values.yaml) - -## Troubleshooting - -Find more information about how to deal with common errors related to Bitnami's Helm charts in [this troubleshooting guide](https://docs.bitnami.com/general/how-to/troubleshoot-helm-chart-issues). - -## Upgrading - -### To 19.0.0 - -This major release bumps the MariaDB version to 11.4. Follow the [upstream instructions](https://mariadb.com/kb/en/upgrading-from-mariadb-11-3-to-mariadb-11-4/) for upgrading from MariaDB 11.3 to 11.4. No major issues are expected during the upgrade. - -### To 18.0.0 - -This major bump changes the following security defaults: - -- `runAsGroup` is changed from `0` to `1001` -- `readOnlyRootFilesystem` is set to `true` -- `resourcesPreset` is changed from `none` to the minimum size working in our test suites (NOTE: `resourcesPreset` is not meant for production usage, but `resources` adapted to your use case). -- `global.compatibility.openshift.adaptSecurityContext` is changed from `disabled` to `auto`. -- The `networkPolicy` section has been normalized amongst all Bitnami charts. Compared to the previous approach, the values section has been simplified (check the Parameters section) and now it set to `enabled=true` by default. Egress traffic is allowed by default and ingress traffic is allowed by all pods but only to the ports set in `containerPorts` and `extraContainerPorts`. - -This could potentially break any customization or init scripts used in your deployment. If this is the case, change the default values to the previous ones. - -Also, this major release bumps the MariaDB chart version to [18.x.x](https://github.com/bitnami/charts/pull/24804); no major issues are expected during the upgrade. - -### To 17.0.0 - -This major release bumps the MariaDB version to 11.2. No major issues are expected during the upgrade. - -### To 16.0.0 - -This major release bumps the MariaDB version to 11.1. No major issues are expected during the upgrade. - -### To 15.0.0 - -This major release bumps the MariaDB version to 11.0. Follow the [upstream instructions](https://mariadb.com/kb/en/upgrading-from-mariadb-10-11-to-mariadb-11-0/) for upgrading from MariaDB 10.11 to 11.0. No major issues are expected during the upgrade. - -### To 14.0.0 - -This major release bumps the MariaDB version to 10.11. Follow the [upstream instructions](https://mariadb.com/kb/en/upgrading-from-mariadb-10-6-to-mariadb-10-11/) for upgrading from MariaDB 10.6 to 10.11. No major issues are expected during the upgrade. - -### To 12.0.0 - -This major release bumps the MariaDB version to 10.6. Follow the [upstream instructions](https://mariadb.com/kb/en/upgrading-from-mariadb-105-to-mariadb-106/) for upgrading from MariaDB 10.5 to 10.6. No major issues are expected during the upgrade. - -### To 11.0.0 - -This major release renames several values in this chart and adds missing features, in order to be inline with the rest of assets in the Bitnami charts repository. - -Affected values: - -- `service.port` was deprecated, we recommend using `service.ports.http` instead. -- `service.httpsPort` was deprecated, we recommend using `service.ports.https` instead. -- `persistence.accessMode` has been deprecated, we recommend using `persistence.accessModes` instead. - -Additionally also updates the MariaDB subchart to it newest major, 10.0.0, which contains similar changes. - -### To 10.0.0 - -[On November 13, 2020, Helm v2 support was formally finished](https://github.com/helm/charts#status-of-the-project), this major version is the result of the required changes applied to the Helm Chart to be able to incorporate the different features added in Helm v3 and to be consistent with the Helm project itself regarding the Helm v2 EOL. - -#### What changes were introduced in this major version? - -- Previous versions of this Helm Chart use `apiVersion: v1` (installable by both Helm 2 and 3), this Helm Chart was updated to `apiVersion: v2` (installable by Helm 3 only). [Here](https://helm.sh/docs/topics/charts/#the-apiversion-field) you can find more information about the `apiVersion` field. -- Move dependency information from the *requirements.yaml* to the *Chart.yaml* -- After running `helm dependency update`, a *Chart.lock* file is generated containing the same structure used in the previous *requirements.lock* -- The different fields present in the *Chart.yaml* file has been ordered alphabetically in a homogeneous way for all the Bitnami Helm Charts - -#### Considerations when upgrading to this version - -- If you want to upgrade to this version from a previous one installed with Helm v3, you shouldn't face any issues -- If you want to upgrade to this version using Helm v2, this scenario is not supported as this version doesn't support Helm v2 anymore -- If you installed the previous version with Helm v2 and wants to upgrade to this version with Helm v3, please refer to the [official Helm documentation](https://helm.sh/docs/topics/v2_v3_migration/#migration-use-cases) about migrating from Helm v2 to v3 - -#### Useful links - -- -- -- - -### To 9.0.0 - -MariaDB dependency version was bumped to a new major version that introduces several incompatilibites. Therefore, backwards compatibility is not guaranteed unless an external database is used. Check [MariaDB Upgrading Notes](https://github.com/bitnami/charts/tree/main/bitnami/mariadb#to-800) for more information. - -To upgrade to `9.0.0`, you have two alternatives: - -- Install a new Drupal chart, and migrate your Drupal site using backup/restore tools such as [Drupal Backup and Migrate](https://www.drupal.org/project/backup_migrate). -- Reuse the PVC used to hold the MariaDB data on your previous release. To do so, follow the instructions below (the following example assumes that the release name is `drupal`): - -> NOTE: Please, create a backup of your database before running any of those actions. The steps below would be only valid if your application (e.g. any plugins or custom code) is compatible with MariaDB 10.5.x - -Obtain the credentials and the name of the PVC used to hold the MariaDB data on your current release: - -```console -export DRUPAL_PASSWORD=$(kubectl get secret --namespace default drupal -o jsonpath="{.data.drupal-password}" | base64 -d) -export MARIADB_ROOT_PASSWORD=$(kubectl get secret --namespace default drupal-mariadb -o jsonpath="{.data.mariadb-root-password}" | base64 -d) -export MARIADB_PASSWORD=$(kubectl get secret --namespace default drupal-mariadb -o jsonpath="{.data.mariadb-password}" | base64 -d) -export MARIADB_PVC=$(kubectl get pvc -l app=mariadb,component=master,release=drupal -o jsonpath="{.items[0].metadata.name}") -``` - -Upgrade your release (maintaining the version) disabling MariaDB and scaling Drupal replicas to 0: - -```console -helm upgrade drupal oci://REGISTRY_NAME/REPOSITORY_NAME/drupal --set drupalPassword=$DRUPAL_PASSWORD --set replicaCount=0 --set mariadb.enabled=false --version 8.2.1 -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -Finally, upgrade you release to 9.0.0 reusing the existing PVC, and enabling back MariaDB: - -```console -helm upgrade drupal oci://REGISTRY_NAME/REPOSITORY_NAME/drupal --set mariadb.primary.persistence.existingClaim=$MARIADB_PVC --set mariadb.auth.rootPassword=$MARIADB_ROOT_PASSWORD --set mariadb.auth.password=$MARIADB_PASSWORD --set drupalPassword=$DRUPAL_PASSWORD -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -You should see the lines below in MariaDB container logs: - -```console -$ kubectl logs $(kubectl get pods -l app.kubernetes.io/instance=drupal,app.kubernetes.io/name=mariadb,app.kubernetes.io/component=primary -o jsonpath="{.items[0].metadata.name}") -... -mariadb 12:13:24.98 INFO ==> Using persisted data -mariadb 12:13:25.01 INFO ==> Running mysql_upgrade -... -``` - -### To 8.0.0 - -The [Bitnami Drupal](https://github.com/bitnami/containers/tree/main/bitnami/drupal) image was migrated to a "non-root" user approach. Previously the container ran as the `root` user and the Apache daemon was started as the `daemon` user. From now on, both the container and the Apache daemon run as user `1001`. You can revert this behavior by setting the parameters `containerSecurityContext.runAsUser` to `root`. - -Consequences: - -- The HTTP/HTTPS ports exposed by the container are now `8080/8443` instead of `80/443`. -- Backwards compatibility is not guaranteed. - -To upgrade to `8.0.0`, backup Drupal data and the previous MariaDB databases, install a new Drupal chart and import the backups and data, ensuring the `1001` user has the appropriate permissions on the migrated volume. - -This upgrade also adapts the chart to the latest Bitnami good practices. Check the Parameters section for more information. - -### To 6.0.0 - -Helm performs a lookup for the object based on its group (apps), version (v1), and kind (Deployment). Also known as its GroupVersionKind, or GVK. Changing the GVK is considered a compatibility breaker from Kubernetes' point of view, so you cannot "upgrade" those objects to the new GVK in-place. Earlier versions of Helm 3 did not perform the lookup correctly which has since been fixed to match the spec. - -In the `apiVersion` of the deployment resources was updated to `apps/v1` in tune with the api's deprecated, resulting in compatibility breakage. - -This major version signifies this change. - -### To 2.0.0 - -Backwards compatibility is not guaranteed unless you modify the labels used on the chart's deployments. -Use the workaround below to upgrade from versions previous to 2.0.0. The following example assumes that the release name is drupal: - -```console -kubectl patch deployment drupal-drupal --type=json -p='[{"op": "remove", "path": "/spec/selector/matchLabels/chart"}]' -kubectl delete statefulset drupal-mariadb --cascade=false -``` - -## License - -Copyright © 2024 Broadcom. The term "Broadcom" refers to Broadcom Inc. and/or its subsidiaries. - -Licensed under the Apache License, Version 2.0 (the "License"); -you may not use this file except in compliance with the License. -You may obtain a copy of the License at - - - -Unless required by applicable law or agreed to in writing, software -distributed under the License is distributed on an "AS IS" BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -See the License for the specific language governing permissions and -limitations under the License. \ No newline at end of file diff --git a/charts/drupal/drupal/templates/NOTES.txt b/charts/drupal/drupal/templates/NOTES.txt deleted file mode 100644 index 95e2c92..0000000 --- a/charts/drupal/drupal/templates/NOTES.txt +++ /dev/null @@ -1,88 +0,0 @@ -CHART NAME: {{ .Chart.Name }} -CHART VERSION: {{ .Chart.Version }} -APP VERSION: {{ .Chart.AppVersion }} - -{{- if or .Values.mariadb.enabled .Values.externalDatabase.host -}} - -** Please be patient while the chart is being deployed ** - -1. Get the Drupal URL: - -{{- if .Values.ingress.enabled }} - - You should be able to access your new Drupal installation through - - http://{{- .Values.ingress.hostname }}/ - -{{- else if eq .Values.service.type "LoadBalancer" }} - - NOTE: It may take a few minutes for the LoadBalancer IP to be available. - Watch the status with: 'kubectl get svc --namespace {{ include "common.names.namespace" . }} -w {{ include "common.names.fullname" . }}' - - export SERVICE_IP=$(kubectl get svc --namespace {{ include "common.names.namespace" . }} {{ include "common.names.fullname" . }} --template "{{ "{{ range (index .status.loadBalancer.ingress 0) }}{{ . }}{{ end }}" }}") - -{{- $port:=(coalesce .Values.service.ports.http .Values.service.port) | toString }} - echo "Drupal URL: http://$SERVICE_IP{{- if ne $port "80" }}:{{ coalesce .Values.service.ports.http .Values.service.port }}{{ end }}/" - -{{- else if eq .Values.service.type "ClusterIP" }} - - echo "Drupal URL: http://127.0.0.1:8080/" - kubectl port-forward --namespace {{ include "common.names.namespace" . }} svc/{{ include "common.names.fullname" . }} 8080:{{ coalesce .Values.service.ports.http .Values.service.port }} - -{{- end }} - -{{- if eq .Values.service.type "NodePort" }} - - Or running: - - export NODE_PORT=$(kubectl get --namespace {{ include "common.names.namespace" . }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ include "common.names.fullname" . }}) - export NODE_IP=$(kubectl get nodes --namespace {{ include "common.names.namespace" . }} -o jsonpath="{.items[0].status.addresses[0].address}") - echo "Drupal URL: http://$NODE_IP:$NODE_PORT/" - -{{- end }} - -2. Get your Drupal login credentials by running: - - echo Username: {{ .Values.drupalUsername }} - echo Password: $(kubectl get secret --namespace {{ include "common.names.namespace" . }} {{ template "drupal.secretName" . }} -o jsonpath="{.data.drupal-password}" | base64 -d) - -{{- else -}} - -######################################################################################## -### ERROR: You did not provide an external database host in your 'helm install' call ### -######################################################################################## - -This deployment will be incomplete until you configure Drupal with a resolvable database -host. To configure Drupal to use and external database host: - -1. Complete your Drupal deployment by running: - - export APP_PASSWORD=$(kubectl get secret --namespace {{ include "common.names.namespace" . }} {{ template "drupal.secretName" . }} -o jsonpath="{.data.drupal-password}" | base64 -d) - - ## PLEASE UPDATE THE EXTERNAL DATABASE CONNECTION PARAMETERS IN THE FOLLOWING COMMAND AS NEEDED ## - - helm upgrade --namespace {{ include "common.names.namespace" . }} {{ .Release.Name }} oci://registry-1.docker.io/bitnamicharts/{{ .Chart.Name }} \ - --set drupalPassword=$APP_PASSWORD,service.type={{ .Values.service.type }},mariadb.enabled=false{{- if not (empty .Values.externalDatabase.user) }},externalDatabase.user={{ .Values.externalDatabase.user }}{{- end }}{{- if not (empty .Values.externalDatabase.password) }},externalDatabase.password={{ .Values.externalDatabase.password }}{{- end }}{{- if not (empty .Values.externalDatabase.database) }},externalDatabase.database={{ .Values.externalDatabase.database }}{{- end }},externalDatabase.host=YOUR_EXTERNAL_DATABASE_HOST{{- if .Values.global }}{{- if .Values.global.imagePullSecrets }},global.imagePullSecrets={{ .Values.global.imagePullSecrets }}{{- end }}{{- end }} - -{{- end }} - -{{- include "common.warnings.rollingTag" .Values.image }} -{{- include "common.warnings.rollingTag" .Values.metrics.image }} -{{- include "common.warnings.rollingTag" .Values.volumePermissions.image }} -{{- include "common.warnings.rollingTag" .Values.certificates.image }} - -{{- $passwordValidationErrors := list -}} -{{- if not .Values.existingSecret -}} - {{- $secretName := include "drupal.secretName" . -}} - {{- $requiredDrupalPassword := dict "valueKey" "drupalPassword" "secret" $secretName "field" "drupal-password" "context" $ -}} - {{- $requiredDrupalPasswordError := include "common.validations.values.single.empty" $requiredDrupalPassword -}} - {{- $passwordValidationErrors = append $passwordValidationErrors $requiredDrupalPasswordError -}} -{{- end -}} - -{{- $mariadbSecretName := include "drupal.databaseSecretName" . -}} -{{- $mariadbPasswordValidationErrors := include "common.validations.values.mariadb.passwords" (dict "secret" $mariadbSecretName "subchart" true "context" $) -}} -{{- $passwordValidationErrors = append $passwordValidationErrors $mariadbPasswordValidationErrors -}} - -{{- include "common.errors.upgrade.passwords.empty" (dict "validationErrors" $passwordValidationErrors "context" $) -}} -{{- include "common.warnings.resources" (dict "sections" (list "metrics" "" "volumePermissions") "context" $) }} -{{- include "common.warnings.modifiedImages" (dict "images" (list .Values.image .Values.volumePermissions.image .Values.metrics.image .Values.certificates.image) "context" $) }} \ No newline at end of file diff --git a/charts/drupal/drupal/templates/_helpers.tpl b/charts/drupal/drupal/templates/_helpers.tpl deleted file mode 100644 index da01ed6..0000000 --- a/charts/drupal/drupal/templates/_helpers.tpl +++ /dev/null @@ -1,144 +0,0 @@ -{{/* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{/* -Create a default fully qualified app name. -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -*/}} -{{- define "drupal.mariadb.fullname" -}} -{{- printf "%s-%s" .Release.Name "mariadb" | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Return the proper certificate image name -*/}} -{{- define "certificates.image" -}} -{{- include "common.images.image" ( dict "imageRoot" .Values.certificates.image "global" .Values.global ) -}} -{{- end -}} - -{{/* -Return the proper Drupal image name -*/}} -{{- define "drupal.image" -}} -{{- include "common.images.image" (dict "imageRoot" .Values.image "global" .Values.global) -}} -{{- end -}} - -{{/* -Return the proper image name (for the metrics image) -*/}} -{{- define "drupal.metrics.image" -}} -{{- include "common.images.image" (dict "imageRoot" .Values.metrics.image "global" .Values.global) -}} -{{- end -}} - -{{/* -Return the proper image name (for the init container volume-permissions image) -*/}} -{{- define "drupal.volumePermissions.image" -}} -{{- include "common.images.image" ( dict "imageRoot" .Values.volumePermissions.image "global" .Values.global ) -}} -{{- end -}} - -{{/* -Return the proper Docker Image Registry Secret Names -*/}} -{{- define "drupal.imagePullSecrets" -}} -{{- include "common.images.pullSecrets" (dict "images" (list .Values.image .Values.metrics.image .Values.volumePermissions.image .Values.certificates.image) "global" .Values.global) -}} -{{- end -}} - -{{/* -Return the proper Storage Class -*/}} -{{- define "drupal.storageClass" -}} -{{- include "common.storage.class" (dict "persistence" .Values.persistence "global" .Values.global) -}} -{{- end -}} - -{{/* -Drupal credential secret name -*/}} -{{- define "drupal.secretName" -}} -{{- coalesce .Values.existingSecret (include "common.names.fullname" .) -}} -{{- end -}} - -{{/* -Return the MariaDB Hostname -*/}} -{{- define "drupal.databaseHost" -}} -{{- if .Values.mariadb.enabled }} - {{- if eq .Values.mariadb.architecture "replication" }} - {{- printf "%s-%s" (include "drupal.mariadb.fullname" .) "primary" | trunc 63 | trimSuffix "-" -}} - {{- else -}} - {{- printf "%s" (include "drupal.mariadb.fullname" .) -}} - {{- end -}} -{{- else -}} - {{- printf "%s" .Values.externalDatabase.host -}} -{{- end -}} -{{- end -}} - -{{/* -Return the MariaDB Port -*/}} -{{- define "drupal.databasePort" -}} -{{- if .Values.mariadb.enabled }} - {{- printf "3306" -}} -{{- else -}} - {{- printf "%d" (.Values.externalDatabase.port | int ) -}} -{{- end -}} -{{- end -}} - -{{/* -Return the MariaDB Database Name -*/}} -{{- define "drupal.databaseName" -}} -{{- if .Values.mariadb.enabled }} - {{- printf "%s" .Values.mariadb.auth.database -}} -{{- else -}} - {{- printf "%s" .Values.externalDatabase.database -}} -{{- end -}} -{{- end -}} - -{{/* -Return the MariaDB User -*/}} -{{- define "drupal.databaseUser" -}} -{{- if .Values.mariadb.enabled }} - {{- printf "%s" .Values.mariadb.auth.username -}} -{{- else -}} - {{- printf "%s" .Values.externalDatabase.user -}} -{{- end -}} -{{- end -}} - -{{/* -Return the MariaDB Secret Name -*/}} -{{- define "drupal.databaseSecretName" -}} -{{- if .Values.mariadb.enabled }} - {{- printf "%s" (include "drupal.mariadb.fullname" .) -}} -{{- else if .Values.externalDatabase.existingSecret -}} - {{- printf "%s" .Values.externalDatabase.existingSecret -}} -{{- else -}} - {{- printf "%s-%s" (include "common.names.fullname" .) "externaldb" -}} -{{- end -}} -{{- end -}} - -{{/* -Return the database password key -*/}} -{{- define "drupal.databasePasswordKey" -}} -{{- if .Values.mariadb.enabled -}} -mariadb-password -{{- else -}} -db-password -{{- end -}} -{{- end -}} - -{{/* -Create the name of the service account to use -*/}} -{{- define "drupal.serviceAccountName" -}} -{{- if .Values.serviceAccount.create -}} - {{ default (include "common.names.fullname" .) .Values.serviceAccount.name }} -{{- else -}} - {{ default "default" .Values.serviceAccount.name }} -{{- end -}} -{{- end -}} diff --git a/charts/drupal/drupal/templates/deployment.yaml b/charts/drupal/drupal/templates/deployment.yaml deleted file mode 100644 index bb36ad4..0000000 --- a/charts/drupal/drupal/templates/deployment.yaml +++ /dev/null @@ -1,384 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -apiVersion: {{ include "common.capabilities.deployment.apiVersion" . }} -kind: Deployment -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.podLabels .Values.commonLabels ) "context" . ) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - {{- if .Values.updateStrategy }} - strategy: {{- toYaml .Values.updateStrategy | nindent 4 }} - {{- end }} - replicas: {{ .Values.replicaCount }} - template: - metadata: - labels: {{- include "common.labels.standard" ( dict "customLabels" $podLabels "context" $ ) | nindent 8 }} - {{- if or .Values.podAnnotations (and .Values.metrics.enabled .Values.metrics.podAnnotations) }} - annotations: - {{- if .Values.podAnnotations }} - {{- include "common.tplvalues.render" (dict "value" .Values.podAnnotations "context" $) | nindent 8 }} - {{- end }} - {{- if and .Values.metrics.enabled .Values.metrics.podAnnotations }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.podAnnotations "context" $) | nindent 8 }} - {{- end }} - {{- end }} - spec: - {{- include "drupal.imagePullSecrets" . | nindent 6 }} - {{- if .Values.podSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.podSecurityContext "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.affinity }} - affinity: {{- include "common.tplvalues.render" (dict "value" .Values.affinity "context" $) | nindent 8 }} - {{- else }} - affinity: - podAffinity: {{- include "common.affinities.pods" (dict "type" .Values.podAffinityPreset "customLabels" $podLabels "context" $) | nindent 10 }} - podAntiAffinity: {{- include "common.affinities.pods" (dict "type" .Values.podAntiAffinityPreset "customLabels" $podLabels "context" $) | nindent 10 }} - nodeAffinity: {{- include "common.affinities.nodes" (dict "type" .Values.nodeAffinityPreset.type "key" .Values.nodeAffinityPreset.key "values" .Values.nodeAffinityPreset.values) | nindent 10 }} - {{- end }} - serviceAccountName: {{ include "drupal.serviceAccountName" . }} - {{- if .Values.schedulerName }} - schedulerName: {{ .Values.schedulerName }} - {{- end }} - {{- if .Values.topologySpreadConstraints }} - topologySpreadConstraints: {{- include "common.tplvalues.render" (dict "value" .Values.topologySpreadConstraints "context" .) | nindent 8 }} - {{- end }} - priorityClassName: {{ .Values.priorityClassName | quote }} - {{- if .Values.nodeSelector }} - nodeSelector: {{- include "common.tplvalues.render" (dict "value" .Values.nodeSelector "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.tolerations }} - tolerations: {{- include "common.tplvalues.render" (dict "value" .Values.tolerations "context" $) | nindent 8 }} - {{- end }} - automountServiceAccountToken: {{ .Values.automountServiceAccountToken }} - {{- if .Values.hostAliases }} - # yamllint disable rule:indentation - hostAliases: {{- include "common.tplvalues.render" (dict "value" .Values.hostAliases "context" $) | nindent 8 }} - # yamllint enable rule:indentation - {{- end }} - initContainers: - - name: prepare-base-dir - image: {{ include "drupal.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy | quote }} - {{- if .Values.resources }} - resources: {{- toYaml .Values.resources | nindent 12 }} - {{- else if ne .Values.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.resourcesPreset) | nindent 12 }} - {{- end }} - {{- if .Values.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - command: - - /bin/bash - args: - - -ec - - | - #!/bin/bash - - . /opt/bitnami/scripts/liblog.sh - . /opt/bitnami/scripts/libfs.sh - - info "Copying base dir to empty dir" - # In order to not break the application functionality (such as upgrades or plugins) we need - # to make the base directory writable, so we need to copy it to an empty dir volume - cp -r --preserve=mode /opt/bitnami/drupal /emptydir/app-base-dir - - info "Copying symlinks to stdout/stderr" - # We copy the logs folder because it has symlinks to stdout and stderr - if ! is_dir_empty /opt/bitnami/apache/logs; then - cp -r /opt/bitnami/apache/logs /emptydir/apache-logs-dir - fi - info "Copy operation completed" - volumeMounts: - - name: empty-dir - mountPath: /emptydir - {{- if and .Values.volumePermissions.enabled .Values.persistence.enabled }} - - name: volume-permissions - image: {{ include "drupal.volumePermissions.image" . }} - imagePullPolicy: {{ .Values.volumePermissions.image.pullPolicy | quote }} - command: - - sh - - -c - - | - mkdir -p "/bitnami/drupal" - chown -R "{{ .Values.containerSecurityContext.runAsUser }}:{{ .Values.podSecurityContext.fsGroup }}" "/bitnami/drupal" - securityContext: - runAsUser: 0 - {{- if .Values.volumePermissions.resources }} - resources: {{- toYaml .Values.volumePermissions.resources | nindent 12 }} - {{- else if ne .Values.volumePermissions.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.volumePermissions.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: drupal-data - mountPath: /bitnami/drupal - {{- end }} - {{- if .Values.certificates.customCAs }} - - name: certificates - image: {{ template "certificates.image" . }} - imagePullPolicy: {{ default .Values.image.pullPolicy .Values.certificates.image.pullPolicy }} - imagePullSecrets: - {{- range (default .Values.image.pullSecrets .Values.certificates.image.pullSecrets) }} - - name: {{ . }} - {{- end }} - command: - {{- if .Values.certificates.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.certificates.command "context" $) | nindent 12 }} - {{- else if .Values.certificates.customCertificate.certificateSecret }} - - sh - - -c - - install_packages ca-certificates openssl - {{- else }} - - sh - - -c - - install_packages ca-certificates openssl - && openssl req -new -x509 -days 3650 -nodes -sha256 - -subj "/CN=$(hostname)" -addext "subjectAltName = DNS:$(hostname)" - -out /etc/ssl/certs/ssl-cert-snakeoil.pem - -keyout /etc/ssl/private/ssl-cert-snakeoil.key -extensions v3_req - {{- end }} - {{- if .Values.certificates.args }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.certificates.args "context" $) | nindent 12 }} - {{- end }} - env: {{- include "common.tplvalues.render" (dict "value" .Values.certificates.extraEnvVars "context" $) | nindent 12 }} - envFrom: - {{- if .Values.certificates.extraEnvVarsCM }} - - configMapRef: - name: {{ include "common.tplvalues.render" (dict "value" .Values.certificates.extraEnvVarsCM "context" $) }} - {{- end }} - {{- if .Values.certificates.extraEnvVarsSecret }} - - secretRef: - name: {{ include "common.tplvalues.render" (dict "value" .Values.certificates.extraEnvVarsSecret "context" $) }} - {{- end }} - volumeMounts: - - name: etc-ssl-certs - mountPath: /etc/ssl/certs - readOnly: false - - name: etc-ssl-private - mountPath: /etc/ssl/private - readOnly: false - - name: custom-ca-certificates - mountPath: /usr/local/share/ca-certificates - readOnly: true - {{- end }} - {{- if .Values.initContainers }} - {{- include "common.tplvalues.render" (dict "value" .Values.initContainers "context" $) | nindent 8 }} - {{- end }} - containers: - - name: {{ include "common.names.fullname" . }} - image: {{ template "drupal.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy | quote }} - {{- if .Values.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.command "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.args }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.args "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - env: - - name: BITNAMI_DEBUG - value: {{ ternary "true" "false" .Values.image.debug | quote }} - - name: ALLOW_EMPTY_PASSWORD - value: {{ ternary "yes" "no" .Values.allowEmptyPassword | quote }} - - name: APACHE_HTTP_PORT_NUMBER - value: {{ .Values.containerPorts.http | quote }} - - name: APACHE_HTTPS_PORT_NUMBER - value: {{ .Values.containerPorts.https | quote }} - - name: DRUPAL_DATABASE_HOST - value: {{ include "drupal.databaseHost" . | quote }} - - name: DRUPAL_DATABASE_PORT_NUMBER - value: {{ include "drupal.databasePort" . | quote }} - - name: DRUPAL_DATABASE_NAME - value: {{ include "drupal.databaseName" . | quote }} - - name: DRUPAL_DATABASE_USER - value: {{ include "drupal.databaseUser" . | quote }} - - name: DRUPAL_DATABASE_PASSWORD - valueFrom: - secretKeyRef: - name: {{ include "drupal.databaseSecretName" . }} - key: {{ include "drupal.databasePasswordKey" . | quote }} - - name: DRUPAL_SKIP_BOOTSTRAP - value: {{ ternary "yes" "no" .Values.drupalSkipInstall | quote }} - - name: DRUPAL_PROFILE - value: {{ .Values.drupalProfile | quote }} - - name: DRUPAL_USERNAME - value: {{ .Values.drupalUsername | quote }} - - name: DRUPAL_PASSWORD - valueFrom: - secretKeyRef: - name: {{ include "drupal.secretName" . }} - key: drupal-password - - name: DRUPAL_EMAIL - value: {{ .Values.drupalEmail | quote }} - {{- if .Values.smtpHost }} - - name: SMTP_HOST - value: {{ .Values.smtpHost | quote }} - {{- end }} - {{- if .Values.smtpPort }} - - name: SMTP_PORT - value: {{ .Values.smtpPort | quote }} - {{- end }} - {{- if .Values.smtpUser }} - - name: SMTP_USER - value: {{ .Values.smtpUser | quote }} - {{- end }} - {{- if .Values.smtpPassword }} - - name: SMTP_PASSWORD - valueFrom: - secretKeyRef: - name: {{ include "drupal.secretName" . }} - key: smtp-password - {{- end }} - {{- if .Values.smtpProtocol }} - - name: SMTP_PROTOCOL - value: {{ .Values.smtpProtocol | quote }} - {{- end }} - {{- if .Values.extraEnvVars }} - {{- include "common.tplvalues.render" (dict "value" .Values.extraEnvVars "context" $) | nindent 12 }} - {{- end }} - envFrom: - {{- if .Values.extraEnvVarsCM }} - - configMapRef: - name: {{ include "common.tplvalues.render" (dict "value" .Values.extraEnvVarsCM "context" $) }} - {{- end }} - {{- if .Values.extraEnvVarsSecret }} - - secretRef: - name: {{ include "common.tplvalues.render" (dict "value" .Values.extraEnvVarsSecret "context" $) }} - {{- end }} - {{- if .Values.lifecycleHooks }} - lifecycle: {{- include "common.tplvalues.render" (dict "value" .Values.lifecycleHooks "context" $) | nindent 12 }} - {{- end }} - ports: - - name: http - containerPort: {{ .Values.containerPorts.http }} - - name: https - containerPort: {{ .Values.containerPorts.https }} - {{- if .Values.extraContainerPorts }} - {{- include "common.tplvalues.render" (dict "value" .Values.extraContainerPorts "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.startupProbe.enabled }} - startupProbe: - httpGet: - path: {{ .Values.startupProbe.path }} - port: http - initialDelaySeconds: {{ .Values.startupProbe.initialDelaySeconds }} - periodSeconds: {{ .Values.startupProbe.periodSeconds }} - timeoutSeconds: {{ .Values.startupProbe.timeoutSeconds }} - successThreshold: {{ .Values.startupProbe.successThreshold }} - failureThreshold: {{ .Values.startupProbe.failureThreshold }} - {{- end }} - {{- if .Values.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.livenessProbe.enabled }} - livenessProbe: - tcpSocket: - port: http - initialDelaySeconds: {{ .Values.livenessProbe.initialDelaySeconds }} - periodSeconds: {{ .Values.livenessProbe.periodSeconds }} - timeoutSeconds: {{ .Values.livenessProbe.timeoutSeconds }} - successThreshold: {{ .Values.livenessProbe.successThreshold }} - failureThreshold: {{ .Values.livenessProbe.failureThreshold }} - {{- end }} - {{- if .Values.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.readinessProbe.enabled }} - readinessProbe: - httpGet: - path: {{ .Values.readinessProbe.path }} - port: http - initialDelaySeconds: {{ .Values.readinessProbe.initialDelaySeconds }} - periodSeconds: {{ .Values.readinessProbe.periodSeconds }} - timeoutSeconds: {{ .Values.readinessProbe.timeoutSeconds }} - successThreshold: {{ .Values.readinessProbe.successThreshold }} - failureThreshold: {{ .Values.readinessProbe.failureThreshold }} - {{- end }} - {{- if .Values.resources }} - resources: {{- toYaml .Values.resources | nindent 12 }} - {{- else if ne .Values.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /opt/bitnami/apache/conf - subPath: apache-conf-dir - - name: empty-dir - mountPath: /opt/bitnami/apache/logs - subPath: apache-logs-dir - - name: empty-dir - mountPath: /opt/bitnami/apache/var/run - subPath: apache-tmp-dir - - name: empty-dir - mountPath: /opt/bitnami/php/etc - subPath: php-conf-dir - - name: empty-dir - mountPath: /opt/bitnami/php/tmp - subPath: php-tmp-dir - - name: empty-dir - mountPath: /opt/bitnami/php/var - subPath: php-var-dir - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: empty-dir - mountPath: /opt/bitnami/drupal - subPath: app-base-dir - - name: drupal-data - mountPath: /bitnami/drupal - {{- if .Values.extraVolumeMounts }} - {{- include "common.tplvalues.render" (dict "value" .Values.extraVolumeMounts "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.metrics.enabled }} - - name: metrics - image: {{ template "drupal.metrics.image" . }} - imagePullPolicy: {{ .Values.metrics.image.pullPolicy | quote }} - command: [ '/bin/apache_exporter', '--scrape_uri', 'http://status.localhost:{{ .Values.containerPorts.http }}/server-status/?auto' ] - ports: - - name: metrics - containerPort: 9117 - livenessProbe: - httpGet: - path: /metrics - port: metrics - initialDelaySeconds: 15 - timeoutSeconds: 5 - readinessProbe: - httpGet: - path: /metrics - port: metrics - initialDelaySeconds: 5 - timeoutSeconds: 1 - {{- if .Values.metrics.resourcesPreset }} - resources: {{- toYaml .Values.metrics.resources | nindent 12 }} - {{- else if ne .Values.metrics.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.metrics.resourcesPreset) | nindent 12 }} - {{- end }} - {{- end }} - {{- if .Values.sidecars }} - {{- include "common.tplvalues.render" (dict "value" .Values.sidecars "context" $) | nindent 8 }} - {{- end }} - volumes: - - name: empty-dir - emptyDir: {} - - name: drupal-data - {{- if .Values.persistence.enabled }} - persistentVolumeClaim: - claimName: {{ .Values.persistence.existingClaim | default (printf "%s-drupal" (include "common.names.fullname" .)) }} - {{- else }} - emptyDir: {} - {{- end }} - {{- if .Values.extraVolumes }} - {{- include "common.tplvalues.render" (dict "value" .Values.extraVolumes "context" $) | nindent 8 }} - {{- end }} diff --git a/charts/drupal/drupal/templates/externaldb-secrets.yaml b/charts/drupal/drupal/templates/externaldb-secrets.yaml deleted file mode 100644 index bfdd266..0000000 --- a/charts/drupal/drupal/templates/externaldb-secrets.yaml +++ /dev/null @@ -1,19 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and (not .Values.mariadb.enabled) (not .Values.externalDatabase.existingSecret) }} -apiVersion: v1 -kind: Secret -metadata: - name: {{ printf "%s-externaldb" (include "common.names.fullname" .) | trunc 63 | trimSuffix "-" }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: Opaque -data: - db-password: {{ default "" .Values.externalDatabase.password | b64enc | quote }} -{{- end }} diff --git a/charts/drupal/drupal/templates/extra-list.yaml b/charts/drupal/drupal/templates/extra-list.yaml deleted file mode 100644 index 329f5c6..0000000 --- a/charts/drupal/drupal/templates/extra-list.yaml +++ /dev/null @@ -1,9 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- range .Values.extraDeploy }} ---- -{{ include "common.tplvalues.render" (dict "value" . "context" $) }} -{{- end }} diff --git a/charts/drupal/drupal/templates/ingress.yaml b/charts/drupal/drupal/templates/ingress.yaml deleted file mode 100644 index a0a3b8e..0000000 --- a/charts/drupal/drupal/templates/ingress.yaml +++ /dev/null @@ -1,66 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.ingress.enabled }} -apiVersion: {{ include "common.capabilities.ingress.apiVersion" . }} -kind: Ingress -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - annotations: - {{- if .Values.ingress.certManager }} - kubernetes.io/tls-acme: "true" - {{- end }} - {{- if or .Values.ingress.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.ingress.annotations .Values.commonAnnotations ) "context" . ) }} - {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - {{- if and .Values.ingress.ingressClassName (eq "true" (include "common.ingress.supportsIngressClassname" .)) }} - ingressClassName: {{ .Values.ingress.ingressClassName | quote }} - {{- end }} - rules: - {{- if .Values.ingress.hostname }} - - host: {{ .Values.ingress.hostname }} - http: - paths: - - path: {{ .Values.ingress.path }} - {{- if eq "true" (include "common.ingress.supportsPathType" .) }} - pathType: {{ .Values.ingress.pathType }} - {{- end }} - backend: {{- include "common.ingress.backend" (dict "serviceName" (include "common.names.fullname" .) "servicePort" "http" "context" $) | nindent 14 }} - {{- if .Values.ingress.extraPaths }} - {{- toYaml .Values.ingress.extraPaths | nindent 10 }} - {{- end }} - {{- end }} - {{- range (coalesce .Values.ingress.extraHosts .Values.ingress.hosts) }} - - host: {{ .name | quote }} - http: - paths: - - path: {{ default "/" .path }} - {{- if eq "true" (include "common.ingress.supportsPathType" $) }} - pathType: {{ default "ImplementationSpecific" .pathType }} - {{- end }} - backend: {{- include "common.ingress.backend" (dict "serviceName" (include "common.names.fullname" $) "servicePort" "http" "context" $) | nindent 14 }} - {{- end }} - {{- if .Values.ingress.extraRules }} - {{- include "common.tplvalues.render" (dict "value" .Values.ingress.extraRules "context" $) | nindent 4 }} - {{- end }} - {{- if or .Values.ingress.tls .Values.ingress.extraTls }} - tls: - {{- if .Values.ingress.tls }} - - hosts: - - {{ .Values.ingress.hostname | quote }} - {{- if and (or (.Values.ingress.tlsWwwPrefix) (eq (index .Values.ingress.annotations "nginx.ingress.kubernetes.io/from-to-www-redirect") "true" )) (not (contains "www." .Values.ingress.hostname)) }} - - {{ printf "www.%s" (tpl .Values.ingress.hostname $) | quote }} - {{- end }} - secretName: {{ printf "%s-tls" .Values.ingress.hostname }} - {{- end }} - {{- if .Values.ingress.extraTls }} - {{- include "common.tplvalues.render" (dict "value" .Values.ingress.extraTls "context" $) | nindent 4 }} - {{- end }} - {{- end }} -{{- end }} diff --git a/charts/drupal/drupal/templates/metrics-svc.yaml b/charts/drupal/drupal/templates/metrics-svc.yaml deleted file mode 100644 index f95e56a..0000000 --- a/charts/drupal/drupal/templates/metrics-svc.yaml +++ /dev/null @@ -1,39 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.metrics.enabled }} -apiVersion: v1 -kind: Service -metadata: - name: {{ printf "%s-metrics" (include "common.names.fullname" .) | trunc 63 | trimSuffix "-" }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: metrics - {{- if or .Values.metrics.service.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.metrics.service.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: {{ .Values.metrics.service.type }} - {{- if eq .Values.metrics.service.type "LoadBalancer" }} - externalTrafficPolicy: {{ .Values.metrics.service.externalTrafficPolicy }} - {{- end }} - {{- if and (eq .Values.metrics.service.type "LoadBalancer") .Values.metrics.service.loadBalancerIP }} - loadBalancerIP: {{ .Values.metrics.service.loadBalancerIP }} - {{- end }} - {{- if and (eq .Values.metrics.service.type "LoadBalancer") .Values.metrics.service.loadBalancerSourceRanges }} - loadBalancerSourceRanges: {{- toYaml .Values.metrics.service.loadBalancerSourceRanges | nindent 4 }} - {{- end }} - ports: - - name: metrics - port: {{ .Values.metrics.service.ports.metrics }} - protocol: TCP - targetPort: metrics - {{- if .Values.metrics.service.extraPorts }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.service.extraPorts "context" $) | nindent 4 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} -{{- end }} diff --git a/charts/drupal/drupal/templates/networkpolicy.yaml b/charts/drupal/drupal/templates/networkpolicy.yaml deleted file mode 100644 index a0c01ac..0000000 --- a/charts/drupal/drupal/templates/networkpolicy.yaml +++ /dev/null @@ -1,80 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.networkPolicy.enabled }} -kind: NetworkPolicy -apiVersion: {{ include "common.capabilities.networkPolicy.apiVersion" . }} -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.podLabels .Values.commonLabels ) "context" . ) }} - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - policyTypes: - - Ingress - - Egress - {{- if .Values.networkPolicy.allowExternalEgress }} - egress: - - {} - {{- else }} - egress: - # Allow dns resolution - - ports: - - port: 53 - protocol: UDP - - port: 53 - protocol: TCP - # Allow outbound connections to MariaDB - - ports: - - port: {{ include "drupal.databasePort" . }} - {{- if .Values.mariadb.enabled }} - to: - - podSelector: - matchLabels: - app.kubernetes.io/name: mariadb - app.kubernetes.io/instance: {{ .Release.Name }} - {{- end }} - {{- if .Values.networkPolicy.extraEgress }} - {{- include "common.tplvalues.render" ( dict "value" .Values.networkPolicy.extraEgress "context" $ ) | nindent 4 }} - {{- end }} - {{- end }} - ingress: - - ports: - - port: {{ .Values.containerPorts.http }} - - port: {{ .Values.containerPorts.https }} - {{- range .Values.extraContainerPorts }} - - port: {{ . }} - {{- end }} - {{- if not .Values.networkPolicy.allowExternal }} - from: - - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 14 }} - - podSelector: - matchLabels: - {{ template "common.names.fullname" . }}-client: "true" - {{- if .Values.networkPolicy.ingressNSMatchLabels }} - - namespaceSelector: - matchLabels: - {{- range $key, $value := .Values.networkPolicy.ingressNSMatchLabels }} - {{ $key | quote }}: {{ $value | quote }} - {{- end }} - {{- if .Values.networkPolicy.ingressNSPodMatchLabels }} - podSelector: - matchLabels: - {{- range $key, $value := .Values.networkPolicy.ingressNSPodMatchLabels }} - {{ $key | quote }}: {{ $value | quote }} - {{- end }} - {{- end }} - {{- end }} - {{- end }} - {{- if .Values.networkPolicy.extraIngress }} - {{- include "common.tplvalues.render" ( dict "value" .Values.networkPolicy.extraIngress "context" $ ) | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/drupal/drupal/templates/pdb.yaml b/charts/drupal/drupal/templates/pdb.yaml deleted file mode 100644 index d28b3ec..0000000 --- a/charts/drupal/drupal/templates/pdb.yaml +++ /dev/null @@ -1,26 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.pdb.create }} -apiVersion: {{ include "common.capabilities.policy.apiVersion" . }} -kind: PodDisruptionBudget -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- if .Values.pdb.minAvailable }} - minAvailable: {{ .Values.pdb.minAvailable }} - {{- end }} - {{- if or .Values.pdb.maxUnavailable (not .Values.pdb.minAvailable) }} - maxUnavailable: {{ .Values.pdb.maxUnavailable | default 1 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" (dict "values" (list .Values.podLabels .Values.commonLabels) "context" .) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} -{{- end }} diff --git a/charts/drupal/drupal/templates/prometheusrule.yaml b/charts/drupal/drupal/templates/prometheusrule.yaml deleted file mode 100644 index 7293adb..0000000 --- a/charts/drupal/drupal/templates/prometheusrule.yaml +++ /dev/null @@ -1,23 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled .Values.metrics.prometheusRule.enabled }} -apiVersion: monitoring.coreos.com/v1 -kind: PrometheusRule -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ default .Release.Namespace .Values.metrics.prometheusRule.namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.metrics.prometheusRule.additionalLabels }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.prometheusRule.additionalLabels "context" $) | nindent 4 }} - {{- end }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - groups: - - name: {{ include "common.names.fullname" . }} - rules: {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.prometheusRule.rules "context" $ ) | nindent 8 }} -{{- end }} diff --git a/charts/drupal/drupal/templates/pv.yaml b/charts/drupal/drupal/templates/pv.yaml deleted file mode 100644 index 95ab4dd..0000000 --- a/charts/drupal/drupal/templates/pv.yaml +++ /dev/null @@ -1,23 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.persistence.enabled .Values.persistence.hostPath (not .Values.persistence.existingClaim) -}} -apiVersion: v1 -kind: PersistentVolume -metadata: - name: {{ printf "%s-drupal" (include "common.names.fullname" .) | trunc 63 | trimSuffix "-" }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - accessModes: - - {{ .Values.persistence.accessMode | quote }} - capacity: - storage: {{ .Values.persistence.size | quote }} - hostPath: - path: {{ .Values.persistence.hostPath | quote }} -{{- end -}} diff --git a/charts/drupal/drupal/templates/pvc.yaml b/charts/drupal/drupal/templates/pvc.yaml deleted file mode 100644 index 923a535..0000000 --- a/charts/drupal/drupal/templates/pvc.yaml +++ /dev/null @@ -1,33 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.persistence.enabled (not .Values.persistence.existingClaim) -}} -kind: PersistentVolumeClaim -apiVersion: v1 -metadata: - name: {{ printf "%s-drupal" (include "common.names.fullname" .) | trunc 63 | trimSuffix "-" }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if or .Values.persistence.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.persistence.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - {{- if .Values.persistence.hostPath }} - storageClassName: "" - {{- end }} - accessModes: - {{- if not (empty .Values.persistence.accessModes) }} - {{- range .Values.persistence.accessModes }} - - {{ . | quote }} - {{- end }} - {{- else }} - - {{ .Values.persistence.accessMode | quote }} - {{- end }} - resources: - requests: - storage: {{ .Values.persistence.size | quote }} - {{- include "drupal.storageClass" . | nindent 2 }} -{{- end -}} diff --git a/charts/drupal/drupal/templates/secrets.yaml b/charts/drupal/drupal/templates/secrets.yaml deleted file mode 100644 index 3a3e3f0..0000000 --- a/charts/drupal/drupal/templates/secrets.yaml +++ /dev/null @@ -1,26 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if not .Values.existingSecret }} -apiVersion: v1 -kind: Secret -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: Opaque -data: - {{- if .Values.drupalPassword }} - drupal-password: {{ default "" .Values.drupalPassword | b64enc | quote }} - {{- else }} - drupal-password: {{ randAlphaNum 10 | b64enc | quote }} - {{- end }} - {{- if .Values.smtpPassword }} - smtp-password: {{ .Values.smtpPassword | b64enc | quote }} - {{- end }} -{{- end }} diff --git a/charts/drupal/drupal/templates/serviceaccount.yaml b/charts/drupal/drupal/templates/serviceaccount.yaml deleted file mode 100644 index ad0ad4a..0000000 --- a/charts/drupal/drupal/templates/serviceaccount.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.serviceAccount.create }} -apiVersion: v1 -kind: ServiceAccount -metadata: - name: {{ include "drupal.serviceAccountName" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if or .Values.serviceAccount.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.serviceAccount.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -automountServiceAccountToken: {{ .Values.serviceAccount.automountServiceAccountToken }} -{{- end }} diff --git a/charts/drupal/drupal/templates/servicemonitor.yaml b/charts/drupal/drupal/templates/servicemonitor.yaml deleted file mode 100644 index 60dd04f..0000000 --- a/charts/drupal/drupal/templates/servicemonitor.yaml +++ /dev/null @@ -1,43 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled .Values.metrics.serviceMonitor.enabled }} -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ default .Release.Namespace .Values.metrics.serviceMonitor.namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.metrics.serviceMonitor.additionalLabels }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.serviceMonitor.additionalLabels "context" $) | nindent 4 }} - {{- end }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - endpoints: - - port: metrics - {{- if .Values.metrics.serviceMonitor.interval }} - interval: {{ .Values.metrics.serviceMonitor.interval }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.scrapeTimeout }} - scrapeTimeout: {{ .Values.metrics.serviceMonitor.scrapeTimeout }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.honorLabels }} - honorLabels: {{ .Values.metrics.serviceMonitor.honorLabels }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.relabellings }} - relabelings: {{- toYaml .Values.metrics.serviceMonitor.relabellings | nindent 6 }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.metricRelabelings }} - metricRelabelings: {{- toYaml .Values.metrics.serviceMonitor.metricRelabelings | nindent 6 }} - {{- end }} - namespaceSelector: - matchNames: - - {{ .Release.Namespace }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: metrics -{{- end }} diff --git a/charts/drupal/drupal/templates/svc.yaml b/charts/drupal/drupal/templates/svc.yaml deleted file mode 100644 index 20b70ef..0000000 --- a/charts/drupal/drupal/templates/svc.yaml +++ /dev/null @@ -1,57 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -apiVersion: v1 -kind: Service -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if or .Values.service.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.service.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: {{ .Values.service.type }} - {{- if .Values.service.sessionAffinity }} - sessionAffinity: {{ .Values.service.sessionAffinity }} - {{- end }} - {{- if .Values.service.sessionAffinityConfig }} - sessionAffinityConfig: {{- include "common.tplvalues.render" (dict "value" .Values.service.sessionAffinityConfig "context" $) | nindent 4 }} - {{- end }} - {{- if and .Values.service.clusterIP (eq .Values.service.type "ClusterIP") }} - clusterIP: {{ .Values.service.clusterIP }} - {{- end }} - {{- if (and (eq .Values.service.type "LoadBalancer") (not (empty .Values.service.loadBalancerIP))) }} - loadBalancerIP: {{ .Values.service.loadBalancerIP }} - {{- end }} - {{- if and (eq .Values.service.type "LoadBalancer") .Values.service.loadBalancerSourceRanges }} - loadBalancerSourceRanges: {{- toYaml .Values.service.loadBalancerSourceRanges | nindent 4 }} - {{- end }} - {{- if or (eq .Values.service.type "LoadBalancer") (eq .Values.service.type "NodePort") }} - externalTrafficPolicy: {{ .Values.service.externalTrafficPolicy | quote }} - {{- end }} - ports: - - name: http - port: {{ coalesce .Values.service.ports.http .Values.service.port }} - targetPort: http - {{- if and (or (eq .Values.service.type "NodePort") (eq .Values.service.type "LoadBalancer")) (not (empty .Values.service.nodePorts.http)) }} - nodePort: {{ .Values.service.nodePorts.http }} - {{- else if eq .Values.service.type "ClusterIP" }} - nodePort: null - {{- end }} - - name: https - port: {{ coalesce .Values.service.ports.https .Values.service.httpsPort }} - targetPort: https - {{- if and (or (eq .Values.service.type "NodePort") (eq .Values.service.type "LoadBalancer")) (not (empty .Values.service.nodePorts.https)) }} - nodePort: {{ .Values.service.nodePorts.https }} - {{- else if eq .Values.service.type "ClusterIP" }} - nodePort: null - {{- end }} - {{- if .Values.service.extraPorts }} - {{- include "common.tplvalues.render" (dict "value" .Values.service.extraPorts "context" $) | nindent 4 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} diff --git a/charts/drupal/drupal/templates/tls-secrets.yaml b/charts/drupal/drupal/templates/tls-secrets.yaml deleted file mode 100644 index 2b2dfd8..0000000 --- a/charts/drupal/drupal/templates/tls-secrets.yaml +++ /dev/null @@ -1,22 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.ingress.enabled }} -{{- range .Values.ingress.secrets }} -apiVersion: v1 -kind: Secret -metadata: - name: {{ .name }} - namespace: {{ include "common.names.namespace" $ | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $.Values.commonLabels "context" $ ) | nindent 4 }} - {{- if $.Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $.Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: kubernetes.io/tls -data: - tls.crt: {{ .certificate | b64enc }} - tls.key: {{ .key | b64enc }} -{{- end }} -{{- end }} diff --git a/charts/drupal/drupal/values.schema.json b/charts/drupal/drupal/values.schema.json deleted file mode 100644 index 072fbcd..0000000 --- a/charts/drupal/drupal/values.schema.json +++ /dev/null @@ -1,233 +0,0 @@ -{ - "$schema": "http://json-schema.org/schema#", - "type": "object", - "properties": { - "drupalUsername": { - "type": "string", - "title": "Username", - "form": true - }, - "drupalPassword": { - "type": "string", - "title": "Password", - "form": true, - "description": "Defaults to a random 10-character alphanumeric string if not set" - }, - "drupalEmail": { - "type": "string", - "title": "Admin email", - "form": true - }, - "persistence": { - "type": "object", - "properties": { - "drupal": { - "type": "object", - "properties": { - "size": { - "type": "string", - "title": "Persistent Volume Size", - "form": true, - "render": "slider", - "sliderMin": 1, - "sliderMax": 100, - "sliderUnit": "Gi" - } - } - } - } - }, - "ingress": { - "type": "object", - "form": true, - "title": "Ingress Configuration", - "properties": { - "enabled": { - "type": "boolean", - "form": true, - "title": "Use a custom hostname", - "description": "Enable the ingress resource that allows you to access the Drupal installation." - }, - "hostname": { - "type": "string", - "form": true, - "title": "Hostname", - "hidden": { - "value": false, - "path": "ingress/enabled" - } - } - } - }, - "service": { - "type": "object", - "form": true, - "title": "Service Configuration", - "properties": { - "type": { - "type": "string", - "form": true, - "title": "Service Type", - "description": "Allowed values: \"ClusterIP\", \"NodePort\" and \"LoadBalancer\"" - } - } - }, - "mariadb": { - "type": "object", - "title": "MariaDB Details", - "form": true, - "properties": { - "enabled": { - "type": "boolean", - "title": "Use a new MariaDB database hosted in the cluster", - "form": true, - "description": "Whether to deploy a mariadb server to satisfy the applications database requirements. To use an external database switch this off and configure the external database details" - }, - "primary": { - "type": "object", - "properties": { - "persistence": { - "type": "object", - "properties": { - "size": { - "type": "string", - "title": "Volume Size", - "form": true, - "hidden": { - "value": false, - "path": "mariadb/enabled" - }, - "render": "slider", - "sliderMin": 1, - "sliderMax": 100, - "sliderUnit": "Gi" - } - } - } - } - } - } - }, - "externalDatabase": { - "type": "object", - "title": "External Database Details", - "description": "If MariaDB is disabled. Use this section to specify the external database details", - "form": true, - "hidden": "mariadb/enabled", - "properties": { - "host": { - "type": "string", - "form": true, - "title": "Database Host" - }, - "user": { - "type": "string", - "form": true, - "title": "Database Username" - }, - "password": { - "type": "string", - "form": true, - "title": "Database Password" - }, - "database": { - "type": "string", - "form": true, - "title": "Database Name" - }, - "port": { - "type": "integer", - "form": true, - "title": "Database Port" - } - } - }, - "resources": { - "type": "object", - "title": "Requested Resources", - "description": "Configure resource requests", - "form": true, - "properties": { - "requests": { - "type": "object", - "properties": { - "memory": { - "type": "string", - "form": true, - "render": "slider", - "title": "Memory Request", - "sliderMin": 10, - "sliderMax": 2048, - "sliderUnit": "Mi" - }, - "cpu": { - "type": "string", - "form": true, - "render": "slider", - "title": "CPU Request", - "sliderMin": 10, - "sliderMax": 2000, - "sliderUnit": "m" - } - } - } - } - }, - "metrics": { - "type": "object", - "properties": { - "enabled": { - "type": "boolean", - "title": "Enable Metrics", - "description": "Prometheus Exporter / Metrics", - "form": true - }, - "serviceMonitor": { - "type": "object", - "properties": { - "enabled": { - "type": "boolean", - "title": "Create Prometheus Operator ServiceMonitor", - "description": "Create a ServiceMonitor to track metrics using Prometheus Operator", - "form": true, - "hidden": { - "value": false, - "path": "metrics/enabled" - } - } - } - } - } - }, - "serviceAccount": { - "type": "object", - "properties": { - "drupal": { - "type": "object", - "properties": { - "create": { - "type": "string", - "title": "Specifies whether a service account should be created", - "form": true - }, - "name": { - "type": "string", - "title": "The name of the service account to use", - "form": true - }, - "annotations": { - "type": "string", - "title": "Add annotations", - "form": true - }, - "automountServiceAccountToken": { - "type": "string", - "title": "Automount API credentials for a service account", - "form": true - } - } - } - } - } - } -} diff --git a/charts/drupal/drupal/values.yaml b/charts/drupal/drupal/values.yaml deleted file mode 100644 index 554dbc3..0000000 --- a/charts/drupal/drupal/values.yaml +++ /dev/null @@ -1,982 +0,0 @@ -# Copyright Broadcom, Inc. All Rights Reserved. -# SPDX-License-Identifier: APACHE-2.0 - -## @section Global parameters -## Global Docker image parameters -## Please, note that this will override the image parameters, including dependencies, configured to use the global value -## Current available global Docker image parameters: imageRegistry, imagePullSecrets and storageClass - -## @param global.imageRegistry Global Docker image registry -## @param global.imagePullSecrets Global Docker registry secret names as an array -## @param global.defaultStorageClass Global default StorageClass for Persistent Volume(s) -## @param global.storageClass DEPRECATED: use global.defaultStorageClass instead -## -global: - imageRegistry: "" - ## E.g. - ## imagePullSecrets: - ## - myRegistryKeySecretName - ## - imagePullSecrets: [] - defaultStorageClass: "" - storageClass: "" - ## Compatibility adaptations for Kubernetes platforms - ## - compatibility: - ## Compatibility adaptations for Openshift - ## - openshift: - ## @param global.compatibility.openshift.adaptSecurityContext Adapt the securityContext sections of the deployment to make them compatible with Openshift restricted-v2 SCC: remove runAsUser, runAsGroup and fsGroup and let the platform use their allowed default IDs. Possible values: auto (apply if the detected running cluster is Openshift), force (perform the adaptation always), disabled (do not perform adaptation) - ## - adaptSecurityContext: auto -## @section Common parameters - -## @param kubeVersion Force target Kubernetes version (using Helm capabilities if not set) -## -kubeVersion: "" -## @param nameOverride String to partially override drupal.fullname template (will maintain the release name) -## -nameOverride: "" -## @param fullnameOverride String to fully override drupal.fullname template -## -fullnameOverride: "" -## @param namespaceOverride String to fully override common.names.namespace -## -namespaceOverride: "" -## @param commonAnnotations Common annotations to add to all Drupal resources (sub-charts are not considered). Evaluated as a template -## -commonAnnotations: {} -## @param commonLabels Common labels to add to all Drupal resources (sub-charts are not considered). Evaluated as a template -## -commonLabels: {} -## @param extraDeploy Array of extra objects to deploy with the release (evaluated as a template). -## -extraDeploy: [] -## @section Drupal parameters - -## Bitnami Drupal image version -## ref: https://hub.docker.com/r/bitnami/drupal/tags/ -## @param image.registry [default: REGISTRY_NAME] Drupal image registry -## @param image.repository [default: REPOSITORY_NAME/drupal] Drupal Image name -## @skip image.tag Drupal Image tag -## @param image.digest Drupal image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag -## @param image.pullPolicy Drupal image pull policy -## @param image.pullSecrets Specify docker-registry secret names as an array -## @param image.debug Specify if debug logs should be enabled -## -image: - registry: docker.io - repository: bitnami/drupal - tag: 11.0.5-debian-12-r1 - digest: "" - ## Specify a imagePullPolicy - ## Defaults to 'Always' if image tag is 'latest', else set to 'IfNotPresent' - ## ref: https://kubernetes.io/docs/concepts/containers/images/#pre-pulled-images - ## - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets. - ## Secrets must be manually created in the namespace. - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## e.g: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## Set to true if you would like to see extra information on logs - ## - debug: false -## @param replicaCount Number of Drupal Pods to run (requires ReadWriteMany PVC support) -## -replicaCount: 1 -## @param drupalProfile Drupal installation profile -## ref: https://github.com/bitnami/containers/tree/main/bitnami/drupal#configuration -## -drupalProfile: standard -## @param drupalSkipInstall Skip Drupal installation wizard. Useful for migrations and restoring from SQL dump -## ref: https://github.com/bitnami/containers/tree/main/bitnami/drupal#configuration -## -drupalSkipInstall: false -## @param drupalUsername User of the application -## ref: https://github.com/bitnami/containers/tree/main/bitnami/drupal#configuration -## -drupalUsername: user -## @param drupalPassword Application password -## Defaults to a random 10-character alphanumeric string if not set -## ref: https://github.com/bitnami/containers/tree/main/bitnami/drupal#configuration -## -drupalPassword: "" -## @param drupalEmail Admin email -## ref: https://github.com/bitnami/containers/tree/main/bitnami/drupal#configuration -## -drupalEmail: user@example.com -## @param allowEmptyPassword Allow DB blank passwords -## ref: https://github.com/bitnami/containers/tree/main/bitnami/drupal#environment-variables -## -allowEmptyPassword: true -## @param command Override default container command (useful when using custom images) -## -command: [] -## @param args Override default container args (useful when using custom images) -## -args: [] -## @param updateStrategy.type Update strategy - only really applicable for deployments with RWO PVs attached -## If replicas = 1, an update can get "stuck", as the previous pod remains attached to the -## PV, and the "incoming" pod can never start. Changing the strategy to "Recreate" will -## terminate the single previous pod, so that the new, incoming pod can attach to the PV -## -updateStrategy: - type: RollingUpdate -## @param priorityClassName Drupal pods' priorityClassName -## -priorityClassName: "" -## @param schedulerName Name of the k8s scheduler (other than default) -## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ -## -schedulerName: "" -## @param topologySpreadConstraints Topology Spread Constraints for pod assignment -## https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/ -## The value is evaluated as a template -## -topologySpreadConstraints: [] -## @param automountServiceAccountToken Mount Service Account token in pod -## -automountServiceAccountToken: false -## @param hostAliases [array] Add deployment host aliases -## https://kubernetes.io/docs/concepts/services-networking/add-entries-to-pod-etc-hosts-with-host-aliases/ -## -hostAliases: - ## Necessary for apache-exporter to work - ## - - ip: "127.0.0.1" - hostnames: - - "status.localhost" -## @param extraEnvVars Extra environment variables -## For example: -## -extraEnvVars: [] -# - name: BEARER_AUTH -# value: true -## @param extraEnvVarsCM ConfigMap containing extra env vars -## -extraEnvVarsCM: "" -## @param extraEnvVarsSecret Secret containing extra env vars (in case of sensitive data) -## -extraEnvVarsSecret: "" -## @param extraVolumes Array of extra volumes to be added to the deployment (evaluated as template). Requires setting `extraVolumeMounts` -## -extraVolumes: [] -## @param extraVolumeMounts Array of extra volume mounts to be added to the container (evaluated as template). Normally used with `extraVolumes`. -## -extraVolumeMounts: [] -## @param initContainers Add additional init containers to the pod (evaluated as a template) -## -initContainers: [] -## Pod Disruption Budget configuration -## ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb -## @param pdb.create Enable/disable a Pod Disruption Budget creation -## @param pdb.minAvailable Minimum number/percentage of pods that should remain scheduled -## @param pdb.maxUnavailable Maximum number/percentage of pods that may be made unavailable. Defaults to `1` if both `pdb.minAvailable` and `pdb.maxUnavailable` are empty. -## -pdb: - create: true - minAvailable: "" - maxUnavailable: "" -## @param sidecars Attach additional containers to the pod (evaluated as a template) -## -sidecars: [] -## @param tolerations Tolerations for pod assignment -## Ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ -## -tolerations: [] -serviceAccount: - ## @param serviceAccount.create Specifies whether a service account should be created - ## - create: true - ## @param serviceAccount.name The name of the service account to use. If not set and create is true, a name is generated using the fullname template - ## - name: "" - ## @param serviceAccount.annotations Add annotations - ## - annotations: {} - ## @param serviceAccount.automountServiceAccountToken Automount API credentials for a service account. - ## - automountServiceAccountToken: false -## @param existingSecret Name of a secret with the application password -## -existingSecret: "" -## SMTP mail delivery configuration -## ref: https://github.com/bitnami/containers/tree/main/bitnami/drupal/#smtp-configuration -## @param smtpHost SMTP host -## @param smtpPort SMTP port -## @param smtpUser SMTP user -## @param smtpPassword SMTP password -## @param smtpProtocol SMTP Protocol (options: ssl,tls, nil) -## -smtpHost: "" -smtpPort: "" -smtpUser: "" -smtpPassword: "" -smtpProtocol: "" -## @param containerPorts [object] Container ports -## -containerPorts: - http: 8080 - https: 8443 -## @param extraContainerPorts Optionally specify extra list of additional ports for Drupal container(s) -## e.g: -## extraContainerPorts: -## - name: myservice -## containerPort: 9090 -## -extraContainerPorts: [] -## @param sessionAffinity Control where client requests go, to the same pod or round-robin. Values: ClientIP or None -## ref: https://kubernetes.io/docs/concepts/services-networking/service/ -## -sessionAffinity: "None" -## Enable persistence using Persistent Volume Claims -## ref: https://kubernetes.io/docs/concepts/storage/persistent-volumes/ -## -persistence: - ## @param persistence.enabled Enable persistence using PVC - ## - enabled: true - ## @param persistence.storageClass PVC Storage Class for Drupal volume - ## If defined, storageClassName: - ## If set to "-", storageClassName: "", which disables dynamic provisioning - ## If undefined (the default) or set to null, no storageClassName spec is - ## set, choosing the default provisioner. (gp2 on AWS, standard on - ## GKE, AWS & OpenStack) - ## - storageClass: "" - ## @param persistence.accessModes PVC Access Mode for Drupal volume - ## Requires persistence.enabled: true - ## If defined, PVC must be created manually before volume will be bound - ## - accessModes: - - ReadWriteOnce - ## @param persistence.size PVC Storage Request for Drupal volume - ## - size: 8Gi - ## @param persistence.existingClaim A manually managed Persistent Volume Claim - ## Requires persistence.enabled: true - ## If defined, PVC must be created manually before volume will be bound - ## - existingClaim: "" - ## @param persistence.hostPath If defined, the drupal-data volume will mount to the specified hostPath. - ## Requires persistence.enabled: true - ## Requires persistence.existingClaim: nil|false - ## Default: nil. - ## - hostPath: "" - ## @param persistence.annotations Persistent Volume Claim annotations - ## - annotations: {} -## @param podAffinityPreset Pod affinity preset. Ignored if `affinity` is set. Allowed values: `soft` or `hard` -## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity -## -podAffinityPreset: "" -## @param podAntiAffinityPreset Pod anti-affinity preset. Ignored if `affinity` is set. Allowed values: `soft` or `hard` -## Ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity -## -podAntiAffinityPreset: soft -## Node affinity preset -## Ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity -## @param nodeAffinityPreset.type Node affinity preset type. Ignored if `affinity` is set. Allowed values: `soft` or `hard` -## @param nodeAffinityPreset.key Node label key to match Ignored if `affinity` is set. -## @param nodeAffinityPreset.values Node label values to match. Ignored if `affinity` is set. -## -nodeAffinityPreset: - type: "" - ## E.g. - ## key: "kubernetes.io/e2e-az-name" - ## - key: "" - ## E.g. - ## values: - ## - e2e-az1 - ## - e2e-az2 - ## - values: [] -## @param affinity Affinity for pod assignment -## Ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity -## Note: podAffinityPreset, podAntiAffinityPreset, and nodeAffinityPreset will be ignored when it's set -## -affinity: {} -## @param nodeSelector Node labels for pod assignment. Evaluated as a template. -## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/ -## -nodeSelector: {} -## Drupal container's resource requests and limits -## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ -## We usually recommend not to specify default resources and to leave this as a conscious -## choice for the user. This also increases chances charts run on environments with little -## resources, such as Minikube. If you do want to specify resources, uncomment the following -## lines, adjust them as necessary, and remove the curly braces after 'resources:'. -## @param resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if resources is set (resources is recommended for production). -## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 -## -resourcesPreset: "micro" -## @param resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) -## Example: -## resources: -## requests: -## cpu: 2 -## memory: 512Mi -## limits: -## cpu: 3 -## memory: 1024Mi -## -resources: {} -## Configure Pods Security Context -## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-pod -## @param podSecurityContext.enabled Enable Drupal pods' Security Context -## @param podSecurityContext.fsGroupChangePolicy Set filesystem group change policy -## @param podSecurityContext.sysctls Set kernel settings using the sysctl interface -## @param podSecurityContext.supplementalGroups Set filesystem extra groups -## @param podSecurityContext.fsGroup Drupal pods' group ID -## -podSecurityContext: - enabled: true - fsGroupChangePolicy: Always - sysctls: [] - supplementalGroups: [] - fsGroup: 1001 -## Configure Container Security Context (only main container) -## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container -## @param containerSecurityContext.enabled Enabled Drupal containers' Security Context -## @param containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container -## @param containerSecurityContext.runAsUser Set Drupal containers' Security Context runAsUser -## @param containerSecurityContext.runAsGroup Set Drupal containers' Security Context runAsGroup -## @param containerSecurityContext.runAsNonRoot Set Controller container's Security Context runAsNonRoot -## @param containerSecurityContext.privileged Set Drupal container's Security Context privileged -## @param containerSecurityContext.allowPrivilegeEscalation Set Drupal container's Security Context allowPrivilegeEscalation -## @param containerSecurityContext.capabilities.drop List of capabilities to be dropped -## @param containerSecurityContext.seccompProfile.type Set container's Security Context seccomp profile -## @param containerSecurityContext.readOnlyRootFilesystem Set Drupal container's Security Context readOnlyRootFilesystem -## -containerSecurityContext: - enabled: true - seLinuxOptions: {} - runAsUser: 1001 - runAsGroup: 1001 - runAsNonRoot: true - privileged: false - allowPrivilegeEscalation: false - readOnlyRootFilesystem: true - capabilities: - drop: ["ALL"] - seccompProfile: - type: "RuntimeDefault" -## Configure extra options for startup probe -## Drupal core exposes /user/login to unauthenticated requests, making it a good -## default startup and readiness path. However, that may not always be the -## case. For example, if the image value is overridden to an image containing a -## module that alters that route, or an image that does not auto-install Drupal. -## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes -## @param startupProbe.enabled Enable startupProbe -## @param startupProbe.path Request path for startupProbe -## @param startupProbe.initialDelaySeconds Initial delay seconds for startupProbe -## @param startupProbe.periodSeconds Period seconds for startupProbe -## @param startupProbe.timeoutSeconds Timeout seconds for startupProbe -## @param startupProbe.failureThreshold Failure threshold for startupProbe -## @param startupProbe.successThreshold Success threshold for startupProbe -## -startupProbe: - enabled: false - path: /user/login - initialDelaySeconds: 600 - periodSeconds: 10 - timeoutSeconds: 5 - failureThreshold: 5 - successThreshold: 1 -## Configure extra options for liveness probe -## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes -## @param livenessProbe.enabled Enable livenessProbe -## @param livenessProbe.initialDelaySeconds Initial delay seconds for livenessProbe -## @param livenessProbe.periodSeconds Period seconds for livenessProbe -## @param livenessProbe.timeoutSeconds Timeout seconds for livenessProbe -## @param livenessProbe.failureThreshold Failure threshold for livenessProbe -## @param livenessProbe.successThreshold Success threshold for livenessProbe -## -livenessProbe: - enabled: true - initialDelaySeconds: 600 - periodSeconds: 10 - timeoutSeconds: 5 - failureThreshold: 5 - successThreshold: 1 -## Configure extra options for readiness probe -## Drupal core exposes /user/login to unauthenticated requests, making it a good -## default startup and readiness path. However, that may not always be the -## case. For example, if the image value is overridden to an image containing a -## module that alters that route, or an image that does not auto-install Drupal. -## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes -## @param readinessProbe.enabled Enable readinessProbe -## @param readinessProbe.path Request path for readinessProbe -## @param readinessProbe.initialDelaySeconds Initial delay seconds for readinessProbe -## @param readinessProbe.periodSeconds Period seconds for readinessProbe -## @param readinessProbe.timeoutSeconds Timeout seconds for readinessProbe -## @param readinessProbe.failureThreshold Failure threshold for readinessProbe -## @param readinessProbe.successThreshold Success threshold for readinessProbe -## -readinessProbe: - enabled: true - path: /user/login - initialDelaySeconds: 30 - periodSeconds: 5 - timeoutSeconds: 1 - failureThreshold: 5 - successThreshold: 1 -## @param customStartupProbe Override default startup probe -## -customStartupProbe: {} -## @param customLivenessProbe Override default liveness probe -## -customLivenessProbe: {} -## @param customReadinessProbe Override default readiness probe -## -customReadinessProbe: {} -## @param lifecycleHooks LifecycleHook to set additional configuration at startup Evaluated as a template -## -lifecycleHooks: {} -## @param podAnnotations Pod annotations -## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ -## -podAnnotations: {} -## @param podLabels Add additional labels to the pod (evaluated as a template) -## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/ -## -podLabels: {} -## @section Traffic Exposure Parameters - -## Kubernetes configuration. For minikube, set this to NodePort, elsewhere use LoadBalancer -## -service: - ## @param service.type Kubernetes Service type - ## - type: LoadBalancer - ## @param service.ports.http Service HTTP port - ## @param service.ports.https Service HTTPS port - ## - ports: - http: 80 - https: 443 - ## @param service.loadBalancerSourceRanges Restricts access for LoadBalancer (only with `service.type: LoadBalancer`) - ## e.g: - ## loadBalancerSourceRanges: - ## - 0.0.0.0/0 - ## - loadBalancerSourceRanges: [] - ## @param service.loadBalancerIP loadBalancerIP for the Drupal Service (optional, cloud specific) - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-loadbalancer - loadBalancerIP: "" - ## @param service.nodePorts [object] Kubernetes node port - ## nodePorts: - ## http: - ## https: - ## - nodePorts: - http: "" - https: "" - ## @param service.externalTrafficPolicy Enable client source IP preservation - ## ref https://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer/#preserving-the-client-source-ip - ## - externalTrafficPolicy: Cluster - ## @param service.clusterIP %%MAIN_CONTAINER_NAME%% service Cluster IP - ## e.g.: - ## clusterIP: None - ## - clusterIP: "" - ## @param service.extraPorts Extra ports to expose (normally used with the `sidecar` value) - ## - extraPorts: [] - ## @param service.annotations Additional custom annotations for %%MAIN_CONTAINER_NAME%% service - ## - annotations: {} - ## @param service.sessionAffinity Session Affinity for Kubernetes service, can be "None" or "ClientIP" - ## If "ClientIP", consecutive client requests will be directed to the same Pod - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies - ## - sessionAffinity: None - ## @param service.sessionAffinityConfig Additional settings for the sessionAffinity - ## sessionAffinityConfig: - ## clientIP: - ## timeoutSeconds: 300 - ## - sessionAffinityConfig: {} -## Configure the ingress resource that allows you to access the -## Drupal installation. Set up the URL -## ref: https://kubernetes.io/docs/concepts/services-networking/ingress/ -## -ingress: - ## @param ingress.enabled Enable ingress controller resource - ## - enabled: false - ## DEPRECATED: Use ingress.annotations instead of ingress.certManager - ## certManager: false - ## - - ## @param ingress.pathType Ingress Path type - ## - pathType: ImplementationSpecific - ## @param ingress.apiVersion Override API Version (automatically detected if not set) - ## - apiVersion: "" - ## @param ingress.ingressClassName IngressClass that will be be used to implement the Ingress (Kubernetes 1.18+) - ## This is supported in Kubernetes 1.18+ and required if you have more than one IngressClass marked as the default for your cluster . - ## ref: https://kubernetes.io/blog/2020/04/02/improvements-to-the-ingress-api-in-kubernetes-1.18/ - ## - ingressClassName: "" - ## @param ingress.hostname Default host for the ingress resource - ## - hostname: drupal.local - ## @param ingress.path The Path to Drupal. You may need to set this to '/*' in order to use this - ## with ALB ingress controllers. - ## - path: / - ## @param ingress.annotations Additional annotations for the Ingress resource. To enable certificate autogeneration, place here your cert-manager annotations. - ## For a full list of possible ingress annotations, please see - ## ref: https://github.com/kubernetes/ingress-nginx/blob/main/docs/user-guide/nginx-configuration/annotations.md - ## Use this parameter to set the required annotations for cert-manager, see - ## ref: https://cert-manager.io/docs/usage/ingress/#supported-annotations - ## - ## e.g: - ## annotations: - ## kubernetes.io/ingress.class: nginx - ## cert-manager.io/cluster-issuer: cluster-issuer-name - ## - annotations: {} - ## @param ingress.tls Enable TLS configuration for the hostname defined at ingress.hostname parameter - ## TLS certificates will be retrieved from a TLS secret with name: {{- printf "%s-tls" .Values.ingress.hostname }} - ## You can use the ingress.secrets parameter to create this TLS secret or relay on cert-manager to create it - ## - tls: false - ## @param ingress.tlsWwwPrefix Adds www subdomain to default cert - ## Creates tls host with ingress.hostname: {{ print "www.%s" .Values.ingress.hostname }} - ## Is enabled if "nginx.ingress.kubernetes.io/from-to-www-redirect" is "true" - tlsWwwPrefix: false - ## @param ingress.extraHosts The list of additional hostnames to be covered with this ingress record. - ## Most likely the hostname above will be enough, but in the event more hosts are needed, this is an array - ## extraHosts: - ## - name: drupal.local - ## path: / - extraHosts: [] - ## @param ingress.extraPaths Any additional arbitrary paths that may need to be added to the ingress under the main host. - ## For example: The ALB ingress controller requires a special rule for handling SSL redirection. - ## extraPaths: - ## - path: /* - ## backend: - ## serviceName: ssl-redirect - ## servicePort: use-annotation - extraPaths: [] - ## @param ingress.extraTls The tls configuration for additional hostnames to be covered with this ingress record. - ## see: https://kubernetes.io/docs/concepts/services-networking/ingress/#tls - ## extraTls: - ## - hosts: - ## - drupal.local - ## secretName: drupal.local-tls - extraTls: [] - ## @param ingress.secrets If you're providing your own certificates, please use this to add the certificates as secrets - ## key and certificate should start with -----BEGIN CERTIFICATE----- or - ## -----BEGIN RSA PRIVATE KEY----- - ## - ## name should line up with a tlsSecret set further up - ## If you're using cert-manager, this is unneeded, as it will create the secret for you if it is not set - ## - ## It is also possible to create and manage the certificates outside of this helm chart - ## Please see README.md for more information - ## Example: - ## - name: drupal.local-tls - ## key: - ## certificate: - ## - secrets: [] - ## @param ingress.extraRules Additional rules to be covered with this ingress record - ## ref: https://kubernetes.io/docs/concepts/services-networking/ingress/#ingress-rules - ## e.g: - ## extraRules: - ## - host: example.local - ## http: - ## path: / - ## backend: - ## service: - ## name: example-svc - ## port: - ## name: http - ## - extraRules: [] -## @section Database parameters - -## MariaDB chart configuration -## https://github.com/bitnami/charts/blob/main/bitnami/mariadb/values.yaml -## -mariadb: - ## @param mariadb.enabled Whether to deploy a mariadb server to satisfy the applications database requirements - ## To use an external database set this to false and configure the externalDatabase parameters - ## - enabled: true - ## @param mariadb.architecture MariaDB architecture (`standalone` or `replication`) - ## - architecture: standalone - ## MariaDB Authentication parameters - ## @param mariadb.auth.rootPassword Password for the MariaDB `root` user - ## @param mariadb.auth.database Database name to create - ## @param mariadb.auth.username Database user to create - ## @param mariadb.auth.password Password for the database - ## - auth: - ## ref: https://github.com/bitnami/containers/tree/main/bitnami/mariadb#setting-the-root-password-on-first-run - ## - rootPassword: "" - ## ref: https://github.com/bitnami/containers/blob/main/bitnami/mariadb/README.md#creating-a-database-on-first-run - ## - database: bitnami_drupal - ## ref: https://github.com/bitnami/containers/blob/main/bitnami/mariadb/README.md#creating-a-database-user-on-first-run - ## - username: bn_drupal - password: "" - primary: - ## Enable persistence using Persistent Volume Claims - ## ref: https://kubernetes.io/docs/concepts/storage/persistent-volumes/ - ## @param mariadb.primary.persistence.enabled Enable database persistence using PVC - ## @param mariadb.primary.persistence.storageClass MariaDB primary persistent volume storage Class - ## @param mariadb.primary.persistence.accessModes Database Persistent Volume Access Modes - ## @param mariadb.primary.persistence.size Database Persistent Volume Size - ## @param mariadb.primary.persistence.hostPath Set path in case you want to use local host path volumes (not recommended in production) - ## @param mariadb.primary.persistence.existingClaim Name of an existing `PersistentVolumeClaim` for MariaDB primary replicas - ## - persistence: - enabled: true - ## mariadb data Persistent Volume Storage Class - ## If defined, storageClassName: - ## If set to "-", storageClassName: "", which disables dynamic provisioning - ## If undefined (the default) or set to null, no storageClassName spec is - ## set, choosing the default provisioner. (gp2 on AWS, standard on - ## GKE, AWS & OpenStack) - ## - storageClass: "" - accessModes: - - ReadWriteOnce - size: 8Gi - hostPath: "" - existingClaim: "" -## External database configuration -## @param externalDatabase.host Host of the existing database -## @param externalDatabase.port Port of the existing database -## @param externalDatabase.user Existing username in the external db -## @param externalDatabase.password Password for the above username. Ignored if existing secret is provided -## @param externalDatabase.database Name of the existing database -## @param externalDatabase.existingSecret Name of a secret with the database password. (externalDatabase.password will be ignored and picked up from this secret). The secret has to contain the key db-password -## -externalDatabase: - host: "" - port: 3306 - user: bn_drupal - password: "" - database: bitnami_drupal - existingSecret: "" -## @section Volume Permissions parameters - -## Init containers parameters: -## volumePermissions: Change the owner and group of the persistent volume mountpoint to runAsUser:fsGroup values from the securityContext section. -## -volumePermissions: - ## @param volumePermissions.enabled Enable init container that changes volume permissions in the data directory (for cases where the default k8s `runAsUser` and `fsUser` values do not work) - ## - enabled: false - ## @param volumePermissions.image.registry [default: REGISTRY_NAME] Init container volume-permissions image registry - ## @param volumePermissions.image.repository [default: REPOSITORY_NAME/os-shell] Init container volume-permissions image name - ## @skip volumePermissions.image.tag Init container volume-permissions image tag - ## @param volumePermissions.image.digest Init container volume-permissions image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag - ## @param volumePermissions.image.pullPolicy Init container volume-permissions image pull policy - ## @param volumePermissions.image.pullSecrets Specify docker-registry secret names as an array - ## - image: - registry: docker.io - repository: bitnami/os-shell - tag: 12-debian-12-r31 - digest: "" - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets. - ## Secrets must be manually created in the namespace. - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## e.g: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## Init containers' resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## We usually recommend not to specify default resources and to leave this as a conscious - ## choice for the user. This also increases chances charts run on environments with little - ## resources, such as Minikube. If you do want to specify resources, uncomment the following - ## lines, adjust them as necessary, and remove the curly braces after 'resources:'. - ## @param volumePermissions.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if volumePermissions.resources is set (volumePermissions.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "none" - ## @param volumePermissions.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} -## @section Metrics parameters - -## Prometheus Exporter / Metrics -## -metrics: - ## @param metrics.enabled Start a exporter side-car - ## - enabled: false - ## @param metrics.image.registry [default: REGISTRY_NAME] Apache exporter image registry - ## @param metrics.image.repository [default: REPOSITORY_NAME/apache-exporter] Apache exporter image repository - ## @skip metrics.image.tag Apache exporter image tag - ## @param metrics.image.digest Apache exporter image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag - ## @param metrics.image.pullPolicy Image pull policy - ## @param metrics.image.pullSecrets Specify docker-registry secret names as an array - ## - image: - registry: docker.io - repository: bitnami/apache-exporter - tag: 1.0.9-debian-12-r1 - digest: "" - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets. - ## Secrets must be manually created in the namespace. - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## e.g: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## @param metrics.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if metrics.resources is set (metrics.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "none" - ## @param metrics.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## - resources: {} - ## - ## @param metrics.podAnnotations [object] Additional annotations for Metrics exporter pod - ## - podAnnotations: - prometheus.io/scrape: "true" - prometheus.io/port: "9117" - ## Drupal exporter service parameters - ## - service: - ## @param metrics.service.type Drupal exporter service type - ## - type: ClusterIP - ## @param metrics.service.ports.metrics Drupal exporter service port - ## - ports: - metrics: 9117 - ## @param metrics.service.externalTrafficPolicy Drupal exporter service external traffic policy - ## ref: https://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer/#preserving-the-client-source-ip - ## - externalTrafficPolicy: Cluster - ## @param metrics.service.extraPorts Extra ports to expose (normally used with the `sidecar` value) - ## - extraPorts: [] - ## @param metrics.service.loadBalancerIP Drupal exporter service Load Balancer IP - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#internal-load-balancer - ## - loadBalancerIP: "" - ## @param metrics.service.loadBalancerSourceRanges Drupal exporter service Load Balancer sources - ## https://kubernetes.io/docs/tasks/access-application-cluster/configure-cloud-provider-firewall/#restrict-access-for-loadbalancer-service - ## e.g. - ## loadBalancerSourceRanges: - ## - 10.10.10.0/24 - ## - loadBalancerSourceRanges: [] - ## @param metrics.service.annotations Additional custom annotations for Drupal exporter service - ## - annotations: {} - ## Prometheus Service Monitor - ## ref: https://github.com/coreos/prometheus-operator - ## https://github.com/coreos/prometheus-operator/blob/master/Documentation/api.md#endpoint - ## - serviceMonitor: - ## @param metrics.serviceMonitor.enabled Create ServiceMonitor resource(s) for scraping metrics using PrometheusOperator - ## - enabled: false - ## @param metrics.serviceMonitor.namespace The namespace in which the ServiceMonitor will be created - ## - namespace: "" - ## @param metrics.serviceMonitor.interval The interval at which metrics should be scraped - ## - interval: 30s - ## @param metrics.serviceMonitor.scrapeTimeout The timeout after which the scrape is ended - ## - scrapeTimeout: "" - ## @param metrics.serviceMonitor.relabellings Metrics RelabelConfigs to apply to samples before scraping. - ## - relabellings: [] - ## @param metrics.serviceMonitor.metricRelabelings Metrics RelabelConfigs to apply to samples before ingestion. - ## - metricRelabelings: [] - ## @param metrics.serviceMonitor.honorLabels Specify honorLabels parameter to add the scrape endpoint - ## - honorLabels: false - ## @param metrics.serviceMonitor.additionalLabels Additional labels that can be used so ServiceMonitor resource(s) can be discovered by Prometheus - ## - additionalLabels: {} - ## Custom PrometheusRule to be defined - ## ref: https://github.com/coreos/prometheus-operator#customresourcedefinitions - ## - prometheusRule: - ## @param metrics.prometheusRule.enabled Create a custom prometheusRule Resource for scraping metrics using PrometheusOperator - ## - enabled: false - ## @param metrics.prometheusRule.namespace The namespace in which the prometheusRule will be created - ## - namespace: "" - ## @param metrics.prometheusRule.additionalLabels Additional labels for the prometheusRule - ## - additionalLabels: {} - ## @param metrics.prometheusRule.rules Custom Prometheus rules - ## e.g: - ## rules: - ## - alert: ApacheDown - ## expr: apache_up{service="{{ template "common.names.fullname" . }}-metrics"} == 0 - ## for: 2m - ## labels: - ## severity: error - ## annotations: - ## summary: Drupal instance {{ "{{ $labels.instance }}" }} down - ## description: Drupal instance {{ "{{ $labels.instance }}" }} is down - ## - rules: [] -## @section Certificate injection parameters - -## Add custom certificates and certificate authorities to drupal container -## -certificates: - ## @param certificates.customCertificate.certificateSecret Secret containing the certificate and key to add - ## @param certificates.customCertificate.chainSecret.name Name of the secret containing the certificate chain - ## @param certificates.customCertificate.chainSecret.key Key of the certificate chain file inside the secret - ## @param certificates.customCertificate.certificateLocation Location in the container to store the certificate - ## @param certificates.customCertificate.keyLocation Location in the container to store the private key - ## @param certificates.customCertificate.chainLocation Location in the container to store the certificate chain - ## - customCertificate: - certificateSecret: "" - chainSecret: - name: secret-name - key: secret-key - certificateLocation: /etc/ssl/certs/ssl-cert-snakeoil.pem - keyLocation: /etc/ssl/private/ssl-cert-snakeoil.key - chainLocation: /etc/ssl/certs/mychain.pem - ## @param certificates.customCAs Defines a list of secrets to import into the container trust store - ## - customCAs: [] - ## @param certificates.command Override default container command (useful when using custom images) - ## - command: [] - ## @param certificates.args Override default container args (useful when using custom images) - ## - args: [] - ## @param certificates.extraEnvVars Container sidecar extra environment variables (eg proxy) - ## - extraEnvVars: [] - ## @param certificates.extraEnvVarsCM ConfigMap containing extra env vars - ## - extraEnvVarsCM: "" - ## @param certificates.extraEnvVarsSecret Secret containing extra env vars (in case of sensitive data) - ## - extraEnvVarsSecret: "" - ## @param certificates.image.registry [default: REGISTRY_NAME] Container sidecar registry - ## @param certificates.image.repository [default: REPOSITORY_NAME/os-shell] Container sidecar image - ## @skip certificates.image.tag Container sidecar image tag - ## @param certificates.image.digest Container sidecar image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag - ## @param certificates.image.pullPolicy Container sidecar image pull policy - ## @param certificates.image.pullSecrets Container sidecar image pull secrets - ## - image: - registry: docker.io - repository: bitnami/os-shell - tag: 12-debian-12-r31 - digest: "" - ## Specify a imagePullPolicy - ## Defaults to 'Always' if image tag is 'latest', else set to 'IfNotPresent' - ## ref: https://kubernetes.io/docs/concepts/containers/images/#pre-pulled-images - ## - pullPolicy: IfNotPresent - ## e.g: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] -## @section NetworkPolicy parameters - -## Network Policy configuration -## ref: https://kubernetes.io/docs/concepts/services-networking/network-policies/ -## -networkPolicy: - ## @param networkPolicy.enabled Specifies whether a NetworkPolicy should be created - ## - enabled: true - ## @param networkPolicy.allowExternal Don't require server label for connections - ## The Policy model to apply. When set to false, only pods with the correct - ## server label will have network access to the ports server is listening - ## on. When true, server will accept connections from any source - ## (with the correct destination port). - ## - allowExternal: true - ## @param networkPolicy.allowExternalEgress Allow the pod to access any range of port and all destinations. - ## - allowExternalEgress: true - ## @param networkPolicy.extraIngress [array] Add extra ingress rules to the NetworkPolicy - ## e.g: - ## extraIngress: - ## - ports: - ## - port: 1234 - ## from: - ## - podSelector: - ## - matchLabels: - ## - role: frontend - ## - podSelector: - ## - matchExpressions: - ## - key: role - ## operator: In - ## values: - ## - frontend - extraIngress: [] - ## @param networkPolicy.extraEgress [array] Add extra ingress rules to the NetworkPolicy - ## e.g: - ## extraEgress: - ## - ports: - ## - port: 1234 - ## to: - ## - podSelector: - ## - matchLabels: - ## - role: frontend - ## - podSelector: - ## - matchExpressions: - ## - key: role - ## operator: In - ## values: - ## - frontend - ## - extraEgress: [] - ## @param networkPolicy.ingressNSMatchLabels [object] Labels to match to allow traffic from other namespaces - ## @param networkPolicy.ingressNSPodMatchLabels [object] Pod labels to match to allow traffic from other namespaces - ## - ingressNSMatchLabels: {} - ingressNSPodMatchLabels: {} diff --git a/charts/drupal/values-overrides.yaml b/charts/drupal/values-overrides.yaml deleted file mode 100644 index 6cc8fd7..0000000 --- a/charts/drupal/values-overrides.yaml +++ /dev/null @@ -1,15 +0,0 @@ -# @see https://artifacthub.io/packages/helm/bitnami/drupal?modal=values - -fullnameOverride: drupal -mariadb: - enabled: false -allowEmptyPassword: true -externalDatabase: - host: "mariadb.futureporn.svc.cluster.local" - port: 3306 - user: drupal - database: drupal - existingSecret: externaldb -image: - debug: true - tag: 10.3.6-debian-12-r1 \ No newline at end of file diff --git a/charts/external-secrets/external-secrets/Chart.lock b/charts/external-secrets/external-secrets/Chart.lock deleted file mode 100644 index 1d198fe..0000000 --- a/charts/external-secrets/external-secrets/Chart.lock +++ /dev/null @@ -1,6 +0,0 @@ -dependencies: -- name: bitwarden-sdk-server - repository: oci://ghcr.io/external-secrets/charts - version: v0.1.4 -digest: sha256:f60d5e4c6ad432fc7efdb0dad33774afaa88e02bd82eb9d5224372828f7d52be -generated: "2024-06-20T10:01:52.49841+02:00" diff --git a/charts/external-secrets/external-secrets/Chart.yaml b/charts/external-secrets/external-secrets/Chart.yaml deleted file mode 100644 index 09f630d..0000000 --- a/charts/external-secrets/external-secrets/Chart.yaml +++ /dev/null @@ -1,20 +0,0 @@ -apiVersion: v2 -appVersion: v0.10.2 -dependencies: -- condition: bitwarden-sdk-server.enabled - name: bitwarden-sdk-server - repository: oci://ghcr.io/external-secrets/charts - version: v0.1.4 -description: External secret management for Kubernetes -home: https://github.com/external-secrets/external-secrets -icon: https://raw.githubusercontent.com/external-secrets/external-secrets/main/assets/eso-logo-large.png -keywords: -- kubernetes-external-secrets -- secrets -kubeVersion: '>= 1.19.0-0' -maintainers: -- email: kellinmcavoy@gmail.com - name: mcavoyk -name: external-secrets -type: application -version: 0.10.2 diff --git a/charts/external-secrets/external-secrets/README.md b/charts/external-secrets/external-secrets/README.md deleted file mode 100644 index a1cd2a8..0000000 --- a/charts/external-secrets/external-secrets/README.md +++ /dev/null @@ -1,225 +0,0 @@ -# External Secrets - -

external-secrets

- -[//]: # (README.md generated by gotmpl. DO NOT EDIT.) - -![Type: application](https://img.shields.io/badge/Type-application-informational?style=flat-square) ![Version: 0.10.2](https://img.shields.io/badge/Version-0.10.2-informational?style=flat-square) - -External secret management for Kubernetes - -## TL;DR -```bash -helm repo add external-secrets https://charts.external-secrets.io -helm install external-secrets external-secrets/external-secrets -``` - -## Installing the Chart -To install the chart with the release name `external-secrets`: -```bash -helm install external-secrets external-secrets/external-secrets -``` - -### Custom Resources -By default, the chart will install external-secrets CRDs, this can be controlled with `installCRDs` value. - -## Uninstalling the Chart -To uninstall the `external-secrets` deployment: -```bash -helm uninstall external-secrets -``` -The command removes all the Kubernetes components associated with the chart and deletes the release. - -## Values - -| Key | Type | Default | Description | -|-----|------|---------|-------------| -| affinity | object | `{}` | | -| bitwarden-sdk-server.enabled | bool | `false` | | -| certController.affinity | object | `{}` | | -| certController.create | bool | `true` | Specifies whether a certificate controller deployment be created. | -| certController.deploymentAnnotations | object | `{}` | Annotations to add to Deployment | -| certController.extraArgs | object | `{}` | | -| certController.extraEnv | list | `[]` | | -| certController.extraVolumeMounts | list | `[]` | | -| certController.extraVolumes | list | `[]` | | -| certController.fullnameOverride | string | `""` | | -| certController.hostNetwork | bool | `false` | Run the certController on the host network | -| certController.image.flavour | string | `""` | | -| certController.image.pullPolicy | string | `"IfNotPresent"` | | -| certController.image.repository | string | `"ghcr.io/external-secrets/external-secrets"` | | -| certController.image.tag | string | `""` | | -| certController.imagePullSecrets | list | `[]` | | -| certController.log | object | `{"level":"info","timeEncoding":"epoch"}` | Specifices Log Params to the Webhook | -| certController.metrics.listen.port | int | `8080` | | -| certController.metrics.service.annotations | object | `{}` | Additional service annotations | -| certController.metrics.service.enabled | bool | `false` | Enable if you use another monitoring tool than Prometheus to scrape the metrics | -| certController.metrics.service.port | int | `8080` | Metrics service port to scrape | -| certController.nameOverride | string | `""` | | -| certController.nodeSelector | object | `{}` | | -| certController.podAnnotations | object | `{}` | Annotations to add to Pod | -| certController.podDisruptionBudget | object | `{"enabled":false,"minAvailable":1}` | Pod disruption budget - for more details see https://kubernetes.io/docs/concepts/workloads/pods/disruptions/ | -| certController.podLabels | object | `{}` | | -| certController.podSecurityContext.enabled | bool | `true` | | -| certController.priorityClassName | string | `""` | Pod priority class name. | -| certController.rbac.create | bool | `true` | Specifies whether role and rolebinding resources should be created. | -| certController.readinessProbe.address | string | `""` | Address for readiness probe | -| certController.readinessProbe.port | int | `8081` | ReadinessProbe port for kubelet | -| certController.replicaCount | int | `1` | | -| certController.requeueInterval | string | `"5m"` | | -| certController.resources | object | `{}` | | -| certController.revisionHistoryLimit | int | `10` | Specifies the amount of historic ReplicaSets k8s should keep (see https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#clean-up-policy) | -| certController.securityContext.allowPrivilegeEscalation | bool | `false` | | -| certController.securityContext.capabilities.drop[0] | string | `"ALL"` | | -| certController.securityContext.enabled | bool | `true` | | -| certController.securityContext.readOnlyRootFilesystem | bool | `true` | | -| certController.securityContext.runAsNonRoot | bool | `true` | | -| certController.securityContext.runAsUser | int | `1000` | | -| certController.securityContext.seccompProfile.type | string | `"RuntimeDefault"` | | -| certController.serviceAccount.annotations | object | `{}` | Annotations to add to the service account. | -| certController.serviceAccount.automount | bool | `true` | Automounts the service account token in all containers of the pod | -| certController.serviceAccount.create | bool | `true` | Specifies whether a service account should be created. | -| certController.serviceAccount.extraLabels | object | `{}` | Extra Labels to add to the service account. | -| certController.serviceAccount.name | string | `""` | The name of the service account to use. If not set and create is true, a name is generated using the fullname template. | -| certController.tolerations | list | `[]` | | -| certController.topologySpreadConstraints | list | `[]` | | -| commonLabels | object | `{}` | Additional labels added to all helm chart resources. | -| concurrent | int | `1` | Specifies the number of concurrent ExternalSecret Reconciles external-secret executes at a time. | -| controllerClass | string | `""` | If set external secrets will filter matching Secret Stores with the appropriate controller values. | -| crds.annotations | object | `{}` | | -| crds.conversion.enabled | bool | `true` | | -| crds.createClusterExternalSecret | bool | `true` | If true, create CRDs for Cluster External Secret. | -| crds.createClusterSecretStore | bool | `true` | If true, create CRDs for Cluster Secret Store. | -| crds.createPushSecret | bool | `true` | If true, create CRDs for Push Secret. | -| createOperator | bool | `true` | Specifies whether an external secret operator deployment be created. | -| deploymentAnnotations | object | `{}` | Annotations to add to Deployment | -| dnsConfig | object | `{}` | Specifies `dnsOptions` to deployment | -| dnsPolicy | string | `"ClusterFirst"` | Specifies `dnsPolicy` to deployment | -| extendedMetricLabels | bool | `false` | If true external secrets will use recommended kubernetes annotations as prometheus metric labels. | -| extraArgs | object | `{}` | | -| extraContainers | list | `[]` | | -| extraEnv | list | `[]` | | -| extraObjects | list | `[]` | | -| extraVolumeMounts | list | `[]` | | -| extraVolumes | list | `[]` | | -| fullnameOverride | string | `""` | | -| global.affinity | object | `{}` | | -| global.compatibility.openshift.adaptSecurityContext | string | `"auto"` | Manages the securityContext properties to make them compatible with OpenShift. Possible values: auto - Apply configurations if it is detected that OpenShift is the target platform. force - Always apply configurations. disabled - No modification applied. | -| global.nodeSelector | object | `{}` | | -| global.tolerations | list | `[]` | | -| global.topologySpreadConstraints | list | `[]` | | -| hostNetwork | bool | `false` | Run the controller on the host network | -| image.flavour | string | `""` | The flavour of tag you want to use There are different image flavours available, like distroless and ubi. Please see GitHub release notes for image tags for these flavors. By default, the distroless image is used. | -| image.pullPolicy | string | `"IfNotPresent"` | | -| image.repository | string | `"ghcr.io/external-secrets/external-secrets"` | | -| image.tag | string | `""` | The image tag to use. The default is the chart appVersion. | -| imagePullSecrets | list | `[]` | | -| installCRDs | bool | `true` | If set, install and upgrade CRDs through helm chart. | -| leaderElect | bool | `false` | If true, external-secrets will perform leader election between instances to ensure no more than one instance of external-secrets operates at a time. | -| log | object | `{"level":"info","timeEncoding":"epoch"}` | Specifices Log Params to the Webhook | -| metrics.listen.port | int | `8080` | | -| metrics.service.annotations | object | `{}` | Additional service annotations | -| metrics.service.enabled | bool | `false` | Enable if you use another monitoring tool than Prometheus to scrape the metrics | -| metrics.service.port | int | `8080` | Metrics service port to scrape | -| nameOverride | string | `""` | | -| namespaceOverride | string | `""` | | -| nodeSelector | object | `{}` | | -| podAnnotations | object | `{}` | Annotations to add to Pod | -| podDisruptionBudget | object | `{"enabled":false,"minAvailable":1}` | Pod disruption budget - for more details see https://kubernetes.io/docs/concepts/workloads/pods/disruptions/ | -| podLabels | object | `{}` | | -| podSecurityContext.enabled | bool | `true` | | -| podSpecExtra | object | `{}` | Any extra pod spec on the deployment | -| priorityClassName | string | `""` | Pod priority class name. | -| processClusterExternalSecret | bool | `true` | if true, the operator will process cluster external secret. Else, it will ignore them. | -| processClusterStore | bool | `true` | if true, the operator will process cluster store. Else, it will ignore them. | -| processPushSecret | bool | `true` | if true, the operator will process push secret. Else, it will ignore them. | -| rbac.create | bool | `true` | Specifies whether role and rolebinding resources should be created. | -| rbac.servicebindings.create | bool | `true` | Specifies whether a clusterrole to give servicebindings read access should be created. | -| replicaCount | int | `1` | | -| resources | object | `{}` | | -| revisionHistoryLimit | int | `10` | Specifies the amount of historic ReplicaSets k8s should keep (see https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#clean-up-policy) | -| scopedNamespace | string | `""` | If set external secrets are only reconciled in the provided namespace | -| scopedRBAC | bool | `false` | Must be used with scopedNamespace. If true, create scoped RBAC roles under the scoped namespace and implicitly disable cluster stores and cluster external secrets | -| securityContext.allowPrivilegeEscalation | bool | `false` | | -| securityContext.capabilities.drop[0] | string | `"ALL"` | | -| securityContext.enabled | bool | `true` | | -| securityContext.readOnlyRootFilesystem | bool | `true` | | -| securityContext.runAsNonRoot | bool | `true` | | -| securityContext.runAsUser | int | `1000` | | -| securityContext.seccompProfile.type | string | `"RuntimeDefault"` | | -| service.ipFamilies | list | `[]` | Sets the families that should be supported and the order in which they should be applied to ClusterIP as well. Can be IPv4 and/or IPv6. | -| service.ipFamilyPolicy | string | `""` | Set the ip family policy to configure dual-stack see [Configure dual-stack](https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services) | -| serviceAccount.annotations | object | `{}` | Annotations to add to the service account. | -| serviceAccount.automount | bool | `true` | Automounts the service account token in all containers of the pod | -| serviceAccount.create | bool | `true` | Specifies whether a service account should be created. | -| serviceAccount.extraLabels | object | `{}` | Extra Labels to add to the service account. | -| serviceAccount.name | string | `""` | The name of the service account to use. If not set and create is true, a name is generated using the fullname template. | -| serviceMonitor.additionalLabels | object | `{}` | Additional labels | -| serviceMonitor.enabled | bool | `false` | Specifies whether to create a ServiceMonitor resource for collecting Prometheus metrics | -| serviceMonitor.honorLabels | bool | `false` | Let prometheus add an exported_ prefix to conflicting labels | -| serviceMonitor.interval | string | `"30s"` | Interval to scrape metrics | -| serviceMonitor.metricRelabelings | list | `[]` | Metric relabel configs to apply to samples before ingestion. [Metric Relabeling](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#metric_relabel_configs) | -| serviceMonitor.namespace | string | `""` | namespace where you want to install ServiceMonitors | -| serviceMonitor.relabelings | list | `[]` | Relabel configs to apply to samples before ingestion. [Relabeling](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#relabel_config) | -| serviceMonitor.scrapeTimeout | string | `"25s"` | Timeout if metrics can't be retrieved in given time interval | -| tolerations | list | `[]` | | -| topologySpreadConstraints | list | `[]` | | -| webhook.affinity | object | `{}` | | -| webhook.certCheckInterval | string | `"5m"` | Specifices the time to check if the cert is valid | -| webhook.certDir | string | `"/tmp/certs"` | | -| webhook.certManager.addInjectorAnnotations | bool | `true` | Automatically add the cert-manager.io/inject-ca-from annotation to the webhooks and CRDs. As long as you have the cert-manager CA Injector enabled, this will automatically setup your webhook's CA to the one used by cert-manager. See https://cert-manager.io/docs/concepts/ca-injector | -| webhook.certManager.cert.annotations | object | `{}` | Add extra annotations to the Certificate resource. | -| webhook.certManager.cert.create | bool | `true` | Create a certificate resource within this chart. See https://cert-manager.io/docs/usage/certificate/ | -| webhook.certManager.cert.duration | string | `"8760h"` | Set the requested duration (i.e. lifetime) of the Certificate. See https://cert-manager.io/docs/reference/api-docs/#cert-manager.io/v1.CertificateSpec One year by default. | -| webhook.certManager.cert.issuerRef | object | `{"group":"cert-manager.io","kind":"Issuer","name":"my-issuer"}` | For the Certificate created by this chart, setup the issuer. See https://cert-manager.io/docs/reference/api-docs/#cert-manager.io/v1.IssuerSpec | -| webhook.certManager.cert.renewBefore | string | `""` | How long before the currently issued certificate’s expiry cert-manager should renew the certificate. See https://cert-manager.io/docs/reference/api-docs/#cert-manager.io/v1.CertificateSpec Note that renewBefore should be greater than .webhook.lookaheadInterval since the webhook will check this far in advance that the certificate is valid. | -| webhook.certManager.enabled | bool | `false` | Enabling cert-manager support will disable the built in secret and switch to using cert-manager (installed separately) to automatically issue and renew the webhook certificate. This chart does not install cert-manager for you, See https://cert-manager.io/docs/ | -| webhook.create | bool | `true` | Specifies whether a webhook deployment be created. | -| webhook.deploymentAnnotations | object | `{}` | Annotations to add to Deployment | -| webhook.extraArgs | object | `{}` | | -| webhook.extraEnv | list | `[]` | | -| webhook.extraVolumeMounts | list | `[]` | | -| webhook.extraVolumes | list | `[]` | | -| webhook.failurePolicy | string | `"Fail"` | Specifies whether validating webhooks should be created with failurePolicy: Fail or Ignore | -| webhook.fullnameOverride | string | `""` | | -| webhook.hostNetwork | bool | `false` | Specifies if webhook pod should use hostNetwork or not. | -| webhook.image.flavour | string | `""` | The flavour of tag you want to use | -| webhook.image.pullPolicy | string | `"IfNotPresent"` | | -| webhook.image.repository | string | `"ghcr.io/external-secrets/external-secrets"` | | -| webhook.image.tag | string | `""` | The image tag to use. The default is the chart appVersion. | -| webhook.imagePullSecrets | list | `[]` | | -| webhook.log | object | `{"level":"info","timeEncoding":"epoch"}` | Specifices Log Params to the Webhook | -| webhook.lookaheadInterval | string | `""` | Specifices the lookaheadInterval for certificate validity | -| webhook.metrics.listen.port | int | `8080` | | -| webhook.metrics.service.annotations | object | `{}` | Additional service annotations | -| webhook.metrics.service.enabled | bool | `false` | Enable if you use another monitoring tool than Prometheus to scrape the metrics | -| webhook.metrics.service.port | int | `8080` | Metrics service port to scrape | -| webhook.nameOverride | string | `""` | | -| webhook.nodeSelector | object | `{}` | | -| webhook.podAnnotations | object | `{}` | Annotations to add to Pod | -| webhook.podDisruptionBudget | object | `{"enabled":false,"minAvailable":1}` | Pod disruption budget - for more details see https://kubernetes.io/docs/concepts/workloads/pods/disruptions/ | -| webhook.podLabels | object | `{}` | | -| webhook.podSecurityContext.enabled | bool | `true` | | -| webhook.port | int | `10250` | The port the webhook will listen to | -| webhook.priorityClassName | string | `""` | Pod priority class name. | -| webhook.rbac.create | bool | `true` | Specifies whether role and rolebinding resources should be created. | -| webhook.readinessProbe.address | string | `""` | Address for readiness probe | -| webhook.readinessProbe.port | int | `8081` | ReadinessProbe port for kubelet | -| webhook.replicaCount | int | `1` | | -| webhook.resources | object | `{}` | | -| webhook.revisionHistoryLimit | int | `10` | Specifies the amount of historic ReplicaSets k8s should keep (see https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#clean-up-policy) | -| webhook.secretAnnotations | object | `{}` | Annotations to add to Secret | -| webhook.securityContext.allowPrivilegeEscalation | bool | `false` | | -| webhook.securityContext.capabilities.drop[0] | string | `"ALL"` | | -| webhook.securityContext.enabled | bool | `true` | | -| webhook.securityContext.readOnlyRootFilesystem | bool | `true` | | -| webhook.securityContext.runAsNonRoot | bool | `true` | | -| webhook.securityContext.runAsUser | int | `1000` | | -| webhook.securityContext.seccompProfile.type | string | `"RuntimeDefault"` | | -| webhook.serviceAccount.annotations | object | `{}` | Annotations to add to the service account. | -| webhook.serviceAccount.automount | bool | `true` | Automounts the service account token in all containers of the pod | -| webhook.serviceAccount.create | bool | `true` | Specifies whether a service account should be created. | -| webhook.serviceAccount.extraLabels | object | `{}` | Extra Labels to add to the service account. | -| webhook.serviceAccount.name | string | `""` | The name of the service account to use. If not set and create is true, a name is generated using the fullname template. | -| webhook.tolerations | list | `[]` | | -| webhook.topologySpreadConstraints | list | `[]` | | diff --git a/charts/external-secrets/external-secrets/templates/NOTES.txt b/charts/external-secrets/external-secrets/templates/NOTES.txt deleted file mode 100644 index ffa0fc7..0000000 --- a/charts/external-secrets/external-secrets/templates/NOTES.txt +++ /dev/null @@ -1,7 +0,0 @@ -external-secrets has been deployed successfully in namespace {{ template "external-secrets.namespace" . }}! - -In order to begin using ExternalSecrets, you will need to set up a SecretStore -or ClusterSecretStore resource (for example, by creating a 'vault' SecretStore). - -More information on the different types of SecretStores and how to configure them -can be found in our Github: {{ .Chart.Home }} diff --git a/charts/external-secrets/external-secrets/templates/_helpers.tpl b/charts/external-secrets/external-secrets/templates/_helpers.tpl deleted file mode 100644 index d5eea07..0000000 --- a/charts/external-secrets/external-secrets/templates/_helpers.tpl +++ /dev/null @@ -1,198 +0,0 @@ -{{/* -Expand the name of the chart. -*/}} -{{- define "external-secrets.name" -}} -{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }} -{{- end }} - -{{/* -Create a default fully qualified app name. -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -If release name contains chart name it will be used as a full name. -*/}} -{{- define "external-secrets.fullname" -}} -{{- if .Values.fullnameOverride }} -{{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" }} -{{- else }} -{{- $name := default .Chart.Name .Values.nameOverride }} -{{- if contains $name .Release.Name }} -{{- .Release.Name | trunc 63 | trimSuffix "-" }} -{{- else }} -{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" }} -{{- end }} -{{- end }} -{{- end }} - -{{/* -Define namespace of chart, useful for multi-namespace deployments -*/}} -{{- define "external-secrets.namespace" -}} -{{- if .Values.namespaceOverride }} -{{- .Values.namespaceOverride }} -{{- else }} -{{- .Release.Namespace }} -{{- end }} -{{- end }} - -{{/* -Create chart name and version as used by the chart label. -*/}} -{{- define "external-secrets.chart" -}} -{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} -{{- end }} - -{{/* -Common labels -*/}} -{{- define "external-secrets.labels" -}} -helm.sh/chart: {{ include "external-secrets.chart" . }} -{{ include "external-secrets.selectorLabels" . }} -{{- if .Chart.AppVersion }} -app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} -{{- end }} -app.kubernetes.io/managed-by: {{ .Release.Service }} -{{- with .Values.commonLabels }} -{{ toYaml . }} -{{- end }} -{{- end }} - -{{- define "external-secrets-webhook.labels" -}} -helm.sh/chart: {{ include "external-secrets.chart" . }} -{{ include "external-secrets-webhook.selectorLabels" . }} -{{- if .Chart.AppVersion }} -app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} -{{- end }} -app.kubernetes.io/managed-by: {{ .Release.Service }} -{{- with .Values.commonLabels }} -{{ toYaml . }} -{{- end }} -{{- end }} - -{{- define "external-secrets-webhook-metrics.labels" -}} -{{ include "external-secrets-webhook.selectorLabels" . }} -app.kubernetes.io/metrics: "webhook" -{{- with .Values.commonLabels }} -{{ toYaml . }} -{{- end }} -{{- end }} - -{{- define "external-secrets-cert-controller.labels" -}} -helm.sh/chart: {{ include "external-secrets.chart" . }} -{{ include "external-secrets-cert-controller.selectorLabels" . }} -{{- if .Chart.AppVersion }} -app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} -{{- end }} -app.kubernetes.io/managed-by: {{ .Release.Service }} -{{- with .Values.commonLabels }} -{{ toYaml . }} -{{- end }} -{{- end }} - -{{- define "external-secrets-cert-controller-metrics.labels" -}} -{{ include "external-secrets-cert-controller.selectorLabels" . }} -app.kubernetes.io/metrics: "cert-controller" -{{- with .Values.commonLabels }} -{{ toYaml . }} -{{- end }} -{{- end }} - -{{/* -Selector labels -*/}} -{{- define "external-secrets.selectorLabels" -}} -app.kubernetes.io/name: {{ include "external-secrets.name" . }} -app.kubernetes.io/instance: {{ .Release.Name }} -{{- end }} -{{- define "external-secrets-webhook.selectorLabels" -}} -app.kubernetes.io/name: {{ include "external-secrets.name" . }}-webhook -app.kubernetes.io/instance: {{ .Release.Name }} -{{- end }} -{{- define "external-secrets-cert-controller.selectorLabels" -}} -app.kubernetes.io/name: {{ include "external-secrets.name" . }}-cert-controller -app.kubernetes.io/instance: {{ .Release.Name }} -{{- end }} -{{/* -Create the name of the service account to use -*/}} -{{- define "external-secrets.serviceAccountName" -}} -{{- if .Values.serviceAccount.create }} -{{- default (include "external-secrets.fullname" .) .Values.serviceAccount.name }} -{{- else }} -{{- default "default" .Values.serviceAccount.name }} -{{- end }} -{{- end }} - -{{/* -Create the name of the service account to use -*/}} -{{- define "external-secrets-webhook.serviceAccountName" -}} -{{- if .Values.webhook.serviceAccount.create }} -{{- default "external-secrets-webhook" .Values.webhook.serviceAccount.name }} -{{- else }} -{{- default "default" .Values.webhook.serviceAccount.name }} -{{- end }} -{{- end }} - -{{/* -Create the name of the service account to use -*/}} -{{- define "external-secrets-cert-controller.serviceAccountName" -}} -{{- if .Values.certController.serviceAccount.create }} -{{- default "external-secrets-cert-controller" .Values.certController.serviceAccount.name }} -{{- else }} -{{- default "default" .Values.certController.serviceAccount.name }} -{{- end }} -{{- end }} - -{{/* -Determine the image to use, including if using a flavour. -*/}} -{{- define "external-secrets.image" -}} -{{- if .image.flavour -}} -{{ printf "%s:%s-%s" .image.repository (.image.tag | default .chartAppVersion) .image.flavour }} -{{- else }} -{{ printf "%s:%s" .image.repository (.image.tag | default .chartAppVersion) }} -{{- end }} -{{- end }} - -{{/* -Renders a complete tree, even values that contains template. -*/}} -{{- define "external-secrets.render" -}} - {{- if typeIs "string" .value }} - {{- tpl .value .context }} - {{ else }} - {{- tpl (.value | toYaml) .context }} - {{- end }} -{{- end -}} - -{{/* -Return true if the OpenShift is the detected platform -Usage: -{{- include "external-secrets.isOpenShift" . -}} -*/}} -{{- define "external-secrets.isOpenShift" -}} -{{- if .Capabilities.APIVersions.Has "security.openshift.io/v1" -}} -{{- true -}} -{{- end -}} -{{- end -}} - -{{/* -Render the securityContext based on the provided securityContext - {{- include "external-secrets.renderSecurityContext" (dict "securityContext" .Values.securityContext "context" $) -}} -*/}} -{{- define "external-secrets.renderSecurityContext" -}} -{{- $adaptedContext := .securityContext -}} -{{- if .context.Values.global.compatibility -}} - {{- if .context.Values.global.compatibility.openshift -}} - {{- if or (eq .context.Values.global.compatibility.openshift.adaptSecurityContext "force") (and (eq .context.Values.global.compatibility.openshift.adaptSecurityContext "auto") (include "external-secrets.isOpenShift" .context)) -}} - {{/* Remove OpenShift managed fields */}} - {{- $adaptedContext = omit $adaptedContext "fsGroup" "runAsUser" "runAsGroup" -}} - {{- if not .securityContext.seLinuxOptions -}} - {{- $adaptedContext = omit $adaptedContext "seLinuxOptions" -}} - {{- end -}} - {{- end -}} - {{- end -}} -{{- end -}} -{{- omit $adaptedContext "enabled" | toYaml -}} -{{- end -}} diff --git a/charts/external-secrets/external-secrets/templates/cert-controller-deployment.yaml b/charts/external-secrets/external-secrets/templates/cert-controller-deployment.yaml deleted file mode 100644 index a843f04..0000000 --- a/charts/external-secrets/external-secrets/templates/cert-controller-deployment.yaml +++ /dev/null @@ -1,124 +0,0 @@ -{{- if and .Values.certController.create (not .Values.webhook.certManager.enabled) }} -apiVersion: apps/v1 -kind: Deployment -metadata: - name: {{ include "external-secrets.fullname" . }}-cert-controller - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets-cert-controller.labels" . | nindent 4 }} - {{- with .Values.certController.deploymentAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - replicas: {{ .Values.certController.replicaCount }} - revisionHistoryLimit: {{ .Values.certController.revisionHistoryLimit }} - selector: - matchLabels: - {{- include "external-secrets-cert-controller.selectorLabels" . | nindent 6 }} - template: - metadata: - {{- with .Values.certController.podAnnotations }} - annotations: - {{- toYaml . | nindent 8 }} - {{- end }} - labels: - {{- include "external-secrets-cert-controller.labels" . | nindent 8 }} - {{- with .Values.certController.podLabels }} - {{- toYaml . | nindent 8 }} - {{- end }} - spec: - {{- with .Values.certController.imagePullSecrets }} - imagePullSecrets: - {{- toYaml . | nindent 8 }} - {{- end }} - serviceAccountName: {{ include "external-secrets-cert-controller.serviceAccountName" . }} - automountServiceAccountToken: {{ .Values.certController.serviceAccount.automount }} - {{- with .Values.certController.podSecurityContext }} - {{- if and (.enabled) (gt (keys . | len) 1) }} - securityContext: - {{- include "external-secrets.renderSecurityContext" (dict "securityContext" . "context" $) | nindent 8 }} - {{- end }} - {{- end }} - hostNetwork: {{ .Values.certController.hostNetwork }} - containers: - - name: cert-controller - {{- with .Values.certController.securityContext }} - {{- if and (.enabled) (gt (keys . | len) 1) }} - securityContext: - {{- include "external-secrets.renderSecurityContext" (dict "securityContext" . "context" $) | nindent 12 }} - {{- end }} - {{- end }} - image: {{ include "external-secrets.image" (dict "chartAppVersion" .Chart.AppVersion "image" .Values.certController.image) | trim }} - imagePullPolicy: {{ .Values.certController.image.pullPolicy }} - args: - - certcontroller - - --crd-requeue-interval={{ .Values.certController.requeueInterval }} - - --service-name={{ include "external-secrets.fullname" . }}-webhook - - --service-namespace={{ template "external-secrets.namespace" . }} - - --secret-name={{ include "external-secrets.fullname" . }}-webhook - - --secret-namespace={{ template "external-secrets.namespace" . }} - - --metrics-addr=:{{ .Values.certController.metrics.listen.port }} - - --healthz-addr={{ .Values.certController.readinessProbe.address }}:{{ .Values.certController.readinessProbe.port }} - - --loglevel={{ .Values.certController.log.level }} - - --zap-time-encoding={{ .Values.certController.log.timeEncoding }} - {{- if not .Values.crds.createClusterSecretStore }} - - --crd-names=externalsecrets.external-secrets.io - - --crd-names=secretstores.external-secrets.io - {{- end }} - {{- if .Values.installCRDs }} - - --enable-partial-cache=true - {{- end }} - {{- range $key, $value := .Values.certController.extraArgs }} - {{- if $value }} - - --{{ $key }}={{ $value }} - {{- else }} - - --{{ $key }} - {{- end }} - {{- end }} - ports: - - containerPort: {{ .Values.certController.metrics.listen.port }} - protocol: TCP - name: metrics - readinessProbe: - httpGet: - port: {{ .Values.certController.readinessProbe.port }} - path: /readyz - initialDelaySeconds: 20 - periodSeconds: 5 - {{- with .Values.certController.extraEnv }} - env: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- with .Values.certController.resources }} - resources: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- if .Values.certController.extraVolumeMounts }} - volumeMounts: - {{- toYaml .Values.certController.extraVolumeMounts | nindent 12 }} - {{- end }} - {{- if .Values.certController.extraVolumes }} - volumes: - {{- toYaml .Values.certController.extraVolumes | nindent 8 }} - {{- end }} - {{- with .Values.certController.nodeSelector | default .Values.global.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.certController.affinity | default .Values.global.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.certController.tolerations | default .Values.global.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.certController.topologySpreadConstraints | default .Values.global.topologySpreadConstraints }} - topologySpreadConstraints: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- if .Values.certController.priorityClassName }} - priorityClassName: {{ .Values.certController.priorityClassName }} - {{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/cert-controller-poddisruptionbudget.yaml b/charts/external-secrets/external-secrets/templates/cert-controller-poddisruptionbudget.yaml deleted file mode 100644 index e61cb8e..0000000 --- a/charts/external-secrets/external-secrets/templates/cert-controller-poddisruptionbudget.yaml +++ /dev/null @@ -1,19 +0,0 @@ -{{- if and .Values.certController.create .Values.certController.podDisruptionBudget.enabled (not .Values.webhook.certManager.enabled) }} -apiVersion: policy/v1 -kind: PodDisruptionBudget -metadata: - name: {{ include "external-secrets.fullname" . }}-cert-controller-pdb - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets-cert-controller.labels" . | nindent 4 }} -spec: - {{- if .Values.certController.podDisruptionBudget.minAvailable }} - minAvailable: {{ .Values.certController.podDisruptionBudget.minAvailable }} - {{- end }} - {{- if .Values.certController.podDisruptionBudget.maxUnavailable }} - maxUnavailable: {{ .Values.certController.podDisruptionBudget.maxUnavailable }} - {{- end }} - selector: - matchLabels: - {{- include "external-secrets-cert-controller.selectorLabels" . | nindent 6 }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/cert-controller-rbac.yaml b/charts/external-secrets/external-secrets/templates/cert-controller-rbac.yaml deleted file mode 100644 index 84a0c11..0000000 --- a/charts/external-secrets/external-secrets/templates/cert-controller-rbac.yaml +++ /dev/null @@ -1,86 +0,0 @@ -{{- if and .Values.certController.create .Values.certController.rbac.create (not .Values.webhook.certManager.enabled) -}} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ include "external-secrets.fullname" . }}-cert-controller - labels: - {{- include "external-secrets-cert-controller.labels" . | nindent 4 }} -rules: - - apiGroups: - - "apiextensions.k8s.io" - resources: - - "customresourcedefinitions" - verbs: - - "get" - - "list" - - "watch" - - "update" - - "patch" - - apiGroups: - - "admissionregistration.k8s.io" - resources: - - "validatingwebhookconfigurations" - verbs: - - "list" - - "watch" - - "get" - - apiGroups: - - "admissionregistration.k8s.io" - resources: - - "validatingwebhookconfigurations" - resourceNames: - - "secretstore-validate" - - "externalsecret-validate" - verbs: - - "update" - - "patch" - - apiGroups: - - "" - resources: - - "endpoints" - verbs: - - "list" - - "get" - - "watch" - - apiGroups: - - "" - resources: - - "events" - verbs: - - "create" - - "patch" - - apiGroups: - - "" - resources: - - "secrets" - verbs: - - "get" - - "list" - - "watch" - - "update" - - "patch" - - apiGroups: - - "coordination.k8s.io" - resources: - - "leases" - verbs: - - "get" - - "create" - - "update" - - "patch" ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ include "external-secrets.fullname" . }}-cert-controller - labels: - {{- include "external-secrets-cert-controller.labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ include "external-secrets.fullname" . }}-cert-controller -subjects: - - name: {{ include "external-secrets-cert-controller.serviceAccountName" . }} - namespace: {{ template "external-secrets.namespace" . }} - kind: ServiceAccount -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/cert-controller-service.yaml b/charts/external-secrets/external-secrets/templates/cert-controller-service.yaml deleted file mode 100644 index 12cb4f4..0000000 --- a/charts/external-secrets/external-secrets/templates/cert-controller-service.yaml +++ /dev/null @@ -1,28 +0,0 @@ -{{- if and .Values.certController.create .Values.certController.metrics.service.enabled (not .Values.webhook.certManager.enabled) }} -apiVersion: v1 -kind: Service -metadata: - name: {{ include "external-secrets.fullname" . }}-cert-controller-metrics - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets.labels" . | nindent 4 }} - {{- with .Values.metrics.service.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - type: ClusterIP - {{- if .Values.service.ipFamilyPolicy }} - ipFamilyPolicy: {{ .Values.service.ipFamilyPolicy }} - {{- end }} - {{- if .Values.service.ipFamilies }} - ipFamilies: {{ .Values.service.ipFamilies | toYaml | nindent 2 }} - {{- end }} - ports: - - port: {{ .Values.certController.metrics.service.port }} - protocol: TCP - targetPort: metrics - name: metrics - selector: - {{- include "external-secrets-cert-controller.selectorLabels" . | nindent 4 }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/cert-controller-serviceaccount.yaml b/charts/external-secrets/external-secrets/templates/cert-controller-serviceaccount.yaml deleted file mode 100644 index 6a36f9d..0000000 --- a/charts/external-secrets/external-secrets/templates/cert-controller-serviceaccount.yaml +++ /dev/null @@ -1,16 +0,0 @@ -{{- if and .Values.certController.create .Values.certController.serviceAccount.create (not .Values.webhook.certManager.enabled) -}} -apiVersion: v1 -kind: ServiceAccount -metadata: - name: {{ include "external-secrets-cert-controller.serviceAccountName" . }} - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets-cert-controller.labels" . | nindent 4 }} - {{- with .Values.certController.serviceAccount.extraLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.certController.serviceAccount.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/crds/acraccesstoken.yaml b/charts/external-secrets/external-secrets/templates/crds/acraccesstoken.yaml deleted file mode 100644 index 7dc3c8e..0000000 --- a/charts/external-secrets/external-secrets/templates/crds/acraccesstoken.yaml +++ /dev/null @@ -1,203 +0,0 @@ -{{- if .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - {{- with .Values.crds.annotations }} - {{- toYaml . | nindent 4}} - {{- end }} - {{- if and .Values.crds.conversion.enabled .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} - controller-gen.kubebuilder.io/version: v0.16.1 - labels: - external-secrets.io/component: controller - name: acraccesstokens.generators.external-secrets.io -spec: - group: generators.external-secrets.io - names: - categories: - - acraccesstoken - kind: ACRAccessToken - listKind: ACRAccessTokenList - plural: acraccesstokens - shortNames: - - acraccesstoken - singular: acraccesstoken - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - ACRAccessToken returns a Azure Container Registry token - that can be used for pushing/pulling images. - Note: by default it will return an ACR Refresh Token with full access - (depending on the identity). - This can be scoped down to the repository level using .spec.scope. - In case scope is defined it will return an ACR Access Token. - - See docs: https://github.com/Azure/acr/blob/main/docs/AAD-OAuth.md - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - ACRAccessTokenSpec defines how to generate the access token - e.g. how to authenticate and which registry to use. - see: https://github.com/Azure/acr/blob/main/docs/AAD-OAuth.md#overview - properties: - auth: - properties: - managedIdentity: - description: ManagedIdentity uses Azure Managed Identity to authenticate with Azure. - properties: - identityId: - description: If multiple Managed Identity is assigned to the pod, you can select the one to be used - type: string - type: object - servicePrincipal: - description: ServicePrincipal uses Azure Service Principal credentials to authenticate with Azure. - properties: - secretRef: - description: |- - Configuration used to authenticate with Azure using static - credentials stored in a Kind=Secret. - properties: - clientId: - description: The Azure clientId of the service principle used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - clientSecret: - description: The Azure ClientSecret of the service principle used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - secretRef - type: object - workloadIdentity: - description: WorkloadIdentity uses Azure Workload Identity to authenticate with Azure. - properties: - serviceAccountRef: - description: |- - ServiceAccountRef specified the service account - that should be used when authenticating with WorkloadIdentity. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - type: object - type: object - environmentType: - default: PublicCloud - description: |- - EnvironmentType specifies the Azure cloud environment endpoints to use for - connecting and authenticating with Azure. By default it points to the public cloud AAD endpoint. - The following endpoints are available, also see here: https://github.com/Azure/go-autorest/blob/main/autorest/azure/environments.go#L152 - PublicCloud, USGovernmentCloud, ChinaCloud, GermanCloud - enum: - - PublicCloud - - USGovernmentCloud - - ChinaCloud - - GermanCloud - type: string - registry: - description: |- - the domain name of the ACR registry - e.g. foobarexample.azurecr.io - type: string - scope: - description: |- - Define the scope for the access token, e.g. pull/push access for a repository. - if not provided it will return a refresh token that has full scope. - Note: you need to pin it down to the repository level, there is no wildcard available. - - examples: - repository:my-repository:pull,push - repository:my-repository:pull - - see docs for details: https://docs.docker.com/registry/spec/auth/scope/ - type: string - tenantId: - description: TenantID configures the Azure Tenant to send requests to. Required for ServicePrincipal auth type. - type: string - required: - - auth - - registry - type: object - type: object - served: true - storage: true - subresources: - status: {} -{{- if .Values.crds.conversion.enabled }} - conversion: - strategy: Webhook - webhook: - conversionReviewVersions: - - v1 - clientConfig: - service: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ .Release.Namespace | quote }} - path: /convert -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/crds/clusterexternalsecret.yaml b/charts/external-secrets/external-secrets/templates/crds/clusterexternalsecret.yaml deleted file mode 100644 index 6fdd831..0000000 --- a/charts/external-secrets/external-secrets/templates/crds/clusterexternalsecret.yaml +++ /dev/null @@ -1,666 +0,0 @@ -{{- if and (.Values.installCRDs) (.Values.crds.createClusterExternalSecret) }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - {{- with .Values.crds.annotations }} - {{- toYaml . | nindent 4}} - {{- end }} - {{- if and .Values.crds.conversion.enabled .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} - controller-gen.kubebuilder.io/version: v0.16.1 - labels: - external-secrets.io/component: controller - name: clusterexternalsecrets.external-secrets.io -spec: - group: external-secrets.io - names: - categories: - - externalsecrets - kind: ClusterExternalSecret - listKind: ClusterExternalSecretList - plural: clusterexternalsecrets - shortNames: - - ces - singular: clusterexternalsecret - scope: Cluster - versions: - - additionalPrinterColumns: - - jsonPath: .spec.externalSecretSpec.secretStoreRef.name - name: Store - type: string - - jsonPath: .spec.refreshTime - name: Refresh Interval - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - name: v1beta1 - schema: - openAPIV3Schema: - description: ClusterExternalSecret is the Schema for the clusterexternalsecrets API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ClusterExternalSecretSpec defines the desired state of ClusterExternalSecret. - properties: - externalSecretMetadata: - description: The metadata of the external secrets to be created - properties: - annotations: - additionalProperties: - type: string - type: object - labels: - additionalProperties: - type: string - type: object - type: object - externalSecretName: - description: The name of the external secrets to be created defaults to the name of the ClusterExternalSecret - type: string - externalSecretSpec: - description: The spec for the ExternalSecrets to be created - properties: - data: - description: Data defines the connection between the Kubernetes Secret keys and the Provider data - items: - description: ExternalSecretData defines the connection between the Kubernetes Secret key (spec.data.) and the Provider data. - properties: - remoteRef: - description: |- - RemoteRef points to the remote secret and defines - which secret (version/property/..) to fetch. - properties: - conversionStrategy: - default: Default - description: Used to define a conversion Strategy - enum: - - Default - - Unicode - type: string - decodingStrategy: - default: None - description: Used to define a decoding Strategy - enum: - - Auto - - Base64 - - Base64URL - - None - type: string - key: - description: Key is the key used in the Provider, mandatory - type: string - metadataPolicy: - default: None - description: Policy for fetching tags/labels from provider secrets, possible options are Fetch, None. Defaults to None - enum: - - None - - Fetch - type: string - property: - description: Used to select a specific property of the Provider value (if a map), if supported - type: string - version: - description: Used to select a specific version of the Provider value, if supported - type: string - required: - - key - type: object - secretKey: - description: |- - SecretKey defines the key in which the controller stores - the value. This is the key in the Kind=Secret - type: string - sourceRef: - description: |- - SourceRef allows you to override the source - from which the value will pulled from. - maxProperties: 1 - properties: - generatorRef: - description: |- - GeneratorRef points to a generator custom resource. - - Deprecated: The generatorRef is not implemented in .data[]. - this will be removed with v1. - properties: - apiVersion: - default: generators.external-secrets.io/v1alpha1 - description: Specify the apiVersion of the generator resource - type: string - kind: - description: Specify the Kind of the resource, e.g. Password, ACRAccessToken etc. - type: string - name: - description: Specify the name of the generator resource - type: string - required: - - kind - - name - type: object - storeRef: - description: SecretStoreRef defines which SecretStore to fetch the ExternalSecret data. - properties: - kind: - description: |- - Kind of the SecretStore resource (SecretStore or ClusterSecretStore) - Defaults to `SecretStore` - type: string - name: - description: Name of the SecretStore resource - type: string - required: - - name - type: object - type: object - required: - - remoteRef - - secretKey - type: object - type: array - dataFrom: - description: |- - DataFrom is used to fetch all properties from a specific Provider data - If multiple entries are specified, the Secret keys are merged in the specified order - items: - properties: - extract: - description: |- - Used to extract multiple key/value pairs from one secret - Note: Extract does not support sourceRef.Generator or sourceRef.GeneratorRef. - properties: - conversionStrategy: - default: Default - description: Used to define a conversion Strategy - enum: - - Default - - Unicode - type: string - decodingStrategy: - default: None - description: Used to define a decoding Strategy - enum: - - Auto - - Base64 - - Base64URL - - None - type: string - key: - description: Key is the key used in the Provider, mandatory - type: string - metadataPolicy: - default: None - description: Policy for fetching tags/labels from provider secrets, possible options are Fetch, None. Defaults to None - enum: - - None - - Fetch - type: string - property: - description: Used to select a specific property of the Provider value (if a map), if supported - type: string - version: - description: Used to select a specific version of the Provider value, if supported - type: string - required: - - key - type: object - find: - description: |- - Used to find secrets based on tags or regular expressions - Note: Find does not support sourceRef.Generator or sourceRef.GeneratorRef. - properties: - conversionStrategy: - default: Default - description: Used to define a conversion Strategy - enum: - - Default - - Unicode - type: string - decodingStrategy: - default: None - description: Used to define a decoding Strategy - enum: - - Auto - - Base64 - - Base64URL - - None - type: string - name: - description: Finds secrets based on the name. - properties: - regexp: - description: Finds secrets base - type: string - type: object - path: - description: A root path to start the find operations. - type: string - tags: - additionalProperties: - type: string - description: Find secrets based on tags. - type: object - type: object - rewrite: - description: |- - Used to rewrite secret Keys after getting them from the secret Provider - Multiple Rewrite operations can be provided. They are applied in a layered order (first to last) - items: - properties: - regexp: - description: |- - Used to rewrite with regular expressions. - The resulting key will be the output of a regexp.ReplaceAll operation. - properties: - source: - description: Used to define the regular expression of a re.Compiler. - type: string - target: - description: Used to define the target pattern of a ReplaceAll operation. - type: string - required: - - source - - target - type: object - transform: - description: |- - Used to apply string transformation on the secrets. - The resulting key will be the output of the template applied by the operation. - properties: - template: - description: |- - Used to define the template to apply on the secret name. - `.value ` will specify the secret name in the template. - type: string - required: - - template - type: object - type: object - type: array - sourceRef: - description: |- - SourceRef points to a store or generator - which contains secret values ready to use. - Use this in combination with Extract or Find pull values out of - a specific SecretStore. - When sourceRef points to a generator Extract or Find is not supported. - The generator returns a static map of values - maxProperties: 1 - properties: - generatorRef: - description: GeneratorRef points to a generator custom resource. - properties: - apiVersion: - default: generators.external-secrets.io/v1alpha1 - description: Specify the apiVersion of the generator resource - type: string - kind: - description: Specify the Kind of the resource, e.g. Password, ACRAccessToken etc. - type: string - name: - description: Specify the name of the generator resource - type: string - required: - - kind - - name - type: object - storeRef: - description: SecretStoreRef defines which SecretStore to fetch the ExternalSecret data. - properties: - kind: - description: |- - Kind of the SecretStore resource (SecretStore or ClusterSecretStore) - Defaults to `SecretStore` - type: string - name: - description: Name of the SecretStore resource - type: string - required: - - name - type: object - type: object - type: object - type: array - refreshInterval: - default: 1h - description: |- - RefreshInterval is the amount of time before the values are read again from the SecretStore provider - Valid time units are "ns", "us" (or "µs"), "ms", "s", "m", "h" - May be set to zero to fetch and create it once. Defaults to 1h. - type: string - secretStoreRef: - description: SecretStoreRef defines which SecretStore to fetch the ExternalSecret data. - properties: - kind: - description: |- - Kind of the SecretStore resource (SecretStore or ClusterSecretStore) - Defaults to `SecretStore` - type: string - name: - description: Name of the SecretStore resource - type: string - required: - - name - type: object - target: - default: - creationPolicy: Owner - deletionPolicy: Retain - description: |- - ExternalSecretTarget defines the Kubernetes Secret to be created - There can be only one target per ExternalSecret. - properties: - creationPolicy: - default: Owner - description: |- - CreationPolicy defines rules on how to create the resulting Secret - Defaults to 'Owner' - enum: - - Owner - - Orphan - - Merge - - None - type: string - deletionPolicy: - default: Retain - description: |- - DeletionPolicy defines rules on how to delete the resulting Secret - Defaults to 'Retain' - enum: - - Delete - - Merge - - Retain - type: string - immutable: - description: Immutable defines if the final secret will be immutable - type: boolean - name: - description: |- - Name defines the name of the Secret resource to be managed - This field is immutable - Defaults to the .metadata.name of the ExternalSecret resource - type: string - template: - description: Template defines a blueprint for the created Secret resource. - properties: - data: - additionalProperties: - type: string - type: object - engineVersion: - default: v2 - description: |- - EngineVersion specifies the template engine version - that should be used to compile/execute the - template specified in .data and .templateFrom[]. - enum: - - v1 - - v2 - type: string - mergePolicy: - default: Replace - enum: - - Replace - - Merge - type: string - metadata: - description: ExternalSecretTemplateMetadata defines metadata fields for the Secret blueprint. - properties: - annotations: - additionalProperties: - type: string - type: object - labels: - additionalProperties: - type: string - type: object - type: object - templateFrom: - items: - properties: - configMap: - properties: - items: - items: - properties: - key: - type: string - templateAs: - default: Values - enum: - - Values - - KeysAndValues - type: string - required: - - key - type: object - type: array - name: - type: string - required: - - items - - name - type: object - literal: - type: string - secret: - properties: - items: - items: - properties: - key: - type: string - templateAs: - default: Values - enum: - - Values - - KeysAndValues - type: string - required: - - key - type: object - type: array - name: - type: string - required: - - items - - name - type: object - target: - default: Data - enum: - - Data - - Annotations - - Labels - type: string - type: object - type: array - type: - type: string - type: object - type: object - type: object - namespaceSelector: - description: |- - The labels to select by to find the Namespaces to create the ExternalSecrets in. - Deprecated: Use NamespaceSelectors instead. - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - x-kubernetes-list-type: atomic - required: - - key - - operator - type: object - type: array - x-kubernetes-list-type: atomic - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - namespaceSelectors: - description: A list of labels to select by to find the Namespaces to create the ExternalSecrets in. The selectors are ORed. - items: - description: |- - A label selector is a label query over a set of resources. The result of matchLabels and - matchExpressions are ANDed. An empty label selector matches all objects. A null - label selector matches no objects. - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - x-kubernetes-list-type: atomic - required: - - key - - operator - type: object - type: array - x-kubernetes-list-type: atomic - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - type: array - namespaces: - description: Choose namespaces by name. This field is ORed with anything that NamespaceSelectors ends up choosing. - items: - type: string - type: array - refreshTime: - description: The time in which the controller should reconcile its objects and recheck namespaces for labels. - type: string - required: - - externalSecretSpec - type: object - status: - description: ClusterExternalSecretStatus defines the observed state of ClusterExternalSecret. - properties: - conditions: - items: - properties: - message: - type: string - status: - type: string - type: - type: string - required: - - status - - type - type: object - type: array - externalSecretName: - description: ExternalSecretName is the name of the ExternalSecrets created by the ClusterExternalSecret - type: string - failedNamespaces: - description: Failed namespaces are the namespaces that failed to apply an ExternalSecret - items: - description: ClusterExternalSecretNamespaceFailure represents a failed namespace deployment and it's reason. - properties: - namespace: - description: Namespace is the namespace that failed when trying to apply an ExternalSecret - type: string - reason: - description: Reason is why the ExternalSecret failed to apply to the namespace - type: string - required: - - namespace - type: object - type: array - provisionedNamespaces: - description: ProvisionedNamespaces are the namespaces where the ClusterExternalSecret has secrets - items: - type: string - type: array - type: object - type: object - served: true - storage: true - subresources: - status: {} -{{- if .Values.crds.conversion.enabled }} - conversion: - strategy: Webhook - webhook: - conversionReviewVersions: - - v1 - clientConfig: - service: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ .Release.Namespace | quote }} - path: /convert -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/crds/clustersecretstore.yaml b/charts/external-secrets/external-secrets/templates/crds/clustersecretstore.yaml deleted file mode 100644 index 0c946b5..0000000 --- a/charts/external-secrets/external-secrets/templates/crds/clustersecretstore.yaml +++ /dev/null @@ -1,4601 +0,0 @@ -{{- if and (.Values.installCRDs) (.Values.crds.createClusterSecretStore) }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - {{- with .Values.crds.annotations }} - {{- toYaml . | nindent 4}} - {{- end }} - {{- if and .Values.crds.conversion.enabled .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} - controller-gen.kubebuilder.io/version: v0.16.1 - labels: - external-secrets.io/component: controller - name: clustersecretstores.external-secrets.io -spec: - group: external-secrets.io - names: - categories: - - externalsecrets - kind: ClusterSecretStore - listKind: ClusterSecretStoreList - plural: clustersecretstores - shortNames: - - css - singular: clustersecretstore - scope: Cluster - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: AGE - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].reason - name: Status - type: string - deprecated: true - name: v1alpha1 - schema: - openAPIV3Schema: - description: ClusterSecretStore represents a secure external location for storing secrets, which can be referenced as part of `storeRef` fields. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: SecretStoreSpec defines the desired state of SecretStore. - properties: - controller: - description: |- - Used to select the correct ESO controller (think: ingress.ingressClassName) - The ESO controller is instantiated with a specific controller name and filters ES based on this property - type: string - provider: - description: Used to configure the provider. Only one provider may be set - maxProperties: 1 - minProperties: 1 - properties: - akeyless: - description: Akeyless configures this store to sync secrets using Akeyless Vault provider - properties: - akeylessGWApiURL: - description: Akeyless GW API Url from which the secrets to be fetched from. - type: string - authSecretRef: - description: Auth configures how the operator authenticates with Akeyless. - properties: - kubernetesAuth: - description: |- - Kubernetes authenticates with Akeyless by passing the ServiceAccount - token stored in the named Secret resource. - properties: - accessID: - description: the Akeyless Kubernetes auth-method access-id - type: string - k8sConfName: - description: Kubernetes-auth configuration name in Akeyless-Gateway - type: string - secretRef: - description: |- - Optional secret field containing a Kubernetes ServiceAccount JWT used - for authenticating with Akeyless. If a name is specified without a key, - `token` is the default. If one is not specified, the one bound to - the controller will be used. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - serviceAccountRef: - description: |- - Optional service account field containing the name of a kubernetes ServiceAccount. - If the service account is specified, the service account secret token JWT will be used - for authenticating with Akeyless. If the service account selector is not supplied, - the secretRef will be used instead. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - accessID - - k8sConfName - type: object - secretRef: - description: |- - Reference to a Secret that contains the details - to authenticate with Akeyless. - properties: - accessID: - description: The SecretAccessID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - accessType: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - accessTypeParam: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - caBundle: - description: |- - PEM/base64 encoded CA bundle used to validate Akeyless Gateway certificate. Only used - if the AkeylessGWApiURL URL is using HTTPS protocol. If not set the system root certificates - are used to validate the TLS connection. - format: byte - type: string - caProvider: - description: The provider for the CA bundle to use to validate Akeyless Gateway certificate. - properties: - key: - description: The key the value inside of the provider type to use, only used with "Secret" type - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: The namespace the Provider type is in. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - required: - - akeylessGWApiURL - - authSecretRef - type: object - alibaba: - description: Alibaba configures this store to sync secrets using Alibaba Cloud provider - properties: - auth: - description: AlibabaAuth contains a secretRef for credentials. - properties: - rrsa: - description: Authenticate against Alibaba using RRSA. - properties: - oidcProviderArn: - type: string - oidcTokenFilePath: - type: string - roleArn: - type: string - sessionName: - type: string - required: - - oidcProviderArn - - oidcTokenFilePath - - roleArn - - sessionName - type: object - secretRef: - description: AlibabaAuthSecretRef holds secret references for Alibaba credentials. - properties: - accessKeyIDSecretRef: - description: The AccessKeyID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - accessKeySecretSecretRef: - description: The AccessKeySecret is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - accessKeyIDSecretRef - - accessKeySecretSecretRef - type: object - type: object - regionID: - description: Alibaba Region to be used for the provider - type: string - required: - - auth - - regionID - type: object - aws: - description: AWS configures this store to sync secrets using AWS Secret Manager provider - properties: - auth: - description: |- - Auth defines the information necessary to authenticate against AWS - if not set aws sdk will infer credentials from your environment - see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials - properties: - jwt: - description: Authenticate against AWS using service account tokens. - properties: - serviceAccountRef: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - type: object - secretRef: - description: |- - AWSAuthSecretRef holds secret references for AWS credentials - both AccessKeyID and SecretAccessKey must be defined in order to properly authenticate. - properties: - accessKeyIDSecretRef: - description: The AccessKeyID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretAccessKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - region: - description: AWS Region to be used for the provider - type: string - role: - description: Role is a Role ARN which the SecretManager provider will assume - type: string - service: - description: Service defines which service should be used to fetch the secrets - enum: - - SecretsManager - - ParameterStore - type: string - required: - - region - - service - type: object - azurekv: - description: AzureKV configures this store to sync secrets using Azure Key Vault provider - properties: - authSecretRef: - description: Auth configures how the operator authenticates with Azure. Required for ServicePrincipal auth type. - properties: - clientId: - description: The Azure clientId of the service principle used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - clientSecret: - description: The Azure ClientSecret of the service principle used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - authType: - default: ServicePrincipal - description: |- - Auth type defines how to authenticate to the keyvault service. - Valid values are: - - "ServicePrincipal" (default): Using a service principal (tenantId, clientId, clientSecret) - - "ManagedIdentity": Using Managed Identity assigned to the pod (see aad-pod-identity) - enum: - - ServicePrincipal - - ManagedIdentity - - WorkloadIdentity - type: string - identityId: - description: If multiple Managed Identity is assigned to the pod, you can select the one to be used - type: string - serviceAccountRef: - description: |- - ServiceAccountRef specified the service account - that should be used when authenticating with WorkloadIdentity. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - tenantId: - description: TenantID configures the Azure Tenant to send requests to. Required for ServicePrincipal auth type. - type: string - vaultUrl: - description: Vault Url from which the secrets to be fetched from. - type: string - required: - - vaultUrl - type: object - fake: - description: Fake configures a store with static key/value pairs - properties: - data: - items: - properties: - key: - type: string - value: - type: string - valueMap: - additionalProperties: - type: string - type: object - version: - type: string - required: - - key - type: object - type: array - required: - - data - type: object - gcpsm: - description: GCPSM configures this store to sync secrets using Google Cloud Platform Secret Manager provider - properties: - auth: - description: Auth defines the information necessary to authenticate against GCP - properties: - secretRef: - properties: - secretAccessKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - workloadIdentity: - properties: - clusterLocation: - type: string - clusterName: - type: string - clusterProjectID: - type: string - serviceAccountRef: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - clusterLocation - - clusterName - - serviceAccountRef - type: object - type: object - projectID: - description: ProjectID project where secret is located - type: string - type: object - gitlab: - description: GitLab configures this store to sync secrets using GitLab Variables provider - properties: - auth: - description: Auth configures how secret-manager authenticates with a GitLab instance. - properties: - SecretRef: - properties: - accessToken: - description: AccessToken is used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - SecretRef - type: object - projectID: - description: ProjectID specifies a project where secrets are located. - type: string - url: - description: URL configures the GitLab instance URL. Defaults to https://gitlab.com/. - type: string - required: - - auth - type: object - ibm: - description: IBM configures this store to sync secrets using IBM Cloud provider - properties: - auth: - description: Auth configures how secret-manager authenticates with the IBM secrets manager. - properties: - secretRef: - properties: - secretApiKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - secretRef - type: object - serviceUrl: - description: ServiceURL is the Endpoint URL that is specific to the Secrets Manager service instance - type: string - required: - - auth - type: object - kubernetes: - description: Kubernetes configures this store to sync secrets using a Kubernetes cluster provider - properties: - auth: - description: Auth configures how secret-manager authenticates with a Kubernetes instance. - maxProperties: 1 - minProperties: 1 - properties: - cert: - description: has both clientCert and clientKey as secretKeySelector - properties: - clientCert: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - clientKey: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - serviceAccount: - description: points to a service account that should be used for authentication - properties: - serviceAccount: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - type: object - token: - description: use static token to authenticate with - properties: - bearerToken: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - remoteNamespace: - default: default - description: Remote namespace to fetch the secrets from - type: string - server: - description: configures the Kubernetes server Address. - properties: - caBundle: - description: CABundle is a base64-encoded CA certificate - format: byte - type: string - caProvider: - description: 'see: https://external-secrets.io/v0.4.1/spec/#external-secrets.io/v1alpha1.CAProvider' - properties: - key: - description: The key the value inside of the provider type to use, only used with "Secret" type - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: The namespace the Provider type is in. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - url: - default: kubernetes.default - description: configures the Kubernetes server Address. - type: string - type: object - required: - - auth - type: object - oracle: - description: Oracle configures this store to sync secrets using Oracle Vault provider - properties: - auth: - description: |- - Auth configures how secret-manager authenticates with the Oracle Vault. - If empty, instance principal is used. Optionally, the authenticating principal type - and/or user data may be supplied for the use of workload identity and user principal. - properties: - secretRef: - description: SecretRef to pass through sensitive information. - properties: - fingerprint: - description: Fingerprint is the fingerprint of the API private key. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - privatekey: - description: PrivateKey is the user's API Signing Key in PEM format, used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - fingerprint - - privatekey - type: object - tenancy: - description: Tenancy is the tenancy OCID where user is located. - type: string - user: - description: User is an access OCID specific to the account. - type: string - required: - - secretRef - - tenancy - - user - type: object - compartment: - description: |- - Compartment is the vault compartment OCID. - Required for PushSecret - type: string - encryptionKey: - description: |- - EncryptionKey is the OCID of the encryption key within the vault. - Required for PushSecret - type: string - principalType: - description: |- - The type of principal to use for authentication. If left blank, the Auth struct will - determine the principal type. This optional field must be specified if using - workload identity. - enum: - - "" - - UserPrincipal - - InstancePrincipal - - Workload - type: string - region: - description: Region is the region where vault is located. - type: string - serviceAccountRef: - description: |- - ServiceAccountRef specified the service account - that should be used when authenticating with WorkloadIdentity. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - vault: - description: Vault is the vault's OCID of the specific vault where secret is located. - type: string - required: - - region - - vault - type: object - passworddepot: - description: Configures a store to sync secrets with a Password Depot instance. - properties: - auth: - description: Auth configures how secret-manager authenticates with a Password Depot instance. - properties: - secretRef: - properties: - credentials: - description: Username / Password is used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - secretRef - type: object - database: - description: Database to use as source - type: string - host: - description: URL configures the Password Depot instance URL. - type: string - required: - - auth - - database - - host - type: object - vault: - description: Vault configures this store to sync secrets using Hashi provider - properties: - auth: - description: Auth configures how secret-manager authenticates with the Vault server. - properties: - appRole: - description: |- - AppRole authenticates with Vault using the App Role auth mechanism, - with the role and secret stored in a Kubernetes Secret resource. - properties: - path: - default: approle - description: |- - Path where the App Role authentication backend is mounted - in Vault, e.g: "approle" - type: string - roleId: - description: |- - RoleID configured in the App Role authentication backend when setting - up the authentication backend in Vault. - type: string - secretRef: - description: |- - Reference to a key in a Secret that contains the App Role secret used - to authenticate with Vault. - The `key` field must be specified and denotes which entry within the Secret - resource is used as the app role secret. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - path - - roleId - - secretRef - type: object - cert: - description: |- - Cert authenticates with TLS Certificates by passing client certificate, private key and ca certificate - Cert authentication method - properties: - clientCert: - description: |- - ClientCert is a certificate to authenticate using the Cert Vault - authentication method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretRef: - description: |- - SecretRef to a key in a Secret resource containing client private key to - authenticate with Vault using the Cert authentication method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - jwt: - description: |- - Jwt authenticates with Vault by passing role and JWT token using the - JWT/OIDC authentication method - properties: - kubernetesServiceAccountToken: - description: |- - Optional ServiceAccountToken specifies the Kubernetes service account for which to request - a token for with the `TokenRequest` API. - properties: - audiences: - description: |- - Optional audiences field that will be used to request a temporary Kubernetes service - account token for the service account referenced by `serviceAccountRef`. - Defaults to a single audience `vault` it not specified. - items: - type: string - type: array - expirationSeconds: - description: |- - Optional expiration time in seconds that will be used to request a temporary - Kubernetes service account token for the service account referenced by - `serviceAccountRef`. - Defaults to 10 minutes. - format: int64 - type: integer - serviceAccountRef: - description: Service account field containing the name of a kubernetes ServiceAccount. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - serviceAccountRef - type: object - path: - default: jwt - description: |- - Path where the JWT authentication backend is mounted - in Vault, e.g: "jwt" - type: string - role: - description: |- - Role is a JWT role to authenticate using the JWT/OIDC Vault - authentication method - type: string - secretRef: - description: |- - Optional SecretRef that refers to a key in a Secret resource containing JWT token to - authenticate with Vault using the JWT/OIDC authentication method. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - path - type: object - kubernetes: - description: |- - Kubernetes authenticates with Vault by passing the ServiceAccount - token stored in the named Secret resource to the Vault server. - properties: - mountPath: - default: kubernetes - description: |- - Path where the Kubernetes authentication backend is mounted in Vault, e.g: - "kubernetes" - type: string - role: - description: |- - A required field containing the Vault Role to assume. A Role binds a - Kubernetes ServiceAccount with a set of Vault policies. - type: string - secretRef: - description: |- - Optional secret field containing a Kubernetes ServiceAccount JWT used - for authenticating with Vault. If a name is specified without a key, - `token` is the default. If one is not specified, the one bound to - the controller will be used. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - serviceAccountRef: - description: |- - Optional service account field containing the name of a kubernetes ServiceAccount. - If the service account is specified, the service account secret token JWT will be used - for authenticating with Vault. If the service account selector is not supplied, - the secretRef will be used instead. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - mountPath - - role - type: object - ldap: - description: |- - Ldap authenticates with Vault by passing username/password pair using - the LDAP authentication method - properties: - path: - default: ldap - description: |- - Path where the LDAP authentication backend is mounted - in Vault, e.g: "ldap" - type: string - secretRef: - description: |- - SecretRef to a key in a Secret resource containing password for the LDAP - user used to authenticate with Vault using the LDAP authentication - method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - username: - description: |- - Username is a LDAP user name used to authenticate using the LDAP Vault - authentication method - type: string - required: - - path - - username - type: object - tokenSecretRef: - description: TokenSecretRef authenticates with Vault by presenting a token. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - caBundle: - description: |- - PEM encoded CA bundle used to validate Vault server certificate. Only used - if the Server URL is using HTTPS protocol. This parameter is ignored for - plain HTTP protocol connection. If not set the system root certificates - are used to validate the TLS connection. - format: byte - type: string - caProvider: - description: The provider for the CA bundle to use to validate Vault server certificate. - properties: - key: - description: The key the value inside of the provider type to use, only used with "Secret" type - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: The namespace the Provider type is in. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - forwardInconsistent: - description: |- - ForwardInconsistent tells Vault to forward read-after-write requests to the Vault - leader instead of simply retrying within a loop. This can increase performance if - the option is enabled serverside. - https://www.vaultproject.io/docs/configuration/replication#allow_forwarding_via_header - type: boolean - namespace: - description: |- - Name of the vault namespace. Namespaces is a set of features within Vault Enterprise that allows - Vault environments to support Secure Multi-tenancy. e.g: "ns1". - More about namespaces can be found here https://www.vaultproject.io/docs/enterprise/namespaces - type: string - path: - description: |- - Path is the mount path of the Vault KV backend endpoint, e.g: - "secret". The v2 KV secret engine version specific "/data" path suffix - for fetching secrets from Vault is optional and will be appended - if not present in specified path. - type: string - readYourWrites: - description: |- - ReadYourWrites ensures isolated read-after-write semantics by - providing discovered cluster replication states in each request. - More information about eventual consistency in Vault can be found here - https://www.vaultproject.io/docs/enterprise/consistency - type: boolean - server: - description: 'Server is the connection address for the Vault server, e.g: "https://vault.example.com:8200".' - type: string - version: - default: v2 - description: |- - Version is the Vault KV secret engine version. This can be either "v1" or - "v2". Version defaults to "v2". - enum: - - v1 - - v2 - type: string - required: - - auth - - server - type: object - webhook: - description: Webhook configures this store to sync secrets using a generic templated webhook - properties: - body: - description: Body - type: string - caBundle: - description: |- - PEM encoded CA bundle used to validate webhook server certificate. Only used - if the Server URL is using HTTPS protocol. This parameter is ignored for - plain HTTP protocol connection. If not set the system root certificates - are used to validate the TLS connection. - format: byte - type: string - caProvider: - description: The provider for the CA bundle to use to validate webhook server certificate. - properties: - key: - description: The key the value inside of the provider type to use, only used with "Secret" type - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: The namespace the Provider type is in. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - headers: - additionalProperties: - type: string - description: Headers - type: object - method: - description: Webhook Method - type: string - result: - description: Result formatting - properties: - jsonPath: - description: Json path of return value - type: string - type: object - secrets: - description: |- - Secrets to fill in templates - These secrets will be passed to the templating function as key value pairs under the given name - items: - properties: - name: - description: Name of this secret in templates - type: string - secretRef: - description: Secret ref to fill in credentials - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - name - - secretRef - type: object - type: array - timeout: - description: Timeout - type: string - url: - description: Webhook url to call - type: string - required: - - result - - url - type: object - yandexlockbox: - description: YandexLockbox configures this store to sync secrets using Yandex Lockbox provider - properties: - apiEndpoint: - description: Yandex.Cloud API endpoint (e.g. 'api.cloud.yandex.net:443') - type: string - auth: - description: Auth defines the information necessary to authenticate against Yandex Lockbox - properties: - authorizedKeySecretRef: - description: The authorized key used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - caProvider: - description: The provider for the CA bundle to use to validate Yandex.Cloud server certificate. - properties: - certSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - auth - type: object - type: object - retrySettings: - description: Used to configure http retries if failed - properties: - maxRetries: - format: int32 - type: integer - retryInterval: - type: string - type: object - required: - - provider - type: object - status: - description: SecretStoreStatus defines the observed state of the SecretStore. - properties: - conditions: - items: - properties: - lastTransitionTime: - format: date-time - type: string - message: - type: string - reason: - type: string - status: - type: string - type: - type: string - required: - - status - - type - type: object - type: array - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: AGE - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].reason - name: Status - type: string - - jsonPath: .status.capabilities - name: Capabilities - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - name: v1beta1 - schema: - openAPIV3Schema: - description: ClusterSecretStore represents a secure external location for storing secrets, which can be referenced as part of `storeRef` fields. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: SecretStoreSpec defines the desired state of SecretStore. - properties: - conditions: - description: Used to constraint a ClusterSecretStore to specific namespaces. Relevant only to ClusterSecretStore - items: - description: |- - ClusterSecretStoreCondition describes a condition by which to choose namespaces to process ExternalSecrets in - for a ClusterSecretStore instance. - properties: - namespaceRegexes: - description: Choose namespaces by using regex matching - items: - type: string - type: array - namespaceSelector: - description: Choose namespace using a labelSelector - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - x-kubernetes-list-type: atomic - required: - - key - - operator - type: object - type: array - x-kubernetes-list-type: atomic - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - namespaces: - description: Choose namespaces by name - items: - type: string - type: array - type: object - type: array - controller: - description: |- - Used to select the correct ESO controller (think: ingress.ingressClassName) - The ESO controller is instantiated with a specific controller name and filters ES based on this property - type: string - provider: - description: Used to configure the provider. Only one provider may be set - maxProperties: 1 - minProperties: 1 - properties: - akeyless: - description: Akeyless configures this store to sync secrets using Akeyless Vault provider - properties: - akeylessGWApiURL: - description: Akeyless GW API Url from which the secrets to be fetched from. - type: string - authSecretRef: - description: Auth configures how the operator authenticates with Akeyless. - properties: - kubernetesAuth: - description: |- - Kubernetes authenticates with Akeyless by passing the ServiceAccount - token stored in the named Secret resource. - properties: - accessID: - description: the Akeyless Kubernetes auth-method access-id - type: string - k8sConfName: - description: Kubernetes-auth configuration name in Akeyless-Gateway - type: string - secretRef: - description: |- - Optional secret field containing a Kubernetes ServiceAccount JWT used - for authenticating with Akeyless. If a name is specified without a key, - `token` is the default. If one is not specified, the one bound to - the controller will be used. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - serviceAccountRef: - description: |- - Optional service account field containing the name of a kubernetes ServiceAccount. - If the service account is specified, the service account secret token JWT will be used - for authenticating with Akeyless. If the service account selector is not supplied, - the secretRef will be used instead. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - accessID - - k8sConfName - type: object - secretRef: - description: |- - Reference to a Secret that contains the details - to authenticate with Akeyless. - properties: - accessID: - description: The SecretAccessID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - accessType: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - accessTypeParam: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - caBundle: - description: |- - PEM/base64 encoded CA bundle used to validate Akeyless Gateway certificate. Only used - if the AkeylessGWApiURL URL is using HTTPS protocol. If not set the system root certificates - are used to validate the TLS connection. - format: byte - type: string - caProvider: - description: The provider for the CA bundle to use to validate Akeyless Gateway certificate. - properties: - key: - description: The key where the CA certificate can be found in the Secret or ConfigMap. - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: |- - The namespace the Provider type is in. - Can only be defined when used in a ClusterSecretStore. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - required: - - akeylessGWApiURL - - authSecretRef - type: object - alibaba: - description: Alibaba configures this store to sync secrets using Alibaba Cloud provider - properties: - auth: - description: AlibabaAuth contains a secretRef for credentials. - properties: - rrsa: - description: Authenticate against Alibaba using RRSA. - properties: - oidcProviderArn: - type: string - oidcTokenFilePath: - type: string - roleArn: - type: string - sessionName: - type: string - required: - - oidcProviderArn - - oidcTokenFilePath - - roleArn - - sessionName - type: object - secretRef: - description: AlibabaAuthSecretRef holds secret references for Alibaba credentials. - properties: - accessKeyIDSecretRef: - description: The AccessKeyID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - accessKeySecretSecretRef: - description: The AccessKeySecret is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - accessKeyIDSecretRef - - accessKeySecretSecretRef - type: object - type: object - regionID: - description: Alibaba Region to be used for the provider - type: string - required: - - auth - - regionID - type: object - aws: - description: AWS configures this store to sync secrets using AWS Secret Manager provider - properties: - additionalRoles: - description: AdditionalRoles is a chained list of Role ARNs which the provider will sequentially assume before assuming the Role - items: - type: string - type: array - auth: - description: |- - Auth defines the information necessary to authenticate against AWS - if not set aws sdk will infer credentials from your environment - see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials - properties: - jwt: - description: Authenticate against AWS using service account tokens. - properties: - serviceAccountRef: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - type: object - secretRef: - description: |- - AWSAuthSecretRef holds secret references for AWS credentials - both AccessKeyID and SecretAccessKey must be defined in order to properly authenticate. - properties: - accessKeyIDSecretRef: - description: The AccessKeyID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretAccessKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - sessionTokenSecretRef: - description: |- - The SessionToken used for authentication - This must be defined if AccessKeyID and SecretAccessKey are temporary credentials - see: https://docs.aws.amazon.com/IAM/latest/UserGuide/id_credentials_temp_use-resources.html - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - externalID: - description: AWS External ID set on assumed IAM roles - type: string - prefix: - description: Prefix adds a prefix to all retrieved values. - type: string - region: - description: AWS Region to be used for the provider - type: string - role: - description: Role is a Role ARN which the provider will assume - type: string - secretsManager: - description: SecretsManager defines how the provider behaves when interacting with AWS SecretsManager - properties: - forceDeleteWithoutRecovery: - description: |- - Specifies whether to delete the secret without any recovery window. You - can't use both this parameter and RecoveryWindowInDays in the same call. - If you don't use either, then by default Secrets Manager uses a 30 day - recovery window. - see: https://docs.aws.amazon.com/secretsmanager/latest/apireference/API_DeleteSecret.html#SecretsManager-DeleteSecret-request-ForceDeleteWithoutRecovery - type: boolean - recoveryWindowInDays: - description: |- - The number of days from 7 to 30 that Secrets Manager waits before - permanently deleting the secret. You can't use both this parameter and - ForceDeleteWithoutRecovery in the same call. If you don't use either, - then by default Secrets Manager uses a 30 day recovery window. - see: https://docs.aws.amazon.com/secretsmanager/latest/apireference/API_DeleteSecret.html#SecretsManager-DeleteSecret-request-RecoveryWindowInDays - format: int64 - type: integer - type: object - service: - description: Service defines which service should be used to fetch the secrets - enum: - - SecretsManager - - ParameterStore - type: string - sessionTags: - description: AWS STS assume role session tags - items: - properties: - key: - type: string - value: - type: string - required: - - key - - value - type: object - type: array - transitiveTagKeys: - description: AWS STS assume role transitive session tags. Required when multiple rules are used with the provider - items: - type: string - type: array - required: - - region - - service - type: object - azurekv: - description: AzureKV configures this store to sync secrets using Azure Key Vault provider - properties: - authSecretRef: - description: Auth configures how the operator authenticates with Azure. Required for ServicePrincipal auth type. Optional for WorkloadIdentity. - properties: - clientCertificate: - description: The Azure ClientCertificate of the service principle used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - clientId: - description: The Azure clientId of the service principle or managed identity used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - clientSecret: - description: The Azure ClientSecret of the service principle used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - tenantId: - description: The Azure tenantId of the managed identity used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - authType: - default: ServicePrincipal - description: |- - Auth type defines how to authenticate to the keyvault service. - Valid values are: - - "ServicePrincipal" (default): Using a service principal (tenantId, clientId, clientSecret) - - "ManagedIdentity": Using Managed Identity assigned to the pod (see aad-pod-identity) - enum: - - ServicePrincipal - - ManagedIdentity - - WorkloadIdentity - type: string - environmentType: - default: PublicCloud - description: |- - EnvironmentType specifies the Azure cloud environment endpoints to use for - connecting and authenticating with Azure. By default it points to the public cloud AAD endpoint. - The following endpoints are available, also see here: https://github.com/Azure/go-autorest/blob/main/autorest/azure/environments.go#L152 - PublicCloud, USGovernmentCloud, ChinaCloud, GermanCloud - enum: - - PublicCloud - - USGovernmentCloud - - ChinaCloud - - GermanCloud - type: string - identityId: - description: If multiple Managed Identity is assigned to the pod, you can select the one to be used - type: string - serviceAccountRef: - description: |- - ServiceAccountRef specified the service account - that should be used when authenticating with WorkloadIdentity. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - tenantId: - description: TenantID configures the Azure Tenant to send requests to. Required for ServicePrincipal auth type. Optional for WorkloadIdentity. - type: string - vaultUrl: - description: Vault Url from which the secrets to be fetched from. - type: string - required: - - vaultUrl - type: object - beyondtrust: - description: Beyondtrust configures this store to sync secrets using Password Safe provider. - properties: - auth: - description: Auth configures how the operator authenticates with Beyondtrust. - properties: - certificate: - description: Content of the certificate (cert.pem) for use when authenticating with an OAuth client Id using a Client Certificate. - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - certificateKey: - description: Certificate private key (key.pem). For use when authenticating with an OAuth client Id - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - clientId: - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - clientSecret: - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - required: - - clientId - - clientSecret - type: object - server: - description: Auth configures how API server works. - properties: - apiUrl: - type: string - clientTimeOutSeconds: - description: Timeout specifies a time limit for requests made by this Client. The timeout includes connection time, any redirects, and reading the response body. Defaults to 45 seconds. - type: integer - retrievalType: - description: The secret retrieval type. SECRET = Secrets Safe (credential, text, file). MANAGED_ACCOUNT = Password Safe account associated with a system. - type: string - separator: - description: A character that separates the folder names. - type: string - verifyCA: - type: boolean - required: - - apiUrl - - verifyCA - type: object - required: - - auth - - server - type: object - bitwardensecretsmanager: - description: BitwardenSecretsManager configures this store to sync secrets using BitwardenSecretsManager provider - properties: - apiURL: - type: string - auth: - description: |- - Auth configures how secret-manager authenticates with a bitwarden machine account instance. - Make sure that the token being used has permissions on the given secret. - properties: - secretRef: - description: BitwardenSecretsManagerSecretRef contains the credential ref to the bitwarden instance. - properties: - credentials: - description: AccessToken used for the bitwarden instance. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - credentials - type: object - required: - - secretRef - type: object - bitwardenServerSDKURL: - type: string - caBundle: - description: |- - Base64 encoded certificate for the bitwarden server sdk. The sdk MUST run with HTTPS to make sure no MITM attack - can be performed. - type: string - caProvider: - description: 'see: https://external-secrets.io/latest/spec/#external-secrets.io/v1alpha1.CAProvider' - properties: - key: - description: The key where the CA certificate can be found in the Secret or ConfigMap. - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: |- - The namespace the Provider type is in. - Can only be defined when used in a ClusterSecretStore. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - identityURL: - type: string - organizationID: - description: OrganizationID determines which organization this secret store manages. - type: string - projectID: - description: ProjectID determines which project this secret store manages. - type: string - required: - - auth - - organizationID - - projectID - type: object - chef: - description: Chef configures this store to sync secrets with chef server - properties: - auth: - description: Auth defines the information necessary to authenticate against chef Server - properties: - secretRef: - description: ChefAuthSecretRef holds secret references for chef server login credentials. - properties: - privateKeySecretRef: - description: SecretKey is the Signing Key in PEM format, used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - privateKeySecretRef - type: object - required: - - secretRef - type: object - serverUrl: - description: ServerURL is the chef server URL used to connect to. If using orgs you should include your org in the url and terminate the url with a "/" - type: string - username: - description: UserName should be the user ID on the chef server - type: string - required: - - auth - - serverUrl - - username - type: object - conjur: - description: Conjur configures this store to sync secrets using conjur provider - properties: - auth: - properties: - apikey: - properties: - account: - type: string - apiKeyRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - userRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - account - - apiKeyRef - - userRef - type: object - jwt: - properties: - account: - type: string - hostId: - description: |- - Optional HostID for JWT authentication. This may be used depending - on how the Conjur JWT authenticator policy is configured. - type: string - secretRef: - description: |- - Optional SecretRef that refers to a key in a Secret resource containing JWT token to - authenticate with Conjur using the JWT authentication method. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - serviceAccountRef: - description: |- - Optional ServiceAccountRef specifies the Kubernetes service account for which to request - a token for with the `TokenRequest` API. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - serviceID: - description: The conjur authn jwt webservice id - type: string - required: - - account - - serviceID - type: object - type: object - caBundle: - type: string - caProvider: - description: |- - Used to provide custom certificate authority (CA) certificates - for a secret store. The CAProvider points to a Secret or ConfigMap resource - that contains a PEM-encoded certificate. - properties: - key: - description: The key where the CA certificate can be found in the Secret or ConfigMap. - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: |- - The namespace the Provider type is in. - Can only be defined when used in a ClusterSecretStore. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - url: - type: string - required: - - auth - - url - type: object - delinea: - description: |- - Delinea DevOps Secrets Vault - https://docs.delinea.com/online-help/products/devops-secrets-vault/current - properties: - clientId: - description: ClientID is the non-secret part of the credential. - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - clientSecret: - description: ClientSecret is the secret part of the credential. - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - tenant: - description: Tenant is the chosen hostname / site name. - type: string - tld: - description: |- - TLD is based on the server location that was chosen during provisioning. - If unset, defaults to "com". - type: string - urlTemplate: - description: |- - URLTemplate - If unset, defaults to "https://%s.secretsvaultcloud.%s/v1/%s%s". - type: string - required: - - clientId - - clientSecret - - tenant - type: object - device42: - description: Device42 configures this store to sync secrets using the Device42 provider - properties: - auth: - description: Auth configures how secret-manager authenticates with a Device42 instance. - properties: - secretRef: - properties: - credentials: - description: Username / Password is used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - secretRef - type: object - host: - description: URL configures the Device42 instance URL. - type: string - required: - - auth - - host - type: object - doppler: - description: Doppler configures this store to sync secrets using the Doppler provider - properties: - auth: - description: Auth configures how the Operator authenticates with the Doppler API - properties: - secretRef: - properties: - dopplerToken: - description: |- - The DopplerToken is used for authentication. - See https://docs.doppler.com/reference/api#authentication for auth token types. - The Key attribute defaults to dopplerToken if not specified. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - dopplerToken - type: object - required: - - secretRef - type: object - config: - description: Doppler config (required if not using a Service Token) - type: string - format: - description: Format enables the downloading of secrets as a file (string) - enum: - - json - - dotnet-json - - env - - yaml - - docker - type: string - nameTransformer: - description: Environment variable compatible name transforms that change secret names to a different format - enum: - - upper-camel - - camel - - lower-snake - - tf-var - - dotnet-env - - lower-kebab - type: string - project: - description: Doppler project (required if not using a Service Token) - type: string - required: - - auth - type: object - fake: - description: Fake configures a store with static key/value pairs - properties: - data: - items: - properties: - key: - type: string - value: - type: string - valueMap: - additionalProperties: - type: string - description: 'Deprecated: ValueMap is deprecated and is intended to be removed in the future, use the `value` field instead.' - type: object - version: - type: string - required: - - key - type: object - type: array - required: - - data - type: object - fortanix: - description: Fortanix configures this store to sync secrets using the Fortanix provider - properties: - apiKey: - description: APIKey is the API token to access SDKMS Applications. - properties: - secretRef: - description: SecretRef is a reference to a secret containing the SDKMS API Key. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - apiUrl: - description: APIURL is the URL of SDKMS API. Defaults to `sdkms.fortanix.com`. - type: string - type: object - gcpsm: - description: GCPSM configures this store to sync secrets using Google Cloud Platform Secret Manager provider - properties: - auth: - description: Auth defines the information necessary to authenticate against GCP - properties: - secretRef: - properties: - secretAccessKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - workloadIdentity: - properties: - clusterLocation: - type: string - clusterName: - type: string - clusterProjectID: - type: string - serviceAccountRef: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - clusterLocation - - clusterName - - serviceAccountRef - type: object - type: object - location: - description: Location optionally defines a location for a secret - type: string - projectID: - description: ProjectID project where secret is located - type: string - type: object - gitlab: - description: GitLab configures this store to sync secrets using GitLab Variables provider - properties: - auth: - description: Auth configures how secret-manager authenticates with a GitLab instance. - properties: - SecretRef: - properties: - accessToken: - description: AccessToken is used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - SecretRef - type: object - environment: - description: Environment environment_scope of gitlab CI/CD variables (Please see https://docs.gitlab.com/ee/ci/environments/#create-a-static-environment on how to create environments) - type: string - groupIDs: - description: GroupIDs specify, which gitlab groups to pull secrets from. Group secrets are read from left to right followed by the project variables. - items: - type: string - type: array - inheritFromGroups: - description: InheritFromGroups specifies whether parent groups should be discovered and checked for secrets. - type: boolean - projectID: - description: ProjectID specifies a project where secrets are located. - type: string - url: - description: URL configures the GitLab instance URL. Defaults to https://gitlab.com/. - type: string - required: - - auth - type: object - ibm: - description: IBM configures this store to sync secrets using IBM Cloud provider - properties: - auth: - description: Auth configures how secret-manager authenticates with the IBM secrets manager. - maxProperties: 1 - minProperties: 1 - properties: - containerAuth: - description: IBM Container-based auth with IAM Trusted Profile. - properties: - iamEndpoint: - type: string - profile: - description: the IBM Trusted Profile - type: string - tokenLocation: - description: Location the token is mounted on the pod - type: string - required: - - profile - type: object - secretRef: - properties: - secretApiKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - serviceUrl: - description: ServiceURL is the Endpoint URL that is specific to the Secrets Manager service instance - type: string - required: - - auth - type: object - infisical: - description: Infisical configures this store to sync secrets using the Infisical provider - properties: - auth: - description: Auth configures how the Operator authenticates with the Infisical API - properties: - universalAuthCredentials: - properties: - clientId: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - clientSecret: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - clientId - - clientSecret - type: object - type: object - hostAPI: - default: https://app.infisical.com/api - type: string - secretsScope: - properties: - environmentSlug: - type: string - projectSlug: - type: string - secretsPath: - default: / - type: string - required: - - environmentSlug - - projectSlug - type: object - required: - - auth - - secretsScope - type: object - keepersecurity: - description: KeeperSecurity configures this store to sync secrets using the KeeperSecurity provider - properties: - authRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - folderID: - type: string - required: - - authRef - - folderID - type: object - kubernetes: - description: Kubernetes configures this store to sync secrets using a Kubernetes cluster provider - properties: - auth: - description: Auth configures how secret-manager authenticates with a Kubernetes instance. - maxProperties: 1 - minProperties: 1 - properties: - cert: - description: has both clientCert and clientKey as secretKeySelector - properties: - clientCert: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - clientKey: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - serviceAccount: - description: points to a service account that should be used for authentication - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - token: - description: use static token to authenticate with - properties: - bearerToken: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - authRef: - description: A reference to a secret that contains the auth information. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - remoteNamespace: - default: default - description: Remote namespace to fetch the secrets from - type: string - server: - description: configures the Kubernetes server Address. - properties: - caBundle: - description: CABundle is a base64-encoded CA certificate - format: byte - type: string - caProvider: - description: 'see: https://external-secrets.io/v0.4.1/spec/#external-secrets.io/v1alpha1.CAProvider' - properties: - key: - description: The key where the CA certificate can be found in the Secret or ConfigMap. - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: |- - The namespace the Provider type is in. - Can only be defined when used in a ClusterSecretStore. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - url: - default: kubernetes.default - description: configures the Kubernetes server Address. - type: string - type: object - type: object - onboardbase: - description: Onboardbase configures this store to sync secrets using the Onboardbase provider - properties: - apiHost: - default: https://public.onboardbase.com/api/v1/ - description: APIHost use this to configure the host url for the API for selfhosted installation, default is https://public.onboardbase.com/api/v1/ - type: string - auth: - description: Auth configures how the Operator authenticates with the Onboardbase API - properties: - apiKeyRef: - description: |- - OnboardbaseAPIKey is the APIKey generated by an admin account. - It is used to recognize and authorize access to a project and environment within onboardbase - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - passcodeRef: - description: OnboardbasePasscode is the passcode attached to the API Key - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - apiKeyRef - - passcodeRef - type: object - environment: - default: development - description: Environment is the name of an environmnent within a project to pull the secrets from - type: string - project: - default: development - description: Project is an onboardbase project that the secrets should be pulled from - type: string - required: - - apiHost - - auth - - environment - - project - type: object - onepassword: - description: OnePassword configures this store to sync secrets using the 1Password Cloud provider - properties: - auth: - description: Auth defines the information necessary to authenticate against OnePassword Connect Server - properties: - secretRef: - description: OnePasswordAuthSecretRef holds secret references for 1Password credentials. - properties: - connectTokenSecretRef: - description: The ConnectToken is used for authentication to a 1Password Connect Server. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - connectTokenSecretRef - type: object - required: - - secretRef - type: object - connectHost: - description: ConnectHost defines the OnePassword Connect Server to connect to - type: string - vaults: - additionalProperties: - type: integer - description: Vaults defines which OnePassword vaults to search in which order - type: object - required: - - auth - - connectHost - - vaults - type: object - oracle: - description: Oracle configures this store to sync secrets using Oracle Vault provider - properties: - auth: - description: |- - Auth configures how secret-manager authenticates with the Oracle Vault. - If empty, use the instance principal, otherwise the user credentials specified in Auth. - properties: - secretRef: - description: SecretRef to pass through sensitive information. - properties: - fingerprint: - description: Fingerprint is the fingerprint of the API private key. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - privatekey: - description: PrivateKey is the user's API Signing Key in PEM format, used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - fingerprint - - privatekey - type: object - tenancy: - description: Tenancy is the tenancy OCID where user is located. - type: string - user: - description: User is an access OCID specific to the account. - type: string - required: - - secretRef - - tenancy - - user - type: object - compartment: - description: |- - Compartment is the vault compartment OCID. - Required for PushSecret - type: string - encryptionKey: - description: |- - EncryptionKey is the OCID of the encryption key within the vault. - Required for PushSecret - type: string - principalType: - description: |- - The type of principal to use for authentication. If left blank, the Auth struct will - determine the principal type. This optional field must be specified if using - workload identity. - enum: - - "" - - UserPrincipal - - InstancePrincipal - - Workload - type: string - region: - description: Region is the region where vault is located. - type: string - serviceAccountRef: - description: |- - ServiceAccountRef specified the service account - that should be used when authenticating with WorkloadIdentity. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - vault: - description: Vault is the vault's OCID of the specific vault where secret is located. - type: string - required: - - region - - vault - type: object - passbolt: - properties: - auth: - description: Auth defines the information necessary to authenticate against Passbolt Server - properties: - passwordSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - privateKeySecretRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - passwordSecretRef - - privateKeySecretRef - type: object - host: - description: Host defines the Passbolt Server to connect to - type: string - required: - - auth - - host - type: object - passworddepot: - description: Configures a store to sync secrets with a Password Depot instance. - properties: - auth: - description: Auth configures how secret-manager authenticates with a Password Depot instance. - properties: - secretRef: - properties: - credentials: - description: Username / Password is used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - secretRef - type: object - database: - description: Database to use as source - type: string - host: - description: URL configures the Password Depot instance URL. - type: string - required: - - auth - - database - - host - type: object - pulumi: - description: Pulumi configures this store to sync secrets using the Pulumi provider - properties: - accessToken: - description: AccessToken is the access tokens to sign in to the Pulumi Cloud Console. - properties: - secretRef: - description: SecretRef is a reference to a secret containing the Pulumi API token. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - apiUrl: - default: https://api.pulumi.com/api/preview - description: APIURL is the URL of the Pulumi API. - type: string - environment: - description: |- - Environment are YAML documents composed of static key-value pairs, programmatic expressions, - dynamically retrieved values from supported providers including all major clouds, - and other Pulumi ESC environments. - To create a new environment, visit https://www.pulumi.com/docs/esc/environments/ for more information. - type: string - organization: - description: |- - Organization are a space to collaborate on shared projects and stacks. - To create a new organization, visit https://app.pulumi.com/ and click "New Organization". - type: string - required: - - accessToken - - environment - - organization - type: object - scaleway: - description: Scaleway - properties: - accessKey: - description: AccessKey is the non-secret part of the api key. - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - apiUrl: - description: APIURL is the url of the api to use. Defaults to https://api.scaleway.com - type: string - projectId: - description: 'ProjectID is the id of your project, which you can find in the console: https://console.scaleway.com/project/settings' - type: string - region: - description: 'Region where your secrets are located: https://developers.scaleway.com/en/quickstart/#region-and-zone' - type: string - secretKey: - description: SecretKey is the non-secret part of the api key. - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - required: - - accessKey - - projectId - - region - - secretKey - type: object - secretserver: - description: |- - SecretServer configures this store to sync secrets using SecretServer provider - https://docs.delinea.com/online-help/secret-server/start.htm - properties: - password: - description: Password is the secret server account password. - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - serverURL: - description: |- - ServerURL - URL to your secret server installation - type: string - username: - description: Username is the secret server account username. - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - required: - - password - - serverURL - - username - type: object - senhasegura: - description: Senhasegura configures this store to sync secrets using senhasegura provider - properties: - auth: - description: Auth defines parameters to authenticate in senhasegura - properties: - clientId: - type: string - clientSecretSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - clientId - - clientSecretSecretRef - type: object - ignoreSslCertificate: - default: false - description: IgnoreSslCertificate defines if SSL certificate must be ignored - type: boolean - module: - description: Module defines which senhasegura module should be used to get secrets - type: string - url: - description: URL of senhasegura - type: string - required: - - auth - - module - - url - type: object - vault: - description: Vault configures this store to sync secrets using Hashi provider - properties: - auth: - description: Auth configures how secret-manager authenticates with the Vault server. - properties: - appRole: - description: |- - AppRole authenticates with Vault using the App Role auth mechanism, - with the role and secret stored in a Kubernetes Secret resource. - properties: - path: - default: approle - description: |- - Path where the App Role authentication backend is mounted - in Vault, e.g: "approle" - type: string - roleId: - description: |- - RoleID configured in the App Role authentication backend when setting - up the authentication backend in Vault. - type: string - roleRef: - description: |- - Reference to a key in a Secret that contains the App Role ID used - to authenticate with Vault. - The `key` field must be specified and denotes which entry within the Secret - resource is used as the app role id. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretRef: - description: |- - Reference to a key in a Secret that contains the App Role secret used - to authenticate with Vault. - The `key` field must be specified and denotes which entry within the Secret - resource is used as the app role secret. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - path - - secretRef - type: object - cert: - description: |- - Cert authenticates with TLS Certificates by passing client certificate, private key and ca certificate - Cert authentication method - properties: - clientCert: - description: |- - ClientCert is a certificate to authenticate using the Cert Vault - authentication method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretRef: - description: |- - SecretRef to a key in a Secret resource containing client private key to - authenticate with Vault using the Cert authentication method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - iam: - description: |- - Iam authenticates with vault by passing a special AWS request signed with AWS IAM credentials - AWS IAM authentication method - properties: - externalID: - description: AWS External ID set on assumed IAM roles - type: string - jwt: - description: Specify a service account with IRSA enabled - properties: - serviceAccountRef: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - type: object - path: - description: 'Path where the AWS auth method is enabled in Vault, e.g: "aws"' - type: string - region: - description: AWS region - type: string - role: - description: This is the AWS role to be assumed before talking to vault - type: string - secretRef: - description: Specify credentials in a Secret object - properties: - accessKeyIDSecretRef: - description: The AccessKeyID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretAccessKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - sessionTokenSecretRef: - description: |- - The SessionToken used for authentication - This must be defined if AccessKeyID and SecretAccessKey are temporary credentials - see: https://docs.aws.amazon.com/IAM/latest/UserGuide/id_credentials_temp_use-resources.html - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - vaultAwsIamServerID: - description: 'X-Vault-AWS-IAM-Server-ID is an additional header used by Vault IAM auth method to mitigate against different types of replay attacks. More details here: https://developer.hashicorp.com/vault/docs/auth/aws' - type: string - vaultRole: - description: Vault Role. In vault, a role describes an identity with a set of permissions, groups, or policies you want to attach a user of the secrets engine - type: string - required: - - vaultRole - type: object - jwt: - description: |- - Jwt authenticates with Vault by passing role and JWT token using the - JWT/OIDC authentication method - properties: - kubernetesServiceAccountToken: - description: |- - Optional ServiceAccountToken specifies the Kubernetes service account for which to request - a token for with the `TokenRequest` API. - properties: - audiences: - description: |- - Optional audiences field that will be used to request a temporary Kubernetes service - account token for the service account referenced by `serviceAccountRef`. - Defaults to a single audience `vault` it not specified. - Deprecated: use serviceAccountRef.Audiences instead - items: - type: string - type: array - expirationSeconds: - description: |- - Optional expiration time in seconds that will be used to request a temporary - Kubernetes service account token for the service account referenced by - `serviceAccountRef`. - Deprecated: this will be removed in the future. - Defaults to 10 minutes. - format: int64 - type: integer - serviceAccountRef: - description: Service account field containing the name of a kubernetes ServiceAccount. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - serviceAccountRef - type: object - path: - default: jwt - description: |- - Path where the JWT authentication backend is mounted - in Vault, e.g: "jwt" - type: string - role: - description: |- - Role is a JWT role to authenticate using the JWT/OIDC Vault - authentication method - type: string - secretRef: - description: |- - Optional SecretRef that refers to a key in a Secret resource containing JWT token to - authenticate with Vault using the JWT/OIDC authentication method. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - path - type: object - kubernetes: - description: |- - Kubernetes authenticates with Vault by passing the ServiceAccount - token stored in the named Secret resource to the Vault server. - properties: - mountPath: - default: kubernetes - description: |- - Path where the Kubernetes authentication backend is mounted in Vault, e.g: - "kubernetes" - type: string - role: - description: |- - A required field containing the Vault Role to assume. A Role binds a - Kubernetes ServiceAccount with a set of Vault policies. - type: string - secretRef: - description: |- - Optional secret field containing a Kubernetes ServiceAccount JWT used - for authenticating with Vault. If a name is specified without a key, - `token` is the default. If one is not specified, the one bound to - the controller will be used. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - serviceAccountRef: - description: |- - Optional service account field containing the name of a kubernetes ServiceAccount. - If the service account is specified, the service account secret token JWT will be used - for authenticating with Vault. If the service account selector is not supplied, - the secretRef will be used instead. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - mountPath - - role - type: object - ldap: - description: |- - Ldap authenticates with Vault by passing username/password pair using - the LDAP authentication method - properties: - path: - default: ldap - description: |- - Path where the LDAP authentication backend is mounted - in Vault, e.g: "ldap" - type: string - secretRef: - description: |- - SecretRef to a key in a Secret resource containing password for the LDAP - user used to authenticate with Vault using the LDAP authentication - method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - username: - description: |- - Username is a LDAP user name used to authenticate using the LDAP Vault - authentication method - type: string - required: - - path - - username - type: object - namespace: - description: |- - Name of the vault namespace to authenticate to. This can be different than the namespace your secret is in. - Namespaces is a set of features within Vault Enterprise that allows - Vault environments to support Secure Multi-tenancy. e.g: "ns1". - More about namespaces can be found here https://www.vaultproject.io/docs/enterprise/namespaces - This will default to Vault.Namespace field if set, or empty otherwise - type: string - tokenSecretRef: - description: TokenSecretRef authenticates with Vault by presenting a token. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - userPass: - description: UserPass authenticates with Vault by passing username/password pair - properties: - path: - default: user - description: |- - Path where the UserPassword authentication backend is mounted - in Vault, e.g: "user" - type: string - secretRef: - description: |- - SecretRef to a key in a Secret resource containing password for the - user used to authenticate with Vault using the UserPass authentication - method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - username: - description: |- - Username is a user name used to authenticate using the UserPass Vault - authentication method - type: string - required: - - path - - username - type: object - type: object - caBundle: - description: |- - PEM encoded CA bundle used to validate Vault server certificate. Only used - if the Server URL is using HTTPS protocol. This parameter is ignored for - plain HTTP protocol connection. If not set the system root certificates - are used to validate the TLS connection. - format: byte - type: string - caProvider: - description: The provider for the CA bundle to use to validate Vault server certificate. - properties: - key: - description: The key where the CA certificate can be found in the Secret or ConfigMap. - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: |- - The namespace the Provider type is in. - Can only be defined when used in a ClusterSecretStore. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - forwardInconsistent: - description: |- - ForwardInconsistent tells Vault to forward read-after-write requests to the Vault - leader instead of simply retrying within a loop. This can increase performance if - the option is enabled serverside. - https://www.vaultproject.io/docs/configuration/replication#allow_forwarding_via_header - type: boolean - headers: - additionalProperties: - type: string - description: Headers to be added in Vault request - type: object - namespace: - description: |- - Name of the vault namespace. Namespaces is a set of features within Vault Enterprise that allows - Vault environments to support Secure Multi-tenancy. e.g: "ns1". - More about namespaces can be found here https://www.vaultproject.io/docs/enterprise/namespaces - type: string - path: - description: |- - Path is the mount path of the Vault KV backend endpoint, e.g: - "secret". The v2 KV secret engine version specific "/data" path suffix - for fetching secrets from Vault is optional and will be appended - if not present in specified path. - type: string - readYourWrites: - description: |- - ReadYourWrites ensures isolated read-after-write semantics by - providing discovered cluster replication states in each request. - More information about eventual consistency in Vault can be found here - https://www.vaultproject.io/docs/enterprise/consistency - type: boolean - server: - description: 'Server is the connection address for the Vault server, e.g: "https://vault.example.com:8200".' - type: string - tls: - description: |- - The configuration used for client side related TLS communication, when the Vault server - requires mutual authentication. Only used if the Server URL is using HTTPS protocol. - This parameter is ignored for plain HTTP protocol connection. - It's worth noting this configuration is different from the "TLS certificates auth method", - which is available under the `auth.cert` section. - properties: - certSecretRef: - description: |- - CertSecretRef is a certificate added to the transport layer - when communicating with the Vault server. - If no key for the Secret is specified, external-secret will default to 'tls.crt'. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - keySecretRef: - description: |- - KeySecretRef to a key in a Secret resource containing client private key - added to the transport layer when communicating with the Vault server. - If no key for the Secret is specified, external-secret will default to 'tls.key'. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - version: - default: v2 - description: |- - Version is the Vault KV secret engine version. This can be either "v1" or - "v2". Version defaults to "v2". - enum: - - v1 - - v2 - type: string - required: - - auth - - server - type: object - webhook: - description: Webhook configures this store to sync secrets using a generic templated webhook - properties: - body: - description: Body - type: string - caBundle: - description: |- - PEM encoded CA bundle used to validate webhook server certificate. Only used - if the Server URL is using HTTPS protocol. This parameter is ignored for - plain HTTP protocol connection. If not set the system root certificates - are used to validate the TLS connection. - format: byte - type: string - caProvider: - description: The provider for the CA bundle to use to validate webhook server certificate. - properties: - key: - description: The key the value inside of the provider type to use, only used with "Secret" type - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: The namespace the Provider type is in. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - headers: - additionalProperties: - type: string - description: Headers - type: object - method: - description: Webhook Method - type: string - result: - description: Result formatting - properties: - jsonPath: - description: Json path of return value - type: string - type: object - secrets: - description: |- - Secrets to fill in templates - These secrets will be passed to the templating function as key value pairs under the given name - items: - properties: - name: - description: Name of this secret in templates - type: string - secretRef: - description: Secret ref to fill in credentials - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - name - - secretRef - type: object - type: array - timeout: - description: Timeout - type: string - url: - description: Webhook url to call - type: string - required: - - result - - url - type: object - yandexcertificatemanager: - description: YandexCertificateManager configures this store to sync secrets using Yandex Certificate Manager provider - properties: - apiEndpoint: - description: Yandex.Cloud API endpoint (e.g. 'api.cloud.yandex.net:443') - type: string - auth: - description: Auth defines the information necessary to authenticate against Yandex Certificate Manager - properties: - authorizedKeySecretRef: - description: The authorized key used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - caProvider: - description: The provider for the CA bundle to use to validate Yandex.Cloud server certificate. - properties: - certSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - auth - type: object - yandexlockbox: - description: YandexLockbox configures this store to sync secrets using Yandex Lockbox provider - properties: - apiEndpoint: - description: Yandex.Cloud API endpoint (e.g. 'api.cloud.yandex.net:443') - type: string - auth: - description: Auth defines the information necessary to authenticate against Yandex Lockbox - properties: - authorizedKeySecretRef: - description: The authorized key used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - caProvider: - description: The provider for the CA bundle to use to validate Yandex.Cloud server certificate. - properties: - certSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - auth - type: object - type: object - refreshInterval: - description: Used to configure store refresh interval in seconds. Empty or 0 will default to the controller config. - type: integer - retrySettings: - description: Used to configure http retries if failed - properties: - maxRetries: - format: int32 - type: integer - retryInterval: - type: string - type: object - required: - - provider - type: object - status: - description: SecretStoreStatus defines the observed state of the SecretStore. - properties: - capabilities: - description: SecretStoreCapabilities defines the possible operations a SecretStore can do. - type: string - conditions: - items: - properties: - lastTransitionTime: - format: date-time - type: string - message: - type: string - reason: - type: string - status: - type: string - type: - type: string - required: - - status - - type - type: object - type: array - type: object - type: object - served: true - storage: true - subresources: - status: {} -{{- if .Values.crds.conversion.enabled }} - conversion: - strategy: Webhook - webhook: - conversionReviewVersions: - - v1 - clientConfig: - service: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ .Release.Namespace | quote }} - path: /convert -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/crds/ecrauthorizationtoken.yaml b/charts/external-secrets/external-secrets/templates/crds/ecrauthorizationtoken.yaml deleted file mode 100644 index f015e11..0000000 --- a/charts/external-secrets/external-secrets/templates/crds/ecrauthorizationtoken.yaml +++ /dev/null @@ -1,177 +0,0 @@ -{{- if .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - {{- with .Values.crds.annotations }} - {{- toYaml . | nindent 4}} - {{- end }} - {{- if and .Values.crds.conversion.enabled .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} - controller-gen.kubebuilder.io/version: v0.16.1 - labels: - external-secrets.io/component: controller - name: ecrauthorizationtokens.generators.external-secrets.io -spec: - group: generators.external-secrets.io - names: - categories: - - ecrauthorizationtoken - kind: ECRAuthorizationToken - listKind: ECRAuthorizationTokenList - plural: ecrauthorizationtokens - shortNames: - - ecrauthorizationtoken - singular: ecrauthorizationtoken - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - ECRAuthorizationTokenSpec uses the GetAuthorizationToken API to retrieve an - authorization token. - The authorization token is valid for 12 hours. - The authorizationToken returned is a base64 encoded string that can be decoded - and used in a docker login command to authenticate to a registry. - For more information, see Registry authentication (https://docs.aws.amazon.com/AmazonECR/latest/userguide/Registries.html#registry_auth) in the Amazon Elastic Container Registry User Guide. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - properties: - auth: - description: Auth defines how to authenticate with AWS - properties: - jwt: - description: Authenticate against AWS using service account tokens. - properties: - serviceAccountRef: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - type: object - secretRef: - description: |- - AWSAuthSecretRef holds secret references for AWS credentials - both AccessKeyID and SecretAccessKey must be defined in order to properly authenticate. - properties: - accessKeyIDSecretRef: - description: The AccessKeyID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretAccessKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - sessionTokenSecretRef: - description: |- - The SessionToken used for authentication - This must be defined if AccessKeyID and SecretAccessKey are temporary credentials - see: https://docs.aws.amazon.com/IAM/latest/UserGuide/id_credentials_temp_use-resources.html - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - region: - description: Region specifies the region to operate in. - type: string - role: - description: |- - You can assume a role before making calls to the - desired AWS service. - type: string - required: - - region - type: object - type: object - served: true - storage: true - subresources: - status: {} -{{- if .Values.crds.conversion.enabled }} - conversion: - strategy: Webhook - webhook: - conversionReviewVersions: - - v1 - clientConfig: - service: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ .Release.Namespace | quote }} - path: /convert -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/crds/externalsecret.yaml b/charts/external-secrets/external-secrets/templates/crds/externalsecret.yaml deleted file mode 100644 index 817690b..0000000 --- a/charts/external-secrets/external-secrets/templates/crds/externalsecret.yaml +++ /dev/null @@ -1,820 +0,0 @@ -{{- if .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - {{- with .Values.crds.annotations }} - {{- toYaml . | nindent 4}} - {{- end }} - {{- if and .Values.crds.conversion.enabled .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} - controller-gen.kubebuilder.io/version: v0.16.1 - labels: - external-secrets.io/component: controller - name: externalsecrets.external-secrets.io -spec: - group: external-secrets.io - names: - categories: - - externalsecrets - kind: ExternalSecret - listKind: ExternalSecretList - plural: externalsecrets - shortNames: - - es - singular: externalsecret - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.secretStoreRef.name - name: Store - type: string - - jsonPath: .spec.refreshInterval - name: Refresh Interval - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].reason - name: Status - type: string - deprecated: true - name: v1alpha1 - schema: - openAPIV3Schema: - description: ExternalSecret is the Schema for the external-secrets API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ExternalSecretSpec defines the desired state of ExternalSecret. - properties: - data: - description: Data defines the connection between the Kubernetes Secret keys and the Provider data - items: - description: ExternalSecretData defines the connection between the Kubernetes Secret key (spec.data.) and the Provider data. - properties: - remoteRef: - description: ExternalSecretDataRemoteRef defines Provider data location. - properties: - conversionStrategy: - default: Default - description: Used to define a conversion Strategy - enum: - - Default - - Unicode - type: string - key: - description: Key is the key used in the Provider, mandatory - type: string - property: - description: Used to select a specific property of the Provider value (if a map), if supported - type: string - version: - description: Used to select a specific version of the Provider value, if supported - type: string - required: - - key - type: object - secretKey: - type: string - required: - - remoteRef - - secretKey - type: object - type: array - dataFrom: - description: |- - DataFrom is used to fetch all properties from a specific Provider data - If multiple entries are specified, the Secret keys are merged in the specified order - items: - description: ExternalSecretDataRemoteRef defines Provider data location. - properties: - conversionStrategy: - default: Default - description: Used to define a conversion Strategy - enum: - - Default - - Unicode - type: string - key: - description: Key is the key used in the Provider, mandatory - type: string - property: - description: Used to select a specific property of the Provider value (if a map), if supported - type: string - version: - description: Used to select a specific version of the Provider value, if supported - type: string - required: - - key - type: object - type: array - refreshInterval: - default: 1h - description: |- - RefreshInterval is the amount of time before the values are read again from the SecretStore provider - Valid time units are "ns", "us" (or "µs"), "ms", "s", "m", "h" - May be set to zero to fetch and create it once. Defaults to 1h. - type: string - secretStoreRef: - description: SecretStoreRef defines which SecretStore to fetch the ExternalSecret data. - properties: - kind: - description: |- - Kind of the SecretStore resource (SecretStore or ClusterSecretStore) - Defaults to `SecretStore` - type: string - name: - description: Name of the SecretStore resource - type: string - required: - - name - type: object - target: - description: |- - ExternalSecretTarget defines the Kubernetes Secret to be created - There can be only one target per ExternalSecret. - properties: - creationPolicy: - default: Owner - description: |- - CreationPolicy defines rules on how to create the resulting Secret - Defaults to 'Owner' - enum: - - Owner - - Merge - - None - type: string - immutable: - description: Immutable defines if the final secret will be immutable - type: boolean - name: - description: |- - Name defines the name of the Secret resource to be managed - This field is immutable - Defaults to the .metadata.name of the ExternalSecret resource - type: string - template: - description: Template defines a blueprint for the created Secret resource. - properties: - data: - additionalProperties: - type: string - type: object - engineVersion: - default: v1 - description: |- - EngineVersion specifies the template engine version - that should be used to compile/execute the - template specified in .data and .templateFrom[]. - enum: - - v1 - - v2 - type: string - metadata: - description: ExternalSecretTemplateMetadata defines metadata fields for the Secret blueprint. - properties: - annotations: - additionalProperties: - type: string - type: object - labels: - additionalProperties: - type: string - type: object - type: object - templateFrom: - items: - maxProperties: 1 - minProperties: 1 - properties: - configMap: - properties: - items: - items: - properties: - key: - type: string - required: - - key - type: object - type: array - name: - type: string - required: - - items - - name - type: object - secret: - properties: - items: - items: - properties: - key: - type: string - required: - - key - type: object - type: array - name: - type: string - required: - - items - - name - type: object - type: object - type: array - type: - type: string - type: object - type: object - required: - - secretStoreRef - - target - type: object - status: - properties: - binding: - description: Binding represents a servicebinding.io Provisioned Service reference to the secret - properties: - name: - default: "" - description: |- - Name of the referent. - This field is effectively required, but due to backwards compatibility is - allowed to be empty. Instances of this type with an empty value here are - almost certainly wrong. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - type: object - x-kubernetes-map-type: atomic - conditions: - items: - properties: - lastTransitionTime: - format: date-time - type: string - message: - type: string - reason: - type: string - status: - type: string - type: - type: string - required: - - status - - type - type: object - type: array - refreshTime: - description: |- - refreshTime is the time and date the external secret was fetched and - the target secret updated - format: date-time - nullable: true - type: string - syncedResourceVersion: - description: SyncedResourceVersion keeps track of the last synced version - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.secretStoreRef.name - name: Store - type: string - - jsonPath: .spec.refreshInterval - name: Refresh Interval - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].reason - name: Status - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - name: v1beta1 - schema: - openAPIV3Schema: - description: ExternalSecret is the Schema for the external-secrets API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ExternalSecretSpec defines the desired state of ExternalSecret. - properties: - data: - description: Data defines the connection between the Kubernetes Secret keys and the Provider data - items: - description: ExternalSecretData defines the connection between the Kubernetes Secret key (spec.data.) and the Provider data. - properties: - remoteRef: - description: |- - RemoteRef points to the remote secret and defines - which secret (version/property/..) to fetch. - properties: - conversionStrategy: - default: Default - description: Used to define a conversion Strategy - enum: - - Default - - Unicode - type: string - decodingStrategy: - default: None - description: Used to define a decoding Strategy - enum: - - Auto - - Base64 - - Base64URL - - None - type: string - key: - description: Key is the key used in the Provider, mandatory - type: string - metadataPolicy: - default: None - description: Policy for fetching tags/labels from provider secrets, possible options are Fetch, None. Defaults to None - enum: - - None - - Fetch - type: string - property: - description: Used to select a specific property of the Provider value (if a map), if supported - type: string - version: - description: Used to select a specific version of the Provider value, if supported - type: string - required: - - key - type: object - secretKey: - description: |- - SecretKey defines the key in which the controller stores - the value. This is the key in the Kind=Secret - type: string - sourceRef: - description: |- - SourceRef allows you to override the source - from which the value will pulled from. - maxProperties: 1 - properties: - generatorRef: - description: |- - GeneratorRef points to a generator custom resource. - - Deprecated: The generatorRef is not implemented in .data[]. - this will be removed with v1. - properties: - apiVersion: - default: generators.external-secrets.io/v1alpha1 - description: Specify the apiVersion of the generator resource - type: string - kind: - description: Specify the Kind of the resource, e.g. Password, ACRAccessToken etc. - type: string - name: - description: Specify the name of the generator resource - type: string - required: - - kind - - name - type: object - storeRef: - description: SecretStoreRef defines which SecretStore to fetch the ExternalSecret data. - properties: - kind: - description: |- - Kind of the SecretStore resource (SecretStore or ClusterSecretStore) - Defaults to `SecretStore` - type: string - name: - description: Name of the SecretStore resource - type: string - required: - - name - type: object - type: object - required: - - remoteRef - - secretKey - type: object - type: array - dataFrom: - description: |- - DataFrom is used to fetch all properties from a specific Provider data - If multiple entries are specified, the Secret keys are merged in the specified order - items: - properties: - extract: - description: |- - Used to extract multiple key/value pairs from one secret - Note: Extract does not support sourceRef.Generator or sourceRef.GeneratorRef. - properties: - conversionStrategy: - default: Default - description: Used to define a conversion Strategy - enum: - - Default - - Unicode - type: string - decodingStrategy: - default: None - description: Used to define a decoding Strategy - enum: - - Auto - - Base64 - - Base64URL - - None - type: string - key: - description: Key is the key used in the Provider, mandatory - type: string - metadataPolicy: - default: None - description: Policy for fetching tags/labels from provider secrets, possible options are Fetch, None. Defaults to None - enum: - - None - - Fetch - type: string - property: - description: Used to select a specific property of the Provider value (if a map), if supported - type: string - version: - description: Used to select a specific version of the Provider value, if supported - type: string - required: - - key - type: object - find: - description: |- - Used to find secrets based on tags or regular expressions - Note: Find does not support sourceRef.Generator or sourceRef.GeneratorRef. - properties: - conversionStrategy: - default: Default - description: Used to define a conversion Strategy - enum: - - Default - - Unicode - type: string - decodingStrategy: - default: None - description: Used to define a decoding Strategy - enum: - - Auto - - Base64 - - Base64URL - - None - type: string - name: - description: Finds secrets based on the name. - properties: - regexp: - description: Finds secrets base - type: string - type: object - path: - description: A root path to start the find operations. - type: string - tags: - additionalProperties: - type: string - description: Find secrets based on tags. - type: object - type: object - rewrite: - description: |- - Used to rewrite secret Keys after getting them from the secret Provider - Multiple Rewrite operations can be provided. They are applied in a layered order (first to last) - items: - properties: - regexp: - description: |- - Used to rewrite with regular expressions. - The resulting key will be the output of a regexp.ReplaceAll operation. - properties: - source: - description: Used to define the regular expression of a re.Compiler. - type: string - target: - description: Used to define the target pattern of a ReplaceAll operation. - type: string - required: - - source - - target - type: object - transform: - description: |- - Used to apply string transformation on the secrets. - The resulting key will be the output of the template applied by the operation. - properties: - template: - description: |- - Used to define the template to apply on the secret name. - `.value ` will specify the secret name in the template. - type: string - required: - - template - type: object - type: object - type: array - sourceRef: - description: |- - SourceRef points to a store or generator - which contains secret values ready to use. - Use this in combination with Extract or Find pull values out of - a specific SecretStore. - When sourceRef points to a generator Extract or Find is not supported. - The generator returns a static map of values - maxProperties: 1 - properties: - generatorRef: - description: GeneratorRef points to a generator custom resource. - properties: - apiVersion: - default: generators.external-secrets.io/v1alpha1 - description: Specify the apiVersion of the generator resource - type: string - kind: - description: Specify the Kind of the resource, e.g. Password, ACRAccessToken etc. - type: string - name: - description: Specify the name of the generator resource - type: string - required: - - kind - - name - type: object - storeRef: - description: SecretStoreRef defines which SecretStore to fetch the ExternalSecret data. - properties: - kind: - description: |- - Kind of the SecretStore resource (SecretStore or ClusterSecretStore) - Defaults to `SecretStore` - type: string - name: - description: Name of the SecretStore resource - type: string - required: - - name - type: object - type: object - type: object - type: array - refreshInterval: - default: 1h - description: |- - RefreshInterval is the amount of time before the values are read again from the SecretStore provider - Valid time units are "ns", "us" (or "µs"), "ms", "s", "m", "h" - May be set to zero to fetch and create it once. Defaults to 1h. - type: string - secretStoreRef: - description: SecretStoreRef defines which SecretStore to fetch the ExternalSecret data. - properties: - kind: - description: |- - Kind of the SecretStore resource (SecretStore or ClusterSecretStore) - Defaults to `SecretStore` - type: string - name: - description: Name of the SecretStore resource - type: string - required: - - name - type: object - target: - default: - creationPolicy: Owner - deletionPolicy: Retain - description: |- - ExternalSecretTarget defines the Kubernetes Secret to be created - There can be only one target per ExternalSecret. - properties: - creationPolicy: - default: Owner - description: |- - CreationPolicy defines rules on how to create the resulting Secret - Defaults to 'Owner' - enum: - - Owner - - Orphan - - Merge - - None - type: string - deletionPolicy: - default: Retain - description: |- - DeletionPolicy defines rules on how to delete the resulting Secret - Defaults to 'Retain' - enum: - - Delete - - Merge - - Retain - type: string - immutable: - description: Immutable defines if the final secret will be immutable - type: boolean - name: - description: |- - Name defines the name of the Secret resource to be managed - This field is immutable - Defaults to the .metadata.name of the ExternalSecret resource - type: string - template: - description: Template defines a blueprint for the created Secret resource. - properties: - data: - additionalProperties: - type: string - type: object - engineVersion: - default: v2 - description: |- - EngineVersion specifies the template engine version - that should be used to compile/execute the - template specified in .data and .templateFrom[]. - enum: - - v1 - - v2 - type: string - mergePolicy: - default: Replace - enum: - - Replace - - Merge - type: string - metadata: - description: ExternalSecretTemplateMetadata defines metadata fields for the Secret blueprint. - properties: - annotations: - additionalProperties: - type: string - type: object - labels: - additionalProperties: - type: string - type: object - type: object - templateFrom: - items: - properties: - configMap: - properties: - items: - items: - properties: - key: - type: string - templateAs: - default: Values - enum: - - Values - - KeysAndValues - type: string - required: - - key - type: object - type: array - name: - type: string - required: - - items - - name - type: object - literal: - type: string - secret: - properties: - items: - items: - properties: - key: - type: string - templateAs: - default: Values - enum: - - Values - - KeysAndValues - type: string - required: - - key - type: object - type: array - name: - type: string - required: - - items - - name - type: object - target: - default: Data - enum: - - Data - - Annotations - - Labels - type: string - type: object - type: array - type: - type: string - type: object - type: object - type: object - status: - properties: - binding: - description: Binding represents a servicebinding.io Provisioned Service reference to the secret - properties: - name: - default: "" - description: |- - Name of the referent. - This field is effectively required, but due to backwards compatibility is - allowed to be empty. Instances of this type with an empty value here are - almost certainly wrong. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - type: object - x-kubernetes-map-type: atomic - conditions: - items: - properties: - lastTransitionTime: - format: date-time - type: string - message: - type: string - reason: - type: string - status: - type: string - type: - type: string - required: - - status - - type - type: object - type: array - refreshTime: - description: |- - refreshTime is the time and date the external secret was fetched and - the target secret updated - format: date-time - nullable: true - type: string - syncedResourceVersion: - description: SyncedResourceVersion keeps track of the last synced version - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} -{{- if .Values.crds.conversion.enabled }} - conversion: - strategy: Webhook - webhook: - conversionReviewVersions: - - v1 - clientConfig: - service: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ .Release.Namespace | quote }} - path: /convert -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/crds/fake.yaml b/charts/external-secrets/external-secrets/templates/crds/fake.yaml deleted file mode 100644 index df66947..0000000 --- a/charts/external-secrets/external-secrets/templates/crds/fake.yaml +++ /dev/null @@ -1,86 +0,0 @@ -{{- if .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - {{- with .Values.crds.annotations }} - {{- toYaml . | nindent 4}} - {{- end }} - {{- if and .Values.crds.conversion.enabled .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} - controller-gen.kubebuilder.io/version: v0.16.1 - labels: - external-secrets.io/component: controller - name: fakes.generators.external-secrets.io -spec: - group: generators.external-secrets.io - names: - categories: - - fake - kind: Fake - listKind: FakeList - plural: fakes - shortNames: - - fake - singular: fake - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - Fake generator is used for testing. It lets you define - a static set of credentials that is always returned. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: FakeSpec contains the static data. - properties: - controller: - description: |- - Used to select the correct ESO controller (think: ingress.ingressClassName) - The ESO controller is instantiated with a specific controller name and filters VDS based on this property - type: string - data: - additionalProperties: - type: string - description: |- - Data defines the static data returned - by this generator. - type: object - type: object - type: object - served: true - storage: true - subresources: - status: {} -{{- if .Values.crds.conversion.enabled }} - conversion: - strategy: Webhook - webhook: - conversionReviewVersions: - - v1 - clientConfig: - service: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ .Release.Namespace | quote }} - path: /convert -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/crds/gcraccesstoken.yaml b/charts/external-secrets/external-secrets/templates/crds/gcraccesstoken.yaml deleted file mode 100644 index 5be5f4e..0000000 --- a/charts/external-secrets/external-secrets/templates/crds/gcraccesstoken.yaml +++ /dev/null @@ -1,138 +0,0 @@ -{{- if .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - {{- with .Values.crds.annotations }} - {{- toYaml . | nindent 4}} - {{- end }} - {{- if and .Values.crds.conversion.enabled .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} - controller-gen.kubebuilder.io/version: v0.16.1 - labels: - external-secrets.io/component: controller - name: gcraccesstokens.generators.external-secrets.io -spec: - group: generators.external-secrets.io - names: - categories: - - gcraccesstoken - kind: GCRAccessToken - listKind: GCRAccessTokenList - plural: gcraccesstokens - shortNames: - - gcraccesstoken - singular: gcraccesstoken - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - GCRAccessToken generates an GCP access token - that can be used to authenticate with GCR. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - properties: - auth: - description: Auth defines the means for authenticating with GCP - properties: - secretRef: - properties: - secretAccessKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - workloadIdentity: - properties: - clusterLocation: - type: string - clusterName: - type: string - clusterProjectID: - type: string - serviceAccountRef: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - clusterLocation - - clusterName - - serviceAccountRef - type: object - type: object - projectID: - description: ProjectID defines which project to use to authenticate with - type: string - required: - - auth - - projectID - type: object - type: object - served: true - storage: true - subresources: - status: {} -{{- if .Values.crds.conversion.enabled }} - conversion: - strategy: Webhook - webhook: - conversionReviewVersions: - - v1 - clientConfig: - service: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ .Release.Namespace | quote }} - path: /convert -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/crds/githubaccesstoken.yaml b/charts/external-secrets/external-secrets/templates/crds/githubaccesstoken.yaml deleted file mode 100644 index 87fde68..0000000 --- a/charts/external-secrets/external-secrets/templates/crds/githubaccesstoken.yaml +++ /dev/null @@ -1,112 +0,0 @@ -{{- if .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - {{- with .Values.crds.annotations }} - {{- toYaml . | nindent 4}} - {{- end }} - {{- if and .Values.crds.conversion.enabled .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} - controller-gen.kubebuilder.io/version: v0.16.1 - labels: - external-secrets.io/component: controller - name: githubaccesstokens.generators.external-secrets.io -spec: - group: generators.external-secrets.io - names: - categories: - - githubaccesstoken - kind: GithubAccessToken - listKind: GithubAccessTokenList - plural: githubaccesstokens - shortNames: - - githubaccesstoken - singular: githubaccesstoken - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: GithubAccessToken generates ghs_ accessToken - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - properties: - appID: - type: string - auth: - description: Auth configures how ESO authenticates with a Github instance. - properties: - privateKey: - properties: - secretRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - secretRef - type: object - required: - - privateKey - type: object - installID: - type: string - url: - description: URL configures the Github instance URL. Defaults to https://github.com/. - type: string - required: - - appID - - auth - - installID - type: object - type: object - served: true - storage: true - subresources: - status: {} -{{- if .Values.crds.conversion.enabled }} - conversion: - strategy: Webhook - webhook: - conversionReviewVersions: - - v1 - clientConfig: - service: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ .Release.Namespace | quote }} - path: /convert -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/crds/password.yaml b/charts/external-secrets/external-secrets/templates/crds/password.yaml deleted file mode 100644 index 4042bcc..0000000 --- a/charts/external-secrets/external-secrets/templates/crds/password.yaml +++ /dev/null @@ -1,108 +0,0 @@ -{{- if .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - {{- with .Values.crds.annotations }} - {{- toYaml . | nindent 4}} - {{- end }} - {{- if and .Values.crds.conversion.enabled .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} - controller-gen.kubebuilder.io/version: v0.16.1 - labels: - external-secrets.io/component: controller - name: passwords.generators.external-secrets.io -spec: - group: generators.external-secrets.io - names: - categories: - - password - kind: Password - listKind: PasswordList - plural: passwords - shortNames: - - password - singular: password - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - Password generates a random password based on the - configuration parameters in spec. - You can specify the length, characterset and other attributes. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: PasswordSpec controls the behavior of the password generator. - properties: - allowRepeat: - default: false - description: set AllowRepeat to true to allow repeating characters. - type: boolean - digits: - description: |- - Digits specifies the number of digits in the generated - password. If omitted it defaults to 25% of the length of the password - type: integer - length: - default: 24 - description: |- - Length of the password to be generated. - Defaults to 24 - type: integer - noUpper: - default: false - description: Set NoUpper to disable uppercase characters - type: boolean - symbolCharacters: - description: |- - SymbolCharacters specifies the special characters that should be used - in the generated password. - type: string - symbols: - description: |- - Symbols specifies the number of symbol characters in the generated - password. If omitted it defaults to 25% of the length of the password - type: integer - required: - - allowRepeat - - length - - noUpper - type: object - type: object - served: true - storage: true - subresources: - status: {} -{{- if .Values.crds.conversion.enabled }} - conversion: - strategy: Webhook - webhook: - conversionReviewVersions: - - v1 - clientConfig: - service: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ .Release.Namespace | quote }} - path: /convert -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/crds/pushsecret.yaml b/charts/external-secrets/external-secrets/templates/crds/pushsecret.yaml deleted file mode 100644 index 7de0c01..0000000 --- a/charts/external-secrets/external-secrets/templates/crds/pushsecret.yaml +++ /dev/null @@ -1,386 +0,0 @@ -{{- if and (.Values.installCRDs) (.Values.crds.createPushSecret) }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - {{- with .Values.crds.annotations }} - {{- toYaml . | nindent 4}} - {{- end }} - {{- if and .Values.crds.conversion.enabled .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} - controller-gen.kubebuilder.io/version: v0.16.1 - name: pushsecrets.external-secrets.io -spec: - group: external-secrets.io - names: - categories: - - pushsecrets - kind: PushSecret - listKind: PushSecretList - plural: pushsecrets - singular: pushsecret - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: AGE - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].reason - name: Status - type: string - name: v1alpha1 - schema: - openAPIV3Schema: - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: PushSecretSpec configures the behavior of the PushSecret. - properties: - data: - description: Secret Data that should be pushed to providers - items: - properties: - conversionStrategy: - default: None - description: Used to define a conversion Strategy for the secret keys - enum: - - None - - ReverseUnicode - type: string - match: - description: Match a given Secret Key to be pushed to the provider. - properties: - remoteRef: - description: Remote Refs to push to providers. - properties: - property: - description: Name of the property in the resulting secret - type: string - remoteKey: - description: Name of the resulting provider secret. - type: string - required: - - remoteKey - type: object - secretKey: - description: Secret Key to be pushed - type: string - required: - - remoteRef - type: object - metadata: - description: |- - Metadata is metadata attached to the secret. - The structure of metadata is provider specific, please look it up in the provider documentation. - x-kubernetes-preserve-unknown-fields: true - required: - - match - type: object - type: array - deletionPolicy: - default: None - description: 'Deletion Policy to handle Secrets in the provider. Possible Values: "Delete/None". Defaults to "None".' - enum: - - Delete - - None - type: string - refreshInterval: - description: The Interval to which External Secrets will try to push a secret definition - type: string - secretStoreRefs: - items: - properties: - kind: - default: SecretStore - description: |- - Kind of the SecretStore resource (SecretStore or ClusterSecretStore) - Defaults to `SecretStore` - type: string - labelSelector: - description: Optionally, sync to secret stores with label selector - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - x-kubernetes-list-type: atomic - required: - - key - - operator - type: object - type: array - x-kubernetes-list-type: atomic - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - name: - description: Optionally, sync to the SecretStore of the given name - type: string - type: object - type: array - selector: - description: The Secret Selector (k8s source) for the Push Secret - properties: - secret: - description: Select a Secret to Push. - properties: - name: - description: Name of the Secret. The Secret must exist in the same namespace as the PushSecret manifest. - type: string - required: - - name - type: object - required: - - secret - type: object - template: - description: Template defines a blueprint for the created Secret resource. - properties: - data: - additionalProperties: - type: string - type: object - engineVersion: - default: v2 - description: |- - EngineVersion specifies the template engine version - that should be used to compile/execute the - template specified in .data and .templateFrom[]. - enum: - - v1 - - v2 - type: string - mergePolicy: - default: Replace - enum: - - Replace - - Merge - type: string - metadata: - description: ExternalSecretTemplateMetadata defines metadata fields for the Secret blueprint. - properties: - annotations: - additionalProperties: - type: string - type: object - labels: - additionalProperties: - type: string - type: object - type: object - templateFrom: - items: - properties: - configMap: - properties: - items: - items: - properties: - key: - type: string - templateAs: - default: Values - enum: - - Values - - KeysAndValues - type: string - required: - - key - type: object - type: array - name: - type: string - required: - - items - - name - type: object - literal: - type: string - secret: - properties: - items: - items: - properties: - key: - type: string - templateAs: - default: Values - enum: - - Values - - KeysAndValues - type: string - required: - - key - type: object - type: array - name: - type: string - required: - - items - - name - type: object - target: - default: Data - enum: - - Data - - Annotations - - Labels - type: string - type: object - type: array - type: - type: string - type: object - updatePolicy: - default: Replace - description: 'UpdatePolicy to handle Secrets in the provider. Possible Values: "Replace/IfNotExists". Defaults to "Replace".' - enum: - - Replace - - IfNotExists - type: string - required: - - secretStoreRefs - - selector - type: object - status: - description: PushSecretStatus indicates the history of the status of PushSecret. - properties: - conditions: - items: - description: PushSecretStatusCondition indicates the status of the PushSecret. - properties: - lastTransitionTime: - format: date-time - type: string - message: - type: string - reason: - type: string - status: - type: string - type: - description: PushSecretConditionType indicates the condition of the PushSecret. - type: string - required: - - status - - type - type: object - type: array - refreshTime: - description: |- - refreshTime is the time and date the external secret was fetched and - the target secret updated - format: date-time - nullable: true - type: string - syncedPushSecrets: - additionalProperties: - additionalProperties: - properties: - conversionStrategy: - default: None - description: Used to define a conversion Strategy for the secret keys - enum: - - None - - ReverseUnicode - type: string - match: - description: Match a given Secret Key to be pushed to the provider. - properties: - remoteRef: - description: Remote Refs to push to providers. - properties: - property: - description: Name of the property in the resulting secret - type: string - remoteKey: - description: Name of the resulting provider secret. - type: string - required: - - remoteKey - type: object - secretKey: - description: Secret Key to be pushed - type: string - required: - - remoteRef - type: object - metadata: - description: |- - Metadata is metadata attached to the secret. - The structure of metadata is provider specific, please look it up in the provider documentation. - x-kubernetes-preserve-unknown-fields: true - required: - - match - type: object - type: object - description: |- - Synced PushSecrets, including secrets that already exist in provider. - Matches secret stores to PushSecretData that was stored to that secret store. - type: object - syncedResourceVersion: - description: SyncedResourceVersion keeps track of the last synced version. - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} -{{- if .Values.crds.conversion.enabled }} - conversion: - strategy: Webhook - webhook: - conversionReviewVersions: - - v1 - clientConfig: - service: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ .Release.Namespace | quote }} - path: /convert -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/crds/secretstore.yaml b/charts/external-secrets/external-secrets/templates/crds/secretstore.yaml deleted file mode 100644 index 6b2fe85..0000000 --- a/charts/external-secrets/external-secrets/templates/crds/secretstore.yaml +++ /dev/null @@ -1,4601 +0,0 @@ -{{- if .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - {{- with .Values.crds.annotations }} - {{- toYaml . | nindent 4}} - {{- end }} - {{- if and .Values.crds.conversion.enabled .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} - controller-gen.kubebuilder.io/version: v0.16.1 - labels: - external-secrets.io/component: controller - name: secretstores.external-secrets.io -spec: - group: external-secrets.io - names: - categories: - - externalsecrets - kind: SecretStore - listKind: SecretStoreList - plural: secretstores - shortNames: - - ss - singular: secretstore - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: AGE - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].reason - name: Status - type: string - deprecated: true - name: v1alpha1 - schema: - openAPIV3Schema: - description: SecretStore represents a secure external location for storing secrets, which can be referenced as part of `storeRef` fields. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: SecretStoreSpec defines the desired state of SecretStore. - properties: - controller: - description: |- - Used to select the correct ESO controller (think: ingress.ingressClassName) - The ESO controller is instantiated with a specific controller name and filters ES based on this property - type: string - provider: - description: Used to configure the provider. Only one provider may be set - maxProperties: 1 - minProperties: 1 - properties: - akeyless: - description: Akeyless configures this store to sync secrets using Akeyless Vault provider - properties: - akeylessGWApiURL: - description: Akeyless GW API Url from which the secrets to be fetched from. - type: string - authSecretRef: - description: Auth configures how the operator authenticates with Akeyless. - properties: - kubernetesAuth: - description: |- - Kubernetes authenticates with Akeyless by passing the ServiceAccount - token stored in the named Secret resource. - properties: - accessID: - description: the Akeyless Kubernetes auth-method access-id - type: string - k8sConfName: - description: Kubernetes-auth configuration name in Akeyless-Gateway - type: string - secretRef: - description: |- - Optional secret field containing a Kubernetes ServiceAccount JWT used - for authenticating with Akeyless. If a name is specified without a key, - `token` is the default. If one is not specified, the one bound to - the controller will be used. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - serviceAccountRef: - description: |- - Optional service account field containing the name of a kubernetes ServiceAccount. - If the service account is specified, the service account secret token JWT will be used - for authenticating with Akeyless. If the service account selector is not supplied, - the secretRef will be used instead. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - accessID - - k8sConfName - type: object - secretRef: - description: |- - Reference to a Secret that contains the details - to authenticate with Akeyless. - properties: - accessID: - description: The SecretAccessID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - accessType: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - accessTypeParam: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - caBundle: - description: |- - PEM/base64 encoded CA bundle used to validate Akeyless Gateway certificate. Only used - if the AkeylessGWApiURL URL is using HTTPS protocol. If not set the system root certificates - are used to validate the TLS connection. - format: byte - type: string - caProvider: - description: The provider for the CA bundle to use to validate Akeyless Gateway certificate. - properties: - key: - description: The key the value inside of the provider type to use, only used with "Secret" type - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: The namespace the Provider type is in. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - required: - - akeylessGWApiURL - - authSecretRef - type: object - alibaba: - description: Alibaba configures this store to sync secrets using Alibaba Cloud provider - properties: - auth: - description: AlibabaAuth contains a secretRef for credentials. - properties: - rrsa: - description: Authenticate against Alibaba using RRSA. - properties: - oidcProviderArn: - type: string - oidcTokenFilePath: - type: string - roleArn: - type: string - sessionName: - type: string - required: - - oidcProviderArn - - oidcTokenFilePath - - roleArn - - sessionName - type: object - secretRef: - description: AlibabaAuthSecretRef holds secret references for Alibaba credentials. - properties: - accessKeyIDSecretRef: - description: The AccessKeyID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - accessKeySecretSecretRef: - description: The AccessKeySecret is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - accessKeyIDSecretRef - - accessKeySecretSecretRef - type: object - type: object - regionID: - description: Alibaba Region to be used for the provider - type: string - required: - - auth - - regionID - type: object - aws: - description: AWS configures this store to sync secrets using AWS Secret Manager provider - properties: - auth: - description: |- - Auth defines the information necessary to authenticate against AWS - if not set aws sdk will infer credentials from your environment - see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials - properties: - jwt: - description: Authenticate against AWS using service account tokens. - properties: - serviceAccountRef: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - type: object - secretRef: - description: |- - AWSAuthSecretRef holds secret references for AWS credentials - both AccessKeyID and SecretAccessKey must be defined in order to properly authenticate. - properties: - accessKeyIDSecretRef: - description: The AccessKeyID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretAccessKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - region: - description: AWS Region to be used for the provider - type: string - role: - description: Role is a Role ARN which the SecretManager provider will assume - type: string - service: - description: Service defines which service should be used to fetch the secrets - enum: - - SecretsManager - - ParameterStore - type: string - required: - - region - - service - type: object - azurekv: - description: AzureKV configures this store to sync secrets using Azure Key Vault provider - properties: - authSecretRef: - description: Auth configures how the operator authenticates with Azure. Required for ServicePrincipal auth type. - properties: - clientId: - description: The Azure clientId of the service principle used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - clientSecret: - description: The Azure ClientSecret of the service principle used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - authType: - default: ServicePrincipal - description: |- - Auth type defines how to authenticate to the keyvault service. - Valid values are: - - "ServicePrincipal" (default): Using a service principal (tenantId, clientId, clientSecret) - - "ManagedIdentity": Using Managed Identity assigned to the pod (see aad-pod-identity) - enum: - - ServicePrincipal - - ManagedIdentity - - WorkloadIdentity - type: string - identityId: - description: If multiple Managed Identity is assigned to the pod, you can select the one to be used - type: string - serviceAccountRef: - description: |- - ServiceAccountRef specified the service account - that should be used when authenticating with WorkloadIdentity. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - tenantId: - description: TenantID configures the Azure Tenant to send requests to. Required for ServicePrincipal auth type. - type: string - vaultUrl: - description: Vault Url from which the secrets to be fetched from. - type: string - required: - - vaultUrl - type: object - fake: - description: Fake configures a store with static key/value pairs - properties: - data: - items: - properties: - key: - type: string - value: - type: string - valueMap: - additionalProperties: - type: string - type: object - version: - type: string - required: - - key - type: object - type: array - required: - - data - type: object - gcpsm: - description: GCPSM configures this store to sync secrets using Google Cloud Platform Secret Manager provider - properties: - auth: - description: Auth defines the information necessary to authenticate against GCP - properties: - secretRef: - properties: - secretAccessKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - workloadIdentity: - properties: - clusterLocation: - type: string - clusterName: - type: string - clusterProjectID: - type: string - serviceAccountRef: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - clusterLocation - - clusterName - - serviceAccountRef - type: object - type: object - projectID: - description: ProjectID project where secret is located - type: string - type: object - gitlab: - description: GitLab configures this store to sync secrets using GitLab Variables provider - properties: - auth: - description: Auth configures how secret-manager authenticates with a GitLab instance. - properties: - SecretRef: - properties: - accessToken: - description: AccessToken is used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - SecretRef - type: object - projectID: - description: ProjectID specifies a project where secrets are located. - type: string - url: - description: URL configures the GitLab instance URL. Defaults to https://gitlab.com/. - type: string - required: - - auth - type: object - ibm: - description: IBM configures this store to sync secrets using IBM Cloud provider - properties: - auth: - description: Auth configures how secret-manager authenticates with the IBM secrets manager. - properties: - secretRef: - properties: - secretApiKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - secretRef - type: object - serviceUrl: - description: ServiceURL is the Endpoint URL that is specific to the Secrets Manager service instance - type: string - required: - - auth - type: object - kubernetes: - description: Kubernetes configures this store to sync secrets using a Kubernetes cluster provider - properties: - auth: - description: Auth configures how secret-manager authenticates with a Kubernetes instance. - maxProperties: 1 - minProperties: 1 - properties: - cert: - description: has both clientCert and clientKey as secretKeySelector - properties: - clientCert: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - clientKey: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - serviceAccount: - description: points to a service account that should be used for authentication - properties: - serviceAccount: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - type: object - token: - description: use static token to authenticate with - properties: - bearerToken: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - remoteNamespace: - default: default - description: Remote namespace to fetch the secrets from - type: string - server: - description: configures the Kubernetes server Address. - properties: - caBundle: - description: CABundle is a base64-encoded CA certificate - format: byte - type: string - caProvider: - description: 'see: https://external-secrets.io/v0.4.1/spec/#external-secrets.io/v1alpha1.CAProvider' - properties: - key: - description: The key the value inside of the provider type to use, only used with "Secret" type - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: The namespace the Provider type is in. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - url: - default: kubernetes.default - description: configures the Kubernetes server Address. - type: string - type: object - required: - - auth - type: object - oracle: - description: Oracle configures this store to sync secrets using Oracle Vault provider - properties: - auth: - description: |- - Auth configures how secret-manager authenticates with the Oracle Vault. - If empty, instance principal is used. Optionally, the authenticating principal type - and/or user data may be supplied for the use of workload identity and user principal. - properties: - secretRef: - description: SecretRef to pass through sensitive information. - properties: - fingerprint: - description: Fingerprint is the fingerprint of the API private key. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - privatekey: - description: PrivateKey is the user's API Signing Key in PEM format, used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - fingerprint - - privatekey - type: object - tenancy: - description: Tenancy is the tenancy OCID where user is located. - type: string - user: - description: User is an access OCID specific to the account. - type: string - required: - - secretRef - - tenancy - - user - type: object - compartment: - description: |- - Compartment is the vault compartment OCID. - Required for PushSecret - type: string - encryptionKey: - description: |- - EncryptionKey is the OCID of the encryption key within the vault. - Required for PushSecret - type: string - principalType: - description: |- - The type of principal to use for authentication. If left blank, the Auth struct will - determine the principal type. This optional field must be specified if using - workload identity. - enum: - - "" - - UserPrincipal - - InstancePrincipal - - Workload - type: string - region: - description: Region is the region where vault is located. - type: string - serviceAccountRef: - description: |- - ServiceAccountRef specified the service account - that should be used when authenticating with WorkloadIdentity. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - vault: - description: Vault is the vault's OCID of the specific vault where secret is located. - type: string - required: - - region - - vault - type: object - passworddepot: - description: Configures a store to sync secrets with a Password Depot instance. - properties: - auth: - description: Auth configures how secret-manager authenticates with a Password Depot instance. - properties: - secretRef: - properties: - credentials: - description: Username / Password is used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - secretRef - type: object - database: - description: Database to use as source - type: string - host: - description: URL configures the Password Depot instance URL. - type: string - required: - - auth - - database - - host - type: object - vault: - description: Vault configures this store to sync secrets using Hashi provider - properties: - auth: - description: Auth configures how secret-manager authenticates with the Vault server. - properties: - appRole: - description: |- - AppRole authenticates with Vault using the App Role auth mechanism, - with the role and secret stored in a Kubernetes Secret resource. - properties: - path: - default: approle - description: |- - Path where the App Role authentication backend is mounted - in Vault, e.g: "approle" - type: string - roleId: - description: |- - RoleID configured in the App Role authentication backend when setting - up the authentication backend in Vault. - type: string - secretRef: - description: |- - Reference to a key in a Secret that contains the App Role secret used - to authenticate with Vault. - The `key` field must be specified and denotes which entry within the Secret - resource is used as the app role secret. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - path - - roleId - - secretRef - type: object - cert: - description: |- - Cert authenticates with TLS Certificates by passing client certificate, private key and ca certificate - Cert authentication method - properties: - clientCert: - description: |- - ClientCert is a certificate to authenticate using the Cert Vault - authentication method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretRef: - description: |- - SecretRef to a key in a Secret resource containing client private key to - authenticate with Vault using the Cert authentication method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - jwt: - description: |- - Jwt authenticates with Vault by passing role and JWT token using the - JWT/OIDC authentication method - properties: - kubernetesServiceAccountToken: - description: |- - Optional ServiceAccountToken specifies the Kubernetes service account for which to request - a token for with the `TokenRequest` API. - properties: - audiences: - description: |- - Optional audiences field that will be used to request a temporary Kubernetes service - account token for the service account referenced by `serviceAccountRef`. - Defaults to a single audience `vault` it not specified. - items: - type: string - type: array - expirationSeconds: - description: |- - Optional expiration time in seconds that will be used to request a temporary - Kubernetes service account token for the service account referenced by - `serviceAccountRef`. - Defaults to 10 minutes. - format: int64 - type: integer - serviceAccountRef: - description: Service account field containing the name of a kubernetes ServiceAccount. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - serviceAccountRef - type: object - path: - default: jwt - description: |- - Path where the JWT authentication backend is mounted - in Vault, e.g: "jwt" - type: string - role: - description: |- - Role is a JWT role to authenticate using the JWT/OIDC Vault - authentication method - type: string - secretRef: - description: |- - Optional SecretRef that refers to a key in a Secret resource containing JWT token to - authenticate with Vault using the JWT/OIDC authentication method. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - path - type: object - kubernetes: - description: |- - Kubernetes authenticates with Vault by passing the ServiceAccount - token stored in the named Secret resource to the Vault server. - properties: - mountPath: - default: kubernetes - description: |- - Path where the Kubernetes authentication backend is mounted in Vault, e.g: - "kubernetes" - type: string - role: - description: |- - A required field containing the Vault Role to assume. A Role binds a - Kubernetes ServiceAccount with a set of Vault policies. - type: string - secretRef: - description: |- - Optional secret field containing a Kubernetes ServiceAccount JWT used - for authenticating with Vault. If a name is specified without a key, - `token` is the default. If one is not specified, the one bound to - the controller will be used. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - serviceAccountRef: - description: |- - Optional service account field containing the name of a kubernetes ServiceAccount. - If the service account is specified, the service account secret token JWT will be used - for authenticating with Vault. If the service account selector is not supplied, - the secretRef will be used instead. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - mountPath - - role - type: object - ldap: - description: |- - Ldap authenticates with Vault by passing username/password pair using - the LDAP authentication method - properties: - path: - default: ldap - description: |- - Path where the LDAP authentication backend is mounted - in Vault, e.g: "ldap" - type: string - secretRef: - description: |- - SecretRef to a key in a Secret resource containing password for the LDAP - user used to authenticate with Vault using the LDAP authentication - method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - username: - description: |- - Username is a LDAP user name used to authenticate using the LDAP Vault - authentication method - type: string - required: - - path - - username - type: object - tokenSecretRef: - description: TokenSecretRef authenticates with Vault by presenting a token. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - caBundle: - description: |- - PEM encoded CA bundle used to validate Vault server certificate. Only used - if the Server URL is using HTTPS protocol. This parameter is ignored for - plain HTTP protocol connection. If not set the system root certificates - are used to validate the TLS connection. - format: byte - type: string - caProvider: - description: The provider for the CA bundle to use to validate Vault server certificate. - properties: - key: - description: The key the value inside of the provider type to use, only used with "Secret" type - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: The namespace the Provider type is in. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - forwardInconsistent: - description: |- - ForwardInconsistent tells Vault to forward read-after-write requests to the Vault - leader instead of simply retrying within a loop. This can increase performance if - the option is enabled serverside. - https://www.vaultproject.io/docs/configuration/replication#allow_forwarding_via_header - type: boolean - namespace: - description: |- - Name of the vault namespace. Namespaces is a set of features within Vault Enterprise that allows - Vault environments to support Secure Multi-tenancy. e.g: "ns1". - More about namespaces can be found here https://www.vaultproject.io/docs/enterprise/namespaces - type: string - path: - description: |- - Path is the mount path of the Vault KV backend endpoint, e.g: - "secret". The v2 KV secret engine version specific "/data" path suffix - for fetching secrets from Vault is optional and will be appended - if not present in specified path. - type: string - readYourWrites: - description: |- - ReadYourWrites ensures isolated read-after-write semantics by - providing discovered cluster replication states in each request. - More information about eventual consistency in Vault can be found here - https://www.vaultproject.io/docs/enterprise/consistency - type: boolean - server: - description: 'Server is the connection address for the Vault server, e.g: "https://vault.example.com:8200".' - type: string - version: - default: v2 - description: |- - Version is the Vault KV secret engine version. This can be either "v1" or - "v2". Version defaults to "v2". - enum: - - v1 - - v2 - type: string - required: - - auth - - server - type: object - webhook: - description: Webhook configures this store to sync secrets using a generic templated webhook - properties: - body: - description: Body - type: string - caBundle: - description: |- - PEM encoded CA bundle used to validate webhook server certificate. Only used - if the Server URL is using HTTPS protocol. This parameter is ignored for - plain HTTP protocol connection. If not set the system root certificates - are used to validate the TLS connection. - format: byte - type: string - caProvider: - description: The provider for the CA bundle to use to validate webhook server certificate. - properties: - key: - description: The key the value inside of the provider type to use, only used with "Secret" type - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: The namespace the Provider type is in. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - headers: - additionalProperties: - type: string - description: Headers - type: object - method: - description: Webhook Method - type: string - result: - description: Result formatting - properties: - jsonPath: - description: Json path of return value - type: string - type: object - secrets: - description: |- - Secrets to fill in templates - These secrets will be passed to the templating function as key value pairs under the given name - items: - properties: - name: - description: Name of this secret in templates - type: string - secretRef: - description: Secret ref to fill in credentials - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - name - - secretRef - type: object - type: array - timeout: - description: Timeout - type: string - url: - description: Webhook url to call - type: string - required: - - result - - url - type: object - yandexlockbox: - description: YandexLockbox configures this store to sync secrets using Yandex Lockbox provider - properties: - apiEndpoint: - description: Yandex.Cloud API endpoint (e.g. 'api.cloud.yandex.net:443') - type: string - auth: - description: Auth defines the information necessary to authenticate against Yandex Lockbox - properties: - authorizedKeySecretRef: - description: The authorized key used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - caProvider: - description: The provider for the CA bundle to use to validate Yandex.Cloud server certificate. - properties: - certSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - auth - type: object - type: object - retrySettings: - description: Used to configure http retries if failed - properties: - maxRetries: - format: int32 - type: integer - retryInterval: - type: string - type: object - required: - - provider - type: object - status: - description: SecretStoreStatus defines the observed state of the SecretStore. - properties: - conditions: - items: - properties: - lastTransitionTime: - format: date-time - type: string - message: - type: string - reason: - type: string - status: - type: string - type: - type: string - required: - - status - - type - type: object - type: array - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: AGE - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].reason - name: Status - type: string - - jsonPath: .status.capabilities - name: Capabilities - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - name: v1beta1 - schema: - openAPIV3Schema: - description: SecretStore represents a secure external location for storing secrets, which can be referenced as part of `storeRef` fields. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: SecretStoreSpec defines the desired state of SecretStore. - properties: - conditions: - description: Used to constraint a ClusterSecretStore to specific namespaces. Relevant only to ClusterSecretStore - items: - description: |- - ClusterSecretStoreCondition describes a condition by which to choose namespaces to process ExternalSecrets in - for a ClusterSecretStore instance. - properties: - namespaceRegexes: - description: Choose namespaces by using regex matching - items: - type: string - type: array - namespaceSelector: - description: Choose namespace using a labelSelector - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - x-kubernetes-list-type: atomic - required: - - key - - operator - type: object - type: array - x-kubernetes-list-type: atomic - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - namespaces: - description: Choose namespaces by name - items: - type: string - type: array - type: object - type: array - controller: - description: |- - Used to select the correct ESO controller (think: ingress.ingressClassName) - The ESO controller is instantiated with a specific controller name and filters ES based on this property - type: string - provider: - description: Used to configure the provider. Only one provider may be set - maxProperties: 1 - minProperties: 1 - properties: - akeyless: - description: Akeyless configures this store to sync secrets using Akeyless Vault provider - properties: - akeylessGWApiURL: - description: Akeyless GW API Url from which the secrets to be fetched from. - type: string - authSecretRef: - description: Auth configures how the operator authenticates with Akeyless. - properties: - kubernetesAuth: - description: |- - Kubernetes authenticates with Akeyless by passing the ServiceAccount - token stored in the named Secret resource. - properties: - accessID: - description: the Akeyless Kubernetes auth-method access-id - type: string - k8sConfName: - description: Kubernetes-auth configuration name in Akeyless-Gateway - type: string - secretRef: - description: |- - Optional secret field containing a Kubernetes ServiceAccount JWT used - for authenticating with Akeyless. If a name is specified without a key, - `token` is the default. If one is not specified, the one bound to - the controller will be used. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - serviceAccountRef: - description: |- - Optional service account field containing the name of a kubernetes ServiceAccount. - If the service account is specified, the service account secret token JWT will be used - for authenticating with Akeyless. If the service account selector is not supplied, - the secretRef will be used instead. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - accessID - - k8sConfName - type: object - secretRef: - description: |- - Reference to a Secret that contains the details - to authenticate with Akeyless. - properties: - accessID: - description: The SecretAccessID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - accessType: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - accessTypeParam: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - caBundle: - description: |- - PEM/base64 encoded CA bundle used to validate Akeyless Gateway certificate. Only used - if the AkeylessGWApiURL URL is using HTTPS protocol. If not set the system root certificates - are used to validate the TLS connection. - format: byte - type: string - caProvider: - description: The provider for the CA bundle to use to validate Akeyless Gateway certificate. - properties: - key: - description: The key where the CA certificate can be found in the Secret or ConfigMap. - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: |- - The namespace the Provider type is in. - Can only be defined when used in a ClusterSecretStore. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - required: - - akeylessGWApiURL - - authSecretRef - type: object - alibaba: - description: Alibaba configures this store to sync secrets using Alibaba Cloud provider - properties: - auth: - description: AlibabaAuth contains a secretRef for credentials. - properties: - rrsa: - description: Authenticate against Alibaba using RRSA. - properties: - oidcProviderArn: - type: string - oidcTokenFilePath: - type: string - roleArn: - type: string - sessionName: - type: string - required: - - oidcProviderArn - - oidcTokenFilePath - - roleArn - - sessionName - type: object - secretRef: - description: AlibabaAuthSecretRef holds secret references for Alibaba credentials. - properties: - accessKeyIDSecretRef: - description: The AccessKeyID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - accessKeySecretSecretRef: - description: The AccessKeySecret is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - accessKeyIDSecretRef - - accessKeySecretSecretRef - type: object - type: object - regionID: - description: Alibaba Region to be used for the provider - type: string - required: - - auth - - regionID - type: object - aws: - description: AWS configures this store to sync secrets using AWS Secret Manager provider - properties: - additionalRoles: - description: AdditionalRoles is a chained list of Role ARNs which the provider will sequentially assume before assuming the Role - items: - type: string - type: array - auth: - description: |- - Auth defines the information necessary to authenticate against AWS - if not set aws sdk will infer credentials from your environment - see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials - properties: - jwt: - description: Authenticate against AWS using service account tokens. - properties: - serviceAccountRef: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - type: object - secretRef: - description: |- - AWSAuthSecretRef holds secret references for AWS credentials - both AccessKeyID and SecretAccessKey must be defined in order to properly authenticate. - properties: - accessKeyIDSecretRef: - description: The AccessKeyID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretAccessKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - sessionTokenSecretRef: - description: |- - The SessionToken used for authentication - This must be defined if AccessKeyID and SecretAccessKey are temporary credentials - see: https://docs.aws.amazon.com/IAM/latest/UserGuide/id_credentials_temp_use-resources.html - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - externalID: - description: AWS External ID set on assumed IAM roles - type: string - prefix: - description: Prefix adds a prefix to all retrieved values. - type: string - region: - description: AWS Region to be used for the provider - type: string - role: - description: Role is a Role ARN which the provider will assume - type: string - secretsManager: - description: SecretsManager defines how the provider behaves when interacting with AWS SecretsManager - properties: - forceDeleteWithoutRecovery: - description: |- - Specifies whether to delete the secret without any recovery window. You - can't use both this parameter and RecoveryWindowInDays in the same call. - If you don't use either, then by default Secrets Manager uses a 30 day - recovery window. - see: https://docs.aws.amazon.com/secretsmanager/latest/apireference/API_DeleteSecret.html#SecretsManager-DeleteSecret-request-ForceDeleteWithoutRecovery - type: boolean - recoveryWindowInDays: - description: |- - The number of days from 7 to 30 that Secrets Manager waits before - permanently deleting the secret. You can't use both this parameter and - ForceDeleteWithoutRecovery in the same call. If you don't use either, - then by default Secrets Manager uses a 30 day recovery window. - see: https://docs.aws.amazon.com/secretsmanager/latest/apireference/API_DeleteSecret.html#SecretsManager-DeleteSecret-request-RecoveryWindowInDays - format: int64 - type: integer - type: object - service: - description: Service defines which service should be used to fetch the secrets - enum: - - SecretsManager - - ParameterStore - type: string - sessionTags: - description: AWS STS assume role session tags - items: - properties: - key: - type: string - value: - type: string - required: - - key - - value - type: object - type: array - transitiveTagKeys: - description: AWS STS assume role transitive session tags. Required when multiple rules are used with the provider - items: - type: string - type: array - required: - - region - - service - type: object - azurekv: - description: AzureKV configures this store to sync secrets using Azure Key Vault provider - properties: - authSecretRef: - description: Auth configures how the operator authenticates with Azure. Required for ServicePrincipal auth type. Optional for WorkloadIdentity. - properties: - clientCertificate: - description: The Azure ClientCertificate of the service principle used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - clientId: - description: The Azure clientId of the service principle or managed identity used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - clientSecret: - description: The Azure ClientSecret of the service principle used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - tenantId: - description: The Azure tenantId of the managed identity used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - authType: - default: ServicePrincipal - description: |- - Auth type defines how to authenticate to the keyvault service. - Valid values are: - - "ServicePrincipal" (default): Using a service principal (tenantId, clientId, clientSecret) - - "ManagedIdentity": Using Managed Identity assigned to the pod (see aad-pod-identity) - enum: - - ServicePrincipal - - ManagedIdentity - - WorkloadIdentity - type: string - environmentType: - default: PublicCloud - description: |- - EnvironmentType specifies the Azure cloud environment endpoints to use for - connecting and authenticating with Azure. By default it points to the public cloud AAD endpoint. - The following endpoints are available, also see here: https://github.com/Azure/go-autorest/blob/main/autorest/azure/environments.go#L152 - PublicCloud, USGovernmentCloud, ChinaCloud, GermanCloud - enum: - - PublicCloud - - USGovernmentCloud - - ChinaCloud - - GermanCloud - type: string - identityId: - description: If multiple Managed Identity is assigned to the pod, you can select the one to be used - type: string - serviceAccountRef: - description: |- - ServiceAccountRef specified the service account - that should be used when authenticating with WorkloadIdentity. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - tenantId: - description: TenantID configures the Azure Tenant to send requests to. Required for ServicePrincipal auth type. Optional for WorkloadIdentity. - type: string - vaultUrl: - description: Vault Url from which the secrets to be fetched from. - type: string - required: - - vaultUrl - type: object - beyondtrust: - description: Beyondtrust configures this store to sync secrets using Password Safe provider. - properties: - auth: - description: Auth configures how the operator authenticates with Beyondtrust. - properties: - certificate: - description: Content of the certificate (cert.pem) for use when authenticating with an OAuth client Id using a Client Certificate. - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - certificateKey: - description: Certificate private key (key.pem). For use when authenticating with an OAuth client Id - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - clientId: - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - clientSecret: - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - required: - - clientId - - clientSecret - type: object - server: - description: Auth configures how API server works. - properties: - apiUrl: - type: string - clientTimeOutSeconds: - description: Timeout specifies a time limit for requests made by this Client. The timeout includes connection time, any redirects, and reading the response body. Defaults to 45 seconds. - type: integer - retrievalType: - description: The secret retrieval type. SECRET = Secrets Safe (credential, text, file). MANAGED_ACCOUNT = Password Safe account associated with a system. - type: string - separator: - description: A character that separates the folder names. - type: string - verifyCA: - type: boolean - required: - - apiUrl - - verifyCA - type: object - required: - - auth - - server - type: object - bitwardensecretsmanager: - description: BitwardenSecretsManager configures this store to sync secrets using BitwardenSecretsManager provider - properties: - apiURL: - type: string - auth: - description: |- - Auth configures how secret-manager authenticates with a bitwarden machine account instance. - Make sure that the token being used has permissions on the given secret. - properties: - secretRef: - description: BitwardenSecretsManagerSecretRef contains the credential ref to the bitwarden instance. - properties: - credentials: - description: AccessToken used for the bitwarden instance. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - credentials - type: object - required: - - secretRef - type: object - bitwardenServerSDKURL: - type: string - caBundle: - description: |- - Base64 encoded certificate for the bitwarden server sdk. The sdk MUST run with HTTPS to make sure no MITM attack - can be performed. - type: string - caProvider: - description: 'see: https://external-secrets.io/latest/spec/#external-secrets.io/v1alpha1.CAProvider' - properties: - key: - description: The key where the CA certificate can be found in the Secret or ConfigMap. - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: |- - The namespace the Provider type is in. - Can only be defined when used in a ClusterSecretStore. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - identityURL: - type: string - organizationID: - description: OrganizationID determines which organization this secret store manages. - type: string - projectID: - description: ProjectID determines which project this secret store manages. - type: string - required: - - auth - - organizationID - - projectID - type: object - chef: - description: Chef configures this store to sync secrets with chef server - properties: - auth: - description: Auth defines the information necessary to authenticate against chef Server - properties: - secretRef: - description: ChefAuthSecretRef holds secret references for chef server login credentials. - properties: - privateKeySecretRef: - description: SecretKey is the Signing Key in PEM format, used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - privateKeySecretRef - type: object - required: - - secretRef - type: object - serverUrl: - description: ServerURL is the chef server URL used to connect to. If using orgs you should include your org in the url and terminate the url with a "/" - type: string - username: - description: UserName should be the user ID on the chef server - type: string - required: - - auth - - serverUrl - - username - type: object - conjur: - description: Conjur configures this store to sync secrets using conjur provider - properties: - auth: - properties: - apikey: - properties: - account: - type: string - apiKeyRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - userRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - account - - apiKeyRef - - userRef - type: object - jwt: - properties: - account: - type: string - hostId: - description: |- - Optional HostID for JWT authentication. This may be used depending - on how the Conjur JWT authenticator policy is configured. - type: string - secretRef: - description: |- - Optional SecretRef that refers to a key in a Secret resource containing JWT token to - authenticate with Conjur using the JWT authentication method. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - serviceAccountRef: - description: |- - Optional ServiceAccountRef specifies the Kubernetes service account for which to request - a token for with the `TokenRequest` API. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - serviceID: - description: The conjur authn jwt webservice id - type: string - required: - - account - - serviceID - type: object - type: object - caBundle: - type: string - caProvider: - description: |- - Used to provide custom certificate authority (CA) certificates - for a secret store. The CAProvider points to a Secret or ConfigMap resource - that contains a PEM-encoded certificate. - properties: - key: - description: The key where the CA certificate can be found in the Secret or ConfigMap. - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: |- - The namespace the Provider type is in. - Can only be defined when used in a ClusterSecretStore. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - url: - type: string - required: - - auth - - url - type: object - delinea: - description: |- - Delinea DevOps Secrets Vault - https://docs.delinea.com/online-help/products/devops-secrets-vault/current - properties: - clientId: - description: ClientID is the non-secret part of the credential. - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - clientSecret: - description: ClientSecret is the secret part of the credential. - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - tenant: - description: Tenant is the chosen hostname / site name. - type: string - tld: - description: |- - TLD is based on the server location that was chosen during provisioning. - If unset, defaults to "com". - type: string - urlTemplate: - description: |- - URLTemplate - If unset, defaults to "https://%s.secretsvaultcloud.%s/v1/%s%s". - type: string - required: - - clientId - - clientSecret - - tenant - type: object - device42: - description: Device42 configures this store to sync secrets using the Device42 provider - properties: - auth: - description: Auth configures how secret-manager authenticates with a Device42 instance. - properties: - secretRef: - properties: - credentials: - description: Username / Password is used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - secretRef - type: object - host: - description: URL configures the Device42 instance URL. - type: string - required: - - auth - - host - type: object - doppler: - description: Doppler configures this store to sync secrets using the Doppler provider - properties: - auth: - description: Auth configures how the Operator authenticates with the Doppler API - properties: - secretRef: - properties: - dopplerToken: - description: |- - The DopplerToken is used for authentication. - See https://docs.doppler.com/reference/api#authentication for auth token types. - The Key attribute defaults to dopplerToken if not specified. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - dopplerToken - type: object - required: - - secretRef - type: object - config: - description: Doppler config (required if not using a Service Token) - type: string - format: - description: Format enables the downloading of secrets as a file (string) - enum: - - json - - dotnet-json - - env - - yaml - - docker - type: string - nameTransformer: - description: Environment variable compatible name transforms that change secret names to a different format - enum: - - upper-camel - - camel - - lower-snake - - tf-var - - dotnet-env - - lower-kebab - type: string - project: - description: Doppler project (required if not using a Service Token) - type: string - required: - - auth - type: object - fake: - description: Fake configures a store with static key/value pairs - properties: - data: - items: - properties: - key: - type: string - value: - type: string - valueMap: - additionalProperties: - type: string - description: 'Deprecated: ValueMap is deprecated and is intended to be removed in the future, use the `value` field instead.' - type: object - version: - type: string - required: - - key - type: object - type: array - required: - - data - type: object - fortanix: - description: Fortanix configures this store to sync secrets using the Fortanix provider - properties: - apiKey: - description: APIKey is the API token to access SDKMS Applications. - properties: - secretRef: - description: SecretRef is a reference to a secret containing the SDKMS API Key. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - apiUrl: - description: APIURL is the URL of SDKMS API. Defaults to `sdkms.fortanix.com`. - type: string - type: object - gcpsm: - description: GCPSM configures this store to sync secrets using Google Cloud Platform Secret Manager provider - properties: - auth: - description: Auth defines the information necessary to authenticate against GCP - properties: - secretRef: - properties: - secretAccessKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - workloadIdentity: - properties: - clusterLocation: - type: string - clusterName: - type: string - clusterProjectID: - type: string - serviceAccountRef: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - clusterLocation - - clusterName - - serviceAccountRef - type: object - type: object - location: - description: Location optionally defines a location for a secret - type: string - projectID: - description: ProjectID project where secret is located - type: string - type: object - gitlab: - description: GitLab configures this store to sync secrets using GitLab Variables provider - properties: - auth: - description: Auth configures how secret-manager authenticates with a GitLab instance. - properties: - SecretRef: - properties: - accessToken: - description: AccessToken is used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - SecretRef - type: object - environment: - description: Environment environment_scope of gitlab CI/CD variables (Please see https://docs.gitlab.com/ee/ci/environments/#create-a-static-environment on how to create environments) - type: string - groupIDs: - description: GroupIDs specify, which gitlab groups to pull secrets from. Group secrets are read from left to right followed by the project variables. - items: - type: string - type: array - inheritFromGroups: - description: InheritFromGroups specifies whether parent groups should be discovered and checked for secrets. - type: boolean - projectID: - description: ProjectID specifies a project where secrets are located. - type: string - url: - description: URL configures the GitLab instance URL. Defaults to https://gitlab.com/. - type: string - required: - - auth - type: object - ibm: - description: IBM configures this store to sync secrets using IBM Cloud provider - properties: - auth: - description: Auth configures how secret-manager authenticates with the IBM secrets manager. - maxProperties: 1 - minProperties: 1 - properties: - containerAuth: - description: IBM Container-based auth with IAM Trusted Profile. - properties: - iamEndpoint: - type: string - profile: - description: the IBM Trusted Profile - type: string - tokenLocation: - description: Location the token is mounted on the pod - type: string - required: - - profile - type: object - secretRef: - properties: - secretApiKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - serviceUrl: - description: ServiceURL is the Endpoint URL that is specific to the Secrets Manager service instance - type: string - required: - - auth - type: object - infisical: - description: Infisical configures this store to sync secrets using the Infisical provider - properties: - auth: - description: Auth configures how the Operator authenticates with the Infisical API - properties: - universalAuthCredentials: - properties: - clientId: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - clientSecret: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - clientId - - clientSecret - type: object - type: object - hostAPI: - default: https://app.infisical.com/api - type: string - secretsScope: - properties: - environmentSlug: - type: string - projectSlug: - type: string - secretsPath: - default: / - type: string - required: - - environmentSlug - - projectSlug - type: object - required: - - auth - - secretsScope - type: object - keepersecurity: - description: KeeperSecurity configures this store to sync secrets using the KeeperSecurity provider - properties: - authRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - folderID: - type: string - required: - - authRef - - folderID - type: object - kubernetes: - description: Kubernetes configures this store to sync secrets using a Kubernetes cluster provider - properties: - auth: - description: Auth configures how secret-manager authenticates with a Kubernetes instance. - maxProperties: 1 - minProperties: 1 - properties: - cert: - description: has both clientCert and clientKey as secretKeySelector - properties: - clientCert: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - clientKey: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - serviceAccount: - description: points to a service account that should be used for authentication - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - token: - description: use static token to authenticate with - properties: - bearerToken: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - type: object - authRef: - description: A reference to a secret that contains the auth information. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - remoteNamespace: - default: default - description: Remote namespace to fetch the secrets from - type: string - server: - description: configures the Kubernetes server Address. - properties: - caBundle: - description: CABundle is a base64-encoded CA certificate - format: byte - type: string - caProvider: - description: 'see: https://external-secrets.io/v0.4.1/spec/#external-secrets.io/v1alpha1.CAProvider' - properties: - key: - description: The key where the CA certificate can be found in the Secret or ConfigMap. - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: |- - The namespace the Provider type is in. - Can only be defined when used in a ClusterSecretStore. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - url: - default: kubernetes.default - description: configures the Kubernetes server Address. - type: string - type: object - type: object - onboardbase: - description: Onboardbase configures this store to sync secrets using the Onboardbase provider - properties: - apiHost: - default: https://public.onboardbase.com/api/v1/ - description: APIHost use this to configure the host url for the API for selfhosted installation, default is https://public.onboardbase.com/api/v1/ - type: string - auth: - description: Auth configures how the Operator authenticates with the Onboardbase API - properties: - apiKeyRef: - description: |- - OnboardbaseAPIKey is the APIKey generated by an admin account. - It is used to recognize and authorize access to a project and environment within onboardbase - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - passcodeRef: - description: OnboardbasePasscode is the passcode attached to the API Key - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - apiKeyRef - - passcodeRef - type: object - environment: - default: development - description: Environment is the name of an environmnent within a project to pull the secrets from - type: string - project: - default: development - description: Project is an onboardbase project that the secrets should be pulled from - type: string - required: - - apiHost - - auth - - environment - - project - type: object - onepassword: - description: OnePassword configures this store to sync secrets using the 1Password Cloud provider - properties: - auth: - description: Auth defines the information necessary to authenticate against OnePassword Connect Server - properties: - secretRef: - description: OnePasswordAuthSecretRef holds secret references for 1Password credentials. - properties: - connectTokenSecretRef: - description: The ConnectToken is used for authentication to a 1Password Connect Server. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - connectTokenSecretRef - type: object - required: - - secretRef - type: object - connectHost: - description: ConnectHost defines the OnePassword Connect Server to connect to - type: string - vaults: - additionalProperties: - type: integer - description: Vaults defines which OnePassword vaults to search in which order - type: object - required: - - auth - - connectHost - - vaults - type: object - oracle: - description: Oracle configures this store to sync secrets using Oracle Vault provider - properties: - auth: - description: |- - Auth configures how secret-manager authenticates with the Oracle Vault. - If empty, use the instance principal, otherwise the user credentials specified in Auth. - properties: - secretRef: - description: SecretRef to pass through sensitive information. - properties: - fingerprint: - description: Fingerprint is the fingerprint of the API private key. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - privatekey: - description: PrivateKey is the user's API Signing Key in PEM format, used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - fingerprint - - privatekey - type: object - tenancy: - description: Tenancy is the tenancy OCID where user is located. - type: string - user: - description: User is an access OCID specific to the account. - type: string - required: - - secretRef - - tenancy - - user - type: object - compartment: - description: |- - Compartment is the vault compartment OCID. - Required for PushSecret - type: string - encryptionKey: - description: |- - EncryptionKey is the OCID of the encryption key within the vault. - Required for PushSecret - type: string - principalType: - description: |- - The type of principal to use for authentication. If left blank, the Auth struct will - determine the principal type. This optional field must be specified if using - workload identity. - enum: - - "" - - UserPrincipal - - InstancePrincipal - - Workload - type: string - region: - description: Region is the region where vault is located. - type: string - serviceAccountRef: - description: |- - ServiceAccountRef specified the service account - that should be used when authenticating with WorkloadIdentity. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - vault: - description: Vault is the vault's OCID of the specific vault where secret is located. - type: string - required: - - region - - vault - type: object - passbolt: - properties: - auth: - description: Auth defines the information necessary to authenticate against Passbolt Server - properties: - passwordSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - privateKeySecretRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - passwordSecretRef - - privateKeySecretRef - type: object - host: - description: Host defines the Passbolt Server to connect to - type: string - required: - - auth - - host - type: object - passworddepot: - description: Configures a store to sync secrets with a Password Depot instance. - properties: - auth: - description: Auth configures how secret-manager authenticates with a Password Depot instance. - properties: - secretRef: - properties: - credentials: - description: Username / Password is used for authentication. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - secretRef - type: object - database: - description: Database to use as source - type: string - host: - description: URL configures the Password Depot instance URL. - type: string - required: - - auth - - database - - host - type: object - pulumi: - description: Pulumi configures this store to sync secrets using the Pulumi provider - properties: - accessToken: - description: AccessToken is the access tokens to sign in to the Pulumi Cloud Console. - properties: - secretRef: - description: SecretRef is a reference to a secret containing the Pulumi API token. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - apiUrl: - default: https://api.pulumi.com/api/preview - description: APIURL is the URL of the Pulumi API. - type: string - environment: - description: |- - Environment are YAML documents composed of static key-value pairs, programmatic expressions, - dynamically retrieved values from supported providers including all major clouds, - and other Pulumi ESC environments. - To create a new environment, visit https://www.pulumi.com/docs/esc/environments/ for more information. - type: string - organization: - description: |- - Organization are a space to collaborate on shared projects and stacks. - To create a new organization, visit https://app.pulumi.com/ and click "New Organization". - type: string - required: - - accessToken - - environment - - organization - type: object - scaleway: - description: Scaleway - properties: - accessKey: - description: AccessKey is the non-secret part of the api key. - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - apiUrl: - description: APIURL is the url of the api to use. Defaults to https://api.scaleway.com - type: string - projectId: - description: 'ProjectID is the id of your project, which you can find in the console: https://console.scaleway.com/project/settings' - type: string - region: - description: 'Region where your secrets are located: https://developers.scaleway.com/en/quickstart/#region-and-zone' - type: string - secretKey: - description: SecretKey is the non-secret part of the api key. - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - required: - - accessKey - - projectId - - region - - secretKey - type: object - secretserver: - description: |- - SecretServer configures this store to sync secrets using SecretServer provider - https://docs.delinea.com/online-help/secret-server/start.htm - properties: - password: - description: Password is the secret server account password. - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - serverURL: - description: |- - ServerURL - URL to your secret server installation - type: string - username: - description: Username is the secret server account username. - properties: - secretRef: - description: SecretRef references a key in a secret that will be used as value. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - value: - description: Value can be specified directly to set a value without using a secret. - type: string - type: object - required: - - password - - serverURL - - username - type: object - senhasegura: - description: Senhasegura configures this store to sync secrets using senhasegura provider - properties: - auth: - description: Auth defines parameters to authenticate in senhasegura - properties: - clientId: - type: string - clientSecretSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - clientId - - clientSecretSecretRef - type: object - ignoreSslCertificate: - default: false - description: IgnoreSslCertificate defines if SSL certificate must be ignored - type: boolean - module: - description: Module defines which senhasegura module should be used to get secrets - type: string - url: - description: URL of senhasegura - type: string - required: - - auth - - module - - url - type: object - vault: - description: Vault configures this store to sync secrets using Hashi provider - properties: - auth: - description: Auth configures how secret-manager authenticates with the Vault server. - properties: - appRole: - description: |- - AppRole authenticates with Vault using the App Role auth mechanism, - with the role and secret stored in a Kubernetes Secret resource. - properties: - path: - default: approle - description: |- - Path where the App Role authentication backend is mounted - in Vault, e.g: "approle" - type: string - roleId: - description: |- - RoleID configured in the App Role authentication backend when setting - up the authentication backend in Vault. - type: string - roleRef: - description: |- - Reference to a key in a Secret that contains the App Role ID used - to authenticate with Vault. - The `key` field must be specified and denotes which entry within the Secret - resource is used as the app role id. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretRef: - description: |- - Reference to a key in a Secret that contains the App Role secret used - to authenticate with Vault. - The `key` field must be specified and denotes which entry within the Secret - resource is used as the app role secret. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - path - - secretRef - type: object - cert: - description: |- - Cert authenticates with TLS Certificates by passing client certificate, private key and ca certificate - Cert authentication method - properties: - clientCert: - description: |- - ClientCert is a certificate to authenticate using the Cert Vault - authentication method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretRef: - description: |- - SecretRef to a key in a Secret resource containing client private key to - authenticate with Vault using the Cert authentication method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - iam: - description: |- - Iam authenticates with vault by passing a special AWS request signed with AWS IAM credentials - AWS IAM authentication method - properties: - externalID: - description: AWS External ID set on assumed IAM roles - type: string - jwt: - description: Specify a service account with IRSA enabled - properties: - serviceAccountRef: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - type: object - path: - description: 'Path where the AWS auth method is enabled in Vault, e.g: "aws"' - type: string - region: - description: AWS region - type: string - role: - description: This is the AWS role to be assumed before talking to vault - type: string - secretRef: - description: Specify credentials in a Secret object - properties: - accessKeyIDSecretRef: - description: The AccessKeyID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretAccessKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - sessionTokenSecretRef: - description: |- - The SessionToken used for authentication - This must be defined if AccessKeyID and SecretAccessKey are temporary credentials - see: https://docs.aws.amazon.com/IAM/latest/UserGuide/id_credentials_temp_use-resources.html - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - vaultAwsIamServerID: - description: 'X-Vault-AWS-IAM-Server-ID is an additional header used by Vault IAM auth method to mitigate against different types of replay attacks. More details here: https://developer.hashicorp.com/vault/docs/auth/aws' - type: string - vaultRole: - description: Vault Role. In vault, a role describes an identity with a set of permissions, groups, or policies you want to attach a user of the secrets engine - type: string - required: - - vaultRole - type: object - jwt: - description: |- - Jwt authenticates with Vault by passing role and JWT token using the - JWT/OIDC authentication method - properties: - kubernetesServiceAccountToken: - description: |- - Optional ServiceAccountToken specifies the Kubernetes service account for which to request - a token for with the `TokenRequest` API. - properties: - audiences: - description: |- - Optional audiences field that will be used to request a temporary Kubernetes service - account token for the service account referenced by `serviceAccountRef`. - Defaults to a single audience `vault` it not specified. - Deprecated: use serviceAccountRef.Audiences instead - items: - type: string - type: array - expirationSeconds: - description: |- - Optional expiration time in seconds that will be used to request a temporary - Kubernetes service account token for the service account referenced by - `serviceAccountRef`. - Deprecated: this will be removed in the future. - Defaults to 10 minutes. - format: int64 - type: integer - serviceAccountRef: - description: Service account field containing the name of a kubernetes ServiceAccount. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - serviceAccountRef - type: object - path: - default: jwt - description: |- - Path where the JWT authentication backend is mounted - in Vault, e.g: "jwt" - type: string - role: - description: |- - Role is a JWT role to authenticate using the JWT/OIDC Vault - authentication method - type: string - secretRef: - description: |- - Optional SecretRef that refers to a key in a Secret resource containing JWT token to - authenticate with Vault using the JWT/OIDC authentication method. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - path - type: object - kubernetes: - description: |- - Kubernetes authenticates with Vault by passing the ServiceAccount - token stored in the named Secret resource to the Vault server. - properties: - mountPath: - default: kubernetes - description: |- - Path where the Kubernetes authentication backend is mounted in Vault, e.g: - "kubernetes" - type: string - role: - description: |- - A required field containing the Vault Role to assume. A Role binds a - Kubernetes ServiceAccount with a set of Vault policies. - type: string - secretRef: - description: |- - Optional secret field containing a Kubernetes ServiceAccount JWT used - for authenticating with Vault. If a name is specified without a key, - `token` is the default. If one is not specified, the one bound to - the controller will be used. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - serviceAccountRef: - description: |- - Optional service account field containing the name of a kubernetes ServiceAccount. - If the service account is specified, the service account secret token JWT will be used - for authenticating with Vault. If the service account selector is not supplied, - the secretRef will be used instead. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - mountPath - - role - type: object - ldap: - description: |- - Ldap authenticates with Vault by passing username/password pair using - the LDAP authentication method - properties: - path: - default: ldap - description: |- - Path where the LDAP authentication backend is mounted - in Vault, e.g: "ldap" - type: string - secretRef: - description: |- - SecretRef to a key in a Secret resource containing password for the LDAP - user used to authenticate with Vault using the LDAP authentication - method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - username: - description: |- - Username is a LDAP user name used to authenticate using the LDAP Vault - authentication method - type: string - required: - - path - - username - type: object - namespace: - description: |- - Name of the vault namespace to authenticate to. This can be different than the namespace your secret is in. - Namespaces is a set of features within Vault Enterprise that allows - Vault environments to support Secure Multi-tenancy. e.g: "ns1". - More about namespaces can be found here https://www.vaultproject.io/docs/enterprise/namespaces - This will default to Vault.Namespace field if set, or empty otherwise - type: string - tokenSecretRef: - description: TokenSecretRef authenticates with Vault by presenting a token. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - userPass: - description: UserPass authenticates with Vault by passing username/password pair - properties: - path: - default: user - description: |- - Path where the UserPassword authentication backend is mounted - in Vault, e.g: "user" - type: string - secretRef: - description: |- - SecretRef to a key in a Secret resource containing password for the - user used to authenticate with Vault using the UserPass authentication - method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - username: - description: |- - Username is a user name used to authenticate using the UserPass Vault - authentication method - type: string - required: - - path - - username - type: object - type: object - caBundle: - description: |- - PEM encoded CA bundle used to validate Vault server certificate. Only used - if the Server URL is using HTTPS protocol. This parameter is ignored for - plain HTTP protocol connection. If not set the system root certificates - are used to validate the TLS connection. - format: byte - type: string - caProvider: - description: The provider for the CA bundle to use to validate Vault server certificate. - properties: - key: - description: The key where the CA certificate can be found in the Secret or ConfigMap. - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: |- - The namespace the Provider type is in. - Can only be defined when used in a ClusterSecretStore. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - forwardInconsistent: - description: |- - ForwardInconsistent tells Vault to forward read-after-write requests to the Vault - leader instead of simply retrying within a loop. This can increase performance if - the option is enabled serverside. - https://www.vaultproject.io/docs/configuration/replication#allow_forwarding_via_header - type: boolean - headers: - additionalProperties: - type: string - description: Headers to be added in Vault request - type: object - namespace: - description: |- - Name of the vault namespace. Namespaces is a set of features within Vault Enterprise that allows - Vault environments to support Secure Multi-tenancy. e.g: "ns1". - More about namespaces can be found here https://www.vaultproject.io/docs/enterprise/namespaces - type: string - path: - description: |- - Path is the mount path of the Vault KV backend endpoint, e.g: - "secret". The v2 KV secret engine version specific "/data" path suffix - for fetching secrets from Vault is optional and will be appended - if not present in specified path. - type: string - readYourWrites: - description: |- - ReadYourWrites ensures isolated read-after-write semantics by - providing discovered cluster replication states in each request. - More information about eventual consistency in Vault can be found here - https://www.vaultproject.io/docs/enterprise/consistency - type: boolean - server: - description: 'Server is the connection address for the Vault server, e.g: "https://vault.example.com:8200".' - type: string - tls: - description: |- - The configuration used for client side related TLS communication, when the Vault server - requires mutual authentication. Only used if the Server URL is using HTTPS protocol. - This parameter is ignored for plain HTTP protocol connection. - It's worth noting this configuration is different from the "TLS certificates auth method", - which is available under the `auth.cert` section. - properties: - certSecretRef: - description: |- - CertSecretRef is a certificate added to the transport layer - when communicating with the Vault server. - If no key for the Secret is specified, external-secret will default to 'tls.crt'. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - keySecretRef: - description: |- - KeySecretRef to a key in a Secret resource containing client private key - added to the transport layer when communicating with the Vault server. - If no key for the Secret is specified, external-secret will default to 'tls.key'. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - version: - default: v2 - description: |- - Version is the Vault KV secret engine version. This can be either "v1" or - "v2". Version defaults to "v2". - enum: - - v1 - - v2 - type: string - required: - - auth - - server - type: object - webhook: - description: Webhook configures this store to sync secrets using a generic templated webhook - properties: - body: - description: Body - type: string - caBundle: - description: |- - PEM encoded CA bundle used to validate webhook server certificate. Only used - if the Server URL is using HTTPS protocol. This parameter is ignored for - plain HTTP protocol connection. If not set the system root certificates - are used to validate the TLS connection. - format: byte - type: string - caProvider: - description: The provider for the CA bundle to use to validate webhook server certificate. - properties: - key: - description: The key the value inside of the provider type to use, only used with "Secret" type - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: The namespace the Provider type is in. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - headers: - additionalProperties: - type: string - description: Headers - type: object - method: - description: Webhook Method - type: string - result: - description: Result formatting - properties: - jsonPath: - description: Json path of return value - type: string - type: object - secrets: - description: |- - Secrets to fill in templates - These secrets will be passed to the templating function as key value pairs under the given name - items: - properties: - name: - description: Name of this secret in templates - type: string - secretRef: - description: Secret ref to fill in credentials - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - name - - secretRef - type: object - type: array - timeout: - description: Timeout - type: string - url: - description: Webhook url to call - type: string - required: - - result - - url - type: object - yandexcertificatemanager: - description: YandexCertificateManager configures this store to sync secrets using Yandex Certificate Manager provider - properties: - apiEndpoint: - description: Yandex.Cloud API endpoint (e.g. 'api.cloud.yandex.net:443') - type: string - auth: - description: Auth defines the information necessary to authenticate against Yandex Certificate Manager - properties: - authorizedKeySecretRef: - description: The authorized key used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - caProvider: - description: The provider for the CA bundle to use to validate Yandex.Cloud server certificate. - properties: - certSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - auth - type: object - yandexlockbox: - description: YandexLockbox configures this store to sync secrets using Yandex Lockbox provider - properties: - apiEndpoint: - description: Yandex.Cloud API endpoint (e.g. 'api.cloud.yandex.net:443') - type: string - auth: - description: Auth defines the information necessary to authenticate against Yandex Lockbox - properties: - authorizedKeySecretRef: - description: The authorized key used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - caProvider: - description: The provider for the CA bundle to use to validate Yandex.Cloud server certificate. - properties: - certSecretRef: - description: |- - A reference to a specific 'key' within a Secret resource, - In some instances, `key` is a required field. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - required: - - auth - type: object - type: object - refreshInterval: - description: Used to configure store refresh interval in seconds. Empty or 0 will default to the controller config. - type: integer - retrySettings: - description: Used to configure http retries if failed - properties: - maxRetries: - format: int32 - type: integer - retryInterval: - type: string - type: object - required: - - provider - type: object - status: - description: SecretStoreStatus defines the observed state of the SecretStore. - properties: - capabilities: - description: SecretStoreCapabilities defines the possible operations a SecretStore can do. - type: string - conditions: - items: - properties: - lastTransitionTime: - format: date-time - type: string - message: - type: string - reason: - type: string - status: - type: string - type: - type: string - required: - - status - - type - type: object - type: array - type: object - type: object - served: true - storage: true - subresources: - status: {} -{{- if .Values.crds.conversion.enabled }} - conversion: - strategy: Webhook - webhook: - conversionReviewVersions: - - v1 - clientConfig: - service: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ .Release.Namespace | quote }} - path: /convert -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/crds/vaultdynamicsecret.yaml b/charts/external-secrets/external-secrets/templates/crds/vaultdynamicsecret.yaml deleted file mode 100644 index 816c3a5..0000000 --- a/charts/external-secrets/external-secrets/templates/crds/vaultdynamicsecret.yaml +++ /dev/null @@ -1,707 +0,0 @@ -{{- if .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - {{- with .Values.crds.annotations }} - {{- toYaml . | nindent 4}} - {{- end }} - {{- if and .Values.crds.conversion.enabled .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} - controller-gen.kubebuilder.io/version: v0.16.1 - labels: - external-secrets.io/component: controller - name: vaultdynamicsecrets.generators.external-secrets.io -spec: - group: generators.external-secrets.io - names: - categories: - - vaultdynamicsecret - kind: VaultDynamicSecret - listKind: VaultDynamicSecretList - plural: vaultdynamicsecrets - shortNames: - - vaultdynamicsecret - singular: vaultdynamicsecret - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - properties: - controller: - description: |- - Used to select the correct ESO controller (think: ingress.ingressClassName) - The ESO controller is instantiated with a specific controller name and filters VDS based on this property - type: string - method: - description: Vault API method to use (GET/POST/other) - type: string - parameters: - description: Parameters to pass to Vault write (for non-GET methods) - x-kubernetes-preserve-unknown-fields: true - path: - description: Vault path to obtain the dynamic secret from - type: string - provider: - description: Vault provider common spec - properties: - auth: - description: Auth configures how secret-manager authenticates with the Vault server. - properties: - appRole: - description: |- - AppRole authenticates with Vault using the App Role auth mechanism, - with the role and secret stored in a Kubernetes Secret resource. - properties: - path: - default: approle - description: |- - Path where the App Role authentication backend is mounted - in Vault, e.g: "approle" - type: string - roleId: - description: |- - RoleID configured in the App Role authentication backend when setting - up the authentication backend in Vault. - type: string - roleRef: - description: |- - Reference to a key in a Secret that contains the App Role ID used - to authenticate with Vault. - The `key` field must be specified and denotes which entry within the Secret - resource is used as the app role id. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretRef: - description: |- - Reference to a key in a Secret that contains the App Role secret used - to authenticate with Vault. - The `key` field must be specified and denotes which entry within the Secret - resource is used as the app role secret. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - path - - secretRef - type: object - cert: - description: |- - Cert authenticates with TLS Certificates by passing client certificate, private key and ca certificate - Cert authentication method - properties: - clientCert: - description: |- - ClientCert is a certificate to authenticate using the Cert Vault - authentication method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretRef: - description: |- - SecretRef to a key in a Secret resource containing client private key to - authenticate with Vault using the Cert authentication method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - iam: - description: |- - Iam authenticates with vault by passing a special AWS request signed with AWS IAM credentials - AWS IAM authentication method - properties: - externalID: - description: AWS External ID set on assumed IAM roles - type: string - jwt: - description: Specify a service account with IRSA enabled - properties: - serviceAccountRef: - description: A reference to a ServiceAccount resource. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - type: object - path: - description: 'Path where the AWS auth method is enabled in Vault, e.g: "aws"' - type: string - region: - description: AWS region - type: string - role: - description: This is the AWS role to be assumed before talking to vault - type: string - secretRef: - description: Specify credentials in a Secret object - properties: - accessKeyIDSecretRef: - description: The AccessKeyID is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - secretAccessKeySecretRef: - description: The SecretAccessKey is used for authentication - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - sessionTokenSecretRef: - description: |- - The SessionToken used for authentication - This must be defined if AccessKeyID and SecretAccessKey are temporary credentials - see: https://docs.aws.amazon.com/IAM/latest/UserGuide/id_credentials_temp_use-resources.html - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - vaultAwsIamServerID: - description: 'X-Vault-AWS-IAM-Server-ID is an additional header used by Vault IAM auth method to mitigate against different types of replay attacks. More details here: https://developer.hashicorp.com/vault/docs/auth/aws' - type: string - vaultRole: - description: Vault Role. In vault, a role describes an identity with a set of permissions, groups, or policies you want to attach a user of the secrets engine - type: string - required: - - vaultRole - type: object - jwt: - description: |- - Jwt authenticates with Vault by passing role and JWT token using the - JWT/OIDC authentication method - properties: - kubernetesServiceAccountToken: - description: |- - Optional ServiceAccountToken specifies the Kubernetes service account for which to request - a token for with the `TokenRequest` API. - properties: - audiences: - description: |- - Optional audiences field that will be used to request a temporary Kubernetes service - account token for the service account referenced by `serviceAccountRef`. - Defaults to a single audience `vault` it not specified. - Deprecated: use serviceAccountRef.Audiences instead - items: - type: string - type: array - expirationSeconds: - description: |- - Optional expiration time in seconds that will be used to request a temporary - Kubernetes service account token for the service account referenced by - `serviceAccountRef`. - Deprecated: this will be removed in the future. - Defaults to 10 minutes. - format: int64 - type: integer - serviceAccountRef: - description: Service account field containing the name of a kubernetes ServiceAccount. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - serviceAccountRef - type: object - path: - default: jwt - description: |- - Path where the JWT authentication backend is mounted - in Vault, e.g: "jwt" - type: string - role: - description: |- - Role is a JWT role to authenticate using the JWT/OIDC Vault - authentication method - type: string - secretRef: - description: |- - Optional SecretRef that refers to a key in a Secret resource containing JWT token to - authenticate with Vault using the JWT/OIDC authentication method. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - required: - - path - type: object - kubernetes: - description: |- - Kubernetes authenticates with Vault by passing the ServiceAccount - token stored in the named Secret resource to the Vault server. - properties: - mountPath: - default: kubernetes - description: |- - Path where the Kubernetes authentication backend is mounted in Vault, e.g: - "kubernetes" - type: string - role: - description: |- - A required field containing the Vault Role to assume. A Role binds a - Kubernetes ServiceAccount with a set of Vault policies. - type: string - secretRef: - description: |- - Optional secret field containing a Kubernetes ServiceAccount JWT used - for authenticating with Vault. If a name is specified without a key, - `token` is the default. If one is not specified, the one bound to - the controller will be used. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - serviceAccountRef: - description: |- - Optional service account field containing the name of a kubernetes ServiceAccount. - If the service account is specified, the service account secret token JWT will be used - for authenticating with Vault. If the service account selector is not supplied, - the secretRef will be used instead. - properties: - audiences: - description: |- - Audience specifies the `aud` claim for the service account token - If the service account uses a well-known annotation for e.g. IRSA or GCP Workload Identity - then this audiences will be appended to the list - items: - type: string - type: array - name: - description: The name of the ServiceAccount resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - required: - - name - type: object - required: - - mountPath - - role - type: object - ldap: - description: |- - Ldap authenticates with Vault by passing username/password pair using - the LDAP authentication method - properties: - path: - default: ldap - description: |- - Path where the LDAP authentication backend is mounted - in Vault, e.g: "ldap" - type: string - secretRef: - description: |- - SecretRef to a key in a Secret resource containing password for the LDAP - user used to authenticate with Vault using the LDAP authentication - method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - username: - description: |- - Username is a LDAP user name used to authenticate using the LDAP Vault - authentication method - type: string - required: - - path - - username - type: object - namespace: - description: |- - Name of the vault namespace to authenticate to. This can be different than the namespace your secret is in. - Namespaces is a set of features within Vault Enterprise that allows - Vault environments to support Secure Multi-tenancy. e.g: "ns1". - More about namespaces can be found here https://www.vaultproject.io/docs/enterprise/namespaces - This will default to Vault.Namespace field if set, or empty otherwise - type: string - tokenSecretRef: - description: TokenSecretRef authenticates with Vault by presenting a token. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - userPass: - description: UserPass authenticates with Vault by passing username/password pair - properties: - path: - default: user - description: |- - Path where the UserPassword authentication backend is mounted - in Vault, e.g: "user" - type: string - secretRef: - description: |- - SecretRef to a key in a Secret resource containing password for the - user used to authenticate with Vault using the UserPass authentication - method - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - username: - description: |- - Username is a user name used to authenticate using the UserPass Vault - authentication method - type: string - required: - - path - - username - type: object - type: object - caBundle: - description: |- - PEM encoded CA bundle used to validate Vault server certificate. Only used - if the Server URL is using HTTPS protocol. This parameter is ignored for - plain HTTP protocol connection. If not set the system root certificates - are used to validate the TLS connection. - format: byte - type: string - caProvider: - description: The provider for the CA bundle to use to validate Vault server certificate. - properties: - key: - description: The key where the CA certificate can be found in the Secret or ConfigMap. - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: |- - The namespace the Provider type is in. - Can only be defined when used in a ClusterSecretStore. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - forwardInconsistent: - description: |- - ForwardInconsistent tells Vault to forward read-after-write requests to the Vault - leader instead of simply retrying within a loop. This can increase performance if - the option is enabled serverside. - https://www.vaultproject.io/docs/configuration/replication#allow_forwarding_via_header - type: boolean - headers: - additionalProperties: - type: string - description: Headers to be added in Vault request - type: object - namespace: - description: |- - Name of the vault namespace. Namespaces is a set of features within Vault Enterprise that allows - Vault environments to support Secure Multi-tenancy. e.g: "ns1". - More about namespaces can be found here https://www.vaultproject.io/docs/enterprise/namespaces - type: string - path: - description: |- - Path is the mount path of the Vault KV backend endpoint, e.g: - "secret". The v2 KV secret engine version specific "/data" path suffix - for fetching secrets from Vault is optional and will be appended - if not present in specified path. - type: string - readYourWrites: - description: |- - ReadYourWrites ensures isolated read-after-write semantics by - providing discovered cluster replication states in each request. - More information about eventual consistency in Vault can be found here - https://www.vaultproject.io/docs/enterprise/consistency - type: boolean - server: - description: 'Server is the connection address for the Vault server, e.g: "https://vault.example.com:8200".' - type: string - tls: - description: |- - The configuration used for client side related TLS communication, when the Vault server - requires mutual authentication. Only used if the Server URL is using HTTPS protocol. - This parameter is ignored for plain HTTP protocol connection. - It's worth noting this configuration is different from the "TLS certificates auth method", - which is available under the `auth.cert` section. - properties: - certSecretRef: - description: |- - CertSecretRef is a certificate added to the transport layer - when communicating with the Vault server. - If no key for the Secret is specified, external-secret will default to 'tls.crt'. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - keySecretRef: - description: |- - KeySecretRef to a key in a Secret resource containing client private key - added to the transport layer when communicating with the Vault server. - If no key for the Secret is specified, external-secret will default to 'tls.key'. - properties: - key: - description: |- - The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be - defaulted, in others it may be required. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - namespace: - description: |- - Namespace of the resource being referred to. Ignored if referent is not cluster-scoped. cluster-scoped defaults - to the namespace of the referent. - type: string - type: object - type: object - version: - default: v2 - description: |- - Version is the Vault KV secret engine version. This can be either "v1" or - "v2". Version defaults to "v2". - enum: - - v1 - - v2 - type: string - required: - - auth - - server - type: object - resultType: - default: Data - description: |- - Result type defines which data is returned from the generator. - By default it is the "data" section of the Vault API response. - When using e.g. /auth/token/create the "data" section is empty but - the "auth" section contains the generated token. - Please refer to the vault docs regarding the result data structure. - enum: - - Data - - Auth - type: string - required: - - path - - provider - type: object - type: object - served: true - storage: true - subresources: - status: {} -{{- if .Values.crds.conversion.enabled }} - conversion: - strategy: Webhook - webhook: - conversionReviewVersions: - - v1 - clientConfig: - service: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ .Release.Namespace | quote }} - path: /convert -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/crds/webhook.yaml b/charts/external-secrets/external-secrets/templates/crds/webhook.yaml deleted file mode 100644 index ac6a859..0000000 --- a/charts/external-secrets/external-secrets/templates/crds/webhook.yaml +++ /dev/null @@ -1,157 +0,0 @@ -{{- if .Values.installCRDs }} -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - {{- with .Values.crds.annotations }} - {{- toYaml . | nindent 4}} - {{- end }} - {{- if and .Values.crds.conversion.enabled .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} - controller-gen.kubebuilder.io/version: v0.16.1 - labels: - external-secrets.io/component: controller - name: webhooks.generators.external-secrets.io -spec: - group: generators.external-secrets.io - names: - categories: - - webhook - kind: Webhook - listKind: WebhookList - plural: webhooks - shortNames: - - webhookl - singular: webhook - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - Webhook connects to a third party API server to handle the secrets generation - configuration parameters in spec. - You can specify the server, the token, and additional body parameters. - See documentation for the full API specification for requests and responses. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: WebhookSpec controls the behavior of the external generator. Any body parameters should be passed to the server through the parameters field. - properties: - body: - description: Body - type: string - caBundle: - description: |- - PEM encoded CA bundle used to validate webhook server certificate. Only used - if the Server URL is using HTTPS protocol. This parameter is ignored for - plain HTTP protocol connection. If not set the system root certificates - are used to validate the TLS connection. - format: byte - type: string - caProvider: - description: The provider for the CA bundle to use to validate webhook server certificate. - properties: - key: - description: The key the value inside of the provider type to use, only used with "Secret" type - type: string - name: - description: The name of the object located at the provider type. - type: string - namespace: - description: The namespace the Provider type is in. - type: string - type: - description: The type of provider to use such as "Secret", or "ConfigMap". - enum: - - Secret - - ConfigMap - type: string - required: - - name - - type - type: object - headers: - additionalProperties: - type: string - description: Headers - type: object - method: - description: Webhook Method - type: string - result: - description: Result formatting - properties: - jsonPath: - description: Json path of return value - type: string - type: object - secrets: - description: |- - Secrets to fill in templates - These secrets will be passed to the templating function as key value pairs under the given name - items: - properties: - name: - description: Name of this secret in templates - type: string - secretRef: - description: Secret ref to fill in credentials - properties: - key: - description: The key where the token is found. - type: string - name: - description: The name of the Secret resource being referred to. - type: string - type: object - required: - - name - - secretRef - type: object - type: array - timeout: - description: Timeout - type: string - url: - description: Webhook url to call - type: string - required: - - result - - url - type: object - type: object - served: true - storage: true - subresources: - status: {} -{{- if .Values.crds.conversion.enabled }} - conversion: - strategy: Webhook - webhook: - conversionReviewVersions: - - v1 - clientConfig: - service: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ .Release.Namespace | quote }} - path: /convert -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/deployment.yaml b/charts/external-secrets/external-secrets/templates/deployment.yaml deleted file mode 100644 index 75a908e..0000000 --- a/charts/external-secrets/external-secrets/templates/deployment.yaml +++ /dev/null @@ -1,146 +0,0 @@ -{{- if .Values.createOperator }} -apiVersion: apps/v1 -kind: Deployment -metadata: - name: {{ include "external-secrets.fullname" . }} - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets.labels" . | nindent 4 }} - {{- with .Values.deploymentAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - replicas: {{ .Values.replicaCount }} - revisionHistoryLimit: {{ .Values.revisionHistoryLimit }} - selector: - matchLabels: - {{- include "external-secrets.selectorLabels" . | nindent 6 }} - template: - metadata: - {{- with .Values.podAnnotations }} - annotations: - {{- toYaml . | nindent 8 }} - {{- end }} - labels: - {{- include "external-secrets.labels" . | nindent 8 }} - {{- with .Values.podLabels }} - {{- toYaml . | nindent 8 }} - {{- end }} - spec: - {{- with .Values.imagePullSecrets }} - imagePullSecrets: - {{- toYaml . | nindent 8 }} - {{- end }} - serviceAccountName: {{ include "external-secrets.serviceAccountName" . }} - automountServiceAccountToken: {{ .Values.serviceAccount.automount }} - {{- with .Values.podSecurityContext }} - {{- if and (.enabled) (gt (keys . | len) 1) }} - securityContext: - {{- include "external-secrets.renderSecurityContext" (dict "securityContext" . "context" $) | nindent 8 }} - {{- end }} - {{- end }} - hostNetwork: {{ .Values.hostNetwork }} - containers: - - name: {{ .Chart.Name }} - {{- with .Values.securityContext }} - {{- if and (.enabled) (gt (keys . | len) 1) }} - securityContext: - {{- include "external-secrets.renderSecurityContext" (dict "securityContext" . "context" $) | nindent 12 }} - {{- end }} - {{- end }} - image: {{ include "external-secrets.image" (dict "chartAppVersion" .Chart.AppVersion "image" .Values.image) | trim }} - imagePullPolicy: {{ .Values.image.pullPolicy }} - {{- if or (.Values.leaderElect) (.Values.scopedNamespace) (.Values.processClusterStore) (.Values.processClusterExternalSecret) (.Values.concurrent) (.Values.extraArgs) }} - args: - {{- if .Values.leaderElect }} - - --enable-leader-election=true - {{- end }} - {{- if .Values.scopedNamespace }} - - --namespace={{ .Values.scopedNamespace }} - {{- end }} - {{- if and .Values.scopedNamespace .Values.scopedRBAC }} - - --enable-cluster-store-reconciler=false - - --enable-cluster-external-secret-reconciler=false - {{- else }} - {{- if not .Values.processClusterStore }} - - --enable-cluster-store-reconciler=false - {{- end }} - {{- if not .Values.processClusterExternalSecret }} - - --enable-cluster-external-secret-reconciler=false - {{- end }} - {{- end }} - {{- if not .Values.processPushSecret }} - - --enable-push-secret-reconciler=false - {{- end }} - {{- if .Values.controllerClass }} - - --controller-class={{ .Values.controllerClass }} - {{- end }} - {{- if .Values.extendedMetricLabels }} - - --enable-extended-metric-labels={{ .Values.extendedMetricLabels }} - {{- end }} - {{- if .Values.concurrent }} - - --concurrent={{ .Values.concurrent }} - {{- end }} - {{- range $key, $value := .Values.extraArgs }} - {{- if $value }} - - --{{ $key }}={{ $value }} - {{- else }} - - --{{ $key }} - {{- end }} - {{- end }} - {{- end }} - - --metrics-addr=:{{ .Values.metrics.listen.port }} - - --loglevel={{ .Values.log.level }} - - --zap-time-encoding={{ .Values.log.timeEncoding }} - ports: - - containerPort: {{ .Values.metrics.listen.port }} - protocol: TCP - name: metrics - {{- with .Values.extraEnv }} - env: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- with .Values.resources }} - resources: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- if .Values.extraVolumeMounts }} - volumeMounts: - {{- toYaml .Values.extraVolumeMounts | nindent 12 }} - {{- end }} - {{- if .Values.extraContainers }} - {{ toYaml .Values.extraContainers | nindent 8}} - {{- end }} - dnsPolicy: {{ .Values.dnsPolicy }} - {{- if .Values.dnsConfig }} - dnsConfig: - {{- toYaml .Values.dnsConfig | nindent 8 }} - {{- end }} - {{- if .Values.extraVolumes }} - volumes: - {{- toYaml .Values.extraVolumes | nindent 8 }} - {{- end }} - {{- with .Values.nodeSelector | default .Values.global.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.affinity | default .Values.global.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.tolerations | default .Values.global.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.topologySpreadConstraints | default .Values.global.topologySpreadConstraints }} - topologySpreadConstraints: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- if .Values.priorityClassName }} - priorityClassName: {{ .Values.priorityClassName }} - {{- end }} - {{- if .Values.podSpecExtra }} - {{- toYaml .Values.podSpecExtra | nindent 6 }} - {{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/extra-manifests.yaml b/charts/external-secrets/external-secrets/templates/extra-manifests.yaml deleted file mode 100644 index 1dfe8f4..0000000 --- a/charts/external-secrets/external-secrets/templates/extra-manifests.yaml +++ /dev/null @@ -1,4 +0,0 @@ -{{- range .Values.extraObjects }} ---- -{{ include "external-secrets.render" (dict "value" . "context" $) }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/poddisruptionbudget.yaml b/charts/external-secrets/external-secrets/templates/poddisruptionbudget.yaml deleted file mode 100644 index 7b75ca3..0000000 --- a/charts/external-secrets/external-secrets/templates/poddisruptionbudget.yaml +++ /dev/null @@ -1,19 +0,0 @@ -{{- if .Values.podDisruptionBudget.enabled }} -apiVersion: policy/v1 -kind: PodDisruptionBudget -metadata: - name: {{ include "external-secrets.fullname" . }}-pdb - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets.labels" . | nindent 4 }} -spec: - {{- if .Values.podDisruptionBudget.minAvailable }} - minAvailable: {{ .Values.podDisruptionBudget.minAvailable }} - {{- end }} - {{- if .Values.podDisruptionBudget.maxUnavailable }} - maxUnavailable: {{ .Values.podDisruptionBudget.maxUnavailable }} - {{- end }} - selector: - matchLabels: - {{- include "external-secrets.selectorLabels" . | nindent 6 }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/rbac.yaml b/charts/external-secrets/external-secrets/templates/rbac.yaml deleted file mode 100644 index 4f4ab48..0000000 --- a/charts/external-secrets/external-secrets/templates/rbac.yaml +++ /dev/null @@ -1,301 +0,0 @@ -{{- if .Values.rbac.create -}} -apiVersion: rbac.authorization.k8s.io/v1 -{{- if and .Values.scopedNamespace .Values.scopedRBAC }} -kind: Role -{{- else }} -kind: ClusterRole -{{- end }} -metadata: - name: {{ include "external-secrets.fullname" . }}-controller - {{- if and .Values.scopedNamespace .Values.scopedRBAC }} - namespace: {{ .Values.scopedNamespace | quote }} - {{- end }} - labels: - {{- include "external-secrets.labels" . | nindent 4 }} -rules: - - apiGroups: - - "external-secrets.io" - resources: - - "secretstores" - - "clustersecretstores" - - "externalsecrets" - - "clusterexternalsecrets" - - "pushsecrets" - verbs: - - "get" - - "list" - - "watch" - - apiGroups: - - "external-secrets.io" - resources: - - "externalsecrets" - - "externalsecrets/status" - - "externalsecrets/finalizers" - - "secretstores" - - "secretstores/status" - - "secretstores/finalizers" - - "clustersecretstores" - - "clustersecretstores/status" - - "clustersecretstores/finalizers" - - "clusterexternalsecrets" - - "clusterexternalsecrets/status" - - "clusterexternalsecrets/finalizers" - - "pushsecrets" - - "pushsecrets/status" - - "pushsecrets/finalizers" - verbs: - - "get" - - "update" - - "patch" - - apiGroups: - - "generators.external-secrets.io" - resources: - - "acraccesstokens" - - "ecrauthorizationtokens" - - "fakes" - - "gcraccesstokens" - - "githubaccesstokens" - - "passwords" - - "vaultdynamicsecrets" - - "webhooks" - verbs: - - "get" - - "list" - - "watch" - - apiGroups: - - "" - resources: - - "serviceaccounts" - - "namespaces" - verbs: - - "get" - - "list" - - "watch" - - apiGroups: - - "" - resources: - - "configmaps" - verbs: - - "get" - - "list" - - "watch" - - apiGroups: - - "" - resources: - - "secrets" - verbs: - - "get" - - "list" - - "watch" - - "create" - - "update" - - "delete" - - "patch" - - apiGroups: - - "" - resources: - - "serviceaccounts/token" - verbs: - - "create" - - apiGroups: - - "" - resources: - - "events" - verbs: - - "create" - - "patch" - - apiGroups: - - "external-secrets.io" - resources: - - "externalsecrets" - verbs: - - "create" - - "update" - - "delete" ---- -apiVersion: rbac.authorization.k8s.io/v1 -{{- if and .Values.scopedNamespace .Values.scopedRBAC }} -kind: Role -{{- else }} -kind: ClusterRole -{{- end }} -metadata: - name: {{ include "external-secrets.fullname" . }}-view - {{- if and .Values.scopedNamespace .Values.scopedRBAC }} - namespace: {{ .Values.scopedNamespace | quote }} - {{- end }} - labels: - {{- include "external-secrets.labels" . | nindent 4 }} - rbac.authorization.k8s.io/aggregate-to-view: "true" - rbac.authorization.k8s.io/aggregate-to-edit: "true" - rbac.authorization.k8s.io/aggregate-to-admin: "true" -rules: - - apiGroups: - - "external-secrets.io" - resources: - - "externalsecrets" - - "secretstores" - - "clustersecretstores" - - "pushsecrets" - verbs: - - "get" - - "watch" - - "list" - - apiGroups: - - "generators.external-secrets.io" - resources: - - "acraccesstokens" - - "ecrauthorizationtokens" - - "fakes" - - "gcraccesstokens" - - "githubaccesstokens" - - "passwords" - - "vaultdynamicsecrets" - - "webhooks" - verbs: - - "get" - - "watch" - - "list" ---- -apiVersion: rbac.authorization.k8s.io/v1 -{{- if and .Values.scopedNamespace .Values.scopedRBAC }} -kind: Role -{{- else }} -kind: ClusterRole -{{- end }} -metadata: - name: {{ include "external-secrets.fullname" . }}-edit - {{- if and .Values.scopedNamespace .Values.scopedRBAC }} - namespace: {{ .Values.scopedNamespace | quote }} - {{- end }} - labels: - {{- include "external-secrets.labels" . | nindent 4 }} - rbac.authorization.k8s.io/aggregate-to-edit: "true" - rbac.authorization.k8s.io/aggregate-to-admin: "true" -rules: - - apiGroups: - - "external-secrets.io" - resources: - - "externalsecrets" - - "secretstores" - - "clustersecretstores" - - "pushsecrets" - verbs: - - "create" - - "delete" - - "deletecollection" - - "patch" - - "update" - - apiGroups: - - "generators.external-secrets.io" - resources: - - "acraccesstokens" - - "ecrauthorizationtokens" - - "fakes" - - "gcraccesstokens" - - "githubaccesstokens" - - "passwords" - - "vaultdynamicsecrets" - - "webhooks" - verbs: - - "create" - - "delete" - - "deletecollection" - - "patch" - - "update" ---- -apiVersion: rbac.authorization.k8s.io/v1 -{{- if and .Values.scopedNamespace .Values.scopedRBAC }} -kind: RoleBinding -{{- else }} -kind: ClusterRoleBinding -{{- end }} -metadata: - name: {{ include "external-secrets.fullname" . }}-controller - {{- if and .Values.scopedNamespace .Values.scopedRBAC }} - namespace: {{ .Values.scopedNamespace | quote }} - {{- end }} - labels: - {{- include "external-secrets.labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - {{- if and .Values.scopedNamespace .Values.scopedRBAC }} - kind: Role - {{- else }} - kind: ClusterRole - {{- end }} - name: {{ include "external-secrets.fullname" . }}-controller -subjects: - - name: {{ include "external-secrets.serviceAccountName" . }} - namespace: {{ template "external-secrets.namespace" . }} - kind: ServiceAccount ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: {{ include "external-secrets.fullname" . }}-leaderelection - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets.labels" . | nindent 4 }} -rules: - - apiGroups: - - "" - resources: - - "configmaps" - resourceNames: - - "external-secrets-controller" - verbs: - - "get" - - "update" - - "patch" - - apiGroups: - - "" - resources: - - "configmaps" - verbs: - - "create" - - apiGroups: - - "coordination.k8s.io" - resources: - - "leases" - verbs: - - "get" - - "create" - - "update" - - "patch" ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: {{ include "external-secrets.fullname" . }}-leaderelection - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets.labels" . | nindent 4 }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: {{ include "external-secrets.fullname" . }}-leaderelection -subjects: - - kind: ServiceAccount - name: {{ include "external-secrets.serviceAccountName" . }} - namespace: {{ template "external-secrets.namespace" . }} -{{- if .Values.rbac.servicebindings.create }} ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ include "external-secrets.fullname" . }}-servicebindings - labels: - servicebinding.io/controller: "true" - {{- include "external-secrets.labels" . | nindent 4 }} -rules: - - apiGroups: - - "external-secrets.io" - resources: - - "externalsecrets" - verbs: - - "get" - - "list" - - "watch" -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/service.yaml b/charts/external-secrets/external-secrets/templates/service.yaml deleted file mode 100644 index 94859a3..0000000 --- a/charts/external-secrets/external-secrets/templates/service.yaml +++ /dev/null @@ -1,28 +0,0 @@ -{{- if .Values.metrics.service.enabled }} -apiVersion: v1 -kind: Service -metadata: - name: {{ include "external-secrets.fullname" . }}-metrics - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets.labels" . | nindent 4 }} - {{- with .Values.metrics.service.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - type: ClusterIP - {{- if .Values.service.ipFamilyPolicy }} - ipFamilyPolicy: {{ .Values.service.ipFamilyPolicy }} - {{- end }} - {{- if .Values.service.ipFamilies }} - ipFamilies: {{ .Values.service.ipFamilies | toYaml | nindent 2 }} - {{- end }} - ports: - - port: {{ .Values.metrics.service.port }} - protocol: TCP - targetPort: metrics - name: metrics - selector: - {{- include "external-secrets.selectorLabels" . | nindent 4 }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/serviceaccount.yaml b/charts/external-secrets/external-secrets/templates/serviceaccount.yaml deleted file mode 100644 index ceaa98e..0000000 --- a/charts/external-secrets/external-secrets/templates/serviceaccount.yaml +++ /dev/null @@ -1,16 +0,0 @@ -{{- if .Values.serviceAccount.create -}} -apiVersion: v1 -kind: ServiceAccount -metadata: - name: {{ include "external-secrets.serviceAccountName" . }} - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets.labels" . | nindent 4 }} - {{- with .Values.serviceAccount.extraLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.serviceAccount.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/servicemonitor.yaml b/charts/external-secrets/external-secrets/templates/servicemonitor.yaml deleted file mode 100644 index 3145179..0000000 --- a/charts/external-secrets/external-secrets/templates/servicemonitor.yaml +++ /dev/null @@ -1,164 +0,0 @@ -{{- if and ( .Capabilities.APIVersions.Has "monitoring.coreos.com/v1" ) .Values.serviceMonitor.enabled -}} -apiVersion: v1 -kind: Service -metadata: - name: {{ include "external-secrets.fullname" . }}-metrics - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets.labels" . | nindent 4 }} -spec: - type: ClusterIP - {{- if .Values.service.ipFamilyPolicy }} - ipFamilyPolicy: {{ .Values.service.ipFamilyPolicy }} - {{- end }} - {{- if .Values.service.ipFamilies }} - ipFamilies: {{ .Values.service.ipFamilies | toYaml | nindent 2 }} - {{- end }} - ports: - - port: {{ .Values.metrics.service.port }} - protocol: TCP - name: metrics - selector: - {{- include "external-secrets.selectorLabels" . | nindent 4 }} ---- -apiVersion: "monitoring.coreos.com/v1" -kind: ServiceMonitor -metadata: - labels: - {{- include "external-secrets.labels" . | nindent 4 }} -{{- if .Values.serviceMonitor.additionalLabels }} -{{ toYaml .Values.serviceMonitor.additionalLabels | indent 4 }} -{{- end }} - name: {{ include "external-secrets.fullname" . }}-metrics - namespace: {{ .Values.serviceMonitor.namespace | default (include "external-secrets.namespace" .) | quote }} -spec: - selector: - matchLabels: - {{- include "external-secrets.selectorLabels" . | nindent 6 }} - namespaceSelector: - matchNames: - - {{ template "external-secrets.namespace" . }} - endpoints: - - port: metrics - interval: {{ .Values.serviceMonitor.interval }} - scrapeTimeout: {{ .Values.serviceMonitor.scrapeTimeout }} - honorLabels: {{ .Values.serviceMonitor.honorLabels }} - {{- with .Values.serviceMonitor.metricRelabelings }} - metricRelabelings: - {{- toYaml . | nindent 6 }} - {{- end }} - {{- with .Values.serviceMonitor.relabelings }} - relabelings: - {{- toYaml . | nindent 6 }} - {{- end }} ---- -{{- if .Values.webhook.create }} -apiVersion: v1 -kind: Service -metadata: - name: {{ include "external-secrets.fullname" . }}-webhook-metrics - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets-webhook-metrics.labels" . | nindent 4 }} -spec: - type: ClusterIP - {{- if .Values.service.ipFamilyPolicy }} - ipFamilyPolicy: {{ .Values.service.ipFamilyPolicy }} - {{- end }} - {{- if .Values.service.ipFamilies }} - ipFamilies: {{ .Values.service.ipFamilies | toYaml | nindent 2 }} - {{- end }} - ports: - - port: {{ .Values.webhook.metrics.service.port }} - protocol: TCP - name: metrics - selector: - {{- include "external-secrets-webhook.selectorLabels" . | nindent 4 }} ---- -apiVersion: "monitoring.coreos.com/v1" -kind: ServiceMonitor -metadata: - labels: - {{- include "external-secrets-webhook.labels" . | nindent 4 }} -{{- if .Values.serviceMonitor.additionalLabels }} -{{ toYaml .Values.serviceMonitor.additionalLabels | indent 4 }} -{{- end }} - name: {{ include "external-secrets.fullname" . }}-webhook-metrics - namespace: {{ .Values.serviceMonitor.namespace | default (include "external-secrets.namespace" .) | quote }} -spec: - selector: - matchLabels: - {{- include "external-secrets-webhook-metrics.labels" . | nindent 6 }} - namespaceSelector: - matchNames: - - {{ template "external-secrets.namespace" . }} - endpoints: - - port: metrics - interval: {{ .Values.serviceMonitor.interval }} - scrapeTimeout: {{ .Values.serviceMonitor.scrapeTimeout }} - honorLabels: {{ .Values.serviceMonitor.honorLabels }} - {{- with .Values.serviceMonitor.metricRelabelings }} - metricRelabelings: - {{- toYaml . | nindent 6 }} - {{- end }} - {{- with .Values.serviceMonitor.relabelings }} - relabelings: - {{- toYaml . | nindent 6 }} - {{- end }} -{{- end }} ---- -{{- if .Values.certController.create }} -apiVersion: v1 -kind: Service -metadata: - name: {{ include "external-secrets.fullname" . }}-cert-controller-metrics - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets-cert-controller-metrics.labels" . | nindent 4 }} -spec: - type: ClusterIP - {{- if .Values.service.ipFamilyPolicy }} - ipFamilyPolicy: {{ .Values.service.ipFamilyPolicy }} - {{- end }} - {{- if .Values.service.ipFamilies }} - ipFamilies: {{ .Values.service.ipFamilies | toYaml | nindent 2 }} - {{- end }} - ports: - - port: {{ .Values.certController.metrics.listen.port }} - protocol: TCP - name: metrics - selector: - {{- include "external-secrets-cert-controller.selectorLabels" . | nindent 4 }} ---- -apiVersion: "monitoring.coreos.com/v1" -kind: ServiceMonitor -metadata: - labels: - {{- include "external-secrets-cert-controller.labels" . | nindent 4 }} -{{- if .Values.serviceMonitor.additionalLabels }} -{{ toYaml .Values.serviceMonitor.additionalLabels | indent 4 }} -{{- end }} - name: {{ include "external-secrets.fullname" . }}-cert-controller-metrics - namespace: {{ .Values.serviceMonitor.namespace | default (include "external-secrets.namespace" .) | quote }} -spec: - selector: - matchLabels: - {{- include "external-secrets-cert-controller-metrics.labels" . | nindent 6 }} - namespaceSelector: - matchNames: - - {{ template "external-secrets.namespace" . }} - endpoints: - - port: metrics - interval: {{ .Values.serviceMonitor.interval }} - scrapeTimeout: {{ .Values.serviceMonitor.scrapeTimeout }} - honorLabels: {{ .Values.serviceMonitor.honorLabels }} - {{- with .Values.serviceMonitor.metricRelabelings }} - metricRelabelings: - {{- toYaml . | nindent 6 }} - {{- end }} - {{- with .Values.serviceMonitor.relabelings }} - relabelings: - {{- toYaml . | nindent 6 }} - {{- end }} -{{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/validatingwebhook.yaml b/charts/external-secrets/external-secrets/templates/validatingwebhook.yaml deleted file mode 100644 index 63b3976..0000000 --- a/charts/external-secrets/external-secrets/templates/validatingwebhook.yaml +++ /dev/null @@ -1,78 +0,0 @@ -{{- if .Values.webhook.create }} -apiVersion: admissionregistration.k8s.io/v1 -kind: ValidatingWebhookConfiguration -metadata: - name: secretstore-validate - labels: - external-secrets.io/component: webhook - {{- with .Values.commonLabels }} - {{ toYaml . | nindent 4 }} - {{- end }} - {{- if and .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - annotations: - cert-manager.io/inject-ca-from: {{ template "external-secrets.namespace" . }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} -webhooks: -- name: "validate.secretstore.external-secrets.io" - rules: - - apiGroups: ["external-secrets.io"] - apiVersions: ["v1beta1"] - operations: ["CREATE", "UPDATE", "DELETE"] - resources: ["secretstores"] - scope: "Namespaced" - clientConfig: - service: - namespace: {{ template "external-secrets.namespace" . }} - name: {{ include "external-secrets.fullname" . }}-webhook - path: /validate-external-secrets-io-v1beta1-secretstore - admissionReviewVersions: ["v1", "v1beta1"] - sideEffects: None - timeoutSeconds: 5 - -- name: "validate.clustersecretstore.external-secrets.io" - rules: - - apiGroups: ["external-secrets.io"] - apiVersions: ["v1beta1"] - operations: ["CREATE", "UPDATE", "DELETE"] - resources: ["clustersecretstores"] - scope: "Cluster" - clientConfig: - service: - namespace: {{ template "external-secrets.namespace" . }} - name: {{ include "external-secrets.fullname" . }}-webhook - path: /validate-external-secrets-io-v1beta1-clustersecretstore - admissionReviewVersions: ["v1", "v1beta1"] - sideEffects: None - timeoutSeconds: 5 ---- -apiVersion: admissionregistration.k8s.io/v1 -kind: ValidatingWebhookConfiguration -metadata: - name: externalsecret-validate - labels: - external-secrets.io/component: webhook - {{- with .Values.commonLabels }} - {{ toYaml . | nindent 4 }} - {{- end }} - {{- if and .Values.webhook.certManager.enabled .Values.webhook.certManager.addInjectorAnnotations }} - annotations: - cert-manager.io/inject-ca-from: {{ template "external-secrets.namespace" . }}/{{ include "external-secrets.fullname" . }}-webhook - {{- end }} -webhooks: -- name: "validate.externalsecret.external-secrets.io" - rules: - - apiGroups: ["external-secrets.io"] - apiVersions: ["v1beta1"] - operations: ["CREATE", "UPDATE", "DELETE"] - resources: ["externalsecrets"] - scope: "Namespaced" - clientConfig: - service: - namespace: {{ template "external-secrets.namespace" . }} - name: {{ include "external-secrets.fullname" . }}-webhook - path: /validate-external-secrets-io-v1beta1-externalsecret - admissionReviewVersions: ["v1", "v1beta1"] - sideEffects: None - timeoutSeconds: 5 - failurePolicy: {{ .Values.webhook.failurePolicy}} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/webhook-certificate.yaml b/charts/external-secrets/external-secrets/templates/webhook-certificate.yaml deleted file mode 100644 index adb19fd..0000000 --- a/charts/external-secrets/external-secrets/templates/webhook-certificate.yaml +++ /dev/null @@ -1,30 +0,0 @@ -{{- if and .Values.webhook.create .Values.webhook.certManager.enabled .Values.webhook.certManager.cert.create }} ---- -apiVersion: cert-manager.io/v1 -kind: Certificate -metadata: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets-webhook.labels" . | nindent 4 }} - external-secrets.io/component: webhook - {{- with .Values.webhook.certManager.cert.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - commonName: {{ include "external-secrets.fullname" . }}-webhook - dnsNames: - - {{ include "external-secrets.fullname" . }}-webhook - - {{ include "external-secrets.fullname" . }}-webhook.{{ template "external-secrets.namespace" . }} - - {{ include "external-secrets.fullname" . }}-webhook.{{ template "external-secrets.namespace" . }}.svc - issuerRef: - {{- toYaml .Values.webhook.certManager.cert.issuerRef | nindent 4 }} - {{- with .Values.webhook.certManager.cert.duration }} - duration: {{ . | quote }} - {{- end }} - {{- with .Values.webhook.certManager.cert.renewBefore }} - renewBefore: {{ . | quote }} - {{- end }} - secretName: {{ include "external-secrets.fullname" . }}-webhook -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/webhook-deployment.yaml b/charts/external-secrets/external-secrets/templates/webhook-deployment.yaml deleted file mode 100644 index 7419a42..0000000 --- a/charts/external-secrets/external-secrets/templates/webhook-deployment.yaml +++ /dev/null @@ -1,128 +0,0 @@ -{{- if .Values.webhook.create }} -apiVersion: apps/v1 -kind: Deployment -metadata: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets-webhook.labels" . | nindent 4 }} - {{- with .Values.webhook.deploymentAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - replicas: {{ .Values.webhook.replicaCount }} - revisionHistoryLimit: {{ .Values.webhook.revisionHistoryLimit }} - selector: - matchLabels: - {{- include "external-secrets-webhook.selectorLabels" . | nindent 6 }} - template: - metadata: - {{- with .Values.webhook.podAnnotations }} - annotations: - {{- toYaml . | nindent 8 }} - {{- end }} - labels: - {{- include "external-secrets-webhook.labels" . | nindent 8 }} - {{- with .Values.webhook.podLabels }} - {{- toYaml . | nindent 8 }} - {{- end }} - spec: - {{- with .Values.webhook.imagePullSecrets }} - imagePullSecrets: - {{- toYaml . | nindent 8 }} - {{- end }} - hostNetwork: {{ .Values.webhook.hostNetwork}} - serviceAccountName: {{ include "external-secrets-webhook.serviceAccountName" . }} - automountServiceAccountToken: {{ .Values.webhook.serviceAccount.automount }} - {{- with .Values.webhook.podSecurityContext }} - {{- if and (.enabled) (gt (keys . | len) 1) }} - securityContext: - {{- include "external-secrets.renderSecurityContext" (dict "securityContext" . "context" $) | nindent 8 }} - {{- end }} - {{- end }} - containers: - - name: webhook - {{- with .Values.webhook.securityContext }} - {{- if and (.enabled) (gt (keys . | len) 1) }} - securityContext: - {{- include "external-secrets.renderSecurityContext" (dict "securityContext" . "context" $) | nindent 12 }} - {{- end }} - {{- end }} - image: {{ include "external-secrets.image" (dict "chartAppVersion" .Chart.AppVersion "image" .Values.webhook.image) | trim }} - imagePullPolicy: {{ .Values.webhook.image.pullPolicy }} - args: - - webhook - - --port={{ .Values.webhook.port }} - - --dns-name={{ include "external-secrets.fullname" . }}-webhook.{{ template "external-secrets.namespace" . }}.svc - - --cert-dir={{ .Values.webhook.certDir }} - - --check-interval={{ .Values.webhook.certCheckInterval }} - - --metrics-addr=:{{ .Values.webhook.metrics.listen.port }} - - --healthz-addr={{ .Values.webhook.readinessProbe.address }}:{{ .Values.webhook.readinessProbe.port }} - - --loglevel={{ .Values.webhook.log.level }} - - --zap-time-encoding={{ .Values.webhook.log.timeEncoding }} - {{- if .Values.webhook.lookaheadInterval }} - - --lookahead-interval={{ .Values.webhook.lookaheadInterval }} - {{- end }} - {{- range $key, $value := .Values.webhook.extraArgs }} - {{- if $value }} - - --{{ $key }}={{ $value }} - {{- else }} - - --{{ $key }} - {{- end }} - {{- end }} - ports: - - containerPort: {{ .Values.webhook.metrics.listen.port }} - protocol: TCP - name: metrics - - containerPort: {{ .Values.webhook.port }} - protocol: TCP - name: webhook - readinessProbe: - httpGet: - port: {{ .Values.webhook.readinessProbe.port }} - path: /readyz - initialDelaySeconds: 20 - periodSeconds: 5 - {{- with .Values.webhook.extraEnv }} - env: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- with .Values.webhook.resources }} - resources: - {{- toYaml . | nindent 12 }} - {{- end }} - volumeMounts: - - name: certs - mountPath: {{ .Values.webhook.certDir }} - readOnly: true - {{- if .Values.webhook.extraVolumeMounts }} - {{- toYaml .Values.webhook.extraVolumeMounts | nindent 12 }} - {{- end }} - volumes: - - name: certs - secret: - secretName: {{ include "external-secrets.fullname" . }}-webhook - {{- if .Values.webhook.extraVolumes }} - {{- toYaml .Values.webhook.extraVolumes | nindent 8 }} - {{- end }} - {{- with .Values.webhook.nodeSelector | default .Values.global.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.webhook.affinity | default .Values.global.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.webhook.tolerations | default .Values.global.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.webhook.topologySpreadConstraints | default .Values.global.topologySpreadConstraints }} - topologySpreadConstraints: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- if .Values.webhook.priorityClassName }} - priorityClassName: {{ .Values.webhook.priorityClassName }} - {{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/webhook-poddisruptionbudget.yaml b/charts/external-secrets/external-secrets/templates/webhook-poddisruptionbudget.yaml deleted file mode 100644 index 58345ba..0000000 --- a/charts/external-secrets/external-secrets/templates/webhook-poddisruptionbudget.yaml +++ /dev/null @@ -1,20 +0,0 @@ -{{- if and .Values.webhook.create .Values.webhook.podDisruptionBudget.enabled }} -apiVersion: policy/v1 -kind: PodDisruptionBudget -metadata: - name: {{ include "external-secrets.fullname" . }}-webhook-pdb - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets-webhook.labels" . | nindent 4 }} - external-secrets.io/component: webhook -spec: - {{- if .Values.webhook.podDisruptionBudget.minAvailable }} - minAvailable: {{ .Values.webhook.podDisruptionBudget.minAvailable }} - {{- end }} - {{- if .Values.webhook.podDisruptionBudget.maxUnavailable }} - maxUnavailable: {{ .Values.webhook.podDisruptionBudget.maxUnavailable }} - {{- end }} - selector: - matchLabels: - {{- include "external-secrets-webhook.selectorLabels" . | nindent 6 }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/webhook-secret.yaml b/charts/external-secrets/external-secrets/templates/webhook-secret.yaml deleted file mode 100644 index fa7760e..0000000 --- a/charts/external-secrets/external-secrets/templates/webhook-secret.yaml +++ /dev/null @@ -1,14 +0,0 @@ -{{- if and .Values.webhook.create (not .Values.webhook.certManager.enabled) }} -apiVersion: v1 -kind: Secret -metadata: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets-webhook.labels" . | nindent 4 }} - external-secrets.io/component: webhook - {{- with .Values.webhook.secretAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/webhook-service.yaml b/charts/external-secrets/external-secrets/templates/webhook-service.yaml deleted file mode 100644 index 59dbddc..0000000 --- a/charts/external-secrets/external-secrets/templates/webhook-service.yaml +++ /dev/null @@ -1,37 +0,0 @@ -{{- if .Values.webhook.create }} -apiVersion: v1 -kind: Service -metadata: - name: {{ include "external-secrets.fullname" . }}-webhook - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets-webhook.labels" . | nindent 4 }} - external-secrets.io/component: webhook - {{- if .Values.webhook.metrics.service.enabled }} - {{- with .Values.webhook.metrics.service.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} - {{- end }} -spec: - type: ClusterIP - {{- if .Values.service.ipFamilyPolicy }} - ipFamilyPolicy: {{ .Values.service.ipFamilyPolicy }} - {{- end }} - {{- if .Values.service.ipFamilies }} - ipFamilies: {{ .Values.service.ipFamilies | toYaml | nindent 2 }} - {{- end }} - ports: - - port: 443 - targetPort: {{ .Values.webhook.port }} - protocol: TCP - name: webhook - {{- if .Values.webhook.metrics.service.enabled }} - - port: {{ .Values.webhook.metrics.service.port }} - protocol: TCP - targetPort: metrics - name: metrics - {{- end }} - selector: - {{- include "external-secrets-webhook.selectorLabels" . | nindent 4 }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/templates/webhook-serviceaccount.yaml b/charts/external-secrets/external-secrets/templates/webhook-serviceaccount.yaml deleted file mode 100644 index 1936218..0000000 --- a/charts/external-secrets/external-secrets/templates/webhook-serviceaccount.yaml +++ /dev/null @@ -1,16 +0,0 @@ -{{- if and .Values.webhook.create .Values.webhook.serviceAccount.create -}} -apiVersion: v1 -kind: ServiceAccount -metadata: - name: {{ include "external-secrets-webhook.serviceAccountName" . }} - namespace: {{ template "external-secrets.namespace" . }} - labels: - {{- include "external-secrets-webhook.labels" . | nindent 4 }} - {{- with .Values.webhook.serviceAccount.extraLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.webhook.serviceAccount.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/external-secrets/external-secrets/values.yaml b/charts/external-secrets/external-secrets/values.yaml deleted file mode 100644 index 19525ad..0000000 --- a/charts/external-secrets/external-secrets/values.yaml +++ /dev/null @@ -1,532 +0,0 @@ -global: - nodeSelector: {} - tolerations: [] - topologySpreadConstraints: [] - affinity: {} - compatibility: - openshift: - # -- Manages the securityContext properties to make them compatible with OpenShift. - # Possible values: - # auto - Apply configurations if it is detected that OpenShift is the target platform. - # force - Always apply configurations. - # disabled - No modification applied. - adaptSecurityContext: auto - -replicaCount: 1 - -bitwarden-sdk-server: - enabled: false - -# -- Specifies the amount of historic ReplicaSets k8s should keep (see https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#clean-up-policy) -revisionHistoryLimit: 10 - -image: - repository: ghcr.io/external-secrets/external-secrets - pullPolicy: IfNotPresent - # -- The image tag to use. The default is the chart appVersion. - tag: "" - # -- The flavour of tag you want to use - # There are different image flavours available, like distroless and ubi. - # Please see GitHub release notes for image tags for these flavors. - # By default, the distroless image is used. - flavour: "" - -# -- If set, install and upgrade CRDs through helm chart. -installCRDs: true - -crds: - # -- If true, create CRDs for Cluster External Secret. - createClusterExternalSecret: true - # -- If true, create CRDs for Cluster Secret Store. - createClusterSecretStore: true - # -- If true, create CRDs for Push Secret. - createPushSecret: true - annotations: {} - conversion: - enabled: true - -imagePullSecrets: [] -nameOverride: "" -fullnameOverride: "" -namespaceOverride: "" - -# -- Additional labels added to all helm chart resources. -commonLabels: {} - -# -- If true, external-secrets will perform leader election between instances to ensure no more -# than one instance of external-secrets operates at a time. -leaderElect: false - -# -- If set external secrets will filter matching -# Secret Stores with the appropriate controller values. -controllerClass: "" - -# -- If true external secrets will use recommended kubernetes -# annotations as prometheus metric labels. -extendedMetricLabels: false - -# -- If set external secrets are only reconciled in the -# provided namespace -scopedNamespace: "" - -# -- Must be used with scopedNamespace. If true, create scoped RBAC roles under the scoped namespace -# and implicitly disable cluster stores and cluster external secrets -scopedRBAC: false - -# -- if true, the operator will process cluster external secret. Else, it will ignore them. -processClusterExternalSecret: true - -# -- if true, the operator will process cluster store. Else, it will ignore them. -processClusterStore: true - -# -- if true, the operator will process push secret. Else, it will ignore them. -processPushSecret: true - -# -- Specifies whether an external secret operator deployment be created. -createOperator: true - -# -- Specifies the number of concurrent ExternalSecret Reconciles external-secret executes at -# a time. -concurrent: 1 -# -- Specifices Log Params to the Webhook -log: - level: info - timeEncoding: epoch -service: - # -- Set the ip family policy to configure dual-stack see [Configure dual-stack](https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services) - ipFamilyPolicy: "" - # -- Sets the families that should be supported and the order in which they should be applied to ClusterIP as well. Can be IPv4 and/or IPv6. - ipFamilies: [] - -serviceAccount: - # -- Specifies whether a service account should be created. - create: true - # -- Automounts the service account token in all containers of the pod - automount: true - # -- Annotations to add to the service account. - annotations: {} - # -- Extra Labels to add to the service account. - extraLabels: {} - # -- The name of the service account to use. - # If not set and create is true, a name is generated using the fullname template. - name: "" - -rbac: - # -- Specifies whether role and rolebinding resources should be created. - create: true - - servicebindings: - # -- Specifies whether a clusterrole to give servicebindings read access should be created. - create: true - -## -- Extra environment variables to add to container. -extraEnv: [] - -## -- Map of extra arguments to pass to container. -extraArgs: {} - -## -- Extra volumes to pass to pod. -extraVolumes: [] - -## -- Extra Kubernetes objects to deploy with the helm chart -extraObjects: [] - -## -- Extra volumes to mount to the container. -extraVolumeMounts: [] - -## -- Extra containers to add to the pod. -extraContainers: [] - -# -- Annotations to add to Deployment -deploymentAnnotations: {} - -# -- Annotations to add to Pod -podAnnotations: {} - -podLabels: {} - -podSecurityContext: - enabled: true - # fsGroup: 2000 - -securityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - enabled: true - readOnlyRootFilesystem: true - runAsNonRoot: true - runAsUser: 1000 - seccompProfile: - type: RuntimeDefault - -resources: {} - # requests: - # cpu: 10m - # memory: 32Mi - -serviceMonitor: - # -- Specifies whether to create a ServiceMonitor resource for collecting Prometheus metrics - enabled: false - - # -- namespace where you want to install ServiceMonitors - namespace: "" - - # -- Additional labels - additionalLabels: {} - - # -- Interval to scrape metrics - interval: 30s - - # -- Timeout if metrics can't be retrieved in given time interval - scrapeTimeout: 25s - - # -- Let prometheus add an exported_ prefix to conflicting labels - honorLabels: false - - # -- Metric relabel configs to apply to samples before ingestion. [Metric Relabeling](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#metric_relabel_configs) - metricRelabelings: [] - # - action: replace - # regex: (.*) - # replacement: $1 - # sourceLabels: - # - exported_namespace - # targetLabel: namespace - - # -- Relabel configs to apply to samples before ingestion. [Relabeling](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#relabel_config) - relabelings: [] - # - sourceLabels: [__meta_kubernetes_pod_node_name] - # separator: ; - # regex: ^(.*)$ - # targetLabel: nodename - # replacement: $1 - # action: replace - -metrics: - - listen: - port: 8080 - - service: - # -- Enable if you use another monitoring tool than Prometheus to scrape the metrics - enabled: false - - # -- Metrics service port to scrape - port: 8080 - - # -- Additional service annotations - annotations: {} - -nodeSelector: {} - -tolerations: [] - -topologySpreadConstraints: [] - -affinity: {} - -# -- Pod priority class name. -priorityClassName: "" - -# -- Pod disruption budget - for more details see https://kubernetes.io/docs/concepts/workloads/pods/disruptions/ -podDisruptionBudget: - enabled: false - minAvailable: 1 - # maxUnavailable: 1 - -# -- Run the controller on the host network -hostNetwork: false - -webhook: - # -- Specifies whether a webhook deployment be created. - create: true - # -- Specifices the time to check if the cert is valid - certCheckInterval: "5m" - # -- Specifices the lookaheadInterval for certificate validity - lookaheadInterval: "" - replicaCount: 1 - # -- Specifices Log Params to the Webhook - log: - level: info - timeEncoding: epoch - # -- Specifies the amount of historic ReplicaSets k8s should keep (see https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#clean-up-policy) - revisionHistoryLimit: 10 - - certDir: /tmp/certs - # -- Specifies whether validating webhooks should be created with failurePolicy: Fail or Ignore - failurePolicy: Fail - # -- Specifies if webhook pod should use hostNetwork or not. - hostNetwork: false - image: - repository: ghcr.io/external-secrets/external-secrets - pullPolicy: IfNotPresent - # -- The image tag to use. The default is the chart appVersion. - tag: "" - # -- The flavour of tag you want to use - flavour: "" - imagePullSecrets: [] - nameOverride: "" - fullnameOverride: "" - # -- The port the webhook will listen to - port: 10250 - rbac: - # -- Specifies whether role and rolebinding resources should be created. - create: true - serviceAccount: - # -- Specifies whether a service account should be created. - create: true - # -- Automounts the service account token in all containers of the pod - automount: true - # -- Annotations to add to the service account. - annotations: {} - # -- Extra Labels to add to the service account. - extraLabels: {} - # -- The name of the service account to use. - # If not set and create is true, a name is generated using the fullname template. - name: "" - nodeSelector: {} - - certManager: - # -- Enabling cert-manager support will disable the built in secret and - # switch to using cert-manager (installed separately) to automatically issue - # and renew the webhook certificate. This chart does not install - # cert-manager for you, See https://cert-manager.io/docs/ - enabled: false - # -- Automatically add the cert-manager.io/inject-ca-from annotation to the - # webhooks and CRDs. As long as you have the cert-manager CA Injector - # enabled, this will automatically setup your webhook's CA to the one used - # by cert-manager. See https://cert-manager.io/docs/concepts/ca-injector - addInjectorAnnotations: true - cert: - # -- Create a certificate resource within this chart. See - # https://cert-manager.io/docs/usage/certificate/ - create: true - # -- For the Certificate created by this chart, setup the issuer. See - # https://cert-manager.io/docs/reference/api-docs/#cert-manager.io/v1.IssuerSpec - issuerRef: - group: cert-manager.io - kind: "Issuer" - name: "my-issuer" - # -- Set the requested duration (i.e. lifetime) of the Certificate. See - # https://cert-manager.io/docs/reference/api-docs/#cert-manager.io/v1.CertificateSpec - # One year by default. - duration: "8760h" - # -- How long before the currently issued certificate’s expiry - # cert-manager should renew the certificate. See - # https://cert-manager.io/docs/reference/api-docs/#cert-manager.io/v1.CertificateSpec - # Note that renewBefore should be greater than .webhook.lookaheadInterval - # since the webhook will check this far in advance that the certificate is - # valid. - renewBefore: "" - # -- Add extra annotations to the Certificate resource. - annotations: {} - - tolerations: [] - - topologySpreadConstraints: [] - - affinity: {} - - # -- Pod priority class name. - priorityClassName: "" - - # -- Pod disruption budget - for more details see https://kubernetes.io/docs/concepts/workloads/pods/disruptions/ - podDisruptionBudget: - enabled: false - minAvailable: 1 - # maxUnavailable: 1 - - metrics: - - listen: - port: 8080 - - service: - # -- Enable if you use another monitoring tool than Prometheus to scrape the metrics - enabled: false - - # -- Metrics service port to scrape - port: 8080 - - # -- Additional service annotations - annotations: {} - - - readinessProbe: - # -- Address for readiness probe - address: "" - # -- ReadinessProbe port for kubelet - port: 8081 - - - ## -- Extra environment variables to add to container. - extraEnv: [] - - ## -- Map of extra arguments to pass to container. - extraArgs: {} - - ## -- Extra volumes to pass to pod. - extraVolumes: [] - - ## -- Extra volumes to mount to the container. - extraVolumeMounts: [] - - # -- Annotations to add to Secret - secretAnnotations: {} - - # -- Annotations to add to Deployment - deploymentAnnotations: {} - - # -- Annotations to add to Pod - podAnnotations: {} - - podLabels: {} - - podSecurityContext: - enabled: true - # fsGroup: 2000 - - securityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - enabled: true - readOnlyRootFilesystem: true - runAsNonRoot: true - runAsUser: 1000 - seccompProfile: - type: RuntimeDefault - - resources: {} - # requests: - # cpu: 10m - # memory: 32Mi - -certController: - # -- Specifies whether a certificate controller deployment be created. - create: true - requeueInterval: "5m" - replicaCount: 1 - # -- Specifices Log Params to the Webhook - log: - level: info - timeEncoding: epoch - # -- Specifies the amount of historic ReplicaSets k8s should keep (see https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#clean-up-policy) - revisionHistoryLimit: 10 - - image: - repository: ghcr.io/external-secrets/external-secrets - pullPolicy: IfNotPresent - tag: "" - flavour: "" - imagePullSecrets: [] - nameOverride: "" - fullnameOverride: "" - rbac: - # -- Specifies whether role and rolebinding resources should be created. - create: true - serviceAccount: - # -- Specifies whether a service account should be created. - create: true - # -- Automounts the service account token in all containers of the pod - automount: true - # -- Annotations to add to the service account. - annotations: {} - # -- Extra Labels to add to the service account. - extraLabels: {} - # -- The name of the service account to use. - # If not set and create is true, a name is generated using the fullname template. - name: "" - nodeSelector: {} - - tolerations: [] - - topologySpreadConstraints: [] - - affinity: {} - - # -- Run the certController on the host network - hostNetwork: false - - # -- Pod priority class name. - priorityClassName: "" - - # -- Pod disruption budget - for more details see https://kubernetes.io/docs/concepts/workloads/pods/disruptions/ - podDisruptionBudget: - enabled: false - minAvailable: 1 - # maxUnavailable: 1 - - metrics: - - listen: - port: 8080 - - service: - # -- Enable if you use another monitoring tool than Prometheus to scrape the metrics - enabled: false - - # -- Metrics service port to scrape - port: 8080 - - # -- Additional service annotations - annotations: {} - - readinessProbe: - # -- Address for readiness probe - address: "" - # -- ReadinessProbe port for kubelet - port: 8081 - - ## -- Extra environment variables to add to container. - extraEnv: [] - - ## -- Map of extra arguments to pass to container. - extraArgs: {} - - - ## -- Extra volumes to pass to pod. - extraVolumes: [] - - ## -- Extra volumes to mount to the container. - extraVolumeMounts: [] - - # -- Annotations to add to Deployment - deploymentAnnotations: {} - - # -- Annotations to add to Pod - podAnnotations: {} - - podLabels: {} - - podSecurityContext: - enabled: true - # fsGroup: 2000 - - securityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - enabled: true - readOnlyRootFilesystem: true - runAsNonRoot: true - runAsUser: 1000 - seccompProfile: - type: RuntimeDefault - - resources: {} - # requests: - # cpu: 10m - # memory: 32Mi - -# -- Specifies `dnsPolicy` to deployment -dnsPolicy: ClusterFirst - -# -- Specifies `dnsOptions` to deployment -dnsConfig: {} - -# -- Any extra pod spec on the deployment -podSpecExtra: {} diff --git a/charts/external-secrets/values.yaml b/charts/external-secrets/values.yaml deleted file mode 100644 index 5b0b68c..0000000 --- a/charts/external-secrets/values.yaml +++ /dev/null @@ -1,5 +0,0 @@ -fullnameOverride: external-secrets -certController: - fullnameOverride: external-secrets -serviceAccount: - create: true \ No newline at end of file diff --git a/charts/fp/Chart.yaml b/charts/fp/Chart.yaml deleted file mode 100644 index 68991b3..0000000 --- a/charts/fp/Chart.yaml +++ /dev/null @@ -1,18 +0,0 @@ -apiVersion: v1 -name: fp -description: The Galaxy's Best VTuber Hentai Site -version: 0.0.54 -keywords: - - fp - - futureporn - - hentai - - porn - - r18 - - projektmelody - - archive - - chaturbate - - fansly - - vod - - nsfw -sources: [] -home: 'https://gitea.futureporn.net/futureporn' diff --git a/charts/fp/README.md b/charts/fp/README.md deleted file mode 100644 index cdbf4eb..0000000 --- a/charts/fp/README.md +++ /dev/null @@ -1,5 +0,0 @@ -# Futureporn helm chart - -This is the chart with templates that define Futureporn kubernetes cluster - -https://helm.sh/docs/topics/charts/ \ No newline at end of file diff --git a/charts/fp/templates-staging/README.md b/charts/fp/templates-staging/README.md deleted file mode 100644 index 77c4fb8..0000000 --- a/charts/fp/templates-staging/README.md +++ /dev/null @@ -1,3 +0,0 @@ -templates that I may or may not use soon. - -(@todo probably delete this entire dir) \ No newline at end of file diff --git a/charts/fp/templates-staging/bot.yaml b/charts/fp/templates-staging/bot.yaml deleted file mode 100644 index b74e2ef..0000000 --- a/charts/fp/templates-staging/bot.yaml +++ /dev/null @@ -1,69 +0,0 @@ ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: bot - namespace: futureporn - labels: - app.kubernetes.io/name: bot -spec: - replicas: {{ .Values.bot.replicas }} - selector: - matchLabels: - app: bot - template: - metadata: - labels: - app: bot - spec: - containers: - - name: bot - image: "{{ .Values.bot.imageName }}" - env: - - name: SCOUT_URL - value: "{{ .Values.scout.url }}" - - name: POSTGREST_URL - value: "{{ .Values.postgrest.url }}" - - name: NODE_ENV - value: production - - name: AUTOMATION_USER_JWT - valueFrom: - secretKeyRef: - name: bot - key: automationUserJwt - - name: DISCORD_TOKEN - valueFrom: - secretKeyRef: - name: bot - key: discordToken - - name: DISCORD_APPLICATION_ID - valueFrom: - secretKeyRef: - name: bot - key: discordApplicationId - - name: DISCORD_CHANNEL_ID - valueFrom: - secretKeyRef: - name: bot - key: discordChannelId - - name: DISCORD_GUILD_ID - valueFrom: - secretKeyRef: - name: bot - key: discordGuildId - - name: WORKER_CONNECTION_STRING - valueFrom: - secretKeyRef: - name: bot - key: workerConnectionString - - name: HTTP_PROXY - valueFrom: - secretKeyRef: - name: capture - key: httpProxy - resources: - limits: - cpu: 150m - memory: 512Mi - restartPolicy: Always - diff --git a/charts/fp/templates-staging/capture.yaml b/charts/fp/templates-staging/capture.yaml deleted file mode 100644 index d8daee3..0000000 --- a/charts/fp/templates-staging/capture.yaml +++ /dev/null @@ -1,108 +0,0 @@ - - ---- -apiVersion: v1 -kind: PersistentVolumeClaim -metadata: - name: capture-worker-cache-pvc - namespace: futureporn -spec: - accessModes: - - ReadWriteOnce - resources: - requests: - storage: {{ .Values.capture.cache.size }} - storageClassName: {{ .Values.storageClassName }} - - ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: capture-worker - namespace: futureporn - labels: - app.kubernetes.io/name: capture-worker -spec: - replicas: {{ .Values.capture.worker.replicas }} - selector: - matchLabels: - app: capture-worker - template: - metadata: - labels: - app: capture-worker - spec: - # IDK if I need this initContainer. - # initContainers: - # - name: capture-worker-init - # image: busybox:latest - # command: - # - "/bin/mkdir" - # args: - # - "-p" - # - "/var/cache/taco-test" - # - "/tmp/test1" - # - "/test123" - # volumeMounts: - # - name: capture-worker-cache - # mountPath: "{{ .Values.capture.cache.dir }}" - containers: - - name: capture-worker - image: "{{ .Values.capture.imageName }}" - volumeMounts: - - name: capture-worker-cache - mountPath: "{{ .Values.capture.cache.dir }}" - env: - # - name: NODE_DEBUG - # value: "stream.onWriteComplete" - - name: SCOUT_URL - value: "{{ .Values.scout.url }}" - - name: FUNCTION - value: worker - - name: WORKER_CONCURRENCY - value: "1" - - name: WORKER_CONNECTION_STRING - valueFrom: - secretKeyRef: - name: capture - key: workerConnectionString - - name: AUTOMATION_USER_JWT - valueFrom: - secretKeyRef: - name: postgrest - key: automationUserJwt - - name: HTTP_PROXY - valueFrom: - secretKeyRef: - name: capture - key: httpProxy - - name: POSTGREST_URL - value: "{{ .Values.postgrest.url }}" - - name: CACHE_DIR - value: "{{ .Values.capture.cache.dir }}" - - name: S3_ENDPOINT - value: "{{ .Values.s3.endpoint }}" - - name: S3_REGION - value: "{{ .Values.s3.region }}" - - name: S3_USC_BUCKET - value: "{{ .Values.s3.buckets.usc }}" - - name: S3_ACCESS_KEY_ID - valueFrom: - secretKeyRef: - name: capture - key: s3AccessKeyId - - name: S3_SECRET_ACCESS_KEY - valueFrom: - secretKeyRef: - name: capture - key: s3SecretAccessKey - resources: - limits: - cpu: 250m - memory: 1024Mi - restartPolicy: Always - volumes: - - name: capture-worker-cache - persistentVolumeClaim: - claimName: capture-worker-cache-pvc \ No newline at end of file diff --git a/charts/fp/templates-staging/chihaya.yaml b/charts/fp/templates-staging/chihaya.yaml deleted file mode 100644 index 0d8ab54..0000000 --- a/charts/fp/templates-staging/chihaya.yaml +++ /dev/null @@ -1,39 +0,0 @@ - -# --- -# apiVersion: v1 -# kind: Service -# metadata: -# name: chihaya -# namespace: futureporn -# annotations: -# external-dns.alpha.kubernetes.io/hostname: "{{ .Values.chihaya.hostname }}" -# spec: -# type: LoadBalancer -# selector: -# app: chihaya -# ports: -# - name: http -# port: 80 -# targetPort: 80 - -# --- -# apiVersion: apps/v1 -# kind: Deployment -# metadata: -# name: chihaya -# spec: -# replicas: 1 -# selector: -# matchLabels: -# app: chihaya -# template: -# metadata: -# labels: -# app: chihaya -# spec: -# containers: -# - name: chihaya -# image: "quay.io/jzelinskie/chihaya-git:latest" -# ports: -# - containerPort: 80 - diff --git a/charts/fp/templates-staging/factory.yaml b/charts/fp/templates-staging/factory.yaml deleted file mode 100644 index 5b8d6f2..0000000 --- a/charts/fp/templates-staging/factory.yaml +++ /dev/null @@ -1,87 +0,0 @@ - ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: factory - namespace: futureporn - labels: - app.kubernetes.io/name: factory -spec: - replicas: {{ .Values.factory.replicas }} - selector: - matchLabels: - app: factory - template: - metadata: - labels: - app: factory - spec: - containers: - - name: factory - image: "{{ .Values.factory.imageName }}" - volumeMounts: - - name: capture-worker-cache - mountPath: "{{ .Values.capture.cache.dir }}" - env: - - name: WORKER_CONNECTION_STRING - valueFrom: - secretKeyRef: - name: postgrest - key: dbUri - - name: AUTOMATION_USER_JWT - valueFrom: - secretKeyRef: - name: postgrest - key: automationUserJwt - - name: PATREON_CREATOR_ACCESS_TOKEN - valueFrom: - secretKeyRef: - name: patreon - key: creatorAccessToken - - name: PATREON_CREATOR_REFRESH_TOKEN - valueFrom: - secretKeyRef: - name: patreon - key: creatorRefreshToken - - name: SUPERTOKENS_URL - value: {{ printf "https://%s" .Values.supertokens.hostname | quote }} - - name: POSTGREST_URL - value: {{ printf "https://%s" .Values.postgrest.hostname | quote }} - - name: SCOUT_URL - value: "{{ .Values.scout.url }}" - - name: CACHE_DIR - value: "{{ .Values.capture.cache.dir }}" - - name: S3_ENDPOINT - value: "{{ .Values.s3.endpoint }}" - - name: S3_REGION - value: "{{ .Values.s3.region }}" - - name: S3_MAIN_BUCKET - value: "{{ .Values.s3.buckets.main }}" - - name: S3_USC_BUCKET - value: "{{ .Values.s3.buckets.usc }}" - - name: S3_ACCESS_KEY_ID - valueFrom: - secretKeyRef: - name: capture - key: s3AccessKeyId - - name: S3_SECRET_ACCESS_KEY - valueFrom: - secretKeyRef: - name: capture - key: s3SecretAccessKey - - name: SUPERTOKENS_API_KEY - valueFrom: - secretKeyRef: - name: supertokens - key: apiKey - resources: - limits: - cpu: 250m - memory: 1Gi - restartPolicy: Always - - volumes: - - name: capture-worker-cache - persistentVolumeClaim: - claimName: capture-worker-cache-pvc \ No newline at end of file diff --git a/charts/fp/templates-staging/mailbox.yaml b/charts/fp/templates-staging/mailbox.yaml deleted file mode 100644 index 58f5841..0000000 --- a/charts/fp/templates-staging/mailbox.yaml +++ /dev/null @@ -1,59 +0,0 @@ -apiVersion: apps/v1 -kind: ReplicaSet -metadata: - name: mailbox - namespace: futureporn - labels: - app: mailbox -spec: - replicas: {{ .Values.mailbox.replicas }} - selector: - matchLabels: - app: mailbox - template: - metadata: - labels: - app: mailbox - spec: - containers: - - name: mailbox - image: "{{ .Values.mailbox.imageName }}" - imagePullPolicy: Always - ports: - - containerPort: 5000 - env: - - name: DATABASE_URL - valueFrom: - secretKeyRef: - name: mailbox - key: databaseUrl - - name: PORT - value: "{{ .Values.mailbox.port }}" - - name: IMAP_SERVER - valueFrom: - secretKeyRef: - name: mailbox - key: imapServer - - name: IMAP_PORT - valueFrom: - secretKeyRef: - name: mailbox - key: imapPort - - name: IMAP_USERNAME - valueFrom: - secretKeyRef: - name: mailbox - key: imapUsername - - name: IMAP_PASSWORD - valueFrom: - secretKeyRef: - name: mailbox - key: imapPassword - resources: - limits: - cpu: "500m" - memory: "512Mi" - requests: - cpu: "250m" - memory: "256Mi" - diff --git a/charts/fp/templates-staging/migrations-data.yaml b/charts/fp/templates-staging/migrations-data.yaml deleted file mode 100644 index 9cd4a9e..0000000 --- a/charts/fp/templates-staging/migrations-data.yaml +++ /dev/null @@ -1,21 +0,0 @@ - ---- -apiVersion: v1 -kind: Pod -metadata: - name: migrations-data - namespace: futureporn - labels: - app.kubernetes.io/name: migrations-data -spec: - containers: - - name: migrations-data - image: "{{ .Values.migrations.data.imageName }}" - resources: {} - env: - - name: DATABASE_PASSWORD - valueFrom: - secretKeyRef: - name: postgresql - key: password - restartPolicy: Never diff --git a/charts/fp/templates-staging/migrations-schema.yaml b/charts/fp/templates-staging/migrations-schema.yaml deleted file mode 100644 index 99f78e4..0000000 --- a/charts/fp/templates-staging/migrations-schema.yaml +++ /dev/null @@ -1,21 +0,0 @@ - ---- -apiVersion: v1 -kind: Pod -metadata: - name: migrations-schema - namespace: futureporn - labels: - app.kubernetes.io/name: migrations-schema -spec: - containers: - - name: migrations-schema - image: "{{ .Values.migrations.schema.imageName }}" - resources: {} - env: - - name: DATABASE_PASSWORD - valueFrom: - secretKeyRef: - name: postgresql - key: password - restartPolicy: Never diff --git a/charts/fp/templates-staging/next.yaml b/charts/fp/templates-staging/next.yaml deleted file mode 100644 index 78df14b..0000000 --- a/charts/fp/templates-staging/next.yaml +++ /dev/null @@ -1,84 +0,0 @@ - - ---- -apiVersion: v1 -kind: Pod -metadata: - name: next - namespace: futureporn - labels: - app.kubernetes.io/name: next -spec: - containers: - - name: next - image: "{{ .Values.next.imageName }}" - env: - - name: HOSTNAME - value: 0.0.0.0 - - name: NEXT_PUBLIC_UPPY_COMPANION_URL - value: "{{ .Values.uppy.url }}" - - name: NEXT_PUBLIC_POSTGREST_URL - value: {{ printf "https://%s" .Values.postgrest.hostname | quote }} - - name: NEXT_PUBLIC_URL - value: {{ printf "https://%s" .Values.next.hostname | quote }} - - name: NEXT_PUBLIC_API_DOMAIN - value: {{ .Values.next.hostname | quote }} - - name: NEXTAUTH_URL - value: {{ printf "https://%s" .Values.next.hostname | quote }} - - name: NEXTAUTH_URL_INTERNAL - value: http://next.futureporn.svc.cluster.local - - name: NEXTAUTH_SECRET - valueFrom: - secretKeyRef: - name: next - key: nextAuthSecret - - name: PATREON_CLIENT_ID - valueFrom: - secretKeyRef: - name: patreon - key: clientId - - name: PATREON_CLIENT_SECRET - valueFrom: - secretKeyRef: - name: patreon - key: clientSecret - ports: - - name: web - containerPort: 3000 - resources: {} - restartPolicy: OnFailure - - ---- -apiVersion: v1 -kind: Service -metadata: - name: next - namespace: futureporn - annotations: - external-dns.alpha.kubernetes.io/hostname: "{{ .Values.next.hostname }}" -spec: - type: LoadBalancer - selector: - app.kubernetes.io/name: next - ports: - - name: web - port: 3000 - targetPort: web - protocol: TCP - - ---- -apiVersion: cert-manager.io/v1 -kind: Certificate -metadata: - name: next - namespace: futureporn -spec: - secretName: next-tls - issuerRef: - name: "{{ .Values.certManager.issuer }}" - kind: ClusterIssuer - dnsNames: - - "{{ .Values.next.hostname }}" - diff --git a/charts/fp/templates-staging/postgres.yaml b/charts/fp/templates-staging/postgres.yaml deleted file mode 100644 index 1cd2082..0000000 --- a/charts/fp/templates-staging/postgres.yaml +++ /dev/null @@ -1,71 +0,0 @@ -apiVersion: v1 -kind: PersistentVolumeClaim -metadata: - namespace: futureporn - name: postgres -annotations: - tilt.dev/down-policy: keep -spec: - accessModes: - - ReadWriteOnce - persistentVolumeReclaimPolicy: Retain - resources: - requests: - storage: 40Gi - # we leave storageClassName blank so the default provisioner gets used - storageClassName: - - ---- -apiVersion: v1 -kind: Service -metadata: - namespace: futureporn - name: postgres -annotations: - tilt.dev/down-policy: keep -spec: - selector: - app.kubernetes.io/name: postgres - ports: - - name: db - protocol: TCP - port: 5432 - targetPort: 5432 -status: - loadBalancer: {} - ---- -apiVersion: v1 -kind: Pod -metadata: - namespace: futureporn - name: postgres - labels: - app.kubernetes.io/name: postgres -annotations: - tilt.dev/down-policy: keep -spec: - containers: - - name: postgres - image: postgres:16.0 - env: - - name: POSTGRES_PASSWORD - valueFrom: - secretKeyRef: - name: postgres - key: password - ports: - - containerPort: 5432 - resources: - limits: - cpu: 500m - memory: 1Gi - volumeMounts: - - name: postgres - mountPath: /data/postgres - restartPolicy: OnFailure - volumes: - - name: postgres - persistentVolumeClaim: - claimName: postgres \ No newline at end of file diff --git a/charts/fp/templates-staging/postgresql.yaml b/charts/fp/templates-staging/postgresql.yaml deleted file mode 100644 index 55e185a..0000000 --- a/charts/fp/templates-staging/postgresql.yaml +++ /dev/null @@ -1,53 +0,0 @@ - - - ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: postgresql - namespace: futureporn - labels: - app: postgresql -spec: - replicas: {{ .Values.postgresql.replicas }} - selector: - matchLabels: - app: postgresql - template: - metadata: - labels: - app: postgresql - spec: - containers: - - name: postgresql - image: "{{ .Values.postgresql.image }}" - ports: - - name: http - containerPort: {{ .Values.postgresql.port }} - env: - - name: POSTGRES_PASSWORD - valueFrom: - secretKeyRef: - name: postgresql - key: password - - name: POSTGRES_USER - value: futureporn - ---- -apiVersion: v1 -kind: Service -metadata: - name: postgresql - namespace: futureporn - annotations: - external-dns.alpha.kubernetes.io/hostname: "{{ .Values.postgresql.hostname }}" -spec: - type: LoadBalancer - selector: - app.kubernetes.io/name: postgresql - ports: - - name: http - port: {{ .Values.postgresql.port }} - targetPort: http - protocol: TCP diff --git a/charts/fp/templates-staging/postgrest.yaml b/charts/fp/templates-staging/postgrest.yaml deleted file mode 100644 index 60c71f4..0000000 --- a/charts/fp/templates-staging/postgrest.yaml +++ /dev/null @@ -1,81 +0,0 @@ - ---- -apiVersion: gateway.networking.k8s.io/v1 -kind: HTTPRoute -metadata: - name: postgrest-httproute - namespace: futureporn -spec: - parentRefs: - - name: example-gateway - hostnames: - - "{{ .Values.postgrest.hostname }}" - rules: - - matches: - - path: - type: PathPrefix - value: / - backendRefs: - - name: postgrest - port: {{ .Values.postgrest.port }} - - ---- -apiVersion: v1 -kind: Service -metadata: - name: postgrest - namespace: futureporn - annotations: - external-dns.alpha.kubernetes.io/hostname: "{{ .Values.postgrest.hostname }}" -spec: - type: LoadBalancer - selector: - app: postgrest - ports: - - name: http - port: {{ .Values.postgrest.port }} - targetPort: {{ .Values.postgrest.port }} - - ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: postgrest - namespace: futureporn - labels: - app: postgrest -spec: - replicas: {{ .Values.postgrest.replicas }} - selector: - matchLabels: - app: postgrest - template: - metadata: - labels: - app: postgrest - spec: - containers: - - name: postgrest - image: "{{ .Values.postgrest.image }}" - ports: - - name: http - containerPort: {{ .Values.postgrest.port }} - env: - - name: PGRST_DB_ANON_ROLE - value: web_anon - - name: PGRST_SERVER_PORT - value: "{{ .Values.postgrest.port }}" - - name: PGRST_DB_SCHEMAS - value: "{{ .Values.postgrest.schemas }}" - - name: PGRST_DB_URI - valueFrom: - secretKeyRef: - name: postgrest - key: dbUri - - name: PGRST_JWT_SECRET - valueFrom: - secretKeyRef: - name: postgrest - key: jwtSecret \ No newline at end of file diff --git a/charts/fp/templates-staging/redis.yaml b/charts/fp/templates-staging/redis.yaml deleted file mode 100644 index 7436057..0000000 --- a/charts/fp/templates-staging/redis.yaml +++ /dev/null @@ -1,48 +0,0 @@ ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: redis - namespace: futureporn - labels: - app.kubernetes.io/name: redis - app: redis -spec: - replicas: {{ .Values.redis.replicas }} - selector: - matchLabels: - app: redis - template: - metadata: - labels: - app: redis - spec: - containers: - - name: redis - image: "{{ .Values.redis.image }}" - ports: - - name: web - containerPort: 6379 - resources: - limits: - cpu: 150m - memory: 128Mi - restartPolicy: Always - - ---- -apiVersion: v1 -kind: Service -metadata: - name: redis - namespace: futureporn -spec: - type: ClusterIP - selector: - app.kubernetes.io/name: redis - ports: - - name: web - port: 6379 - targetPort: web - protocol: TCP - diff --git a/charts/fp/templates-staging/scout.yaml b/charts/fp/templates-staging/scout.yaml deleted file mode 100644 index 33dd94f..0000000 --- a/charts/fp/templates-staging/scout.yaml +++ /dev/null @@ -1,67 +0,0 @@ ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: scout - namespace: futureporn - labels: - app.kubernetes.io/name: scout -spec: - replicas: {{ .Values.scout.replicas }} - selector: - matchLabels: - app: scout - template: - metadata: - labels: - app: scout - spec: - containers: - - name: scout - image: "{{ .Values.scout.imageName }}" - ports: - - name: http - containerPort: {{ .Values.scout.port }} - env: - - name: POSTGREST_URL - value: "{{ .Values.postgrest.url }}" - - name: NODE_ENV - value: production - - name: AUTOMATION_USER_JWT - valueFrom: - secretKeyRef: - name: bot - key: automationUserJwt - - name: WORKER_CONNECTION_STRING - valueFrom: - secretKeyRef: - name: bot - key: workerConnectionString - - name: HTTP_PROXY - valueFrom: - secretKeyRef: - name: capture - key: httpProxy - - name: PORT - value: "{{ .Values.scout.port }}" - resources: - limits: - cpu: 150m - memory: 1024Mi - restartPolicy: Always - - ---- -apiVersion: v1 -kind: Service -metadata: - name: scout - namespace: futureporn -spec: - type: ClusterIP - selector: - app: scout - ports: - - name: http - port: {{ .Values.scout.port }} - targetPort: http diff --git a/charts/fp/templates-staging/supertokens.yaml b/charts/fp/templates-staging/supertokens.yaml deleted file mode 100644 index 1d44a12..0000000 --- a/charts/fp/templates-staging/supertokens.yaml +++ /dev/null @@ -1,141 +0,0 @@ - -# --- -# apiVersion: traefik.io/v1alpha1 -# kind: IngressRoute -# metadata: -# name: supertokens-http -# namespace: futureporn -# spec: -# entryPoints: -# - web -# routes: -# - match: Host(`auth.fp.sbtp.xyz`) -# kind: Rule -# middlewares: -# - name: redirect -# namespace: futureporn -# services: -# - name: supertokens -# port: 3567 - -# --- -# apiVersion: traefik.io/v1alpha1 -# kind: IngressRoute -# metadata: -# name: supertokens-https -# namespace: futureporn -# annotations: -# cert-manager.io/cluster-issuer: "{{ .Values.certManager.issuer }}" -# spec: -# entryPoints: -# - websecure -# routes: -# - match: Host(`auth.fp.sbtp.xyz`) -# kind: Rule -# services: -# - name: supertokens -# namespace: futureporn -# port: 3567 -# tls: -# secretName: supertokens-tls - -# --- -# apiVersion: batch/v1 -# kind: Job -# metadata: -# name: supertokens-database-seed -# namespace: futureporn -# spec: -# template: -# spec: -# containers: -# - name: supertokens-database-seed -# image: postgres:latest -# imagePullPolicy: IfNotPresent -# env: -# - name: DB_DATABASE -# value: postgres - -# # valueFrom: -# # secretKeyRef: -# # name: supertokens -# # key: postgresqlUri -# command: -# - /bin/sh -# - -c -# - psql -a -f CREATE DATABASE supertokens; -# restartPolicy: OnFailure - ---- -apiVersion: v1 -kind: Service -metadata: - name: supertokens - namespace: futureporn - annotations: - external-dns.alpha.kubernetes.io/hostname: "{{ .Values.supertokens.hostname }}" -spec: - type: LoadBalancer - selector: - app: supertokens - ports: - - name: http - port: {{ .Values.supertokens.port }} - targetPort: {{ .Values.supertokens.port }} - ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: supertokens - namespace: futureporn -spec: - replicas: {{ .Values.supertokens.replicas }} - selector: - matchLabels: - app: supertokens - template: - metadata: - labels: - app: supertokens - spec: - containers: - - name: supertokens - image: registry.supertokens.io/supertokens/supertokens-postgresql - ports: - - name: http - containerPort: {{ .Values.supertokens.port }} - # livenessProbe: - # httpGet: - # path: /hello - # port: http - # failureThreshold: 3 - # initialDelaySeconds: 10 - # periodSeconds: 10 - # readinessProbe: - # httpGet: - # path: /hello - # port: http - # failureThreshold: 3 - # initialDelaySeconds: 10 - # periodSeconds: 10 - # timeoutSeconds: 30 - env: - ## @see https://github.com/supertokens/supertokens-docker-postgresql/blob/master/README.md - - name: DISABLE_TELEMETRY - value: "true" - - name: SUPERTOKENS_PORT - value: "{{ .Values.supertokens.port }}" - - name: SUPERTOKENS_URL - value: {{ printf "https://%s" .Values.supertokens.hostname | quote }} - - name: POSTGRESQL_CONNECTION_URI - valueFrom: - secretKeyRef: - name: supertokens - key: postgresqlUri - - name: API_KEYS - valueFrom: - secretKeyRef: - name: supertokens - key: apiKeys - diff --git a/charts/fp/templates-staging/trigger.yaml b/charts/fp/templates-staging/trigger.yaml deleted file mode 100644 index fada408..0000000 --- a/charts/fp/templates-staging/trigger.yaml +++ /dev/null @@ -1,110 +0,0 @@ - -apiVersion: apps/v1 -kind: Deployment -metadata: - name: trigger - namespace: futureporn -spec: - replicas: {{ .Values.trigger.worker.replicas }} - minReadySeconds: 5 - strategy: - type: RollingUpdate - rollingUpdate: - maxSurge: {{ .Values.trigger.worker.replicas }} - maxUnavailable: 1 - selector: - matchLabels: - app: trigger - template: - metadata: - labels: - app: trigger - spec: - containers: - - name: trigger - image: {{ .Values.trigger.imageName }} - ## We override the entrypoint in order to apply a database schema workaround. - ## The problem is that trigger.dev uses the public postgres schema, which is a postgres default. - ## We don't want trigger.dev to use this default because there might be multiple system components using that default. - ## In order to avoid accidental data loss when two components are writing data there, we want trigger to use a non-public schema. - ## @see https://github.com/triggerdotdev/trigger.dev/issues/1026 - command: - - sh - - -c - # - echo fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ fuck @ - - sed -i s/public/triggerdotdev/ ./packages/database/prisma/migrations/20240130165343_add_composite_index_to_job_run_for_job_id_and_created_at/migration.sql && ./scripts/entrypoint.sh - imagePullPolicy: IfNotPresent - resources: - limits: - memory: 150Mi - requests: - memory: 100Mi - env: - - name: TRIGGER_TELEMETRY_DISABLED - value: "1" - - name: REDIS_PORT - value: "6379" - - name: REDIS_HOST - value: redis-master.futureporn.svc.cluster.local - - name: DATABASE_URL - valueFrom: - secretKeyRef: - name: trigger - key: databaseUrl - - name: DIRECT_URL - valueFrom: - secretKeyRef: - name: trigger - key: databaseUrl - - name: MAGIC_LINK_SECRET - valueFrom: - secretKeyRef: - name: trigger - key: magicLinkSecret - - name: SESSION_SECRET - valueFrom: - secretKeyRef: - name: trigger - key: sessionSecret - - name: ENCRYPTION_KEY - valueFrom: - secretKeyRef: - name: trigger - key: encryptionKey - - name: PROVIDER_SECRET - valueFrom: - secretKeyRef: - name: trigger - key: providerSecret - - name: COORDINATOR_SECRET - valueFrom: - secretKeyRef: - name: trigger - key: coordinatorSecret - ports: - - containerPort: 3030 - volumeMounts: - - name: trigger-data - mountPath: /mnt/trigger-data - volumes: - - name: trigger-data - emptyDir: {} - - ---- -apiVersion: v1 -kind: Service -metadata: - name: trigger - namespace: futureporn - annotations: - external-dns.alpha.kubernetes.io/hostname: "{{ .Values.trigger.hostname }}" -spec: - type: LoadBalancer - ports: - - port: 3030 - targetPort: 3030 - protocol: TCP - selector: - app: trigger - diff --git a/charts/fp/templates/.gitignore b/charts/fp/templates/.gitignore deleted file mode 100644 index df2b9aa..0000000 --- a/charts/fp/templates/.gitignore +++ /dev/null @@ -1 +0,0 @@ -strapi-app.yaml \ No newline at end of file diff --git a/charts/fp/templates/bright.yaml b/charts/fp/templates/bright.yaml deleted file mode 100644 index a90fde7..0000000 --- a/charts/fp/templates/bright.yaml +++ /dev/null @@ -1,87 +0,0 @@ - - ---- -apiVersion: v1 -kind: Pod -metadata: - name: bright - namespace: futureporn - labels: - app.kubernetes.io/name: bright -spec: - containers: - - name: bright - image: {{ .Values.bright.imageName | quote }} - env: - - name: DATABASE_URL - valueFrom: - secretKeyRef: - name: bright - key: databaseUrl - - name: SECRET_KEY_BASE - valueFrom: - secretKeyRef: - name: bright - key: secretKeyBase - - name: DATABASE_HOST - value: postgresql-primary.futureporn.svc.cluster.local - - name: PORT - value: {{ .Values.bright.port | quote }} - - name: SUPERSTREAMER_URL - value: {{ .Values.superstreamer.api.localUrl | quote }} - - name: PUBLIC_S3_ENDPOINT - value: {{ .Values.bright.s3.endpoint | quote }} - - name: SUPERSTREAMER_AUTH_TOKEN - valueFrom: - secretKeyRef: - name: superstreamer - key: authToken - - name: DATABASE_PASSWORD - valueFrom: - secretKeyRef: - name: postgresql - key: password - - name: DATABASE - value: bright - - name: MIX_ENV - value: dev - ports: - - name: web - containerPort: {{ .Values.bright.port }} - resources: {} - restartPolicy: OnFailure - - ---- -apiVersion: v1 -kind: Service -metadata: - name: bright - namespace: futureporn - annotations: - external-dns.alpha.kubernetes.io/hostname: "{{ .Values.bright.hostname }}" -spec: - type: LoadBalancer - selector: - app.kubernetes.io/name: bright - ports: - - name: web - port: {{ .Values.bright.port }} - targetPort: web - protocol: TCP - - ---- -apiVersion: cert-manager.io/v1 -kind: Certificate -metadata: - name: bright - namespace: futureporn -spec: - secretName: bright-tls - issuerRef: - name: "{{ .Values.certManager.issuer }}" - kind: ClusterIssuer - dnsNames: - - "{{ .Values.bright.hostname }}" - diff --git a/charts/fp/templates/cert-manager.yaml b/charts/fp/templates/cert-manager.yaml deleted file mode 100644 index c3bc9b4..0000000 --- a/charts/fp/templates/cert-manager.yaml +++ /dev/null @@ -1,101 +0,0 @@ -## roles to deal with the following error -## exoscale.acme.exoscale.com is forbidden: User "system:serviceaccount:cert-manager:cert-manager" cannot create resource "exoscale" -## cannot create resource "exoscale" in API group "acme.exoscale.com" at the cluster scope - -# I don't think I need the following commented yaml. I think the problem I was seeing was actually caused by the lack of cert-manager-webhook-exoscale - -# --- -# apiVersion: rbac.authorization.k8s.io/v1 -# kind: ClusterRole -# metadata: -# name: cert-manager-webhook-exoscale -# namespace: cert-manager -# labels: -# app: webhook -# rules: -# - apiGroups: -# - acme.exoscale.com -# resources: -# - '*' -# verbs: -# - 'create' - -# --- -# apiVersion: rbac.authorization.k8s.io/v1 -# kind: ClusterRoleBinding -# metadata: -# name: cert-manager-webhook-exoscale -# namespace: cert-manager -# labels: -# app: webhook -# roleRef: -# apiGroup: rbac.authorization.k8s.io -# kind: ClusterRole -# name: cert-manager-webhook-exoscale -# subjects: -# - apiGroup: "" -# kind: ServiceAccount -# name: cert-manager -# namespace: cert-manager - - -## Role & RoleBinding to give cert-manager access to the exoscale secret - ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: cert-manager-webhook-exoscale:secret-reader - namespace: cert-manager -rules: -- apiGroups: [""] - resources: ["secrets"] - resourceNames: ["exoscale"] - verbs: ["get", "watch"] - ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: cert-manager-webhook-exoscale:secret-reader - namespace: cert-manager -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: cert-manager-webhook-exoscale:secret-reader -subjects: - - apiGroup: "" - kind: ServiceAccount - name: cert-manager-webhook-exoscale - ---- -{{ if eq .Values.environment "development" }} -apiVersion: cert-manager.io/v1 -kind: ClusterIssuer -metadata: - name: letsencrypt-staging -spec: - acme: - # You must replace this email address with your own. - # Let's Encrypt will use this to contact you about expiring - # certificates, and issues related to your account. - email: {{ .Values.adminEmail }} - server: https://acme-staging-v02.api.letsencrypt.org/directory - privateKeySecretRef: - # Secret resource that will be used to store the account's private key. - name: letsencrypt-staging - solvers: - - dns01: - webhook: - groupName: acme.exoscale.com - solverName: exoscale - config: - apiKeyRef: - name: exoscale - key: apiKey - apiSecretRef: - name: exoscale - key: apiSecret -{{ end }} - -## in staging and production, ClusterIssuer is created by Flux2/Kustomize. (not Helm) \ No newline at end of file diff --git a/charts/fp/templates/chisel.yaml b/charts/fp/templates/chisel.yaml deleted file mode 100644 index e6b3a13..0000000 --- a/charts/fp/templates/chisel.yaml +++ /dev/null @@ -1,14 +0,0 @@ -{{ if ne .Values.environment "production" }} - ---- -apiVersion: chisel-operator.io/v1 -kind: ExitNode -metadata: - name: vault-exit-node - namespace: futureporn -spec: - host: {{ quote .Values.chisel.exitNode.host }} - port: 9090 - auth: chisel - -{{ end }} \ No newline at end of file diff --git a/charts/fp/templates/exoscale-webhook.yaml b/charts/fp/templates/exoscale-webhook.yaml deleted file mode 100644 index a02807f..0000000 --- a/charts/fp/templates/exoscale-webhook.yaml +++ /dev/null @@ -1,321 +0,0 @@ -## copied from https://github.com/exoscale/cert-manager-webhook-exoscale - ---- -# Source: exoscale-webhook/templates/rbac.yaml -apiVersion: v1 -kind: ServiceAccount -metadata: - name: cert-manager-webhook-exoscale - namespace: "cert-manager" - labels: - app: exoscale-webhook - chart: exoscale-webhook-0.3.0 - release: exoscale-webhook - heritage: Helm ---- -# Source: exoscale-webhook/templates/rbac.yaml -# Grant cert-manager permission to validate using our apiserver -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: cert-manager-webhook-exoscale:domain-solver - labels: - app: exoscale-webhook - chart: exoscale-webhook-0.3.0 - release: exoscale-webhook - heritage: Helm -rules: - - apiGroups: - - acme.exoscale.com - resources: - - '*' - verbs: - - 'create' ---- -# Source: exoscale-webhook/templates/rbac.yaml -# apiserver gets the auth-delegator role to delegate auth decisions to -# the core apiserver -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: cert-manager-webhook-exoscale:auth-delegator - labels: - app: exoscale-webhook - chart: exoscale-webhook-0.3.0 - release: exoscale-webhook - heritage: Helm -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: system:auth-delegator -subjects: - - apiGroup: "" - kind: ServiceAccount - name: cert-manager-webhook-exoscale - namespace: cert-manager ---- -# Source: exoscale-webhook/templates/rbac.yaml -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: cert-manager-webhook-exoscale:domain-solver - labels: - app: exoscale-webhook - chart: exoscale-webhook-0.3.0 - release: exoscale-webhook - heritage: Helm -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: cert-manager-webhook-exoscale:domain-solver -subjects: - - apiGroup: "" - kind: ServiceAccount - name: cert-manager - namespace: cert-manager ---- -# Source: exoscale-webhook/templates/rbac.yaml -# Grant the webhook permission to read the secrets containing the credentials -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: cert-manager-webhook-exoscale:secrets-reader - namespace: cert-manager - labels: - app: exoscale-webhook - chart: exoscale-webhook-0.3.0 - release: exoscale-webhook - heritage: Helm -rules: - - apiGroups: - - '' - resources: - - 'secrets' - verbs: - - 'get' ---- -# Source: exoscale-webhook/templates/rbac.yaml -# Grant the webhook permission to read the ConfigMap containing the Kubernetes -# apiserver's requestheader-ca-certificate. -# This ConfigMap is automatically created by the Kubernetes apiserver. -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: cert-manager-webhook-exoscale:webhook-authentication-reader - namespace: kube-system - labels: - app: exoscale-webhook - chart: exoscale-webhook-0.3.0 - release: exoscale-webhook - heritage: Helm -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: extension-apiserver-authentication-reader -subjects: - - apiGroup: "" - kind: ServiceAccount - name: cert-manager-webhook-exoscale - namespace: cert-manager ---- -# Source: exoscale-webhook/templates/rbac.yaml -# Grant the webhook permission to read the secrets containing the credentials -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: cert-manager-webhook-exoscale:secrets-reader - namespace: cert-manager - labels: - app: exoscale-webhook - chart: exoscale-webhook-0.3.0 - release: exoscale-webhook - heritage: Helm -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: cert-manager-webhook-exoscale:secrets-reader -subjects: - - apiGroup: "" - kind: ServiceAccount - name: cert-manager-webhook-exoscale - namespace: cert-manager ---- -# Source: exoscale-webhook/templates/service.yaml -apiVersion: v1 -kind: Service -metadata: - name: cert-manager-webhook-exoscale - namespace: "cert-manager" - labels: - app: exoscale-webhook - chart: exoscale-webhook-0.3.0 - release: exoscale-webhook - heritage: Helm -spec: - type: ClusterIP - ports: - - port: 443 - targetPort: https - protocol: TCP - name: https - selector: - app: exoscale-webhook - release: exoscale-webhook ---- -# Source: exoscale-webhook/templates/deployment.yaml -apiVersion: apps/v1 -kind: Deployment -metadata: - name: cert-manager-webhook-exoscale - namespace: "cert-manager" - labels: - app: exoscale-webhook - chart: exoscale-webhook-0.3.0 - release: exoscale-webhook - heritage: Helm -spec: - replicas: - selector: - matchLabels: - app: exoscale-webhook - release: exoscale-webhook - template: - metadata: - labels: - app: exoscale-webhook - release: exoscale-webhook - spec: - serviceAccountName: cert-manager-webhook-exoscale - containers: - - name: exoscale-webhook - image: "exoscale/cert-manager-webhook-exoscale:latest" - imagePullPolicy: IfNotPresent - args: - - --tls-cert-file=/tls/tls.crt - - --tls-private-key-file=/tls/tls.key - env: - - name: GROUP_NAME - value: "acme.exoscale.com" - - name: EXOSCALE_DEBUG - value: "" - - name: EXOSCALE_API_TRACE - value: "" - ports: - - name: https - containerPort: 443 - protocol: TCP - livenessProbe: - httpGet: - scheme: HTTPS - path: /healthz - port: https - readinessProbe: - httpGet: - scheme: HTTPS - path: /healthz - port: https - volumeMounts: - - name: certs - mountPath: /tls - readOnly: true - resources: - {} - volumes: - - name: certs - secret: - secretName: cert-manager-webhook-exoscale-webhook-tls ---- -# Source: exoscale-webhook/templates/apiservice.yaml -apiVersion: apiregistration.k8s.io/v1 -kind: APIService -metadata: - name: v1alpha1.acme.exoscale.com - labels: - app: exoscale-webhook - chart: exoscale-webhook-0.3.0 - release: exoscale-webhook - heritage: Helm - annotations: - cert-manager.io/inject-ca-from: "cert-manager/cert-manager-webhook-exoscale-webhook-tls" -spec: - group: acme.exoscale.com - groupPriorityMinimum: 1000 - versionPriority: 15 - service: - name: cert-manager-webhook-exoscale - namespace: cert-manager - version: v1alpha1 ---- -# Source: exoscale-webhook/templates/pki.yaml -# Generate a CA Certificate used to sign certificates for the webhook -apiVersion: cert-manager.io/v1 -kind: Certificate -metadata: - name: cert-manager-webhook-exoscale-ca - namespace: "cert-manager" - labels: - app: exoscale-webhook - chart: exoscale-webhook-0.3.0 - release: exoscale-webhook - heritage: Helm -spec: - secretName: cert-manager-webhook-exoscale-ca - duration: 43800h # 5y - issuerRef: - name: cert-manager-webhook-exoscale-selfsign - commonName: "ca.exoscale-webhook.cert-manager" - isCA: true ---- -# Source: exoscale-webhook/templates/pki.yaml -# Finally, generate a serving certificate for the webhook to use -apiVersion: cert-manager.io/v1 -kind: Certificate -metadata: - name: cert-manager-webhook-exoscale-webhook-tls - namespace: "cert-manager" - labels: - app: exoscale-webhook - chart: exoscale-webhook-0.3.0 - release: exoscale-webhook - heritage: Helm -spec: - secretName: cert-manager-webhook-exoscale-webhook-tls - duration: 8760h # 1y - issuerRef: - name: cert-manager-webhook-exoscale-ca - dnsNames: - - cert-manager-webhook-exoscale - - cert-manager-webhook-exoscale.cert-manager - - cert-manager-webhook-exoscale.cert-manager.svc ---- -# Source: exoscale-webhook/templates/pki.yaml -# Create a selfsigned Issuer, in order to create a root CA certificate for -# signing webhook serving certificates -apiVersion: cert-manager.io/v1 -kind: Issuer -metadata: - name: cert-manager-webhook-exoscale-selfsign - namespace: "cert-manager" - labels: - app: exoscale-webhook - chart: exoscale-webhook-0.3.0 - release: exoscale-webhook - heritage: Helm -spec: - selfSigned: {} ---- -# Source: exoscale-webhook/templates/pki.yaml -# Create an Issuer that uses the above generated CA certificate to issue certs -apiVersion: cert-manager.io/v1 -kind: Issuer -metadata: - name: cert-manager-webhook-exoscale-ca - namespace: "cert-manager" - labels: - app: exoscale-webhook - chart: exoscale-webhook-0.3.0 - release: exoscale-webhook - heritage: Helm -spec: - ca: - secretName: cert-manager-webhook-exoscale-ca \ No newline at end of file diff --git a/charts/fp/templates/external-dns.yaml b/charts/fp/templates/external-dns.yaml deleted file mode 100644 index 2823acb..0000000 --- a/charts/fp/templates/external-dns.yaml +++ /dev/null @@ -1,87 +0,0 @@ ---- -apiVersion: v1 -kind: ServiceAccount -metadata: - name: external-dns - namespace: futureporn - annotations: - meta.helm.sh/release-name: fp - meta.helm.sh/release-namespace: futureporn - labels: - app.kubernetes.io/managed-by: Helm - ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: external-dns -rules: -- apiGroups: [""] - resources: ["services","endpoints","pods"] - verbs: ["get","watch","list"] -- apiGroups: ["extensions","networking.k8s.io"] - resources: ["ingresses"] - verbs: ["get","watch","list"] -- apiGroups: [""] - resources: ["nodes"] - verbs: ["list"] - ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: external-dns-viewer -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: external-dns -subjects: -- kind: ServiceAccount - name: external-dns - namespace: futureporn - ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: external-dns - namespace: futureporn -spec: - strategy: - type: Recreate - selector: - matchLabels: - app: external-dns - template: - metadata: - labels: - app: external-dns - spec: - serviceAccountName: external-dns - containers: - - name: external-dns - resources: - limits: - memory: "1024Mi" - cpu: "100m" - image: registry.k8s.io/external-dns/external-dns:v0.14.1 - args: - - --source=service - - --source=ingress - - --provider=exoscale - - --domain-filter=.sbtp.xyz - - --policy=sync # if you want DNS entries to get deleted as well - - --txt-owner-id=futureporn - - --exoscale-apizone=de-fra-1 - - --exoscale-apienv=api - env: - - name: EXTERNAL_DNS_EXOSCALE_APIKEY - valueFrom: - secretKeyRef: - name: exoscale - key: apiKey - - name: EXTERNAL_DNS_EXOSCALE_APISECRET - valueFrom: - secretKeyRef: - name: exoscale - key: apiSecret diff --git a/charts/fp/templates/gateway.yaml b/charts/fp/templates/gateway.yaml deleted file mode 100644 index 3e0b13b..0000000 --- a/charts/fp/templates/gateway.yaml +++ /dev/null @@ -1,12 +0,0 @@ ---- -apiVersion: gateway.networking.k8s.io/v1 -kind: Gateway -metadata: - name: example-gateway - namespace: futureporn -spec: - gatewayClassName: example-class - listeners: - - name: http - protocol: HTTP - port: 80 diff --git a/charts/fp/templates/pgadmin4.yaml b/charts/fp/templates/pgadmin4.yaml deleted file mode 100644 index 4d38fe4..0000000 --- a/charts/fp/templates/pgadmin4.yaml +++ /dev/null @@ -1,55 +0,0 @@ -## Resist the urge to delete this container! We use it again and again! - -apiVersion: apps/v1 -kind: Deployment -metadata: - name: pgadmin4 - namespace: futureporn -spec: - selector: - matchLabels: - app: pgadmin4 - template: - metadata: - labels: - app: pgadmin4 - spec: - containers: - - name: pgadmin4 - image: docker.io/dpage/pgadmin4:latest - imagePullPolicy: IfNotPresent - resources: - limits: - memory: 256Mi - requests: - memory: 100Mi - env: - - name: PGADMIN_DEFAULT_EMAIL - valueFrom: - secretKeyRef: - name: pgadmin4 - key: email - - name: PGADMIN_DEFAULT_PASSWORD - valueFrom: - secretKeyRef: - name: pgadmin4 - key: password - - ---- -apiVersion: v1 -kind: Service -metadata: - name: pgadmin4 - namespace: futureporn -spec: - type: ClusterIP - ports: - - port: 5050 - targetPort: 80 - protocol: TCP - selector: - app: pgadmin4 - - - diff --git a/charts/fp/templates/roles.yaml b/charts/fp/templates/roles.yaml deleted file mode 100644 index 183ef63..0000000 --- a/charts/fp/templates/roles.yaml +++ /dev/null @@ -1,28 +0,0 @@ -{{ if eq .Values.environment "development" }} - -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: cert-manager-webhook-vultr-secret-reader - namespace: cert-manager -rules: -- apiGroups: [""] - resources: ["secrets"] - verbs: ["get", "watch", "list"] - ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: cert-manager-webhook-vultr-secret-reader-binding - namespace: cert-manager -subjects: -- kind: ServiceAccount - name: cert-manager-webhook-vultr - namespace: cert-manager -roleRef: - kind: Role - name: cert-manager-webhook-vultr-secret-reader - apiGroup: rbac.authorization.k8s.io - -{{ end }} \ No newline at end of file diff --git a/charts/fp/templates/superstreamer.yaml b/charts/fp/templates/superstreamer.yaml deleted file mode 100644 index d02533a..0000000 --- a/charts/fp/templates/superstreamer.yaml +++ /dev/null @@ -1,339 +0,0 @@ - -## we don't use this because I don't know of a good way to sync the image tag with that of the postgres pod. -## It's more foolproof to use a script activated by a button in Tilt UI -# --- -# apiVersion: batch/v1 -# kind: Job -# metadata: -# name: superstreamer-database-seed -# namespace: futureporn -# spec: -# template: -# spec: -# restartPolicy: Never -# containers: -# - name: postgres-client -# image: postgres:latest -# command: ["sh", "-c"] -# args: -# - | -# psql -h postgresql-primary.futureporn.svc.cluster.local \ -# -U postgres \ -# -c "CREATE DATABASE sprs"; -# env: -# - name: PGPASSWORD -# valueFrom: -# secretKeyRef: -# name: postgresql -# key: password - - ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: superstreamer-app - namespace: futureporn -spec: - replicas: 1 - selector: - matchLabels: - app: superstreamer-app - template: - metadata: - labels: - app: superstreamer-app - spec: - containers: - - name: superstreamer-app - image: {{ .Values.superstreamer.app.image | quote }} - ports: - - containerPort: 52000 - env: - - name: PUBLIC_API_ENDPOINT - value: http://localhost:52001 - - name: PUBLIC_STITCHER_ENDPOINT - value: http://localhost:52002 - - name: DATABASE_URI - valueFrom: - secretKeyRef: - name: superstreamer - key: databaseUri - - name: S3_ENDPOINT - valueFrom: - secretKeyRef: - name: superstreamer - key: s3Endpoint - - name: S3_REGION - valueFrom: - secretKeyRef: - name: superstreamer - key: s3Region - - name: S3_ACCESS_KEY - valueFrom: - secretKeyRef: - name: superstreamer - key: s3AccessKey - - name: S3_SECRET_KEY - valueFrom: - secretKeyRef: - name: superstreamer - key: s3SecretKey - - name: S3_BUCKET - valueFrom: - secretKeyRef: - name: superstreamer - key: s3SecretKey - - name: PUBLIC_S3_ENDPOINT - valueFrom: - secretKeyRef: - name: superstreamer - key: publicS3Endpoint - - name: SUPER_SECRET - valueFrom: - secretKeyRef: - name: superstreamer - key: superSecret ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: superstreamer-api - namespace: futureporn -spec: - replicas: 1 - selector: - matchLabels: - app: superstreamer-api - template: - metadata: - labels: - app: superstreamer-api - spec: - containers: - - name: superstreamer-api - image: {{ .Values.superstreamer.api.image | quote }} - ports: - - containerPort: 52001 - env: - - name: REDIS_HOST - value: {{ .Values.superstreamer.redisUrl | quote }} - - name: REDIS_PORT - value: {{ .Values.superstreamer.redisPort | quote }} - - name: DATABASE_URI - valueFrom: - secretKeyRef: - name: superstreamer - key: databaseUri - - name: S3_ENDPOINT - valueFrom: - secretKeyRef: - name: superstreamer - key: s3Endpoint - - name: S3_REGION - valueFrom: - secretKeyRef: - name: superstreamer - key: s3Region - - name: S3_ACCESS_KEY - valueFrom: - secretKeyRef: - name: superstreamer - key: s3AccessKey - - name: S3_SECRET_KEY - valueFrom: - secretKeyRef: - name: superstreamer - key: s3SecretKey - - name: S3_BUCKET - valueFrom: - secretKeyRef: - name: superstreamer - key: s3Bucket - - name: PUBLIC_S3_ENDPOINT - valueFrom: - secretKeyRef: - name: superstreamer - key: publicS3Endpoint - - name: SUPER_SECRET - valueFrom: - secretKeyRef: - name: superstreamer - key: superSecret - - ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: superstreamer-stitcher - namespace: futureporn -spec: - replicas: 1 - selector: - matchLabels: - app: superstreamer-stitcher - template: - metadata: - labels: - app: superstreamer-stitcher - spec: - containers: - - name: superstreamer-stitcher - image: {{ .Values.superstreamer.stitcher.image | quote }} - ports: - - containerPort: 52002 - env: - - name: REDIS_HOST - value: {{ .Values.superstreamer.redisUrl | quote }} - - name: REDIS_PORT - value: {{ .Values.superstreamer.redisPort | quote }} - - name: PUBLIC_API_ENDPOINT - value: "http://localhost:52001" - - name: PUBLIC_STITCHER_ENDPOINT - value: "http://localhost:52002" - - name: DATABASE_URI - valueFrom: - secretKeyRef: - name: superstreamer - key: databaseUri - - name: S3_ENDPOINT - valueFrom: - secretKeyRef: - name: superstreamer - key: s3Endpoint - - name: S3_REGION - valueFrom: - secretKeyRef: - name: superstreamer - key: s3Region - - name: S3_ACCESS_KEY - valueFrom: - secretKeyRef: - name: superstreamer - key: s3AccessKey - - name: S3_SECRET_KEY - valueFrom: - secretKeyRef: - name: superstreamer - key: s3SecretKey - - name: S3_BUCKET - valueFrom: - secretKeyRef: - name: superstreamer - key: s3Bucket - - name: PUBLIC_S3_ENDPOINT - valueFrom: - secretKeyRef: - name: superstreamer - key: publicS3Endpoint - - name: SUPER_SECRET - valueFrom: - secretKeyRef: - name: superstreamer - key: superSecret ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: superstreamer-artisan - namespace: futureporn -spec: - replicas: 1 - selector: - matchLabels: - app: superstreamer-artisan - template: - metadata: - labels: - app: superstreamer-artisan - spec: - containers: - - name: superstreamer-artisan - image: {{ .Values.superstreamer.artisan.image | quote }} - env: - - name: REDIS_HOST - value: {{ .Values.superstreamer.redisUrl | quote }} - - name: REDIS_PORT - value: {{ .Values.superstreamer.redisPort | quote }} - - name: DATABASE_URI - valueFrom: - secretKeyRef: - name: superstreamer - key: databaseUri - - name: S3_ENDPOINT - valueFrom: - secretKeyRef: - name: superstreamer - key: s3Endpoint - - name: S3_REGION - valueFrom: - secretKeyRef: - name: superstreamer - key: s3Region - - name: S3_ACCESS_KEY - valueFrom: - secretKeyRef: - name: superstreamer - key: s3AccessKey - - name: S3_SECRET_KEY - valueFrom: - secretKeyRef: - name: superstreamer - key: s3SecretKey - - name: S3_BUCKET - valueFrom: - secretKeyRef: - name: superstreamer - key: s3Bucket - - name: PUBLIC_S3_ENDPOINT - valueFrom: - secretKeyRef: - name: superstreamer - key: publicS3Endpoint - - name: SUPER_SECRET - valueFrom: - secretKeyRef: - name: superstreamer - key: superSecret ---- -apiVersion: v1 -kind: Service -metadata: - name: superstreamer-app - namespace: futureporn -spec: - selector: - app: superstreamer-app - ports: - - protocol: TCP - port: 52000 - targetPort: 52000 - ---- -apiVersion: v1 -kind: Service -metadata: - name: superstreamer-api - namespace: futureporn -spec: - selector: - app: superstreamer-api - ports: - - protocol: TCP - port: 52001 - targetPort: 52001 - ---- -apiVersion: v1 -kind: Service -metadata: - name: superstreamer-stitcher - namespace: futureporn -spec: - selector: - app: superstreamer-stitcher - ports: - - protocol: TCP - port: 52002 - targetPort: 52002 diff --git a/charts/fp/templates/traefik.yaml b/charts/fp/templates/traefik.yaml deleted file mode 100644 index f87411a..0000000 --- a/charts/fp/templates/traefik.yaml +++ /dev/null @@ -1,115 +0,0 @@ -# --- -# apiVersion: traefik.io/v1alpha1 -# kind: Middleware -# metadata: -# name: redirect -# namespace: futureporn - -# spec: -# redirectScheme: -# scheme: https -# permanent: true - - -# --- -# apiVersion: traefik.io/v1alpha1 -# kind: Middleware -# metadata: -# name: traefik-dashboard-basicauth -# namespace: futureporn - -# spec: -# basicAuth: -# secret: traefik-dashboard-auth - -# --- -# apiVersion: traefik.io/v1alpha1 -# kind: IngressRoute -# metadata: -# name: traefik-dashboard-custom -# namespace: futureporn - -# spec: -# entryPoints: -# - websecure -# - web -# routes: -# - match: PathPrefix(`/board`) -# kind: Rule -# # priority: 2147483647 -# # middlewares: -# # - name: traefik-dashboard-basicauth -# # namespace: traefik -# services: -# - name: dashboard@internal -# kind: TraefikService - -# --- -# apiVersion: traefik.io/v1alpha1 -# kind: IngressRoute -# metadata: -# name: traefik-api-custom -# namespace: futureporn - -# spec: -# entryPoints: -# - websecure -# - web -# routes: -# - match: PathPrefix(`/api`) -# kind: Rule -# # priority: 2147483647 -# # middlewares: -# # - name: traefik-dashboard-basicauth -# # namespace: traefik -# services: -# - name: api@internal -# kind: TraefikService - - ---- -apiVersion: gateway.networking.k8s.io/v1 -kind: HTTPRoute -metadata: - name: whoami-httproute - namespace: futureporn -spec: - parentRefs: - - name: traefik-gateway - hostnames: - - traefik.futureporn.svc.cluster.local - rules: - - matches: - - path: - type: PathPrefix - value: /whoami - filters: - - type: ResponseHeaderModifier - responseHeaderModifier: - add: - - name: x-cj-was-here - value: "true" - backendRefs: - - name: whoami - port: 8888 - - matches: - - path: - type: PathPrefix - value: /postgrest - backendRefs: - - name: postgrest - port: 9000 - -# --- -# apiVersion: gateway.networking.k8s.io/v1 -# kind: HTTPRoute -# metadata: -# name: whoami-httproute -# namespace: futureporn -# spec: -# parentRefs: -# - name: traefik-gateway -# hostnames: -# - traefik.futureporn.svc.cluster.local -# rules: -# - matches: diff --git a/charts/fp/templates/whoami.yaml b/charts/fp/templates/whoami.yaml deleted file mode 100644 index 31a3d7d..0000000 --- a/charts/fp/templates/whoami.yaml +++ /dev/null @@ -1,70 +0,0 @@ -# an http service for testing purposes - - -{{ if ne .Values.environment "production" }} - - -# --- -# apiVersion: gateway.networking.k8s.io/v1 -# kind: HTTPRoute -# metadata: -# name: whoami-httproute -# namespace: futureporn -# spec: -# parentRefs: -# - name: traefik-gateway -# hostnames: -# - traefik.futureporn -# rules: -# - matches: -# - path: -# type: PathPrefix -# value: /whoami -# backendRefs: -# - name: whoami -# port: {{ .Values.whoami.port }} - - ---- -apiVersion: v1 -kind: Service -metadata: - name: whoami - namespace: futureporn - annotations: - external-dns.alpha.kubernetes.io/hostname: "{{ .Values.whoami.hostname }}" -spec: - type: LoadBalancer - selector: - app: whoami - ports: - - port: {{ .Values.whoami.port }} - targetPort: {{ .Values.whoami.port }} - - ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: whoami - namespace: futureporn -spec: - replicas: 1 - selector: - matchLabels: - app: whoami - template: - metadata: - labels: - app: whoami - spec: - containers: - - name: whoami - image: containous/whoami - args: - - --port={{ .Values.whoami.port }} - ports: - - containerPort: {{ .Values.whoami.port }} - - -{{ end }} \ No newline at end of file diff --git a/charts/fp/values.yaml b/charts/fp/values.yaml deleted file mode 100644 index da4a2cc..0000000 --- a/charts/fp/values.yaml +++ /dev/null @@ -1,138 +0,0 @@ -## values.yaml -## this file is for development values only. -## these values are loaded by Tilt. -## For production values, see /flux/clusters/production - -environment: development -# storageClassName: csi-hostpath-sc # used by minikube -storageClassName: standard # used by Kind -s3: - endpoint: https://futureporn.hel1.your-objectstorage.com - region: hel1 - buckets: - main: futureporn - usc: fp-usc-dev - backup: futureporn-db-backup-dev -link2cid: - imageName: fp/link2cid -next: - imageName: fp/next - hostname: next.fp.sbtp.xyz - ingressClassName: nginx - nodeExtraCaCerts: /app/certs/letsencrypt-stg-root-x1.pem -capture: - imageName: fp/capture - worker: - replicas: 1 - cache: - size: 20Gi - dir: /var/cache/capture-worker -mailbox: - imageName: fp/mailbox - replicas: 1 - cdnBucketUrl: https://futureporn.hel1.your-objectstorage.com - s3BucketName: futureporn - port: 5000 -factory: - replicas: 1 - imageName: fp/factory -strapi: - replicas: 1 - imageName: fp/strapi - port: 1339 - url: https://strapi.fp.sbtp.xyz - hostname: strapi.fp.sbtp.xyz - # hostname: strapi.futureporn.svc.cluster.local ## https://gitea.futureporn.net/futureporn/pm/issues/126 - ingressClassName: nginx - nodeExtraCaCerts: /usr/src/app/certs/letsencrypt-stg-root-x1.pem -ngrok: - hostname: mint-star-crappie.ngrok-free.app -realtime: - imageName: fp/realtime -adminEmail: cj@futureporn.net -echo: - hostname: echo.fp.sbtp.xyz -superstreamer: - redisUrl: redis-master.futureporn.svc.cluster.local - redisPort: 6379 - app: - #image: fp/superstreamer-app - image: "superstreamerapp/app:alpha" - api: - #image: fp/superstreamer-api - localUrl: http://superstreamer-api.futureporn.svc.cluster.local:52001 - image: "superstreamerapp/api:alpha" - artisan: - #image: fp/superstreamer-artisan - image: "superstreamerapp/artisan:alpha" - stitcher: - #image: fp/superstreamer-stitcher - image: "superstreamerapp/stitcher:alpha" -uppy: - replicas: 3 - hostname: uppy.fp.sbtp.xyz - imageName: fp/uppy - redisUrl: redis-master.futureporn.svc.cluster.local - s3: - endpoint: your-objectstorage.com - bucket: futureporn - region: hel1 - clientOrigins: next.fp.sbtp.xyz - domain: uppy.fp.sbtp.xyz - uploadUrls: https://uppy.fp.sbtp.xyz/files - url: https://uppy.fp.sbtp.xyz -certManager: - issuer: letsencrypt-staging -bot: - discordChannelId: "1185024773231759481" - discordGuildId: "1084674137391374338" - imageName: fp/bot - replicas: 1 -chihaya: - hostname: tracker.fp.sbtp.xyz -scout: - imageName: fp/scout - replicas: 1 - port: 8134 - url: http://scout.fp.sbtp.xyz -postgrest: - hostname: postgrest.fp.sbtp.xyz - image: postgrest/postgrest - replicas: 1 - port: 9000 - schemas: api -chisel: - exitNode: - host: "155.138.254.201" - port: 9090 -game2048: - hostname: game-2048.fp.sbtp.xyz -supertokens: - port: 3348 - hostname: supertokens.fp.sbtp.xyz - replicas: 1 - -logto: - admin: - port: 3002 - hostname: logto-admin.fp.sbtp.xyz - main: - port: 3001 - hostname: logto.fp.sbtp.xyz -whoami: - hostname: whoami.fp.sbtp.xyz - port: 8888 -migrations: - schema: - imageName: fp/migrations-schema - data: - imageName: fp/migrations-data -authentik: - replias: 1 - hostname: auth.fp.sbtp.xyz -bright: - imageName: fp/bright - hostname: bright.fp.sbtp.xyz - port: 4000 - s3: - endpoint: https://fp-dev.b-cdn.net \ No newline at end of file diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/.helmignore b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/.helmignore deleted file mode 100644 index faa1198..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/.helmignore +++ /dev/null @@ -1,25 +0,0 @@ -# Source: https://github.com/helm/helm/blob/main/pkg/repo/repotest/testdata/examplechart/.helmignore -# 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 - -# helmtest plugin tests -tests diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/CHANGELOG.md b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/CHANGELOG.md deleted file mode 100644 index 1d3da7a..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/CHANGELOG.md +++ /dev/null @@ -1,156 +0,0 @@ -# Changelog - -All notable changes to this project will be documented in this file. - -The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), -and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). - -## 0.14.0 -**Full Changelog**: https://github.com/ngrok/kubernetes-ingress-controller/compare/helm-chart-0.13.0...helm-chart-0.14.0 - -### Added - -- feat: Auto-provision domain for TLS Edges [#386]( https://github.com/ngrok/kubernetes-ingress-controller/pull/386) -- feat: Support for Load Balancer services [#387](https://github.com/ngrok/kubernetes-ingress-controller/pull/387) -- feat: Support TLS termination in modulesets for Load Balancer Services [388](https://github.com/ngrok/kubernetes-ingress-controller/pull/388) - -## 0.13.0 - -**Full Changelog**: https://github.com/ngrok/kubernetes-ingress-controller/compare/helm-chart-0.12.4...helm-chart-0.13.0 - -**Important**: If you are upgrading from a previous version and are using `helm install` or `helm upgrade`, you will need to manually apply the changes to the CRDs. This is because the CRDs are not [updated automatically when the chart is updated](https://helm.sh/docs/chart_best_practices/custom_resource_definitions/#some-caveats-and-explanations). To do this, apply the contents of the `crds` directory in the chart to your cluster. - -Ex (from the root of the repository): -```shell -kubectl apply -f ./helm/ingress-controller/templates/crds/ -``` - -### Added - -- root-cas setting [#371](https://github.com/ngrok/kubernetes-ingress-controller/pull/371) - Takes an install option for `--set rootCAs=host` and plumb the isHostCA check into the caCerts for it to just get the host certs. -- feat: Add support for mutualTLS [#373](https://github.com/ngrok/kubernetes-ingress-controller/pull/373) - -### Changed - -- Update nix flake, go version, and Makefile dep versions [#379](https://github.com/ngrok/kubernetes-ingress-controller/pull/379) - -## 0.12.4 - -- Add the `apiURL` value. - This sets the ngrok API endpoint used by the controller. - It corresponds to the `--api-url` argument to the manager binary. - -- Update to version 0.10.4 of the ingress controller. - See its changes [here](../../CHANGELOG.md#0104). - -## 0.12.1 - -- Update to version 0.10.1 of the ingress controller, which includes: - - IPPolicy controller wasn't applying the attached rules, leaving the IP policy in its current state [#315](https://github.com/ngrok/kubernetes-ingress-controller/pull/315) - -## 0.12.0 - -- Update to version 0.10.0 of the ingress controller, this includes: - - TLSEdge support - see the [TCP and TLS Edges Guide](https://github.com/ngrok/kubernetes-ingress-controller/blob/main/docs/user-guide/tcp-tls-edges.md) for more details. - - A fix for renegotiating TLS backends - -## 0.11.0 - -** Important ** This version of the controller changes the ownership model for https edge and tunnel CRs. To ease out the transition to the new ownership, make sure to run `migrate-edges.sh` and `migrate-tunnels.sh` scripts before installing the new version. - -### Changed -- Specify IPPolicyRule action as an enum of (allow,deny) as part of [#260](https://github.com/ngrok/kubernetes-ingress-controller/pull/260) -- Handle special case for changing auth types that causes an error during state transition [#259](https://github.com/ngrok/kubernetes-ingress-controller/pull/259) -- Better handling when changing pathType between 'Exact' and 'Prefix' [#262](https://github.com/ngrok/kubernetes-ingress-controller/pull/262) -- Update ngrok-go to 1.4.0 [#298](https://github.com/ngrok/kubernetes-ingress-controller/pull/298) -- Tunnels are now unique in their respective namespace, not across the cluster [#281](https://github.com/ngrok/kubernetes-ingress-controller/pull/281) -- The CRs that ingress controller creates are uniquely marked and managed by it. Other CRs created manually are no longer deleted when the ingress controller is not using them [#267](https://github.com/ngrok/kubernetes-ingress-controller/issues/267); fixed for tunnel in [#285](https://github.com/ngrok/kubernetes-ingress-controller/pull/285) and for https edges in [#286](https://github.com/ngrok/kubernetes-ingress-controller/pull/286) -- Better error handling and retry, specifically for the case where we try to create an https edge for a domain which is not created yet [#283](https://github.com/ngrok/kubernetes-ingress-controller/issues/283); fixed in [#288](https://github.com/ngrok/kubernetes-ingress-controller/pull/288) -- Watch and apply ngrok module set CR changes [#287](https://github.com/ngrok/kubernetes-ingress-controller/issues/287); fixed in [#290](https://github.com/ngrok/kubernetes-ingress-controller/pull/290) -- Label https edges and tunnels with service UID to make them more unique within ngrok [#291](https://github.com/ngrok/kubernetes-ingress-controller/issues/291); fixed in [#293](https://github.com/ngrok/kubernetes-ingress-controller/pull/293) and [#302](https://github.com/ngrok/kubernetes-ingress-controller/pull/302) - -### Added -- Add support for configuring pod affinities, pod disruption budget, and priorityClassName [#258](https://github.com/ngrok/kubernetes-ingress-controller/pull/258) -- The controller stopping at the first resource create [#270](https://github.com/ngrok/kubernetes-ingress-controller/pull/270) -- Using `make deploy` now requires `NGROK_AUTHTOKEN` and `NGROK_API_KEY` to be set [#292](https://github.com/ngrok/kubernetes-ingress-controller/pull/292) - -## 0.10.0 - -### Added -- Support HTTPS backends via service annotation [#238](https://github.com/ngrok/kubernetes-ingress-controller/pull/238) - -### Changed -- Normalize all ngrok `.io` TLD to `.app` TLD [#240](https://github.com/ngrok/kubernetes-ingress-controller/pull/240) -- Chart Icon - -### Fixed -- Add namespace to secret [#244](https://github.com/ngrok/kubernetes-ingress-controller/pull/244). Thank you for the contribution, @vincetse! - -## 0.9.0 -### Added -- Add a 'podLabels' option to the helm chart [#212](https://github.com/ngrok/kubernetes-ingress-controller/pull/212). -- Permission to `get`,`list`, and `watch` `services` [#222](https://github.com/ngrok-kubernetes-ingress-controller/pull/222). - -## 0.8.0 -### Changed -- Log Level configuration to helm chart [#199](https://github.com/ngrok/kubernetes-ingress-controller/pull/199). -- Bump default controller image to use `0.6.0` release [#204](https://github.com/ngrok/kubernetes-ingress-controller/pull/204). - -### Fixed -- update default-container annotation so logs work correctly [#197](https://github.com/ngrok/kubernetes-ingress-controller/pull/197) - -## 0.7.0 - -### Added -- Update `NgrokModuleSet` and `HTTPSEdge` CRD to support SAML and OAuth - -### Changed -- Update appVersion to `0.5.0` to match the latest release of the controller. - -## 0.6.1 -### Fixed -- Default the image tag to the chart's `appVersion` for predictable installs. Previously, the helm chart would default to the `latest` image tag which can have breaking changes, notably with CRDs. - -## 0.6.0 -### Changed -- Ingress Class has Default set to false [#109](https://github.com/ngrok/kubernetes-ingress-controller/pull/109) - -### Added -- Allow controller name to be configured to support multiple ngrok ingress classes [#159](https://github.com/ngrok/kubernetes-ingress-controller/pull/159) -- Allow the controller to be configured to only watch a single namespace [#157](https://github.com/ngrok/kubernetes-ingress-controller/pull/157) -- Pass key/value pairs to helm that get added as json string metadata in ngrok api resources [#156](https://github.com/ngrok/kubernetes-ingress-controller/pull/156) -- Add IP Policy CRD and IP Policy Route Module [#120](https://github.com/ngrok/kubernetes-ingress-controller/pull/120) -- Load certs from the directory `"/etc/ssl/certs/ngrok/"` for ngrok-go if present [#111](https://github.com/ngrok/kubernetes-ingress-controller/pull/111) - -## 0.5.0 -### Changed -- Renamed chart from `ngrok-ingress-controller` to `kubernetes-ingress-controller`. -- Added CRDs for `domains`, `tcpedges`, and `httpsedges`. - -## 0.4.0 -### Added -- `serverAddr` flag to override the ngrok tunnel server address -- `extraVolumes` to add an arbitrary set of volumes to the controller pod -- `extraVolumeMounts` to add an arbitrary set of volume mounts to the controller container - -## 0.3.1 -### Fixed -- Fixes rendering of `NOTES.txt` when installing via helm - -## 0.3.0 -### Changed - -- Moved from calling ngrok-agent sidecar to using the ngrok-go library in the controller process. -- Moved `apiKey` and `authtoken` to `credentials.apiKey` and `credentials.authtoken` respectively. -- `credentialSecrets.name` is now `credentials.secret.name` -- Changed replicas to 1 by default to work better for default/demo setup. - -## 0.2.0 -### Added - -- Support for different values commonly found in helm charts - -# 0.1.0 - -TODO diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/Chart.lock b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/Chart.lock deleted file mode 100644 index f02dfe1..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/Chart.lock +++ /dev/null @@ -1,6 +0,0 @@ -dependencies: -- name: common - repository: https://charts.bitnami.com/bitnami - version: 2.19.3 -digest: sha256:b0c5b947a6622accb48678c6034a1180e1fb58d0d8c8d2fa44cb7e9c3b1a4255 -generated: "2024-05-21T13:06:30.412702362-05:00" diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/Chart.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/Chart.yaml deleted file mode 100644 index e0d3398..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/Chart.yaml +++ /dev/null @@ -1,21 +0,0 @@ -apiVersion: v2 -appVersion: 0.12.0 -dependencies: -- name: common - repository: https://charts.bitnami.com/bitnami - tags: - - bitnami-common - version: 2.x.x -description: A Kubernetes ingress controller built using ngrok. -home: https://ngrok.com -icon: https://ngrok.github.io/kubernetes-ingress-controller/ngrok-favicon.svg -keywords: -- ngrok -- networking -- ingress -- edge -- api gateway -name: kubernetes-ingress-controller -sources: -- https://github.com/ngrok/kubernetes-ingress-controller -version: 0.14.0 diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/README.md b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/README.md deleted file mode 100644 index c4d0c06..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/README.md +++ /dev/null @@ -1,93 +0,0 @@ -# ngrok Ingress Controller - -This is the helm chart to install the ngrok ingress controller - -# Usage - -## Prerequisites - -The cluster Must be setup with a secret named `ngrok-ingress-controller-credentials` with the following keys: -* AUTHTOKEN -* API\_KEY - -## Install the controller with helm - -[Helm](https://helm.sh) must be installed to use the charts. Please refer to -Helm's [documentation](https://helm.sh/docs) to get started. - -Once Helm has been set up correctly, add the repo as follows: - -`helm repo add ngrok https://ngrok.github.io/kubernetes-ingress-controller` - -If you had already added this repo earlier, run `helm repo update` to retrieve -the latest versions of the packages. You can then run `helm search repo ngrok` to see the charts. - -To install the ngrok-ingress-controller chart: - -`helm install my-ngrok-ingress-controller ngrok/kubernetes-ingress-controller` - -To uninstall the chart: - -`helm delete my-ngrok-ingress-controller` - - -## Parameters - -### Common parameters - -| Name | Description | Value | -| ------------------- | ----------------------------------------------------- | ----- | -| `nameOverride` | String to partially override generated resource names | `""` | -| `fullnameOverride` | String to fully override generated resource names | `""` | -| `commonLabels` | Labels to add to all deployed objects | `{}` | -| `commonAnnotations` | Annotations to add to all deployed objects | `{}` | - - -### Controller parameters - -| Name | Description | Value | -| ------------------------------------ | --------------------------------------------------------------------------------------------------------------------- | ------------------------------------- | -| `podAnnotations` | Used to apply custom annotations to the ingress pods. | `{}` | -| `podLabels` | Used to apply custom labels to the ingress pods. | `{}` | -| `replicaCount` | The number of controllers to run. | `1` | -| `image.registry` | The ngrok ingress controller image registry. | `docker.io` | -| `image.repository` | The ngrok ingress controller image repository. | `ngrok/kubernetes-ingress-controller` | -| `image.tag` | The ngrok ingress controller image tag. Defaults to the chart's appVersion if not specified | `""` | -| `image.pullPolicy` | The ngrok ingress controller image pull policy. | `IfNotPresent` | -| `image.pullSecrets` | An array of imagePullSecrets to be used when pulling the image. | `[]` | -| `ingressClass.name` | The name of the ingress class to use. | `ngrok` | -| `ingressClass.create` | Whether to create the ingress class. | `true` | -| `ingressClass.default` | Whether to set the ingress class as default. | `false` | -| `controllerName` | The name of the controller to look for matching ingress classes | `k8s.ngrok.com/ingress-controller` | -| `watchNamespace` | The namespace to watch for ingress resources. Defaults to all | `""` | -| `credentials.secret.name` | The name of the secret the credentials are in. If not provided, one will be generated using the helm release name. | `""` | -| `credentials.apiKey` | Your ngrok API key. If provided, it will be will be written to the secret and the authtoken must be provided as well. | `""` | -| `credentials.authtoken` | Your ngrok authtoken. If provided, it will be will be written to the secret and the apiKey must be provided as well. | `""` | -| `region` | ngrok region to create tunnels in. Defaults to connect to the closest geographical region. | `""` | -| `rootCAs` | Set to "trusted" for the ngrok agent CA or "host" to trust the host's CA. Defaults to "trusted". | `""` | -| `serverAddr` | This is the address of the ngrok server to connect to. You should set this if you are using a custom ingress address. | `""` | -| `apiURL` | This is the URL of the ngrok API. You should set this if you are using a custom API URL. | `""` | -| `metaData` | This is a map of key/value pairs that will be added as meta data to all ngrok api resources created | `{}` | -| `affinity` | Affinity for the controller pod assignment | `{}` | -| `podAffinityPreset` | Pod affinity preset. Ignored if `affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `podAntiAffinityPreset` | Pod anti-affinity preset. Ignored if `affinity` is set. Allowed values: `soft` or `hard` | `soft` | -| `nodeAffinityPreset.type` | Node affinity preset type. Ignored if `affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `nodeAffinityPreset.key` | Node label key to match. Ignored if `affinity` is set. | `""` | -| `nodeAffinityPreset.values` | Node label values to match. Ignored if `affinity` is set. | `[]` | -| `priorityClassName` | Priority class for pod scheduling | `""` | -| `podDisruptionBudget.create` | Enable a Pod Disruption Budget creation | `false` | -| `podDisruptionBudget.minAvailable` | Minimum number/percentage of pods that should remain scheduled | `""` | -| `podDisruptionBudget.maxUnavailable` | Maximum number/percentage of pods that may be made unavailable | `1` | -| `resources.limits` | The resources limits for the container | `{}` | -| `resources.requests` | The requested resources for the container | `{}` | -| `extraVolumes` | An array of extra volumes to add to the controller. | `[]` | -| `extraVolumeMounts` | An array of extra volume mounts to add to the controller. | `[]` | -| `extraEnv` | an object of extra environment variables to add to the controller. | `{}` | -| `serviceAccount.create` | Specifies whether a ServiceAccount should be created | `true` | -| `serviceAccount.name` | The name of the ServiceAccount to use. | `""` | -| `serviceAccount.annotations` | Additional annotations to add to the ServiceAccount | `{}` | -| `log.level` | The level to log at. One of 'debug', 'info', or 'error'. | `info` | -| `log.stacktraceLevel` | The level to report stacktrace logs one of 'info' or 'error'. | `error` | -| `log.format` | The log format to use. One of console, json. | `json` | -| `lifecycle` | an object containing lifecycle configuration | `{}` | - diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/NOTES.txt b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/NOTES.txt deleted file mode 100644 index 3b5bf28..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/NOTES.txt +++ /dev/null @@ -1,53 +0,0 @@ -================================================================================ -The ngrok Ingress controller has been deployed as a Deployment type to your -cluster. - -If you haven't yet, create some Ingress resources in your cluster and they will -be automatically configured on the internet using ngrok. - - -{{- $found := false }} -{{- range $svcIndex, $service := (lookup "v1" "Service" "" "").items }} - {{- if not $found }} - {{- range $portMapIdx, $portMap := $service.spec.ports }} - {{- if eq $portMap.port 80 443 }} - {{- if ne $service.metadata.name "kubernetes" }} - {{- $found = true -}} - {{- $randomStr := randAlphaNum 8 | lower }} - -One example, taken from your cluster, is the Service: - {{ $service.metadata.name | quote }} - -You can make this accessible via ngrok with the following manifest: --------------------------------------------------------------------------------- -apiVersion: networking.k8s.io/v1 -kind: Ingress -metadata: - name: {{ $service.metadata.name }} - namespace: {{ $service.metadata.namespace }} -spec: - ingressClassName: ngrok - rules: - - host: {{ $service.metadata.name -}}-{{- $randomStr -}}.ngrok.app - http: - paths: - - path: / - pathType: Prefix - backend: - service: - name: {{ $service.metadata.name }} - port: - number: {{ $portMap.port }} --------------------------------------------------------------------------------- -Applying this manifest will make the service {{ $service.metadata.name | quote }} -available on the public internet at "https://{{ $service.metadata.name -}}-{{- $randomStr -}}.ngrok.app/". - {{- end }} - {{- end }} - {{- end }} - {{- end}} -{{- end }} - -Once done, view your edges in the Dashboard https://dashboard.ngrok.com/cloud-edge/edges -Find the tunnels running in your cluster here https://dashboard.ngrok.com/tunnels/agents - -If you have any questions or feedback, please join us in https://ngrok.com/slack and let us know! diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/_helpers.tpl b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/_helpers.tpl deleted file mode 100644 index c5be41c..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/_helpers.tpl +++ /dev/null @@ -1,87 +0,0 @@ -{{/* vim: set filetype=mustache: */}} -{{/* -Expand the name of the chart. -*/}} -{{- define "kubernetes-ingress-controller.name" -}} -{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Create chart name and version as used by the chart label. -*/}} -{{- define "kubernetes-ingress-controller.chart" -}} -{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Create a default fully qualified app name. -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -*/}} -{{- define "kubernetes-ingress-controller.fullname" -}} -{{- if .Values.fullnameOverride -}} -{{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" -}} -{{- else -}} -{{- $name := default .Chart.Name .Values.nameOverride -}} -{{- if contains $name .Release.Name -}} -{{- .Release.Name | trunc 63 | trimSuffix "-" -}} -{{- else -}} -{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} -{{- end -}} -{{- end -}} -{{- end -}} - -{{/* -Create a default name for the credentials secret name using the helm release -*/}} -{{- define "kubernetes-ingress-controller.credentialsSecretName" -}} -{{- if .Values.credentials.secret.name -}} -{{- .Values.credentials.secret.name -}} -{{- else -}} -{{- printf "%s-credentials" (include "kubernetes-ingress-controller.fullname" .) -}} -{{- end -}} -{{- end -}} - -{{/* -Common labels -*/}} -{{- define "kubernetes-ingress-controller.labels" -}} -helm.sh/chart: {{ include "kubernetes-ingress-controller.chart" . }} -{{ include "kubernetes-ingress-controller.selectorLabels" . }} -{{- if .Chart.AppVersion }} -app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} -{{- end }} -app.kubernetes.io/part-of: {{ template "kubernetes-ingress-controller.name" . }} -app.kubernetes.io/managed-by: {{ .Release.Service }} -{{- if .Values.commonLabels}} -{{ toYaml .Values.commonLabels }} -{{- end }} -{{- end -}} - -{{/* -Selector labels -*/}} -{{- define "kubernetes-ingress-controller.selectorLabels" -}} -app.kubernetes.io/name: {{ include "kubernetes-ingress-controller.name" . }} -app.kubernetes.io/instance: {{ .Release.Name }} -{{- end -}} - -{{/* -Create the name of the controller service account to use -*/}} -{{- define "kubernetes-ingress-controller.serviceAccountName" -}} -{{- if .Values.serviceAccount.create -}} - {{ default (include "kubernetes-ingress-controller.fullname" .) .Values.serviceAccount.name }} -{{- else -}} - {{ default "default" .Values.serviceAccount.name }} -{{- end -}} -{{- end -}} - -{{/* -Return the ngrok/ingress-controller image name -*/}} -{{- define "kubernetes-ingress-controller.image" -}} -{{- $registryName := .Values.image.registry -}} -{{- $repositoryName := .Values.image.repository -}} -{{- $tag := .Values.image.tag | default .Chart.AppVersion | toString -}} -{{- printf "%s/%s:%s" $registryName $repositoryName $tag -}} -{{- end -}} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-cm.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-cm.yaml deleted file mode 100644 index 8872834..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-cm.yaml +++ /dev/null @@ -1,16 +0,0 @@ -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ include "kubernetes-ingress-controller.fullname" . }}-manager-config - namespace: {{ .Release.Namespace }} -data: - controller_manager_config.yaml: | - apiVersion: controller-runtime.sigs.k8s.io/v1alpha1 - kind: ControllerManagerConfig - health: - healthProbeBindAddress: :8081 - metrics: - bindAddress: 127.0.0.1:8080 - leaderElection: - leaderElect: true - resourceName: {{ include "kubernetes-ingress-controller.fullname" . }}-leader diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-deployment.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-deployment.yaml deleted file mode 100644 index 0bb1b98..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-deployment.yaml +++ /dev/null @@ -1,143 +0,0 @@ -{{- $component := "controller" }} -apiVersion: apps/v1 -kind: Deployment -metadata: - labels: - {{- include "kubernetes-ingress-controller.labels" . | nindent 4 }} - app.kubernetes.io/component: {{ $component }} - name: {{ include "kubernetes-ingress-controller.fullname" . }}-manager - namespace: {{ .Release.Namespace }} - annotations: - checksum/controller-role: {{ include (print $.Template.BasePath "/rbac/role.yaml") . | sha256sum }} - checksum/rbac: {{ include (print $.Template.BasePath "/controller-rbac.yaml") . | sha256sum }} -spec: - replicas: {{.Values.replicaCount}} - selector: - matchLabels: - {{- include "kubernetes-ingress-controller.selectorLabels" . | nindent 6 }} - {{- if .Values.podLabels }} - {{- toYaml .Values.podLabels | nindent 6 }} - {{- end }} - app.kubernetes.io/component: {{ $component }} - template: - metadata: - annotations: - {{- if .Values.podAnnotations }} - {{- toYaml .Values.podAnnotations | nindent 8 }} - {{- end }} - prometheus.io/path: /metrics - prometheus.io/port: '8080' - prometheus.io/scrape: 'true' - checksum/controller-role: {{ include (print $.Template.BasePath "/rbac/role.yaml") . | sha256sum }} - checksum/rbac: {{ include (print $.Template.BasePath "/controller-rbac.yaml") . | sha256sum }} - checksum/secret: {{ include (print $.Template.BasePath "/credentials-secret.yaml") . | sha256sum }} - labels: - {{- include "kubernetes-ingress-controller.selectorLabels" . | nindent 8 }} - {{- if .Values.podLabels }} - {{- toYaml .Values.podLabels | nindent 8 }} - {{- end }} - app.kubernetes.io/component: {{ $component }} - spec: - {{- if .Values.priorityClassName }} - priorityClassName: {{ .Values.priorityClassName | quote }} - {{- end }} - {{- if .Values.affinity }} - affinity: {{- include "common.tplvalues.render" (dict "value" .Values.affinity "context" $) | nindent 8 }} - {{- else }} - affinity: - podAffinity: {{- include "common.affinities.pods" (dict "type" .Values.podAffinityPreset "component" $component "context" $) | nindent 10 }} - podAntiAffinity: {{- include "common.affinities.pods" (dict "type" .Values.podAntiAffinityPreset "component" $component "context" $) | nindent 10 }} - nodeAffinity: {{- include "common.affinities.nodes" (dict "type" .Values.nodeAffinityPreset.type "key" .Values.nodeAffinityPreset.key "values" .Values.nodeAffinityPreset.values) | nindent 10 }} - {{- end }} - serviceAccountName: {{ template "kubernetes-ingress-controller.serviceAccountName" . }} - {{- if .Values.image.pullSecrets }} - imagePullSecrets: - {{- toYaml .Values.image.pullSecrets | nindent 8 }} - {{- end }} - containers: - - name: ngrok-ingress-controller - image: {{ include "kubernetes-ingress-controller.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy }} - command: - - /manager - args: - {{- if .Values.region }} - - --region={{ .Values.region}} - {{- end }} - {{- if .Values.apiURL }} - - --api-url={{ .Values.apiURL}} - {{- end }} - {{- if .Values.rootCAs }} - - --root-cas={{ .Values.rootCAs}} - {{- end }} - {{- if .Values.serverAddr }} - - --server-addr={{ .Values.serverAddr}} - {{- end }} - {{- if .Values.metaData }} - - --metadata={{- $metadataArgs := list -}} - {{- range $key, $value := .Values.metaData }} - {{- $metadataArgs = append $metadataArgs (printf "%s=%s" $key $value) -}} - {{- end }} - {{- $metadataArgs | join "," }} - {{- end }} - - --controller-name={{ .Values.controllerName }} - {{- if .Values.watchNamespace }} - - --watch-namespace={{ .Values.watchNamespace}} - {{- end }} - {{- if .Values.useExperimentalGatewayApi }} - - --use-experimental-gateway-api={{ .Values.useExperimentalGatewayApi }} - {{- end }} - - --zap-log-level={{ .Values.log.level }} - - --zap-stacktrace-level={{ .Values.log.stacktraceLevel }} - - --zap-encoder={{ .Values.log.format }} - - --health-probe-bind-address=:8081 - - --metrics-bind-address=:8080 - - --election-id={{ include "kubernetes-ingress-controller.fullname" . }}-leader - - --manager-name={{ include "kubernetes-ingress-controller.fullname" . }}-manager - securityContext: - allowPrivilegeEscalation: false - env: - - name: NGROK_API_KEY - valueFrom: - secretKeyRef: - key: API_KEY - name: {{ include "kubernetes-ingress-controller.credentialsSecretName" . }} - - name: NGROK_AUTHTOKEN - valueFrom: - secretKeyRef: - key: AUTHTOKEN - name: {{ include "kubernetes-ingress-controller.credentialsSecretName" . }} - - name: POD_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - {{- range $key, $value := .Values.extraEnv }} - - name: {{ $key }} - value: {{- toYaml $value | nindent 12 }} - {{- end }} - {{- if .Values.extraVolumeMounts }} - volumeMounts: - {{ toYaml .Values.extraVolumeMounts | nindent 10 }} - {{- end }} - {{- if .Values.lifecycle }} - lifecycle: - {{ toYaml .Values.lifecycle | nindent 10 }} - {{- end }} - livenessProbe: - httpGet: - path: /healthz - port: 8081 - initialDelaySeconds: 15 - periodSeconds: 20 - readinessProbe: - httpGet: - path: /readyz - port: 8081 - initialDelaySeconds: 5 - periodSeconds: 10 - resources: - {{- toYaml .Values.resources | nindent 10 }} - {{- if .Values.extraVolumes }} - volumes: - {{ toYaml .Values.extraVolumes | nindent 6 }} - {{- end }} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-pdb.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-pdb.yaml deleted file mode 100644 index 7046631..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-pdb.yaml +++ /dev/null @@ -1,26 +0,0 @@ -{{- if .Values.podDisruptionBudget.create }} -{{ $component := "controller"}} ---- -apiVersion: policy/v1 -kind: PodDisruptionBudget -metadata: - name: {{ include "kubernetes-ingress-controller.fullname" . }}-controller-pdb - namespace: {{ .Release.Namespace | quote }} - labels: - {{- include "kubernetes-ingress-controller.labels" . | nindent 4 }} - app.kubernetes.io/component: {{ $component }} -spec: - {{- if .Values.podDisruptionBudget.minAvailable }} - minAvailable: {{ .Values.podDisruptionBudget.minAvailable }} - {{- end }} - {{- if .Values.podDisruptionBudget.maxUnavailable }} - maxUnavailable: {{ .Values.podDisruptionBudget.maxUnavailable }} - {{- end }} - selector: - matchLabels: - {{- include "kubernetes-ingress-controller.selectorLabels" . | nindent 6 }} - {{- if .Values.podLabels }} - {{- toYaml .Values.podLabels | nindent 6 }} - {{- end }} - app.kubernetes.io/component: {{ $component }} -{{- end }} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-rbac.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-rbac.yaml deleted file mode 100644 index 82fade5..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-rbac.yaml +++ /dev/null @@ -1,96 +0,0 @@ ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: ngrok-ingress-controller-leader-election-role - namespace: {{ .Release.Namespace }} -rules: -- apiGroups: - - "" - resources: - - configmaps - verbs: - - get - - list - - watch - - create - - update - - patch - - delete -- apiGroups: - - coordination.k8s.io - resources: - - leases - verbs: - - get - - list - - watch - - create - - update - - patch - - delete -- apiGroups: - - "" - resources: - - events - verbs: - - create - - patch ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: ngrok-ingress-controller-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: RoleBinding -metadata: - name: ngrok-ingress-controller-leader-election-rolebinding - namespace: {{ .Release.Namespace }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: ngrok-ingress-controller-leader-election-role -subjects: -- kind: ServiceAccount - name: {{ template "kubernetes-ingress-controller.serviceAccountName" . }} - namespace: {{ .Release.Namespace }} ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: ngrok-ingress-controller-manager-rolebinding -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: ngrok-ingress-controller-manager-role -subjects: -- kind: ServiceAccount - name: {{ template "kubernetes-ingress-controller.serviceAccountName" . }} - namespace: {{ .Release.Namespace }} ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: ngrok-ingress-controller-proxy-rolebinding -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: ngrok-ingress-controller-proxy-role -subjects: -- kind: ServiceAccount - name: {{ template "kubernetes-ingress-controller.serviceAccountName" . }} - namespace: {{ .Release.Namespace }} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-serviceaccount.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-serviceaccount.yaml deleted file mode 100644 index d80a5d8..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/controller-serviceaccount.yaml +++ /dev/null @@ -1,15 +0,0 @@ -{{- if .Values.serviceAccount.create -}} ---- -apiVersion: v1 -kind: ServiceAccount -metadata: - name: {{ template "kubernetes-ingress-controller.serviceAccountName" . }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "kubernetes-ingress-controller.labels" . | nindent 4 }} - app.kubernetes.io/component: controller - {{- if .Values.serviceAccount.annotations }} - annotations: - {{- toYaml .Values.serviceAccount.annotations | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_domains.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_domains.yaml deleted file mode 100644 index ab02a95..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_domains.yaml +++ /dev/null @@ -1,105 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: domains.ingress.k8s.ngrok.com -spec: - group: ingress.k8s.ngrok.com - names: - kind: Domain - listKind: DomainList - plural: domains - singular: domain - scope: Namespaced - versions: - - additionalPrinterColumns: - - description: Domain ID - jsonPath: .status.id - name: ID - type: string - - description: Region - jsonPath: .status.region - name: Region - type: string - - description: Domain - jsonPath: .status.domain - name: Domain - type: string - - description: CNAME Target - jsonPath: .status.cnameTarget - name: CNAME Target - type: string - - description: Age - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: Domain is the Schema for the domains API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: DomainSpec defines the desired state of Domain - properties: - description: - default: Created by kubernetes-ingress-controller - description: Description is a human-readable description of the object - in the ngrok API/Dashboard - type: string - domain: - description: Domain is the domain name to reserve - type: string - metadata: - default: '{"owned-by":"kubernetes-ingress-controller"}' - description: Metadata is a string of arbitrary data associated with - the object in the ngrok API/Dashboard - type: string - region: - description: Region is the region in which to reserve the domain - type: string - required: - - domain - type: object - status: - description: DomainStatus defines the observed state of Domain - properties: - cnameTarget: - description: CNAMETarget is the CNAME target for the domain - type: string - domain: - description: Domain is the domain that was reserved - type: string - id: - description: ID is the unique identifier of the domain - type: string - region: - description: Region is the region in which the domain was created - type: string - uri: - description: URI of the reserved domain API resource - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_httpsedges.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_httpsedges.yaml deleted file mode 100644 index ffafa42..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_httpsedges.yaml +++ /dev/null @@ -1,1062 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: httpsedges.ingress.k8s.ngrok.com -spec: - group: ingress.k8s.ngrok.com - names: - kind: HTTPSEdge - listKind: HTTPSEdgeList - plural: httpsedges - singular: httpsedge - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: HTTPSEdge is the Schema for the httpsedges API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HTTPSEdgeSpec defines the desired state of HTTPSEdge - properties: - description: - default: Created by kubernetes-ingress-controller - description: Description is a human-readable description of the object - in the ngrok API/Dashboard - type: string - hostports: - description: Hostports is a list of hostports served by this edge - items: - type: string - type: array - metadata: - default: '{"owned-by":"kubernetes-ingress-controller"}' - description: Metadata is a string of arbitrary data associated with - the object in the ngrok API/Dashboard - type: string - mutualTLS: - properties: - certificateAuthorities: - description: |- - List of CA IDs that will be used to validate incoming connections to the - edge. - items: - type: string - type: array - type: object - routes: - description: Routes is a list of routes served by this edge - items: - properties: - backend: - description: |- - Backend is the definition for the tunnel group backend - that serves traffic for this edge - properties: - description: - default: Created by kubernetes-ingress-controller - description: Description is a human-readable description - of the object in the ngrok API/Dashboard - type: string - labels: - additionalProperties: - type: string - description: Labels to watch for tunnels on this backend - type: object - metadata: - default: '{"owned-by":"kubernetes-ingress-controller"}' - description: Metadata is a string of arbitrary data associated - with the object in the ngrok API/Dashboard - type: string - type: object - circuitBreaker: - description: CircuitBreaker is a circuit breaker configuration - to apply to this route - properties: - errorThresholdPercentage: - anyOf: - - type: integer - - type: string - description: Error threshold percentage should be between - 0 - 1.0, not 0-100.0 - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - numBuckets: - description: Integer number of buckets into which metrics - are retained. Max 128. - format: int32 - maximum: 128 - minimum: 1 - type: integer - rollingWindow: - description: Statistical rolling window duration that metrics - are retained for. - format: duration - type: string - trippedDuration: - description: Duration after which the circuit is tripped - to wait before re-evaluating upstream health - format: duration - type: string - volumeThreshold: - description: |- - Integer number of requests in a rolling window that will trip the circuit. - Helpful if traffic volume is low. - format: int32 - type: integer - type: object - compression: - description: Compression is whether or not to enable compression - for this route - properties: - enabled: - description: Enabled is whether or not to enable compression - for this endpoint - type: boolean - type: object - description: - default: Created by kubernetes-ingress-controller - description: Description is a human-readable description of - the object in the ngrok API/Dashboard - type: string - headers: - description: Headers are request/response headers to apply to - this route - properties: - request: - description: Request headers are the request headers module - configuration or null - properties: - add: - additionalProperties: - type: string - description: |- - a map of header key to header value that will be injected into the HTTP Request - before being sent to the upstream application server - type: object - remove: - description: |- - a list of header names that will be removed from the HTTP Request before being - sent to the upstream application server - items: - type: string - type: array - type: object - response: - description: Response headers are the response headers module - configuration or null - properties: - add: - additionalProperties: - type: string - description: |- - a map of header key to header value that will be injected into the HTTP Response - returned to the HTTP client - type: object - remove: - description: |- - a list of header names that will be removed from the HTTP Response returned to - the HTTP client - items: - type: string - type: array - type: object - type: object - ipRestriction: - description: IPRestriction is an IPRestriction to apply to this - route - properties: - policies: - items: - type: string - type: array - type: object - match: - description: Match is the value to match against the request - path - type: string - matchType: - description: 'MatchType is the type of match to use for this - route. Valid values are:' - enum: - - exact_path - - path_prefix - type: string - metadata: - default: '{"owned-by":"kubernetes-ingress-controller"}' - description: Metadata is a string of arbitrary data associated - with the object in the ngrok API/Dashboard - type: string - oauth: - description: OAuth configuration to apply to this route - properties: - amazon: - description: configuration for using amazon as the identity - provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - type: object - facebook: - description: configuration for using facebook as the identity - provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - type: object - github: - description: configuration for using github as the identity - provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - organizations: - description: |- - a list of github org identifiers. users who are members of any of the listed - organizations will be allowed access. identifiers should be the organization's - 'slug' - items: - type: string - type: array - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - teams: - description: |- - a list of github teams identifiers. users will be allowed access to the endpoint - if they are a member of any of these teams. identifiers should be in the 'slug' - format qualified with the org name, e.g. org-name/team-name - items: - type: string - type: array - type: object - gitlab: - description: configuration for using gitlab as the identity - provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - type: object - google: - description: configuration for using google as the identity - provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - type: object - linkedin: - description: configuration for using linkedin as the identity - provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - type: object - microsoft: - description: configuration for using microsoft as the identity - provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - type: object - twitch: - description: configuration for using twitch as the identity - provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - type: object - type: object - oidc: - description: OIDC is the OpenID Connect configuration to apply - to this route - properties: - clientId: - description: The OIDC app's client ID and OIDC audience. - type: string - clientSecret: - description: The OIDC app's client secret. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - issuer: - description: URL of the OIDC "OpenID provider". This is - the base URL used for discovery. - type: string - maximumDuration: - description: |- - The maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: The set of scopes to request from the OIDC - identity provider. - items: - type: string - type: array - type: object - policy: - description: raw json policy string that was applied to the - ngrok API - type: object - x-kubernetes-preserve-unknown-fields: true - saml: - description: SAML is the SAML configuration to apply to this - route - properties: - allowIdpInitiated: - description: |- - If true, the IdP may initiate a login directly (e.g. the user does not need to - visit the endpoint first and then be redirected). The IdP should set the - RelayState parameter to the target URL of the resource they want the user to be - redirected to after the SAML login assertion has been processed. - type: boolean - authorizedGroups: - description: |- - If present, only users who are a member of one of the listed groups may access - the target endpoint. - items: - type: string - type: array - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - forceAuthn: - description: |- - If true, indicates that whenever we redirect a user to the IdP for - authentication that the IdP must prompt the user for authentication credentials - even if the user already has a valid session with the IdP. - type: boolean - idpMetadata: - description: |- - The full XML IdP EntityDescriptor. Your IdP may provide this to you as a a file - to download or as a URL. - type: string - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - The maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - nameidFormat: - description: |- - Defines the name identifier format the SP expects the IdP to use in its - assertions to identify subjects. If unspecified, a default value of - urn:oasis:names:tc:SAML:2.0:nameid-format:persistent will be used. A subset of - the allowed values enumerated by the SAML specification are supported. - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - type: object - webhookVerification: - description: WebhookVerification is webhook verification configuration - to apply to this route - properties: - provider: - description: |- - a string indicating which webhook provider will be sending webhooks to this - endpoint. Value must be one of the supported providers defined at - https://ngrok.com/docs/http/webhook-verification/#supported-providers - type: string - secret: - description: |- - SecretRef is a reference to a secret containing the secret used to validate - requests from the given provider. All providers except AWS SNS require a secret - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - type: object - required: - - match - - matchType - type: object - type: array - tlsTermination: - description: TLSTermination is the TLS termination configuration for - this edge - properties: - minVersion: - description: MinVersion is the minimum TLS version to allow for - connections to the edge - type: string - type: object - type: object - status: - description: HTTPSEdgeStatus defines the observed state of HTTPSEdge - properties: - id: - description: ID is the unique identifier for this edge - type: string - routes: - items: - properties: - backend: - description: |- - Backend stores the status of the tunnel group backend, - mainly the ID of the backend - properties: - id: - description: ID is the unique identifier for this backend - type: string - type: object - id: - description: ID is the unique identifier for this route - type: string - match: - type: string - matchType: - type: string - uri: - description: URI is the URI for this route - type: string - type: object - type: array - uri: - description: URI is the URI for this edge - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_ippolicies.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_ippolicies.yaml deleted file mode 100644 index 1508f76..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_ippolicies.yaml +++ /dev/null @@ -1,109 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: ippolicies.ingress.k8s.ngrok.com -spec: - group: ingress.k8s.ngrok.com - names: - kind: IPPolicy - listKind: IPPolicyList - plural: ippolicies - singular: ippolicy - scope: Namespaced - versions: - - additionalPrinterColumns: - - description: IPPolicy ID - jsonPath: .status.id - name: ID - type: string - - description: Age - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: IPPolicy is the Schema for the ippolicies API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: IPPolicySpec defines the desired state of IPPolicy - properties: - description: - default: Created by kubernetes-ingress-controller - description: Description is a human-readable description of the object - in the ngrok API/Dashboard - type: string - metadata: - default: '{"owned-by":"kubernetes-ingress-controller"}' - description: Metadata is a string of arbitrary data associated with - the object in the ngrok API/Dashboard - type: string - rules: - description: Rules is a list of rules that belong to the policy - items: - properties: - action: - enum: - - allow - - deny - type: string - cidr: - type: string - description: - default: Created by kubernetes-ingress-controller - description: Description is a human-readable description of - the object in the ngrok API/Dashboard - type: string - metadata: - default: '{"owned-by":"kubernetes-ingress-controller"}' - description: Metadata is a string of arbitrary data associated - with the object in the ngrok API/Dashboard - type: string - type: object - type: array - type: object - status: - description: IPPolicyStatus defines the observed state of IPPolicy - properties: - id: - description: |- - INSERT ADDITIONAL STATUS FIELD - define observed state of cluster - Important: Run "make" to regenerate code after modifying this file - type: string - rules: - items: - properties: - action: - type: string - cidr: - type: string - id: - type: string - type: object - type: array - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_ngrokmodulesets.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_ngrokmodulesets.yaml deleted file mode 100644 index ef4a5a8..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_ngrokmodulesets.yaml +++ /dev/null @@ -1,1008 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: ngrokmodulesets.ingress.k8s.ngrok.com -spec: - group: ingress.k8s.ngrok.com - names: - kind: NgrokModuleSet - listKind: NgrokModuleSetList - plural: ngrokmodulesets - singular: ngrokmoduleset - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: NgrokModuleSet is the Schema for the ngrokmodules API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - modules: - properties: - circuitBreaker: - description: CircuitBreaker configuration for this module set - properties: - errorThresholdPercentage: - anyOf: - - type: integer - - type: string - description: Error threshold percentage should be between 0 - - 1.0, not 0-100.0 - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - numBuckets: - description: Integer number of buckets into which metrics are - retained. Max 128. - format: int32 - maximum: 128 - minimum: 1 - type: integer - rollingWindow: - description: Statistical rolling window duration that metrics - are retained for. - format: duration - type: string - trippedDuration: - description: Duration after which the circuit is tripped to wait - before re-evaluating upstream health - format: duration - type: string - volumeThreshold: - description: |- - Integer number of requests in a rolling window that will trip the circuit. - Helpful if traffic volume is low. - format: int32 - type: integer - type: object - compression: - description: Compression configuration for this module set - properties: - enabled: - description: Enabled is whether or not to enable compression for - this endpoint - type: boolean - type: object - headers: - description: Header configuration for this module set - properties: - request: - description: Request headers are the request headers module configuration - or null - properties: - add: - additionalProperties: - type: string - description: |- - a map of header key to header value that will be injected into the HTTP Request - before being sent to the upstream application server - type: object - remove: - description: |- - a list of header names that will be removed from the HTTP Request before being - sent to the upstream application server - items: - type: string - type: array - type: object - response: - description: Response headers are the response headers module - configuration or null - properties: - add: - additionalProperties: - type: string - description: |- - a map of header key to header value that will be injected into the HTTP Response - returned to the HTTP client - type: object - remove: - description: |- - a list of header names that will be removed from the HTTP Response returned to - the HTTP client - items: - type: string - type: array - type: object - type: object - ipRestriction: - description: IPRestriction configuration for this module set - properties: - policies: - items: - type: string - type: array - type: object - mutualTLS: - description: MutualTLS configuration for this module set - properties: - certificateAuthorities: - description: |- - List of CA IDs that will be used to validate incoming connections to the - edge. - items: - type: string - type: array - type: object - oauth: - description: OAuth configuration for this module set - properties: - amazon: - description: configuration for using amazon as the identity provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - type: object - facebook: - description: configuration for using facebook as the identity - provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - type: object - github: - description: configuration for using github as the identity provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - organizations: - description: |- - a list of github org identifiers. users who are members of any of the listed - organizations will be allowed access. identifiers should be the organization's - 'slug' - items: - type: string - type: array - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - teams: - description: |- - a list of github teams identifiers. users will be allowed access to the endpoint - if they are a member of any of these teams. identifiers should be in the 'slug' - format qualified with the org name, e.g. org-name/team-name - items: - type: string - type: array - type: object - gitlab: - description: configuration for using gitlab as the identity provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - type: object - google: - description: configuration for using google as the identity provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - type: object - linkedin: - description: configuration for using linkedin as the identity - provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - type: object - microsoft: - description: configuration for using microsoft as the identity - provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - type: object - twitch: - description: configuration for using twitch as the identity provider - properties: - authCheckInterval: - description: |- - Duration after which ngrok guarantees it will refresh user - state from the identity provider and recheck whether the user is still - authorized to access the endpoint. This is the preferred tunable to use to - enforce a minimum amount of time after which a revoked user will no longer be - able to access the resource. - format: duration - type: string - clientId: - description: |- - the OAuth app client ID. retrieve it from the identity provider's dashboard - where you created your own OAuth app. optional. if unspecified, ngrok will use - its own managed oauth application which has additional restrictions. see the - OAuth module docs for more details. if present, clientSecret must be present as - well. - type: string - clientSecret: - description: |- - the OAuth app client secret. retrieve if from the identity provider's dashboard - where you created your own OAuth app. optional, see all of the caveats in the - docs for clientId. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - emailAddresses: - description: |- - a list of email addresses of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - emailDomains: - description: |- - a list of email domains of users authenticated by identity provider who are - allowed access to the endpoint - items: - type: string - type: array - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - Integer number of seconds of the maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: |- - a list of provider-specific OAuth scopes with the permissions your OAuth app - would like to ask for. these may not be set if you are using the ngrok-managed - oauth app (i.e. you must pass both client_id and client_secret to set scopes) - items: - type: string - type: array - type: object - type: object - oidc: - description: OIDC configuration for this module set - properties: - clientId: - description: The OIDC app's client ID and OIDC audience. - type: string - clientSecret: - description: The OIDC app's client secret. - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - issuer: - description: URL of the OIDC "OpenID provider". This is the base - URL used for discovery. - type: string - maximumDuration: - description: |- - The maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - scopes: - description: The set of scopes to request from the OIDC identity - provider. - items: - type: string - type: array - type: object - policy: - description: Policy configuration for this module set - properties: - enabled: - description: Determines if the rule will be applied to traffic - type: boolean - inbound: - description: Inbound traffic rule - items: - properties: - actions: - description: Actions - items: - properties: - config: - type: object - x-kubernetes-preserve-unknown-fields: true - type: - type: string - type: object - type: array - expressions: - description: Expressions - items: - type: string - type: array - name: - description: Name - type: string - type: object - type: array - outbound: - description: Outbound traffic rule - items: - properties: - actions: - description: Actions - items: - properties: - config: - type: object - x-kubernetes-preserve-unknown-fields: true - type: - type: string - type: object - type: array - expressions: - description: Expressions - items: - type: string - type: array - name: - description: Name - type: string - type: object - type: array - type: object - saml: - description: SAML configuration for this module set - properties: - allowIdpInitiated: - description: |- - If true, the IdP may initiate a login directly (e.g. the user does not need to - visit the endpoint first and then be redirected). The IdP should set the - RelayState parameter to the target URL of the resource they want the user to be - redirected to after the SAML login assertion has been processed. - type: boolean - authorizedGroups: - description: |- - If present, only users who are a member of one of the listed groups may access - the target endpoint. - items: - type: string - type: array - cookiePrefix: - description: |- - the prefix of the session cookie that ngrok sets on the http client to cache - authentication. default is 'ngrok.' - type: string - forceAuthn: - description: |- - If true, indicates that whenever we redirect a user to the IdP for - authentication that the IdP must prompt the user for authentication credentials - even if the user already has a valid session with the IdP. - type: boolean - idpMetadata: - description: |- - The full XML IdP EntityDescriptor. Your IdP may provide this to you as a a file - to download or as a URL. - type: string - inactivityTimeout: - description: |- - Duration of inactivity after which if the user has not accessed - the endpoint, their session will time out and they will be forced to - reauthenticate. - format: duration - type: string - maximumDuration: - description: |- - The maximum duration of an authenticated session. - After this period is exceeded, a user must reauthenticate. - format: duration - type: string - nameidFormat: - description: |- - Defines the name identifier format the SP expects the IdP to use in its - assertions to identify subjects. If unspecified, a default value of - urn:oasis:names:tc:SAML:2.0:nameid-format:persistent will be used. A subset of - the allowed values enumerated by the SAML specification are supported. - type: string - optionsPassthrough: - description: |- - Do not enforce authentication on HTTP OPTIONS requests. necessary if you are - supporting CORS. - type: boolean - type: object - tlsTermination: - description: TLSTermination configuration for this module set - properties: - minVersion: - description: MinVersion is the minimum TLS version to allow for - connections to the edge - type: string - terminateAt: - description: |- - TerminateAt determines where the TLS connection should be terminated. - "edge" if the ngrok edge should terminate TLS traffic, "upstream" if TLS - traffic should be passed through to the upstream ngrok agent / - application server for termination. - type: string - type: object - webhookVerification: - description: WebhookVerification configuration for this module set - properties: - provider: - description: |- - a string indicating which webhook provider will be sending webhooks to this - endpoint. Value must be one of the supported providers defined at - https://ngrok.com/docs/http/webhook-verification/#supported-providers - type: string - secret: - description: |- - SecretRef is a reference to a secret containing the secret used to validate - requests from the given provider. All providers except AWS SNS require a secret - properties: - key: - description: Key in the secret to use - type: string - name: - description: Name of the Kubernetes secret - type: string - type: object - type: object - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_tcpedges.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_tcpedges.yaml deleted file mode 100644 index 0201620..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_tcpedges.yaml +++ /dev/null @@ -1,132 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: tcpedges.ingress.k8s.ngrok.com -spec: - group: ingress.k8s.ngrok.com - names: - kind: TCPEdge - listKind: TCPEdgeList - plural: tcpedges - singular: tcpedge - scope: Namespaced - versions: - - additionalPrinterColumns: - - description: Domain ID - jsonPath: .status.id - name: ID - type: string - - description: Hostports - jsonPath: .status.hostports - name: Hostports - type: string - - description: Tunnel Group Backend ID - jsonPath: .status.backend.id - name: Backend ID - type: string - - description: Age - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: TCPEdge is the Schema for the tcpedges API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: TCPEdgeSpec defines the desired state of TCPEdge - properties: - backend: - description: |- - Backend is the definition for the tunnel group backend - that serves traffic for this edge - properties: - description: - default: Created by kubernetes-ingress-controller - description: Description is a human-readable description of the - object in the ngrok API/Dashboard - type: string - labels: - additionalProperties: - type: string - description: Labels to watch for tunnels on this backend - type: object - metadata: - default: '{"owned-by":"kubernetes-ingress-controller"}' - description: Metadata is a string of arbitrary data associated - with the object in the ngrok API/Dashboard - type: string - type: object - description: - default: Created by kubernetes-ingress-controller - description: Description is a human-readable description of the object - in the ngrok API/Dashboard - type: string - ipRestriction: - description: IPRestriction is an IPRestriction to apply to this edge - properties: - policies: - items: - type: string - type: array - type: object - metadata: - default: '{"owned-by":"kubernetes-ingress-controller"}' - description: Metadata is a string of arbitrary data associated with - the object in the ngrok API/Dashboard - type: string - policy: - description: raw json policy string that was applied to the ngrok - API - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - status: - description: TCPEdgeStatus defines the observed state of TCPEdge - properties: - backend: - description: |- - Backend stores the status of the tunnel group backend, - mainly the ID of the backend - properties: - id: - description: ID is the unique identifier for this backend - type: string - type: object - hostports: - description: Hostports served by this edge - items: - type: string - type: array - id: - description: ID is the unique identifier for this edge - type: string - uri: - description: URI is the URI of the edge - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_tlsedges.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_tlsedges.yaml deleted file mode 100644 index 4aa99d3..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_tlsedges.yaml +++ /dev/null @@ -1,166 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: tlsedges.ingress.k8s.ngrok.com -spec: - group: ingress.k8s.ngrok.com - names: - kind: TLSEdge - listKind: TLSEdgeList - plural: tlsedges - singular: tlsedge - scope: Namespaced - versions: - - additionalPrinterColumns: - - description: Domain ID - jsonPath: .status.id - name: ID - type: string - - description: Hostports - jsonPath: .status.hostports - name: Hostports - type: string - - description: Tunnel Group Backend ID - jsonPath: .status.backend.id - name: Backend ID - type: string - - description: Age - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: TLSEdge is the Schema for the tlsedges API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: TLSEdgeSpec defines the desired state of TLSEdge - properties: - backend: - description: |- - Backend is the definition for the tunnel group backend - that serves traffic for this edge - properties: - description: - default: Created by kubernetes-ingress-controller - description: Description is a human-readable description of the - object in the ngrok API/Dashboard - type: string - labels: - additionalProperties: - type: string - description: Labels to watch for tunnels on this backend - type: object - metadata: - default: '{"owned-by":"kubernetes-ingress-controller"}' - description: Metadata is a string of arbitrary data associated - with the object in the ngrok API/Dashboard - type: string - type: object - description: - default: Created by kubernetes-ingress-controller - description: Description is a human-readable description of the object - in the ngrok API/Dashboard - type: string - hostports: - description: Hostports is a list of hostports served by this edge - items: - type: string - type: array - ipRestriction: - description: IPRestriction is an IPRestriction to apply to this edge - properties: - policies: - items: - type: string - type: array - type: object - metadata: - default: '{"owned-by":"kubernetes-ingress-controller"}' - description: Metadata is a string of arbitrary data associated with - the object in the ngrok API/Dashboard - type: string - mutualTls: - properties: - certificateAuthorities: - description: |- - List of CA IDs that will be used to validate incoming connections to the - edge. - items: - type: string - type: array - type: object - policy: - description: raw json policy string that was applied to the ngrok - API - type: object - x-kubernetes-preserve-unknown-fields: true - tlsTermination: - properties: - minVersion: - description: MinVersion is the minimum TLS version to allow for - connections to the edge - type: string - terminateAt: - description: |- - TerminateAt determines where the TLS connection should be terminated. - "edge" if the ngrok edge should terminate TLS traffic, "upstream" if TLS - traffic should be passed through to the upstream ngrok agent / - application server for termination. - type: string - type: object - type: object - status: - description: TLSEdgeStatus defines the observed state of TLSEdge - properties: - backend: - description: |- - Backend stores the status of the tunnel group backend, - mainly the ID of the backend - properties: - id: - description: ID is the unique identifier for this backend - type: string - type: object - cnameTargets: - additionalProperties: - type: string - description: Map of hostports to the ngrok assigned CNAME targets - type: object - hostports: - description: Hostports served by this edge - items: - type: string - type: array - id: - description: ID is the unique identifier for this edge - type: string - uri: - description: URI is the URI of the edge - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_tunnels.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_tunnels.yaml deleted file mode 100644 index e440dbd..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ingress.k8s.ngrok.com_tunnels.yaml +++ /dev/null @@ -1,78 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: tunnels.ingress.k8s.ngrok.com -spec: - group: ingress.k8s.ngrok.com - names: - kind: Tunnel - listKind: TunnelList - plural: tunnels - singular: tunnel - scope: Namespaced - versions: - - additionalPrinterColumns: - - description: Service/port to forward to - jsonPath: .spec.forwardsTo - name: ForwardsTo - type: string - - description: Age - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: Tunnel is the Schema for the tunnels API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: TunnelSpec defines the desired state of Tunnel - properties: - appProtocol: - description: The appProtocol for the backend. Currently only supports - `http2` - type: string - backend: - description: The configuration for backend connections to services - properties: - protocol: - type: string - type: object - forwardsTo: - description: ForwardsTo is the name and port of the service to forward - traffic to - type: string - labels: - additionalProperties: - type: string - description: Labels are key/value pairs that are attached to the tunnel - type: object - type: object - status: - description: TunnelStatus defines the observed state of Tunnel - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ngrok.k8s.ngrok.com_ngroktrafficpolicies.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ngrok.k8s.ngrok.com_ngroktrafficpolicies.yaml deleted file mode 100644 index 463f927..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/crds/ngrok.k8s.ngrok.com_ngroktrafficpolicies.yaml +++ /dev/null @@ -1,62 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: ngroktrafficpolicies.ngrok.k8s.ngrok.com -spec: - group: ngrok.k8s.ngrok.com - names: - kind: NgrokTrafficPolicy - listKind: NgrokTrafficPolicyList - plural: ngroktrafficpolicies - singular: ngroktrafficpolicy - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: NgrokTrafficPolicy is the Schema for the ngroktrafficpolicies - API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: NgrokTrafficPolicySpec defines the desired state of NgrokTrafficPolicy - properties: - policy: - description: The raw json encoded policy that was applied to the ngrok - API - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - status: - description: NgrokTrafficPolicyStatus defines the observed state of NgrokTrafficPolicy - properties: - policy: - description: The raw json encoded policy that was applied to the ngrok - API - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/credentials-secret.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/credentials-secret.yaml deleted file mode 100644 index c6c7286..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/credentials-secret.yaml +++ /dev/null @@ -1,11 +0,0 @@ -{{- if or (not (empty .Values.credentials.apiKey)) (not (empty .Values.credentials.authtoken)) }} -apiVersion: v1 -kind: Secret -metadata: - name: {{ include "kubernetes-ingress-controller.credentialsSecretName" .}} - namespace: {{ .Release.Namespace }} -type: Opaque -data: - API_KEY: {{ required "An ngrok API key is required" .Values.credentials.apiKey | b64enc }} - AUTHTOKEN: {{ required "An ngrok Authtoken is required" .Values.credentials.authtoken | b64enc }} -{{ end }} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/ingress-class.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/ingress-class.yaml deleted file mode 100644 index 0932b47..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/ingress-class.yaml +++ /dev/null @@ -1,15 +0,0 @@ -{{- if .Values.ingressClass.create -}} -apiVersion: networking.k8s.io/v1 -kind: IngressClass -metadata: - labels: - {{- include "kubernetes-ingress-controller.labels" . | nindent 4 }} - app.kubernetes.io/component: controller - name: {{ .Values.ingressClass.name }} - {{- if .Values.ingressClass.default }} - annotations: - ingressclass.kubernetes.io/is-default-class: "true" - {{- end }} -spec: - controller: {{ .Values.controllerName }} -{{- end}} diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/domain_editor_role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/domain_editor_role.yaml deleted file mode 100644 index 88e527a..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/domain_editor_role.yaml +++ /dev/null @@ -1,27 +0,0 @@ -# permissions for end users to edit domains. -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - {{- include "kubernetes-ingress-controller.labels" . | nindent 4 }} - app.kubernetes.io/component: rbac - name: {{ include "kubernetes-ingress-controller.fullname" . }}-domain-editor-role -rules: -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - domains - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - domains/status - verbs: - - get diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/domain_viewer_role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/domain_viewer_role.yaml deleted file mode 100644 index bf5f519..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/domain_viewer_role.yaml +++ /dev/null @@ -1,23 +0,0 @@ -# permissions for end users to view domains. -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - {{- include "kubernetes-ingress-controller.labels" . | nindent 4 }} - app.kubernetes.io/component: rbac - name: {{ include "kubernetes-ingress-controller.fullname" . }}-domain-viewer-role -rules: -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - domains - verbs: - - get - - list - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - domains/status - verbs: - - get diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/httpsedge_editor_role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/httpsedge_editor_role.yaml deleted file mode 100644 index d4d3831..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/httpsedge_editor_role.yaml +++ /dev/null @@ -1,31 +0,0 @@ -# permissions for end users to edit httpsedges. -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/name: clusterrole - app.kubernetes.io/instance: httpsedge-editor-role - app.kubernetes.io/component: rbac - app.kubernetes.io/created-by: ngrok-ingress-controller - app.kubernetes.io/part-of: ngrok-ingress-controller - app.kubernetes.io/managed-by: kustomize - name: httpsedge-editor-role -rules: -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - httpsedges - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - httpsedges/status - verbs: - - get diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/httpsedge_viewer_role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/httpsedge_viewer_role.yaml deleted file mode 100644 index 8d01cd7..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/httpsedge_viewer_role.yaml +++ /dev/null @@ -1,27 +0,0 @@ -# permissions for end users to view httpsedges. -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/name: clusterrole - app.kubernetes.io/instance: httpsedge-viewer-role - app.kubernetes.io/component: rbac - app.kubernetes.io/created-by: ngrok-ingress-controller - app.kubernetes.io/part-of: ngrok-ingress-controller - app.kubernetes.io/managed-by: kustomize - name: httpsedge-viewer-role -rules: -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - httpsedges - verbs: - - get - - list - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - httpsedges/status - verbs: - - get diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ippolicy_editor_role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ippolicy_editor_role.yaml deleted file mode 100644 index a8aa5eb..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ippolicy_editor_role.yaml +++ /dev/null @@ -1,31 +0,0 @@ -# permissions for end users to edit ippolicies. -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/name: clusterrole - app.kubernetes.io/instance: ippolicy-editor-role - app.kubernetes.io/component: rbac - app.kubernetes.io/created-by: ngrok-ingress-controller - app.kubernetes.io/part-of: ngrok-ingress-controller - app.kubernetes.io/managed-by: kustomize - name: ippolicy-editor-role -rules: -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - ippolicies - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - ippolicies/status - verbs: - - get diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ippolicy_viewer_role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ippolicy_viewer_role.yaml deleted file mode 100644 index a83a34a..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ippolicy_viewer_role.yaml +++ /dev/null @@ -1,27 +0,0 @@ -# permissions for end users to view ippolicies. -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/name: clusterrole - app.kubernetes.io/instance: ippolicy-viewer-role - app.kubernetes.io/component: rbac - app.kubernetes.io/created-by: ngrok-ingress-controller - app.kubernetes.io/part-of: ngrok-ingress-controller - app.kubernetes.io/managed-by: kustomize - name: ippolicy-viewer-role -rules: -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - ippolicies - verbs: - - get - - list - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - ippolicies/status - verbs: - - get diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ngrokmoduleset_editor_role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ngrokmoduleset_editor_role.yaml deleted file mode 100644 index c8d5f16..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ngrokmoduleset_editor_role.yaml +++ /dev/null @@ -1,31 +0,0 @@ -# permissions for end users to edit ngrokmodulesets. -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/name: clusterrole - app.kubernetes.io/instance: ngrokmoduleset-editor-role - app.kubernetes.io/component: rbac - app.kubernetes.io/created-by: kubernetes-ingress-controller - app.kubernetes.io/part-of: kubernetes-ingress-controller - app.kubernetes.io/managed-by: kustomize - name: ngrokmoduleset-editor-role -rules: -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - ngrokmodulesets - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - ngrokmodulesets/status - verbs: - - get diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ngrokmoduleset_viewer_role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ngrokmoduleset_viewer_role.yaml deleted file mode 100644 index a9948ad..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/ngrokmoduleset_viewer_role.yaml +++ /dev/null @@ -1,27 +0,0 @@ -# permissions for end users to view ngrokmodulesets. -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/name: clusterrole - app.kubernetes.io/instance: ngrokmoduleset-viewer-role - app.kubernetes.io/component: rbac - app.kubernetes.io/created-by: kubernetes-ingress-controller - app.kubernetes.io/part-of: kubernetes-ingress-controller - app.kubernetes.io/managed-by: kustomize - name: ngrokmoduleset-viewer-role -rules: -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - ngrokmodulesets - verbs: - - get - - list - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - ngrokmodulesets/status - verbs: - - get diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/role.yaml deleted file mode 100644 index 4b05ac6..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/role.yaml +++ /dev/null @@ -1,330 +0,0 @@ ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: ngrok-ingress-controller-manager-role -rules: -- apiGroups: - - "" - resources: - - configmaps - verbs: - - create - - delete - - get - - list - - update - - watch -- apiGroups: - - "" - resources: - - events - verbs: - - create - - patch -- apiGroups: - - "" - resources: - - namespaces - verbs: - - get - - list - - update - - watch -- apiGroups: - - "" - resources: - - secrets - verbs: - - get - - list - - watch -- apiGroups: - - "" - resources: - - services - verbs: - - get - - list - - update - - watch -- apiGroups: - - "" - resources: - - services/status - verbs: - - get - - list - - patch - - update - - watch -- apiGroups: - - gateway.networking.k8s.io - resources: - - gatewayclasses - verbs: - - get - - list - - update - - watch -- apiGroups: - - gateway.networking.k8s.io - resources: - - gatewayclasses/status - verbs: - - get - - list - - update - - watch -- apiGroups: - - gateway.networking.k8s.io - resources: - - gateways - verbs: - - get - - list - - update - - watch -- apiGroups: - - gateway.networking.k8s.io - resources: - - gateways/status - verbs: - - get - - list - - update - - watch -- apiGroups: - - gateway.networking.k8s.io - resources: - - httproutes - verbs: - - get - - list - - update - - watch -- apiGroups: - - gateway.networking.k8s.io - resources: - - httproutes/status - verbs: - - get - - list - - update - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - domains - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - domains/finalizers - verbs: - - update -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - domains/status - verbs: - - get - - patch - - update -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - httpsedges - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - httpsedges/finalizers - verbs: - - update -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - httpsedges/status - verbs: - - get - - patch - - update -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - ippolicies - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - ippolicies/finalizers - verbs: - - update -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - ippolicies/status - verbs: - - get - - patch - - update -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - ngrokmodulesets - verbs: - - get - - list - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tcpedges - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tcpedges/finalizers - verbs: - - update -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tcpedges/status - verbs: - - get - - patch - - update -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tlsedges - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tlsedges/finalizers - verbs: - - update -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tlsedges/status - verbs: - - get - - patch - - update -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tunnels - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tunnels/finalizers - verbs: - - update -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tunnels/status - verbs: - - get - - patch - - update -- apiGroups: - - networking.k8s.io - resources: - - ingressclasses - verbs: - - get - - list - - watch -- apiGroups: - - networking.k8s.io - resources: - - ingresses - verbs: - - get - - list - - update - - watch -- apiGroups: - - networking.k8s.io - resources: - - ingresses/status - verbs: - - get - - list - - update - - watch -- apiGroups: - - ngrok.k8s.ngrok.com - resources: - - ngroktrafficpolicies - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - ngrok.k8s.ngrok.com - resources: - - ngroktrafficpolicies/finalizers - verbs: - - update -- apiGroups: - - ngrok.k8s.ngrok.com - resources: - - ngroktrafficpolicies/status - verbs: - - get - - patch - - update diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tcpedge_editor_role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tcpedge_editor_role.yaml deleted file mode 100644 index 2e49847..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tcpedge_editor_role.yaml +++ /dev/null @@ -1,31 +0,0 @@ -# permissions for end users to edit tcpedges. -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/name: clusterrole - app.kubernetes.io/instance: tcpedge-editor-role - app.kubernetes.io/component: rbac - app.kubernetes.io/created-by: ngrok-ingress-controller - app.kubernetes.io/part-of: ngrok-ingress-controller - app.kubernetes.io/managed-by: kustomize - name: tcpedge-editor-role -rules: -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tcpedges - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tcpedges/status - verbs: - - get diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tcpedge_viewer_role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tcpedge_viewer_role.yaml deleted file mode 100644 index b8eb5ef..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tcpedge_viewer_role.yaml +++ /dev/null @@ -1,27 +0,0 @@ -# permissions for end users to view tcpedges. -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/name: clusterrole - app.kubernetes.io/instance: tcpedge-viewer-role - app.kubernetes.io/component: rbac - app.kubernetes.io/created-by: ngrok-ingress-controller - app.kubernetes.io/part-of: ngrok-ingress-controller - app.kubernetes.io/managed-by: kustomize - name: tcpedge-viewer-role -rules: -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tcpedges - verbs: - - get - - list - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tcpedges/status - verbs: - - get diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tlsedge_editor_role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tlsedge_editor_role.yaml deleted file mode 100644 index 9d05231..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tlsedge_editor_role.yaml +++ /dev/null @@ -1,31 +0,0 @@ -# permissions for end users to edit tlsedges. -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/name: clusterrole - app.kubernetes.io/instance: tlsedge-editor-role - app.kubernetes.io/component: rbac - app.kubernetes.io/created-by: ngrok-ingress-controller - app.kubernetes.io/part-of: ngrok-ingress-controller - app.kubernetes.io/managed-by: kustomize - name: tlsedge-editor-role -rules: -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tlsedges - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tlsedges/status - verbs: - - get diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tlsedge_viewer_role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tlsedge_viewer_role.yaml deleted file mode 100644 index a9eb99d..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tlsedge_viewer_role.yaml +++ /dev/null @@ -1,27 +0,0 @@ -# permissions for end users to view tlsedges. -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/name: clusterrole - app.kubernetes.io/instance: tlsedge-viewer-role - app.kubernetes.io/component: rbac - app.kubernetes.io/created-by: ngrok-ingress-controller - app.kubernetes.io/part-of: ngrok-ingress-controller - app.kubernetes.io/managed-by: kustomize - name: tlsedge-viewer-role -rules: -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tlsedges - verbs: - - get - - list - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tlsedges/status - verbs: - - get diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tunnel_editor_role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tunnel_editor_role.yaml deleted file mode 100644 index ab72751..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tunnel_editor_role.yaml +++ /dev/null @@ -1,27 +0,0 @@ -# permissions for end users to edit tunnels. -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - {{- include "kubernetes-ingress-controller.labels" . | nindent 4 }} - app.kubernetes.io/component: rbac - name: {{ include "kubernetes-ingress-controller.fullname" . }}-tunnel-editor-role -rules: -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tunnels - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tunnels/status - verbs: - - get diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tunnel_viewer_role.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tunnel_viewer_role.yaml deleted file mode 100644 index dfdb4b6..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/templates/rbac/tunnel_viewer_role.yaml +++ /dev/null @@ -1,23 +0,0 @@ -# permissions for end users to view tunnels. -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - {{- include "kubernetes-ingress-controller.labels" . | nindent 4 }} - app.kubernetes.io/component: rbac - name: {{ include "kubernetes-ingress-controller.fullname" . }}-tunnel-viewer-role -rules: -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tunnels - verbs: - - get - - list - - watch -- apiGroups: - - ingress.k8s.ngrok.com - resources: - - tunnels/status - verbs: - - get diff --git a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/values.yaml b/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/values.yaml deleted file mode 100644 index 88c0b7c..0000000 --- a/charts/kubernetes-ingress-controller/kubernetes-ingress-controller/values.yaml +++ /dev/null @@ -1,202 +0,0 @@ -## @section Common parameters -## - -## @param nameOverride String to partially override generated resource names -## @param fullnameOverride String to fully override generated resource names -## @param commonLabels Labels to add to all deployed objects -## @param commonAnnotations Annotations to add to all deployed objects -## -nameOverride: "" -fullnameOverride: "" -commonLabels: {} -commonAnnotations: {} - -## @section Controller parameters -## - -## @param podAnnotations Used to apply custom annotations to the ingress pods. -## @param podLabels Used to apply custom labels to the ingress pods. -## -podAnnotations: {} -podLabels: {} - -## @param replicaCount The number of controllers to run. -## A minimum of 2 is recommended in production for HA. -## -replicaCount: 1 - -## @param image.registry The ngrok ingress controller image registry. -## @param image.repository The ngrok ingress controller image repository. -## @param image.tag The ngrok ingress controller image tag. Defaults to the chart's appVersion if not specified -## @param image.pullPolicy The ngrok ingress controller image pull policy. -## @param image.pullSecrets An array of imagePullSecrets to be used when pulling the image. -image: - registry: docker.io - repository: ngrok/kubernetes-ingress-controller - tag: "" - pullPolicy: IfNotPresent - ## Example - ## pullSecrets: - ## - name: my-imagepull-secret - ## - pullSecrets: [] - -## @param ingressClass.name The name of the ingress class to use. -## @param ingressClass.create Whether to create the ingress class. -## @param ingressClass.default Whether to set the ingress class as default. -ingressClass: - name: ngrok - create: true - default: false - -## @param controllerName The name of the controller to look for matching ingress classes -controllerName: "k8s.ngrok.com/ingress-controller" - -## @param watchNamespace The namespace to watch for ingress resources. Defaults to all -watchNamespace: "" - -## @param credentials.secret.name The name of the secret the credentials are in. If not provided, one will be generated using the helm release name. -## @param credentials.apiKey Your ngrok API key. If provided, it will be will be written to the secret and the authtoken must be provided as well. -## @param credentials.authtoken Your ngrok authtoken. If provided, it will be will be written to the secret and the apiKey must be provided as well. -credentials: - secret: - name: "" - apiKey: "" - authtoken: "" - -## @param region ngrok region to create tunnels in. Defaults to connect to the closest geographical region. -region: "" - -## @param rootCAs Set to "trusted" for the ngrok agent CA or "host" to trust the host's CA. Defaults to "trusted". -rootCAs: "" - -## @param serverAddr This is the address of the ngrok server to connect to. You should set this if you are using a custom ingress address. -serverAddr: "" - -## @param apiURL This is the URL of the ngrok API. You should set this if you are using a custom API URL. -apiURL: "" - -## @param metaData This is a map of key/value pairs that will be added as meta data to all ngrok api resources created -metaData: {} - -## @param affinity Affinity for the controller pod assignment -## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity -## Note: podAffinityPreset, podAntiAffinityPreset, and nodeAffinityPreset will be ignored when it's set -## -affinity: {} -## @param podAffinityPreset Pod affinity preset. Ignored if `affinity` is set. Allowed values: `soft` or `hard` -## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity -## -podAffinityPreset: "" -## @param podAntiAffinityPreset Pod anti-affinity preset. Ignored if `affinity` is set. Allowed values: `soft` or `hard` -## Ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity -## -podAntiAffinityPreset: soft -## Node affinity preset -## Ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity -## -nodeAffinityPreset: - ## @param nodeAffinityPreset.type Node affinity preset type. Ignored if `affinity` is set. Allowed values: `soft` or `hard` - ## - type: "" - ## @param nodeAffinityPreset.key Node label key to match. Ignored if `affinity` is set. - ## E.g. - ## key: "kubernetes.io/e2e-az-name" - ## - key: "" - ## @param nodeAffinityPreset.values Node label values to match. Ignored if `affinity` is set. - ## E.g. - ## values: - ## - e2e-az1 - ## - e2e-az2 - ## - values: [] - -## @param priorityClassName Priority class for pod scheduling -## ref: https://kubernetes.io/docs/concepts/configuration/pod-priority-preemption/#priorityclass -priorityClassName: "" - -## Pod Disruption Budget configuration -## ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb/ -## @param podDisruptionBudget.create Enable a Pod Disruption Budget creation -## @param podDisruptionBudget.minAvailable Minimum number/percentage of pods that should remain scheduled -## @param podDisruptionBudget.maxUnavailable Maximum number/percentage of pods that may be made unavailable -## -podDisruptionBudget: - create: false - minAvailable: "" - maxUnavailable: 1 - -## Controller container resource requests and limits -## ref: https://kubernetes.io/docs/user-guide/compute-resources/ -## We usually recommend not to specify default resources and to leave this as a conscious -## choice for the user. This also increases chances charts run on environments with little -## resources, such as Minikube. If you do want to specify resources, uncomment the following -## lines, adjust them as necessary, and remove the curly braces after 'resources:'. -## @param resources.limits The resources limits for the container -## @param resources.requests The requested resources for the container -## -resources: - ## Example: - ## limits: - ## cpu: 100m - ## memory: 128Mi - ## - limits: {} - ## Examples: - ## requests: - ## cpu: 100m - ## memory: 128Mi - ## - requests: {} - - -## @param extraVolumes An array of extra volumes to add to the controller. -extraVolumes: [] -## @param extraVolumeMounts An array of extra volume mounts to add to the controller. -extraVolumeMounts: [] -## -## Example: -## -## extraVolumes: -## - name: test-volume -## emptyDir: {} -## extraVolumeMounts: -## - name: test-volume -## mountPath: /test-volume - - -## @param extraEnv an object of extra environment variables to add to the controller. -extraEnv: {} -## Example: -## MY_VAR: test -## MY_SECRET_VAR: -## secretKeyRef: -## key: test-key -## value: test-value - -## Controller Service Account Settings -## @param serviceAccount.create Specifies whether a ServiceAccount should be created -## @param serviceAccount.name The name of the ServiceAccount to use. -## If not set and create is true, a name is generated using the fullname template -## @param serviceAccount.annotations Additional annotations to add to the ServiceAccount -## -serviceAccount: - create: true - name: "" - annotations: {} - - -## Logging configuration -## @param log.level The level to log at. One of 'debug', 'info', or 'error'. -## @param log.stacktraceLevel The level to report stacktrace logs one of 'info' or 'error'. -## @param log.format The log format to use. One of console, json. -log: - format: json - level: info - stacktraceLevel: error - -## @param lifecycle an object containing lifecycle configuration -## ref: https://kubernetes.io/docs/concepts/containers/container-lifecycle-hooks/ -## -lifecycle: {} diff --git a/charts/kubernetes-ingress-controller/values-overrides.yaml b/charts/kubernetes-ingress-controller/values-overrides.yaml deleted file mode 100644 index b9bbae4..0000000 --- a/charts/kubernetes-ingress-controller/values-overrides.yaml +++ /dev/null @@ -1,7 +0,0 @@ -fullnameOverride: ngrok -credentials: - secret: - name: ngrok -serviceAccount: - create: true - name: ngrok \ No newline at end of file diff --git a/charts/mariadb/mariadb/.helmignore b/charts/mariadb/mariadb/.helmignore deleted file mode 100644 index 207983f..0000000 --- a/charts/mariadb/mariadb/.helmignore +++ /dev/null @@ -1,25 +0,0 @@ -# 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 -# img folder -img/ -# Changelog -CHANGELOG.md diff --git a/charts/mariadb/mariadb/Chart.lock b/charts/mariadb/mariadb/Chart.lock deleted file mode 100644 index 029083d..0000000 --- a/charts/mariadb/mariadb/Chart.lock +++ /dev/null @@ -1,6 +0,0 @@ -dependencies: -- name: common - repository: oci://registry-1.docker.io/bitnamicharts - version: 2.23.0 -digest: sha256:fbd6439f12ded949c04553b9c52a4c8153a8f2790147d972b314ddcd46921a14 -generated: "2024-09-23T12:24:39.221888112Z" diff --git a/charts/mariadb/mariadb/Chart.yaml b/charts/mariadb/mariadb/Chart.yaml deleted file mode 100644 index 4a84e3f..0000000 --- a/charts/mariadb/mariadb/Chart.yaml +++ /dev/null @@ -1,36 +0,0 @@ -annotations: - category: Database - images: | - - name: mariadb - image: docker.io/bitnami/mariadb:11.4.3-debian-12-r1 - - name: mysqld-exporter - image: docker.io/bitnami/mysqld-exporter:0.15.1-debian-12-r34 - - name: os-shell - image: docker.io/bitnami/os-shell:12-debian-12-r30 - licenses: Apache-2.0 -apiVersion: v2 -appVersion: 11.4.3 -dependencies: -- name: common - repository: oci://registry-1.docker.io/bitnamicharts - tags: - - bitnami-common - version: 2.x.x -description: MariaDB is an open source, community-developed SQL database server that - is widely in use around the world due to its enterprise features, flexibility, and - collaboration with leading tech firms. -home: https://bitnami.com -icon: https://bitnami.com/assets/stacks/mariadb/img/mariadb-stack-220x234.png -keywords: -- mariadb -- mysql -- database -- sql -- prometheus -maintainers: -- name: Broadcom, Inc. All Rights Reserved. - url: https://github.com/bitnami/charts -name: mariadb -sources: -- https://github.com/bitnami/charts/tree/main/bitnami/mariadb -version: 19.1.0 diff --git a/charts/mariadb/mariadb/README.md b/charts/mariadb/mariadb/README.md deleted file mode 100644 index 5eace34..0000000 --- a/charts/mariadb/mariadb/README.md +++ /dev/null @@ -1,650 +0,0 @@ - - -# Bitnami package for MariaDB - -MariaDB is an open source, community-developed SQL database server that is widely in use around the world due to its enterprise features, flexibility, and collaboration with leading tech firms. - -[Overview of MariaDB](https://mariadb.org/) - -Trademarks: This software listing is packaged by Bitnami. The respective trademarks mentioned in the offering are owned by the respective companies, and use of them does not imply any affiliation or endorsement. - -## TL;DR - -```console -helm install my-release oci://registry-1.docker.io/bitnamicharts/mariadb -``` - -Looking to use MariaDB in production? Try [VMware Tanzu Application Catalog](https://bitnami.com/enterprise), the commercial edition of the Bitnami catalog. - -## Introduction - -This chart bootstraps a [MariaDB](https://github.com/bitnami/containers/tree/main/bitnami/mariadb) replication cluster deployment on a [Kubernetes](https://kubernetes.io) cluster using the [Helm](https://helm.sh) package manager. - -MariaDB is developed as open source software and as a relational database it provides an SQL interface for accessing data. The latest versions of MariaDB also include GIS and JSON features. - -Bitnami charts can be used with [Kubeapps](https://kubeapps.dev/) for deployment and management of Helm Charts in clusters. - -## Prerequisites - -- Kubernetes 1.23+ -- Helm 3.8.0+ -- PV provisioner support in the underlying infrastructure - -## Installing the Chart - -To install the chart with the release name `my-release`: - -```console -helm install my-release oci://REGISTRY_NAME/REPOSITORY_NAME/mariadb -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -The command deploys MariaDB on the Kubernetes cluster in the default configuration. The [Parameters](#parameters) section lists the parameters that can be configured during installation. - -> **Tip**: List all releases using `helm list` - -## Configuration and installation details - -### Resource requests and limits - -Bitnami charts allow setting resource requests and limits for all containers inside the chart deployment. These are inside the `resources` value (check parameter table). Setting requests is essential for production workloads and these should be adapted to your specific use case. - -To make this process easier, the chart contains the `resourcesPreset` values, which automatically sets the `resources` section according to different presets. Check these presets in [the bitnami/common chart](https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15). However, in production workloads using `resourcesPreset` is discouraged as it may not fully adapt to your specific needs. Find more information on container resource management in the [official Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/). - -### [Rolling VS Immutable tags](https://docs.vmware.com/en/VMware-Tanzu-Application-Catalog/services/tutorials/GUID-understand-rolling-tags-containers-index.html) - -It is strongly recommended to use immutable tags in a production environment. This ensures your deployment does not change automatically if the same tag is updated with a different image. - -Bitnami will release a new chart updating its containers if a new version of the main container, significant changes, or critical vulnerabilities exist. - -### Change MariaDB version - -To modify the MariaDB version used in this chart you can specify a [valid image tag](https://hub.docker.com/r/bitnami/mariadb/tags/) using the `image.tag` parameter. For example, `image.tag=X.Y.Z`. This approach is also applicable to other images like exporters. - -### Initialize a fresh instance - -The [Bitnami MariaDB](https://github.com/bitnami/containers/tree/main/bitnami/mariadb) image allows you to use your custom scripts to initialize a fresh instance. Custom scripts may be specified using the `initdbScripts` parameter. Alternatively, an external ConfigMap may be created with all the initialization scripts and the ConfigMap passed to the chart via the `initdbScriptsConfigMap` parameter. Note that this will override the `initdbScripts` parameter. - -The allowed extensions are `.sh`, `.sql` and `.sql.gz`. - -These scripts are treated differently depending on their extension. While `.sh` scripts are executed on all the nodes, `.sql` and `.sql.gz` scripts are only executed on the primary nodes. This is because `.sh` scripts support conditional tests to identify the type of node they are running on, while such tests are not supported in `.sql` or `.sql.gz` files. - -When using a `.sh` script, you may wish to perform a "one-time" action like creating a database. This can be achieved by adding a condition in the script to ensure that it is executed only on one node, as shown in the example below: - -```yaml -initdbScripts: - my_init_script.sh: | - #!/bin/sh - if [[ $(hostname) == *primary* ]]; then - echo "Primary node" - mysql -P 3306 -uroot -prandompassword -e "create database new_database"; - else - echo "No primary node" - fi -``` - -### Sidecars and Init Containers - -If additional containers are needed in the same pod as MariaDB (such as additional metrics or logging exporters), they can be defined using the `sidecars` parameter. - -```yaml -sidecars: -- name: your-image-name - image: your-image - imagePullPolicy: Always - ports: - - name: portname - containerPort: 1234 -``` - -If these sidecars export extra ports, extra port definitions can be added using the `service.extraPorts` parameter (where available), as shown in the example below: - -```yaml -service: - extraPorts: - - name: extraPort - port: 11311 - targetPort: 11311 -``` - -> NOTE: This Helm chart already includes sidecar containers for the Prometheus exporters (where applicable). These can be activated by adding the `--enable-metrics=true` parameter at deployment time. The `sidecars` parameter should therefore only be used for any extra sidecar containers. - -If additional init containers are needed in the same pod, they can be defined using the `initContainers` parameter. Here is an example: - -```yaml -initContainers: - - name: your-image-name - image: your-image - imagePullPolicy: Always - ports: - - name: portname - containerPort: 1234 -``` - -Learn more about [sidecar containers](https://kubernetes.io/docs/concepts/workloads/pods/) and [init containers](https://kubernetes.io/docs/concepts/workloads/pods/init-containers/). - -## Persistence - -The [Bitnami MariaDB](https://github.com/bitnami/containers/tree/main/bitnami/mariadb) image stores the MariaDB data and configurations at the `/bitnami/mariadb` path of the container. - -The chart mounts a [Persistent Volume](https://kubernetes.io/docs/concepts/storage/persistent-volumes/) volume at this location. The volume is created using dynamic volume provisioning, by default. An existing PersistentVolumeClaim can also be defined. - -If you encounter errors when working with persistent volumes, refer to our [troubleshooting guide for persistent volumes](https://docs.bitnami.com/kubernetes/faq/troubleshooting/troubleshooting-persistence-volumes/). - -### Adjust permissions of persistent volume mountpoint - -As the image run as non-root by default, it is necessary to adjust the ownership of the persistent volume so that the container can write data into it. - -By default, the chart is configured to use Kubernetes Security Context to automatically change the ownership of the volume. However, this feature does not work in all Kubernetes distributions. - -As an alternative, this chart supports using an initContainer to change the ownership of the volume before mounting it in the final destination. You can enable this initContainer by setting `volumePermissions.enabled` to `true`. - -## Parameters - -### Global parameters - -| Name | Description | Value | -| ----------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------ | -| `global.imageRegistry` | Global Docker Image registry | `""` | -| `global.imagePullSecrets` | Global Docker registry secret names as an array | `[]` | -| `global.defaultStorageClass` | Global default StorageClass for Persistent Volume(s) | `""` | -| `global.storageClass` | DEPRECATED: use global.defaultStorageClass instead | `""` | -| `global.compatibility.openshift.adaptSecurityContext` | Adapt the securityContext sections of the deployment to make them compatible with Openshift restricted-v2 SCC: remove runAsUser, runAsGroup and fsGroup and let the platform use their allowed default IDs. Possible values: auto (apply if the detected running cluster is Openshift), force (perform the adaptation always), disabled (do not perform adaptation) | `auto` | - -### Common parameters - -| Name | Description | Value | -| ------------------------- | --------------------------------------------------------------------------------------- | --------------- | -| `kubeVersion` | Force target Kubernetes version (using Helm capabilities if not set) | `""` | -| `nameOverride` | String to partially override mariadb.fullname | `""` | -| `fullnameOverride` | String to fully override mariadb.fullname | `""` | -| `clusterDomain` | Default Kubernetes cluster domain | `cluster.local` | -| `commonAnnotations` | Common annotations to add to all MariaDB resources (sub-charts are not considered) | `{}` | -| `commonLabels` | Common labels to add to all MariaDB resources (sub-charts are not considered) | `{}` | -| `schedulerName` | Name of the scheduler (other than default) to dispatch pods | `""` | -| `runtimeClassName` | Name of the Runtime Class for all MariaDB pods | `""` | -| `extraDeploy` | Array of extra objects to deploy with the release (evaluated as a template) | `[]` | -| `diagnosticMode.enabled` | Enable diagnostic mode (all probes will be disabled and the command will be overridden) | `false` | -| `diagnosticMode.command` | Command to override all containers in the deployment | `["sleep"]` | -| `diagnosticMode.args` | Args to override all containers in the deployment | `["infinity"]` | -| `serviceBindings.enabled` | Create secret for service binding (Experimental) | `false` | - -### MariaDB common parameters - -| Name | Description | Value | -| -------------------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------------- | -| `image.registry` | MariaDB image registry | `REGISTRY_NAME` | -| `image.repository` | MariaDB image repository | `REPOSITORY_NAME/mariadb` | -| `image.digest` | MariaDB image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `image.pullPolicy` | MariaDB image pull policy | `IfNotPresent` | -| `image.pullSecrets` | Specify docker-registry secret names as an array | `[]` | -| `image.debug` | Specify if debug logs should be enabled | `false` | -| `architecture` | MariaDB architecture (`standalone` or `replication`) | `standalone` | -| `auth.rootPassword` | Password for the `root` user. Ignored if existing secret is provided. | `""` | -| `auth.database` | Name for a custom database to create | `my_database` | -| `auth.username` | Name for a custom user to create | `""` | -| `auth.password` | Password for the new user. Ignored if existing secret is provided | `""` | -| `auth.replicationUser` | MariaDB replication user | `replicator` | -| `auth.replicationPassword` | MariaDB replication user password. Ignored if existing secret is provided | `""` | -| `auth.existingSecret` | Use existing secret for password details (`auth.rootPassword`, `auth.password`, `auth.replicationPassword` will be ignored and picked up from this secret). The secret has to contain the keys `mariadb-root-password`, `mariadb-replication-password` and `mariadb-password` | `""` | -| `auth.forcePassword` | Force users to specify required passwords | `false` | -| `auth.usePasswordFiles` | Mount credentials as files instead of using environment variables | `false` | -| `auth.customPasswordFiles` | Use custom password files when `auth.usePasswordFiles` is set to `true`. Define path for keys `root` and `user`, also define `replicator` if `architecture` is set to `replication` | `{}` | -| `initdbScripts` | Dictionary of initdb scripts | `{}` | -| `initdbScriptsConfigMap` | ConfigMap with the initdb scripts (Note: Overrides `initdbScripts`) | `""` | - -### MariaDB Primary parameters - -| Name | Description | Value | -| ----------------------------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------- | -| `primary.name` | Name of the primary database (eg primary, master, leader, ...) | `primary` | -| `primary.command` | Override default container command on MariaDB Primary container(s) (useful when using custom images) | `[]` | -| `primary.args` | Override default container args on MariaDB Primary container(s) (useful when using custom images) | `[]` | -| `primary.lifecycleHooks` | for the MariaDB Primary container(s) to automate configuration before or after startup | `{}` | -| `primary.automountServiceAccountToken` | Mount Service Account token in pod | `false` | -| `primary.hostAliases` | Add deployment host aliases | `[]` | -| `primary.containerPorts.mysql` | Container port for mysql | `3306` | -| `primary.configuration` | MariaDB Primary configuration to be injected as ConfigMap | `""` | -| `primary.existingConfigmap` | Name of existing ConfigMap with MariaDB Primary configuration. | `""` | -| `primary.updateStrategy.type` | MariaDB primary statefulset strategy type | `RollingUpdate` | -| `primary.rollingUpdatePartition` | Partition update strategy for Mariadb Primary statefulset | `""` | -| `primary.podAnnotations` | Additional pod annotations for MariaDB primary pods | `{}` | -| `primary.podLabels` | Extra labels for MariaDB primary pods | `{}` | -| `primary.podAffinityPreset` | MariaDB primary pod affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `primary.podAntiAffinityPreset` | MariaDB primary pod anti-affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `soft` | -| `primary.nodeAffinityPreset.type` | MariaDB primary node affinity preset type. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `primary.nodeAffinityPreset.key` | MariaDB primary node label key to match Ignored if `primary.affinity` is set. | `""` | -| `primary.nodeAffinityPreset.values` | MariaDB primary node label values to match. Ignored if `primary.affinity` is set. | `[]` | -| `primary.affinity` | Affinity for MariaDB primary pods assignment | `{}` | -| `primary.nodeSelector` | Node labels for MariaDB primary pods assignment | `{}` | -| `primary.tolerations` | Tolerations for MariaDB primary pods assignment | `[]` | -| `primary.schedulerName` | Name of the k8s scheduler (other than default) | `""` | -| `primary.podManagementPolicy` | podManagementPolicy to manage scaling operation of MariaDB primary pods | `""` | -| `primary.topologySpreadConstraints` | Topology Spread Constraints for MariaDB primary pods assignment | `[]` | -| `primary.priorityClassName` | Priority class for MariaDB primary pods assignment | `""` | -| `primary.runtimeClassName` | Runtime Class for MariaDB primary pods | `""` | -| `primary.podSecurityContext.enabled` | Enable security context for MariaDB primary pods | `true` | -| `primary.podSecurityContext.fsGroupChangePolicy` | Set filesystem group change policy | `Always` | -| `primary.podSecurityContext.sysctls` | Set kernel settings using the sysctl interface | `[]` | -| `primary.podSecurityContext.supplementalGroups` | Set filesystem extra groups | `[]` | -| `primary.podSecurityContext.fsGroup` | Group ID for the mounted volumes' filesystem | `1001` | -| `primary.containerSecurityContext.enabled` | MariaDB primary container securityContext | `true` | -| `primary.containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `primary.containerSecurityContext.runAsUser` | User ID for the MariaDB primary container | `1001` | -| `primary.containerSecurityContext.runAsGroup` | Group ID for the MariaDB primary container | `1001` | -| `primary.containerSecurityContext.runAsNonRoot` | Set primary container's Security Context runAsNonRoot | `true` | -| `primary.containerSecurityContext.privileged` | Set primary container's Security Context privileged | `false` | -| `primary.containerSecurityContext.allowPrivilegeEscalation` | Set primary container's Security Context allowPrivilegeEscalation | `false` | -| `primary.containerSecurityContext.readOnlyRootFilesystem` | Set container's Security Context readOnlyRootFilesystem | `true` | -| `primary.containerSecurityContext.capabilities.drop` | List of capabilities to be dropped | `["ALL"]` | -| `primary.containerSecurityContext.seccompProfile.type` | Set container's Security Context seccomp profile | `RuntimeDefault` | -| `primary.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if primary.resources is set (primary.resources is recommended for production). | `micro` | -| `primary.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `primary.startupProbe.enabled` | Enable startupProbe | `false` | -| `primary.startupProbe.initialDelaySeconds` | Initial delay seconds for startupProbe | `120` | -| `primary.startupProbe.periodSeconds` | Period seconds for startupProbe | `15` | -| `primary.startupProbe.timeoutSeconds` | Timeout seconds for startupProbe | `5` | -| `primary.startupProbe.failureThreshold` | Failure threshold for startupProbe | `10` | -| `primary.startupProbe.successThreshold` | Success threshold for startupProbe | `1` | -| `primary.livenessProbe.enabled` | Enable livenessProbe | `true` | -| `primary.livenessProbe.initialDelaySeconds` | Initial delay seconds for livenessProbe | `120` | -| `primary.livenessProbe.periodSeconds` | Period seconds for livenessProbe | `10` | -| `primary.livenessProbe.timeoutSeconds` | Timeout seconds for livenessProbe | `1` | -| `primary.livenessProbe.failureThreshold` | Failure threshold for livenessProbe | `3` | -| `primary.livenessProbe.successThreshold` | Success threshold for livenessProbe | `1` | -| `primary.readinessProbe.enabled` | Enable readinessProbe | `true` | -| `primary.readinessProbe.initialDelaySeconds` | Initial delay seconds for readinessProbe | `30` | -| `primary.readinessProbe.periodSeconds` | Period seconds for readinessProbe | `10` | -| `primary.readinessProbe.timeoutSeconds` | Timeout seconds for readinessProbe | `1` | -| `primary.readinessProbe.failureThreshold` | Failure threshold for readinessProbe | `3` | -| `primary.readinessProbe.successThreshold` | Success threshold for readinessProbe | `1` | -| `primary.customStartupProbe` | Override default startup probe for MariaDB primary containers | `{}` | -| `primary.customLivenessProbe` | Override default liveness probe for MariaDB primary containers | `{}` | -| `primary.customReadinessProbe` | Override default readiness probe for MariaDB primary containers | `{}` | -| `primary.startupWaitOptions` | Override default builtin startup wait check options for MariaDB primary containers | `{}` | -| `primary.extraFlags` | MariaDB primary additional command line flags | `""` | -| `primary.extraEnvVars` | Extra environment variables to be set on MariaDB primary containers | `[]` | -| `primary.extraEnvVarsCM` | Name of existing ConfigMap containing extra env vars for MariaDB primary containers | `""` | -| `primary.extraEnvVarsSecret` | Name of existing Secret containing extra env vars for MariaDB primary containers | `""` | -| `primary.persistence.enabled` | Enable persistence on MariaDB primary replicas using a `PersistentVolumeClaim`. If false, use emptyDir | `true` | -| `primary.persistence.existingClaim` | Name of an existing `PersistentVolumeClaim` for MariaDB primary replicas | `""` | -| `primary.persistence.subPath` | Subdirectory of the volume to mount at | `""` | -| `primary.persistence.storageClass` | MariaDB primary persistent volume storage Class | `""` | -| `primary.persistence.labels` | Labels for the PVC | `{}` | -| `primary.persistence.annotations` | MariaDB primary persistent volume claim annotations | `{}` | -| `primary.persistence.accessModes` | MariaDB primary persistent volume access Modes | `["ReadWriteOnce"]` | -| `primary.persistence.size` | MariaDB primary persistent volume size | `8Gi` | -| `primary.persistence.selector` | Selector to match an existing Persistent Volume | `{}` | -| `primary.extraVolumes` | Optionally specify extra list of additional volumes to the MariaDB Primary pod(s) | `[]` | -| `primary.extraVolumeMounts` | Optionally specify extra list of additional volumeMounts for the MariaDB Primary container(s) | `[]` | -| `primary.initContainers` | Add additional init containers for the MariaDB Primary pod(s) | `[]` | -| `primary.sidecars` | Add additional sidecar containers for the MariaDB Primary pod(s) | `[]` | -| `primary.service.type` | MariaDB Primary Kubernetes service type | `ClusterIP` | -| `primary.service.ports.mysql` | MariaDB Primary Kubernetes service port for MariaDB | `3306` | -| `primary.service.ports.metrics` | MariaDB Primary Kubernetes service port for metrics | `9104` | -| `primary.service.nodePorts.mysql` | MariaDB Primary Kubernetes service node port | `""` | -| `primary.service.clusterIP` | MariaDB Primary Kubernetes service clusterIP IP | `""` | -| `primary.service.loadBalancerIP` | MariaDB Primary loadBalancerIP if service type is `LoadBalancer` | `""` | -| `primary.service.externalTrafficPolicy` | Enable client source IP preservation | `Cluster` | -| `primary.service.loadBalancerSourceRanges` | Address that are allowed when MariaDB Primary service is LoadBalancer | `[]` | -| `primary.service.extraPorts` | Extra ports to expose (normally used with the `sidecar` value) | `[]` | -| `primary.service.annotations` | Provide any additional annotations which may be required | `{}` | -| `primary.service.sessionAffinity` | Session Affinity for Kubernetes service, can be "None" or "ClientIP" | `None` | -| `primary.service.sessionAffinityConfig` | Additional settings for the sessionAffinity | `{}` | -| `primary.pdb.create` | Enable/disable a Pod Disruption Budget creation for MariaDB primary pods | `true` | -| `primary.pdb.minAvailable` | Minimum number/percentage of MariaDB primary pods that must still be available after the eviction | `""` | -| `primary.pdb.maxUnavailable` | Maximum number/percentage of MariaDB primary pods that can be unavailable after the eviction. Defaults to `1` if both `primary.pdb.minAvailable` and `primary.pdb.maxUnavailable` are empty. | `""` | -| `primary.revisionHistoryLimit` | Maximum number of revisions that will be maintained in the StatefulSet | `10` | - -### MariaDB Secondary parameters - -| Name | Description | Value | -| ------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------- | -| `secondary.name` | Name of the secondary database (eg secondary, slave, ...) | `secondary` | -| `secondary.replicaCount` | Number of MariaDB secondary replicas | `1` | -| `secondary.command` | Override default container command on MariaDB Secondary container(s) (useful when using custom images) | `[]` | -| `secondary.args` | Override default container args on MariaDB Secondary container(s) (useful when using custom images) | `[]` | -| `secondary.lifecycleHooks` | for the MariaDB Secondary container(s) to automate configuration before or after startup | `{}` | -| `secondary.automountServiceAccountToken` | Mount Service Account token in pod | `false` | -| `secondary.hostAliases` | Add deployment host aliases | `[]` | -| `secondary.containerPorts.mysql` | Container port for mysql | `3306` | -| `secondary.configuration` | MariaDB Secondary configuration to be injected as ConfigMap | `""` | -| `secondary.existingConfigmap` | Name of existing ConfigMap with MariaDB Secondary configuration. | `""` | -| `secondary.updateStrategy.type` | MariaDB secondary statefulset strategy type | `RollingUpdate` | -| `secondary.rollingUpdatePartition` | Partition update strategy for Mariadb Secondary statefulset | `""` | -| `secondary.podAnnotations` | Additional pod annotations for MariaDB secondary pods | `{}` | -| `secondary.podLabels` | Extra labels for MariaDB secondary pods | `{}` | -| `secondary.podAffinityPreset` | MariaDB secondary pod affinity preset. Ignored if `secondary.affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `secondary.podAntiAffinityPreset` | MariaDB secondary pod anti-affinity preset. Ignored if `secondary.affinity` is set. Allowed values: `soft` or `hard` | `soft` | -| `secondary.nodeAffinityPreset.type` | MariaDB secondary node affinity preset type. Ignored if `secondary.affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `secondary.nodeAffinityPreset.key` | MariaDB secondary node label key to match Ignored if `secondary.affinity` is set. | `""` | -| `secondary.nodeAffinityPreset.values` | MariaDB secondary node label values to match. Ignored if `secondary.affinity` is set. | `[]` | -| `secondary.affinity` | Affinity for MariaDB secondary pods assignment | `{}` | -| `secondary.nodeSelector` | Node labels for MariaDB secondary pods assignment | `{}` | -| `secondary.tolerations` | Tolerations for MariaDB secondary pods assignment | `[]` | -| `secondary.topologySpreadConstraints` | Topology Spread Constraints for MariaDB secondary pods assignment | `[]` | -| `secondary.priorityClassName` | Priority class for MariaDB secondary pods assignment | `""` | -| `secondary.runtimeClassName` | Runtime Class for MariaDB secondary pods | `""` | -| `secondary.schedulerName` | Name of the k8s scheduler (other than default) | `""` | -| `secondary.podManagementPolicy` | podManagementPolicy to manage scaling operation of MariaDB secondary pods | `""` | -| `secondary.podSecurityContext.enabled` | Enable security context for MariaDB secondary pods | `true` | -| `secondary.podSecurityContext.fsGroupChangePolicy` | Set filesystem group change policy | `Always` | -| `secondary.podSecurityContext.sysctls` | Set kernel settings using the sysctl interface | `[]` | -| `secondary.podSecurityContext.supplementalGroups` | Set filesystem extra groups | `[]` | -| `secondary.podSecurityContext.fsGroup` | Group ID for the mounted volumes' filesystem | `1001` | -| `secondary.containerSecurityContext.enabled` | MariaDB secondary container securityContext | `true` | -| `secondary.containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `secondary.containerSecurityContext.runAsUser` | User ID for the MariaDB secondary container | `1001` | -| `secondary.containerSecurityContext.runAsGroup` | Group ID for the MariaDB secondary container | `1001` | -| `secondary.containerSecurityContext.runAsNonRoot` | Set secondary container's Security Context runAsNonRoot | `true` | -| `secondary.containerSecurityContext.privileged` | Set secondary container's Security Context privileged | `false` | -| `secondary.containerSecurityContext.allowPrivilegeEscalation` | Set secondary container's Security Context allowPrivilegeEscalation | `false` | -| `secondary.containerSecurityContext.readOnlyRootFilesystem` | Set container's Security Context readOnlyRootFilesystem | `true` | -| `secondary.containerSecurityContext.capabilities.drop` | List of capabilities to be dropped | `["ALL"]` | -| `secondary.containerSecurityContext.seccompProfile.type` | Set container's Security Context seccomp profile | `RuntimeDefault` | -| `secondary.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if secondary.resources is set (secondary.resources is recommended for production). | `micro` | -| `secondary.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `secondary.startupProbe.enabled` | Enable startupProbe | `false` | -| `secondary.startupProbe.initialDelaySeconds` | Initial delay seconds for startupProbe | `120` | -| `secondary.startupProbe.periodSeconds` | Period seconds for startupProbe | `15` | -| `secondary.startupProbe.timeoutSeconds` | Timeout seconds for startupProbe | `5` | -| `secondary.startupProbe.failureThreshold` | Failure threshold for startupProbe | `10` | -| `secondary.startupProbe.successThreshold` | Success threshold for startupProbe | `1` | -| `secondary.livenessProbe.enabled` | Enable livenessProbe | `true` | -| `secondary.livenessProbe.initialDelaySeconds` | Initial delay seconds for livenessProbe | `120` | -| `secondary.livenessProbe.periodSeconds` | Period seconds for livenessProbe | `10` | -| `secondary.livenessProbe.timeoutSeconds` | Timeout seconds for livenessProbe | `1` | -| `secondary.livenessProbe.failureThreshold` | Failure threshold for livenessProbe | `3` | -| `secondary.livenessProbe.successThreshold` | Success threshold for livenessProbe | `1` | -| `secondary.readinessProbe.enabled` | Enable readinessProbe | `true` | -| `secondary.readinessProbe.initialDelaySeconds` | Initial delay seconds for readinessProbe | `30` | -| `secondary.readinessProbe.periodSeconds` | Period seconds for readinessProbe | `10` | -| `secondary.readinessProbe.timeoutSeconds` | Timeout seconds for readinessProbe | `1` | -| `secondary.readinessProbe.failureThreshold` | Failure threshold for readinessProbe | `3` | -| `secondary.readinessProbe.successThreshold` | Success threshold for readinessProbe | `1` | -| `secondary.customStartupProbe` | Override default startup probe for MariaDB secondary containers | `{}` | -| `secondary.customLivenessProbe` | Override default liveness probe for MariaDB secondary containers | `{}` | -| `secondary.customReadinessProbe` | Override default readiness probe for MariaDB secondary containers | `{}` | -| `secondary.startupWaitOptions` | Override default builtin startup wait check options for MariaDB secondary containers | `{}` | -| `secondary.extraFlags` | MariaDB secondary additional command line flags | `""` | -| `secondary.extraEnvVars` | Extra environment variables to be set on MariaDB secondary containers | `[]` | -| `secondary.extraEnvVarsCM` | Name of existing ConfigMap containing extra env vars for MariaDB secondary containers | `""` | -| `secondary.extraEnvVarsSecret` | Name of existing Secret containing extra env vars for MariaDB secondary containers | `""` | -| `secondary.persistence.enabled` | Enable persistence on MariaDB secondary replicas using a `PersistentVolumeClaim` | `true` | -| `secondary.persistence.subPath` | Subdirectory of the volume to mount at | `""` | -| `secondary.persistence.storageClass` | MariaDB secondary persistent volume storage Class | `""` | -| `secondary.persistence.labels` | Labels for the PVC | `{}` | -| `secondary.persistence.annotations` | MariaDB secondary persistent volume claim annotations | `{}` | -| `secondary.persistence.accessModes` | MariaDB secondary persistent volume access Modes | `["ReadWriteOnce"]` | -| `secondary.persistence.size` | MariaDB secondary persistent volume size | `8Gi` | -| `secondary.persistence.selector` | Selector to match an existing Persistent Volume | `{}` | -| `secondary.extraVolumes` | Optionally specify extra list of additional volumes to the MariaDB secondary pod(s) | `[]` | -| `secondary.extraVolumeMounts` | Optionally specify extra list of additional volumeMounts for the MariaDB secondary container(s) | `[]` | -| `secondary.initContainers` | Add additional init containers for the MariaDB secondary pod(s) | `[]` | -| `secondary.sidecars` | Add additional sidecar containers for the MariaDB secondary pod(s) | `[]` | -| `secondary.service.type` | MariaDB secondary Kubernetes service type | `ClusterIP` | -| `secondary.service.ports.mysql` | MariaDB secondary Kubernetes service port for MariaDB | `3306` | -| `secondary.service.ports.metrics` | MariaDB secondary Kubernetes service port for metrics | `9104` | -| `secondary.service.nodePorts.mysql` | MariaDB secondary Kubernetes service node port | `""` | -| `secondary.service.clusterIP` | MariaDB secondary Kubernetes service clusterIP IP | `""` | -| `secondary.service.loadBalancerIP` | MariaDB secondary loadBalancerIP if service type is `LoadBalancer` | `""` | -| `secondary.service.externalTrafficPolicy` | Enable client source IP preservation | `Cluster` | -| `secondary.service.loadBalancerSourceRanges` | Address that are allowed when MariaDB secondary service is LoadBalancer | `[]` | -| `secondary.service.extraPorts` | Extra ports to expose (normally used with the `sidecar` value) | `[]` | -| `secondary.service.annotations` | Provide any additional annotations which may be required | `{}` | -| `secondary.service.sessionAffinity` | Session Affinity for Kubernetes service, can be "None" or "ClientIP" | `None` | -| `secondary.service.sessionAffinityConfig` | Additional settings for the sessionAffinity | `{}` | -| `secondary.pdb.create` | Enable/disable a Pod Disruption Budget creation for MariaDB secondary pods | `true` | -| `secondary.pdb.minAvailable` | Minimum number/percentage of MariaDB secondary pods that should remain scheduled | `""` | -| `secondary.pdb.maxUnavailable` | Maximum number/percentage of MariaDB secondary pods that may be made unavailable. Defaults to `1` if both `secondary.pdb.minAvailable` and `secondary.pdb.maxUnavailable` are empty. | `""` | -| `secondary.revisionHistoryLimit` | Maximum number of revisions that will be maintained in the StatefulSet | `10` | - -### RBAC parameters - -| Name | Description | Value | -| --------------------------------------------- | -------------------------------------------------------------- | ------- | -| `serviceAccount.create` | Enable the creation of a ServiceAccount for MariaDB pods | `true` | -| `serviceAccount.name` | Name of the created ServiceAccount | `""` | -| `serviceAccount.annotations` | Annotations for MariaDB Service Account | `{}` | -| `serviceAccount.automountServiceAccountToken` | Automount service account token for the server service account | `false` | -| `rbac.create` | Whether to create and use RBAC resources or not | `false` | - -### Volume Permissions parameters - -| Name | Description | Value | -| ------------------------------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | -------------------------- | -| `volumePermissions.enabled` | Enable init container that changes the owner and group of the persistent volume(s) mountpoint to `runAsUser:fsGroup` | `false` | -| `volumePermissions.image.registry` | Init container volume-permissions image registry | `REGISTRY_NAME` | -| `volumePermissions.image.repository` | Init container volume-permissions image repository | `REPOSITORY_NAME/os-shell` | -| `volumePermissions.image.digest` | Init container volume-permissions image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `volumePermissions.image.pullPolicy` | Init container volume-permissions image pull policy | `IfNotPresent` | -| `volumePermissions.image.pullSecrets` | Specify docker-registry secret names as an array | `[]` | -| `volumePermissions.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if volumePermissions.resources is set (volumePermissions.resources is recommended for production). | `nano` | -| `volumePermissions.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | - -### Metrics parameters - -| Name | Description | Value | -| ----------------------------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | --------------------------------- | -| `metrics.enabled` | Start a side-car prometheus exporter | `false` | -| `metrics.image.registry` | Exporter image registry | `REGISTRY_NAME` | -| `metrics.image.repository` | Exporter image repository | `REPOSITORY_NAME/mysqld-exporter` | -| `metrics.image.digest` | Exporter image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `metrics.image.pullPolicy` | Exporter image pull policy | `IfNotPresent` | -| `metrics.image.pullSecrets` | Specify docker-registry secret names as an array | `[]` | -| `metrics.annotations` | Annotations for the Exporter pod | `{}` | -| `metrics.extraArgs` | Extra args to be passed to mysqld_exporter | `{}` | -| `metrics.extraVolumeMounts` | Optionally specify extra list of additional volumeMounts for the MariaDB metrics container(s) | `{}` | -| `metrics.containerPorts.http` | Container port for http | `9104` | -| `metrics.containerSecurityContext.enabled` | Enable security context for MariaDB metrics container | `false` | -| `metrics.containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `metrics.containerSecurityContext.runAsUser` | User ID for the MariaDB metrics container | `1001` | -| `metrics.containerSecurityContext.runAsGroup` | Group ID for the MariaDB metrics container | `1001` | -| `metrics.containerSecurityContext.runAsNonRoot` | Set metrics container's Security Context runAsNonRoot | `true` | -| `metrics.containerSecurityContext.privileged` | Set metrics container's Security Context privileged | `false` | -| `metrics.containerSecurityContext.allowPrivilegeEscalation` | Set metrics container's Security Context allowPrivilegeEscalation | `false` | -| `metrics.containerSecurityContext.readOnlyRootFilesystem` | Set container's Security Context readOnlyRootFilesystem | `true` | -| `metrics.containerSecurityContext.capabilities.drop` | List of capabilities to be dropped | `["ALL"]` | -| `metrics.containerSecurityContext.seccompProfile.type` | Set container's Security Context seccomp profile | `RuntimeDefault` | -| `metrics.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if metrics.resources is set (metrics.resources is recommended for production). | `nano` | -| `metrics.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `metrics.livenessProbe.enabled` | Enable livenessProbe | `true` | -| `metrics.livenessProbe.initialDelaySeconds` | Initial delay seconds for livenessProbe | `120` | -| `metrics.livenessProbe.periodSeconds` | Period seconds for livenessProbe | `10` | -| `metrics.livenessProbe.timeoutSeconds` | Timeout seconds for livenessProbe | `1` | -| `metrics.livenessProbe.failureThreshold` | Failure threshold for livenessProbe | `3` | -| `metrics.livenessProbe.successThreshold` | Success threshold for livenessProbe | `1` | -| `metrics.readinessProbe.enabled` | Enable readinessProbe | `true` | -| `metrics.readinessProbe.initialDelaySeconds` | Initial delay seconds for readinessProbe | `30` | -| `metrics.readinessProbe.periodSeconds` | Period seconds for readinessProbe | `10` | -| `metrics.readinessProbe.timeoutSeconds` | Timeout seconds for readinessProbe | `1` | -| `metrics.readinessProbe.failureThreshold` | Failure threshold for readinessProbe | `3` | -| `metrics.readinessProbe.successThreshold` | Success threshold for readinessProbe | `1` | -| `metrics.serviceMonitor.enabled` | Create ServiceMonitor Resource for scraping metrics using PrometheusOperator | `false` | -| `metrics.serviceMonitor.namespace` | Namespace which Prometheus is running in | `""` | -| `metrics.serviceMonitor.jobLabel` | The name of the label on the target service to use as the job name in prometheus. | `""` | -| `metrics.serviceMonitor.interval` | Interval at which metrics should be scraped | `30s` | -| `metrics.serviceMonitor.scrapeTimeout` | Specify the timeout after which the scrape is ended | `""` | -| `metrics.serviceMonitor.relabelings` | RelabelConfigs to apply to samples before scraping | `[]` | -| `metrics.serviceMonitor.metricRelabelings` | MetricRelabelConfigs to apply to samples before ingestion | `[]` | -| `metrics.serviceMonitor.honorLabels` | honorLabels chooses the metric's labels on collisions with target labels | `false` | -| `metrics.serviceMonitor.selector` | ServiceMonitor selector labels | `{}` | -| `metrics.serviceMonitor.labels` | Extra labels for the ServiceMonitor | `{}` | -| `metrics.prometheusRule.enabled` | if `true`, creates a Prometheus Operator PrometheusRule (also requires `metrics.enabled` to be `true` and `metrics.prometheusRule.rules`) | `false` | -| `metrics.prometheusRule.namespace` | Namespace for the PrometheusRule Resource (defaults to the Release Namespace) | `""` | -| `metrics.prometheusRule.additionalLabels` | Additional labels that can be used so PrometheusRule will be discovered by Prometheus | `{}` | -| `metrics.prometheusRule.rules` | Prometheus Rule definitions | `[]` | - -### NetworkPolicy parameters - -| Name | Description | Value | -| --------------------------------------- | --------------------------------------------------------------- | ------ | -| `networkPolicy.enabled` | Enable creation of NetworkPolicy resources | `true` | -| `networkPolicy.allowExternal` | The Policy model to apply | `true` | -| `networkPolicy.allowExternalEgress` | Allow the pod to access any range of port and all destinations. | `true` | -| `networkPolicy.extraIngress` | Add extra ingress rules to the NetworkPolicy | `[]` | -| `networkPolicy.extraEgress` | Add extra ingress rules to the NetworkPolicy | `[]` | -| `networkPolicy.ingressNSMatchLabels` | Labels to match to allow traffic from other namespaces | `{}` | -| `networkPolicy.ingressNSPodMatchLabels` | Pod labels to match to allow traffic from other namespaces | `{}` | - -The above parameters map to the env variables defined in [bitnami/mariadb](https://github.com/bitnami/containers/tree/main/bitnami/mariadb). For more information please refer to the [bitnami/mariadb](https://github.com/bitnami/containers/tree/main/bitnami/mariadb) image documentation. - -Specify each parameter using the `--set key=value[,key=value]` argument to `helm install`. For example, - -```console -helm install my-release \ - --set auth.rootPassword=secretpassword,auth.database=app_database \ - oci://REGISTRY_NAME/REPOSITORY_NAME/mariadb -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -The above command sets the MariaDB `root` account password to `secretpassword`. Additionally it creates a database named `my_database`. - -> NOTE: Once this chart is deployed, it is not possible to change the application's access credentials, such as usernames or passwords, using Helm. To change these application credentials after deployment, delete any persistent volumes (PVs) used by the chart and re-deploy it, or use the application's built-in administrative tools if available. - -Alternatively, a YAML file that specifies the values for the parameters can be provided while installing the chart. For example, - -```console -helm install my-release -f values.yaml oci://REGISTRY_NAME/REPOSITORY_NAME/mariadb -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. -> **Tip**: You can use the default [values.yaml](https://github.com/bitnami/charts/tree/main/bitnami/mariadb/values.yaml) - -## Troubleshooting - -Find more information about how to deal with common errors related to Bitnami's Helm charts in [this troubleshooting guide](https://docs.bitnami.com/general/how-to/troubleshoot-helm-chart-issues). - -## Upgrading - -It's necessary to set the `auth.rootPassword` parameter when upgrading for readiness/liveness probes to work properly. When you install this chart for the first time, some notes will be displayed providing the credentials you must use under the 'Administrator credentials' section. Please note down the password and run the command below to upgrade your chart: - -```console -helm upgrade my-release oci://REGISTRY_NAME/REPOSITORY_NAME/mariadb --set auth.rootPassword=[ROOT_PASSWORD] -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -| Note: you need to substitute the placeholder _[ROOT_PASSWORD]_ with the value obtained in the installation notes. - -### To 17.0.0 - -This major bump changes the following security defaults: - -- `runAsGroup` is changed from `0` to `1001` -- `readOnlyRootFilesystem` is set to `true` -- `resourcesPreset` is changed from `none` to the minimum size working in our test suites (NOTE: `resourcesPreset` is not meant for production usage, but `resources` adapted to your use case). -- `global.compatibility.openshift.adaptSecurityContext` is changed from `disabled` to `auto`. - -This could potentially break any customization or init scripts used in your deployment. If this is the case, change the default values to the previous ones. - -### To 16.0.0 - -This section enables NetworkPolicies by default to increase security of the application. It also adapts the values in the `networkPolicy` section to the current Bitnami standards. The removed sections are `networkPolicy.metrics.*`, `networkPolicy.ingressRules.*` and `networkPolicy.egressRules.*`. Check the Parameters table for the new structure. - -### To 14.0.0 - -This major release bumps the MariaDB version to 11.1. Follow the [upstream instructions](https://mariadb.com/kb/en/upgrading-between-minor-versions-on-linux/) for upgrading from MariaDB 11.0 to 11.1. No major issues are expected during the upgrade. - -### To 13.0.0 - -This major release bumps the MariaDB version to 11.0. Follow the [upstream instructions](https://mariadb.com/kb/en/upgrading-from-mariadb-10-11-to-mariadb-11-0/) for upgrading from MariaDB 10.11 to 11.0. No major issues are expected during the upgrade. - -### To 11.0.0 - -This major release bumps default MariaDB branch to 10.6. Follow the [official instructions](https://mariadb.com/kb/en/upgrading-from-mariadb-105-to-mariadb-106/) from upgrading between 10.5 and 10.6. - -No major issues are expected during the upgrade. - -### To 10.0.0 - -This major release renames several values in this chart and adds missing features, in order to be inline with the rest of assets in the Bitnami charts repository. - -Affected values: - -- `primary.service.port` was deprecated, we recommend using `primary.service.ports.mysql` instead. -- `primary.service.nodePort` was deprecated, we recommend using `primary.service.nodePorts.mysql` instead. -- `secondary.service.port` was deprecated, we recommend using `secondary.service.ports.mysql` instead. -- `secondary.service.nodePort` was deprecated, we recommend using `secondary.service.nodePorts.mysql` instead. -- `metrics.serviceMonitor.additionalLabels` was deprecated, we recommend using `metrics.serviceMonitor.selector` instead. -- `primary.pdb.enabled` renamed as `primary.pdb.create`. -- `secondary.pdb.enabled` renamed as `secondary.pdb.create`. -- `primary.updateStrategy` changed from String type (previously default to 'rollingUpdate') to Object type, allowing users to configure other updateStrategy parameters, similar to other charts. -- Removed value `primary.rollingUpdatePartition`, now configured using `primary.updateStrategy` setting `primary.updateStrategy.rollingUpdate.partition`. -- `secondary.updateStrategy` changed from String type (previously default to 'rollingUpdate') to Object type, allowing users to configure other updateStrategy parameters, similar to other charts. -- Removed value `secondary.rollingUpdatePartition`, now configured using `secondary.updateStrategy` setting `secondary.updateStrategy.rollingUpdate.partition`. -- `metrics.serviceMonitor.relabellings`, previously used to configure ServiceMonitor metricRelabelings, has been replaced with the value `metrics.serviceMonitor.metricRelabelings`, and new value `metrics.serviceMonitor.relabelings` can be used to set ServiceMonitor relabelings parameter - -### To 9.0.0 - -[On November 13, 2020, Helm v2 support was formally finished](https://github.com/helm/charts#status-of-the-project), this major version is the result of the required changes applied to the Helm Chart to be able to incorporate the different features added in Helm v3 and to be consistent with the Helm project itself regarding the Helm v2 EOL. - -### To 8.0.0 - -- Several parameters were renamed or disappeared in favor of new ones on this major version: - - The terms _master_ and _slave_ have been replaced by the terms _primary_ and _secondary_. Therefore, parameters prefixed with `master` or `slave` are now prefixed with `primary` or `secondary`, respectively. - - `securityContext.*` is deprecated in favor of `primary.podSecurityContext`, `primary.containerSecurityContext`, `secondary.podSecurityContext`, and `secondary.containerSecurityContext`. - - Credentials parameter are reorganized under the `auth` parameter. - - `replication.enabled` parameter is deprecated in favor of `architecture` parameter that accepts two values: `standalone` and `replication`. -- The default MariaDB version was updated from 10.3 to 10.5. According to the official documentation, upgrading from 10.3 should be painless. However, there are some things that have changed which could affect an upgrade: - - [Incompatible changes upgrading from MariaDB 10.3 to MariaDB 10.4](https://mariadb.com/kb/en/upgrading-from-mariadb-103-to-mariadb-104/#incompatible-changes-between-103-and-104). - - [Incompatible changes upgrading from MariaDB 10.4 to MariaDB 10.5](https://mariadb.com/kb/en/upgrading-from-mariadb-104-to-mariadb-105/#incompatible-changes-between-104-and-105). -- Chart labels were adapted to follow the [Helm charts standard labels](https://helm.sh/docs/chart_best_practices/labels/#standard-labels). -- This version also introduces `bitnami/common`, a [library chart](https://helm.sh/docs/topics/library_charts/#helm) as a dependency. More documentation about this new utility could be found [here](https://github.com/bitnami/charts/tree/main/bitnami/common#bitnami-common-library-chart). Please, make sure that you have updated the chart dependencies before executing any upgrade. - -Consequences: - -Backwards compatibility is not guaranteed. To upgrade to `8.0.0`, install a new release of the MariaDB chart, and migrate the data from your previous release. You have 2 alternatives to do so: - -- Create a backup of the database, and restore it on the new release using tools such as [mysqldump](https://mariadb.com/kb/en/mysqldump/). -- Reuse the PVC used to hold the master data on your previous release. To do so, use the `primary.persistence.existingClaim` parameter. The following example assumes that the release name is `mariadb`: - -```console -helm install mariadb oci://REGISTRY_NAME/REPOSITORY_NAME/mariadb --set auth.rootPassword=[ROOT_PASSWORD] --set primary.persistence.existingClaim=[EXISTING_PVC] -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -| Note: you need to substitute the placeholder _[EXISTING_PVC]_ with the name of the PVC used on your previous release, and _[ROOT_PASSWORD]_ with the root password used in your previous release. - -### To 7.0.0 - -Helm performs a lookup for the object based on its group (apps), version (v1), and kind (Deployment). Also known as its GroupVersionKind, or GVK. Changing the GVK is considered a compatibility breaker from Kubernetes' point of view, so you cannot "upgrade" those objects to the new GVK in-place. Earlier versions of Helm 3 did not perform the lookup correctly which has since been fixed to match the spec. - -In the `apiVersion` of the statefulset resources was updated to `apps/v1` in tune with the api's deprecated, resulting in compatibility breakage. - -This major version bump signifies this change. - -### To 6.0.0 - -MariaDB version was updated from 10.1 to 10.3, there are no changes in the chart itself. According to the official documentation, upgrading from 10.1 should be painless. However, there are some things that have changed which could affect an upgrade: - -- [Incompatible changes upgrading from MariaDB 10.1 to MariaDB 10.2](https://mariadb.com/kb/en/library/upgrading-from-mariadb-101-to-mariadb-102//#incompatible-changes-between-101-and-102) -- [Incompatible changes upgrading from MariaDB 10.2 to MariaDB 10.3](https://mariadb.com/kb/en/library/upgrading-from-mariadb-102-to-mariadb-103/#incompatible-changes-between-102-and-103) - -### To 5.0.0 - -Backwards compatibility is not guaranteed unless you modify the labels used on the chart's deployments. -Use the workaround below to upgrade from versions previous to 5.0.0. The following example assumes that the release name is mariadb: - -```console -kubectl delete statefulset opencart-mariadb --cascade=false -``` - -## License - -Copyright © 2024 Broadcom. The term "Broadcom" refers to Broadcom Inc. and/or its subsidiaries. - -Licensed under the Apache License, Version 2.0 (the "License"); -you may not use this file except in compliance with the License. -You may obtain a copy of the License at - - - -Unless required by applicable law or agreed to in writing, software -distributed under the License is distributed on an "AS IS" BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -See the License for the specific language governing permissions and -limitations under the License. \ No newline at end of file diff --git a/charts/mariadb/mariadb/templates/NOTES.txt b/charts/mariadb/mariadb/templates/NOTES.txt deleted file mode 100644 index c5e1110..0000000 --- a/charts/mariadb/mariadb/templates/NOTES.txt +++ /dev/null @@ -1,77 +0,0 @@ -CHART NAME: {{ .Chart.Name }} -CHART VERSION: {{ .Chart.Version }} -APP VERSION: {{ .Chart.AppVersion }} - -** Please be patient while the chart is being deployed ** - -{{- if .Values.diagnosticMode.enabled }} -The chart has been deployed in diagnostic mode. All probes have been disabled and the command has been overwritten with: - - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 4 }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 4 }} - -Get the list of pods by executing: - - kubectl get pods --namespace {{ .Release.Namespace }} -l app.kubernetes.io/instance={{ .Release.Name }} - -Access the pod you want to debug by executing - - kubectl exec --namespace {{ .Release.Namespace }} -ti -- bash - -In order to replicate the container startup scripts execute this command: - - /opt/bitnami/scripts/mariadb/entrypoint.sh /opt/bitnami/scripts/mariadb/run.sh - -{{- else }} - -Tip: - - Watch the deployment status using the command: kubectl get pods -w --namespace {{ .Release.Namespace }} -l app.kubernetes.io/instance={{ .Release.Name }} - -Services: - - echo Primary: {{ include "mariadb.primary.fullname" . }}.{{ .Release.Namespace }}.svc.{{ .Values.clusterDomain }}:{{ coalesce .Values.primary.service.ports.mysql .Values.primary.service.port }} -{{- if eq .Values.architecture "replication" }} - echo Secondary: {{ include "mariadb.secondary.fullname" . }}.{{ .Release.Namespace }}.svc.{{ .Values.clusterDomain }}:{{ coalesce .Values.secondary.service.ports.mysql .Values.secondary.service.port }} -{{- end }} - -Administrator credentials: - - Username: root - Password : $(kubectl get secret --namespace {{ .Release.Namespace }} {{ template "mariadb.secretName" . }} -o jsonpath="{.data.mariadb-root-password}" | base64 -d) - -To connect to your database: - - 1. Run a pod that you can use as a client: - - kubectl run {{ include "common.names.fullname" . }}-client --rm --tty -i --restart='Never' --image {{ template "mariadb.image" . }} --namespace {{ .Release.Namespace }} --command -- bash - - 2. To connect to primary service (read/write): - - mysql -h {{ include "mariadb.primary.fullname" . }}.{{ .Release.Namespace }}.svc.{{ .Values.clusterDomain }} -uroot -p {{ .Values.auth.database }} - -{{- if eq .Values.architecture "replication" }} - - 3. To connect to secondary service (read-only): - - mysql -h {{ include "mariadb.secondary.fullname" . }}.{{ .Release.Namespace }}.svc.{{ .Values.clusterDomain }} -uroot -p {{ .Values.auth.database }} -{{- end }} - -To upgrade this helm chart: - - 1. Obtain the password as described on the 'Administrator credentials' section and set the 'auth.rootPassword' parameter as shown below: - - ROOT_PASSWORD=$(kubectl get secret --namespace {{ .Release.Namespace }} {{ template "mariadb.secretName" . }} -o jsonpath="{.data.mariadb-root-password}" | base64 -d) - helm upgrade --namespace {{ .Release.Namespace }} {{ .Release.Name }} oci://registry-1.docker.io/bitnamicharts/mariadb --set auth.rootPassword=$ROOT_PASSWORD - -{{- include "common.warnings.rollingTag" .Values.image }} -{{- include "common.warnings.rollingTag" .Values.metrics.image }} -{{- include "common.warnings.rollingTag" .Values.volumePermissions.image }} -{{- include "mariadb.validateValues" . }} -{{- if not .Values.auth.customPasswordFiles -}} - {{- $passwordValidationErrors := include "common.validations.values.mariadb.passwords" (dict "secret" (include "common.names.fullname" .) "context" $) -}} - {{- include "common.errors.upgrade.passwords.empty" (dict "validationErrors" (list $passwordValidationErrors) "context" $) -}} -{{- end }} -{{- end }} -{{- include "common.warnings.resources" (dict "sections" (list "metrics" "primary" "secondary" "volumePermissions") "context" $) }} -{{- include "common.warnings.modifiedImages" (dict "images" (list .Values.image .Values.volumePermissions.image .Values.metrics.image) "context" $) }} \ No newline at end of file diff --git a/charts/mariadb/mariadb/templates/_helpers.tpl b/charts/mariadb/mariadb/templates/_helpers.tpl deleted file mode 100644 index e871624..0000000 --- a/charts/mariadb/mariadb/templates/_helpers.tpl +++ /dev/null @@ -1,164 +0,0 @@ -{{/* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{/* vim: set filetype=mustache: */}} - -{{- define "mariadb.primary.fullname" -}} -{{- if eq .Values.architecture "replication" }} -{{- printf "%s-%s" (include "common.names.fullname" .) .Values.primary.name | trunc 63 | trimSuffix "-" -}} -{{- else -}} -{{- include "common.names.fullname" . -}} -{{- end -}} -{{- end -}} - -{{- define "mariadb.secondary.fullname" -}} -{{- printf "%s-%s" (include "common.names.fullname" .) .Values.secondary.name | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Return the proper MariaDB image name -*/}} -{{- define "mariadb.image" -}} -{{ include "common.images.image" (dict "imageRoot" .Values.image "global" .Values.global) }} -{{- end -}} - -{{/* -Return the proper metrics image name -*/}} -{{- define "mariadb.metrics.image" -}} -{{ include "common.images.image" (dict "imageRoot" .Values.metrics.image "global" .Values.global) }} -{{- end -}} - -{{/* -Return the proper image name (for the init container volume-permissions image) -*/}} -{{- define "mariadb.volumePermissions.image" -}} -{{ include "common.images.image" (dict "imageRoot" .Values.volumePermissions.image "global" .Values.global) }} -{{- end -}} - -{{/* -Return the proper Docker Image Registry Secret Names -*/}} -{{- define "mariadb.imagePullSecrets" -}} -{{ include "common.images.pullSecrets" (dict "images" (list .Values.image .Values.metrics.image .Values.volumePermissions.image) "global" .Values.global) }} -{{- end -}} - -{{ template "mariadb.initdbScriptsCM" . }} -{{/* -Get the initialization scripts ConfigMap name. -*/}} -{{- define "mariadb.initdbScriptsCM" -}} -{{- if .Values.initdbScriptsConfigMap -}} -{{- printf "%s" (tpl .Values.initdbScriptsConfigMap $) -}} -{{- else -}} -{{- printf "%s-init-scripts" (include "mariadb.primary.fullname" .) -}} -{{- end -}} -{{- end -}} - -{{/* -Create the name of the service account to use -*/}} -{{- define "mariadb.serviceAccountName" -}} -{{- if .Values.serviceAccount.create -}} - {{ default (include "common.names.fullname" .) .Values.serviceAccount.name }} -{{- else -}} - {{ default "default" .Values.serviceAccount.name }} -{{- end -}} -{{- end -}} - -{{/* -Return the configmap with the MariaDB Primary configuration -*/}} -{{- define "mariadb.primary.configmapName" -}} -{{- if .Values.primary.existingConfigmap -}} - {{- printf "%s" (tpl .Values.primary.existingConfigmap $) -}} -{{- else -}} - {{- printf "%s" (include "mariadb.primary.fullname" .) -}} -{{- end -}} -{{- end -}} - -{{/* -Return true if a configmap object should be created for MariaDB Secondary -*/}} -{{- define "mariadb.primary.createConfigmap" -}} -{{- if and .Values.primary.configuration (not .Values.primary.existingConfigmap) }} - {{- true -}} -{{- else -}} -{{- end -}} -{{- end -}} - -{{/* -Return the configmap with the MariaDB Primary configuration -*/}} -{{- define "mariadb.secondary.configmapName" -}} -{{- if .Values.secondary.existingConfigmap -}} - {{- printf "%s" (tpl .Values.secondary.existingConfigmap $) -}} -{{- else -}} - {{- printf "%s" (include "mariadb.secondary.fullname" .) -}} -{{- end -}} -{{- end -}} - -{{/* -Return true if a configmap object should be created for MariaDB Secondary -*/}} -{{- define "mariadb.secondary.createConfigmap" -}} -{{- if and (eq .Values.architecture "replication") .Values.secondary.configuration (not .Values.secondary.existingConfigmap) }} - {{- true -}} -{{- else -}} -{{- end -}} -{{- end -}} - -{{/* -Return the secret with MariaDB credentials -*/}} -{{- define "mariadb.secretName" -}} - {{- if .Values.auth.existingSecret -}} - {{- printf "%s" (tpl .Values.auth.existingSecret $) -}} - {{- else -}} - {{- printf "%s" (include "common.names.fullname" .) -}} - {{- end -}} -{{- end -}} - -{{/* -Return true if a secret object should be created for MariaDB -*/}} -{{- define "mariadb.createSecret" -}} -{{- if not (or .Values.auth.existingSecret .Values.auth.customPasswordFiles) }} - {{- true -}} -{{- end -}} -{{- end -}} - -{{/* -Compile all warnings into a single message, and call fail. -*/}} -{{- define "mariadb.validateValues" -}} -{{- $messages := list -}} -{{- $messages := append $messages (include "mariadb.validateValues.architecture" .) -}} -{{- $messages := without $messages "" -}} -{{- $message := join "\n" $messages -}} - -{{- if $message -}} -{{- printf "\nVALUES VALIDATION:\n%s" $message | fail -}} -{{- end -}} -{{- end -}} - -{{/* Validate values of MariaDB - must provide a valid architecture */}} -{{- define "mariadb.validateValues.architecture" -}} -{{- if and (ne .Values.architecture "standalone") (ne .Values.architecture "replication") -}} -mariadb: architecture - Invalid architecture selected. Valid values are "standalone" and - "replication". Please set a valid architecture (--set architecture="xxxx") -{{- end -}} -{{- end -}} - -{{/* -Get existing password to access MariaDB -*/}} -{{- define "mariadb.secret.existPassword" -}} -{{- $secret := (lookup "v1" "Secret" .Release.Namespace (include "mariadb.secretName" .)).data -}} -{{- if hasKey $secret "mariadb-password" }} - {{- true -}} -{{- end -}} -{{- end -}} diff --git a/charts/mariadb/mariadb/templates/extra-list.yaml b/charts/mariadb/mariadb/templates/extra-list.yaml deleted file mode 100644 index 329f5c6..0000000 --- a/charts/mariadb/mariadb/templates/extra-list.yaml +++ /dev/null @@ -1,9 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- range .Values.extraDeploy }} ---- -{{ include "common.tplvalues.render" (dict "value" . "context" $) }} -{{- end }} diff --git a/charts/mariadb/mariadb/templates/networkpolicy.yaml b/charts/mariadb/mariadb/templates/networkpolicy.yaml deleted file mode 100644 index c7f6053..0000000 --- a/charts/mariadb/mariadb/templates/networkpolicy.yaml +++ /dev/null @@ -1,76 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.networkPolicy.enabled }} -kind: NetworkPolicy -apiVersion: {{ template "common.capabilities.networkPolicy.apiVersion" . }} -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - podSelector: - matchLabels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 6 }} - policyTypes: - - Ingress - - Egress - {{- if .Values.networkPolicy.allowExternalEgress }} - egress: - - {} - {{- else }} - egress: - # Allow dns resolution - - ports: - - port: 53 - protocol: UDP - - port: 53 - protocol: TCP - # Allow connection to other cluster pods - - ports: - - port: {{ .Values.primary.containerPorts.mysql }} - - port: {{ .Values.secondary.containerPorts.mysql }} - to: - - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 14 }} - {{- if .Values.networkPolicy.extraEgress }} - {{- include "common.tplvalues.render" ( dict "value" .Values.rts.networkPolicy.extraEgress "context" $ ) | nindent 4 }} - {{- end }} - {{- end }} - ingress: - - ports: - - port: {{ .Values.primary.containerPorts.mysql }} - - port: {{ .Values.secondary.containerPorts.mysql }} - {{- if .Values.metrics.enabled }} - - port: {{ .Values.metrics.containerPorts.http }} - {{- end }} - {{- if not .Values.networkPolicy.allowExternal }} - from: - - podSelector: - matchLabels: - {{ template "common.names.fullname" . }}-client: "true" - - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 14 }} - {{- if .Values.networkPolicy.ingressNSMatchLabels }} - - namespaceSelector: - matchLabels: - {{- range $key, $value := .Values.networkPolicy.ingressNSMatchLabels }} - {{ $key | quote }}: {{ $value | quote }} - {{- end }} - {{- if .Values.networkPolicy.ingressNSPodMatchLabels }} - podSelector: - matchLabels: - {{- range $key, $value := .Values.networkPolicy.ingressNSPodMatchLabels }} - {{ $key | quote }}: {{ $value | quote }} - {{- end }} - {{- end }} - {{- end }} - {{- end }} - {{- if .Values.networkPolicy.extraIngress }} - {{- include "common.tplvalues.render" ( dict "value" .Values.networkPolicy.extraIngress "context" $ ) | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/mariadb/mariadb/templates/primary/configmap.yaml b/charts/mariadb/mariadb/templates/primary/configmap.yaml deleted file mode 100644 index 37136a9..0000000 --- a/charts/mariadb/mariadb/templates/primary/configmap.yaml +++ /dev/null @@ -1,20 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if (include "mariadb.primary.createConfigmap" .) }} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ include "mariadb.primary.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: - my.cnf: |- - {{- include "common.tplvalues.render" ( dict "value" .Values.primary.configuration "context" $ ) | nindent 4 }} -{{- end -}} diff --git a/charts/mariadb/mariadb/templates/primary/initialization-configmap.yaml b/charts/mariadb/mariadb/templates/primary/initialization-configmap.yaml deleted file mode 100644 index eab9863..0000000 --- a/charts/mariadb/mariadb/templates/primary/initialization-configmap.yaml +++ /dev/null @@ -1,19 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.initdbScripts (not .Values.initdbScriptsConfigMap) }} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ printf "%s-init-scripts" (include "mariadb.primary.fullname" .) }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: -{{- include "common.tplvalues.render" (dict "value" .Values.initdbScripts "context" .) | nindent 2 }} -{{ end }} diff --git a/charts/mariadb/mariadb/templates/primary/pdb.yaml b/charts/mariadb/mariadb/templates/primary/pdb.yaml deleted file mode 100644 index 5f4f652..0000000 --- a/charts/mariadb/mariadb/templates/primary/pdb.yaml +++ /dev/null @@ -1,28 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.primary.pdb.create }} -apiVersion: {{ include "common.capabilities.policy.apiVersion" . }} -kind: PodDisruptionBudget -metadata: - name: {{ include "mariadb.primary.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- if .Values.primary.pdb.minAvailable }} - minAvailable: {{ .Values.primary.pdb.minAvailable }} - {{- end }} - {{- if or .Values.primary.pdb.maxUnavailable (not .Values.primary.pdb.minAvailable) }} - maxUnavailable: {{ .Values.primary.pdb.maxUnavailable | default 1 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.podLabels .Values.commonLabels ) "context" . ) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: primary -{{- end }} diff --git a/charts/mariadb/mariadb/templates/primary/statefulset.yaml b/charts/mariadb/mariadb/templates/primary/statefulset.yaml deleted file mode 100644 index 9b66234..0000000 --- a/charts/mariadb/mariadb/templates/primary/statefulset.yaml +++ /dev/null @@ -1,443 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -apiVersion: {{ include "common.capabilities.statefulset.apiVersion" . }} -kind: StatefulSet -metadata: - name: {{ include "mariadb.primary.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - replicas: 1 - revisionHistoryLimit: {{ .Values.primary.revisionHistoryLimit }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.podLabels .Values.commonLabels ) "context" . ) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: primary - serviceName: {{ include "mariadb.primary.fullname" . }} - {{- if .Values.primary.updateStrategy }} - updateStrategy: {{- toYaml .Values.primary.updateStrategy | nindent 4 }} - {{- end }} - template: - metadata: - annotations: - {{- if (include "mariadb.primary.createConfigmap" .) }} - checksum/configuration: {{ include (print $.Template.BasePath "/primary/configmap.yaml") . | sha256sum }} - {{- end }} - {{- if .Values.primary.podAnnotations }} - {{- include "common.tplvalues.render" (dict "value" .Values.primary.podAnnotations "context" $) | nindent 8 }} - {{- end }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $podLabels "context" $ ) | nindent 8 }} - app.kubernetes.io/component: primary - spec: - {{- include "mariadb.imagePullSecrets" . | nindent 6 }} - automountServiceAccountToken: {{ .Values.primary.automountServiceAccountToken }} - {{- if .Values.primary.hostAliases }} - hostAliases: {{- include "common.tplvalues.render" (dict "value" .Values.primary.hostAliases "context" $) | nindent 8 }} - {{- end }} - {{- if or .Values.primary.schedulerName .Values.schedulerName }} - schedulerName: {{ (coalesce .Values.primary.schedulerName .Values.schedulerName) | quote }} - {{- end }} - serviceAccountName: {{ template "mariadb.serviceAccountName" . }} - {{- if .Values.primary.affinity }} - affinity: {{- include "common.tplvalues.render" (dict "value" .Values.primary.affinity "context" $) | nindent 8 }} - {{- else }} - affinity: - podAffinity: {{- include "common.affinities.pods" (dict "type" .Values.primary.podAffinityPreset "component" "primary" "customLabels" $podLabels "context" $) | nindent 10 }} - podAntiAffinity: {{- include "common.affinities.pods" (dict "type" .Values.primary.podAntiAffinityPreset "component" "primary" "customLabels" $podLabels "context" $) | nindent 10 }} - nodeAffinity: {{- include "common.affinities.nodes" (dict "type" .Values.primary.nodeAffinityPreset.type "key" .Values.primary.nodeAffinityPreset.key "values" .Values.primary.nodeAffinityPreset.values) | nindent 10 }} - {{- end }} - {{- if .Values.primary.nodeSelector }} - nodeSelector: {{- include "common.tplvalues.render" (dict "value" .Values.primary.nodeSelector "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.primary.tolerations }} - tolerations: {{- include "common.tplvalues.render" (dict "value" .Values.primary.tolerations "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.primary.schedulerName }} - schedulerName: {{ .Values.primary.schedulerName }} - {{- end }} - {{- if .Values.primary.topologySpreadConstraints }} - topologySpreadConstraints: {{- include "common.tplvalues.render" (dict "value" .Values.primary.topologySpreadConstraints "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.primary.priorityClassName }} - priorityClassName: {{ .Values.primary.priorityClassName | quote }} - {{- else if .Values.priorityClassName }} - priorityClassName: {{ .Values.priorityClassName | quote }} - {{- end }} - {{- if .Values.primary.runtimeClassName }} - runtimeClassName: {{ .Values.primary.runtimeClassName | quote }} - {{- else if .Values.runtimeClassName }} - runtimeClassName: {{ .Values.runtimeClassName | quote }} - {{- end }} - {{- if .Values.primary.podSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.primary.podSecurityContext "context" $) | nindent 8 }} - {{- end }} - initContainers: - - name: preserve-logs-symlinks - image: {{ include "mariadb.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy | quote }} - {{- if .Values.primary.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.primary.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.primary.resources }} - resources: {{ toYaml .Values.primary.resources | nindent 12 }} - {{- else if ne .Values.primary.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.primary.resourcesPreset) | nindent 12 }} - {{- end }} - command: - - /bin/bash - args: - - -ec - - | - #!/bin/bash - - . /opt/bitnami/scripts/libfs.sh - # We copy the logs folder because it has symlinks to stdout and stderr - if ! is_dir_empty /opt/bitnami/mariadb/logs; then - cp -r /opt/bitnami/mariadb/logs /emptydir/app-logs-dir - fi - volumeMounts: - - name: empty-dir - mountPath: /emptydir - {{- if and .Values.primary.podSecurityContext.enabled .Values.volumePermissions.enabled .Values.primary.persistence.enabled }} - - name: volume-permissions - image: {{ include "mariadb.volumePermissions.image" . }} - imagePullPolicy: {{ .Values.volumePermissions.image.pullPolicy | quote }} - command: - - /bin/bash - - -ec - - | - chown -R {{ .Values.primary.containerSecurityContext.runAsUser }}:{{ .Values.primary.podSecurityContext.fsGroup }} /bitnami/mariadb - securityContext: - runAsUser: 0 - {{- if .Values.volumePermissions.resources }} - resources: {{- toYaml .Values.volumePermissions.resources | nindent 12 }} - {{- else if ne .Values.volumePermissions.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.volumePermissions.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: data - mountPath: /bitnami/mariadb - {{- if .Values.primary.persistence.subPath }} - subPath: {{ .Values.primary.persistence.subPath }} - {{- end }} - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - {{- end }} - {{- if .Values.primary.initContainers }} - {{- include "common.tplvalues.render" (dict "value" .Values.primary.initContainers "context" $) | nindent 8 }} - {{- end }} - containers: - - name: mariadb - image: {{ include "mariadb.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy | quote }} - {{- if .Values.primary.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.primary.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - {{- else if .Values.primary.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.primary.command "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- else if .Values.primary.args }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.primary.args "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.primary.lifecycleHooks }} - lifecycle: {{- include "common.tplvalues.render" (dict "value" .Values.primary.lifecycleHooks "context" $) | nindent 12 }} - {{- end }} - env: - - name: BITNAMI_DEBUG - value: {{ ternary "true" "false" (or .Values.image.debug .Values.diagnosticMode.enabled) | quote }} - {{- if .Values.auth.usePasswordFiles }} - - name: MARIADB_ROOT_PASSWORD_FILE - value: {{ default "/opt/bitnami/mariadb/secrets/mariadb-root-password" .Values.auth.customPasswordFiles.root }} - {{- else }} - - name: MARIADB_ROOT_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "mariadb.secretName" . }} - key: mariadb-root-password - {{- end }} - {{- if not (empty .Values.auth.username) }} - - name: MARIADB_USER - value: {{ .Values.auth.username | quote }} - {{- if .Values.auth.usePasswordFiles }} - - name: MARIADB_PASSWORD_FILE - value: {{ default "/opt/bitnami/mariadb/secrets/mariadb-password" .Values.auth.customPasswordFiles.user }} - {{- else }} - - name: MARIADB_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "mariadb.secretName" . }} - key: mariadb-password - {{- end }} - {{- end }} - - name: MARIADB_DATABASE - value: {{ .Values.auth.database | quote }} - {{- if eq .Values.architecture "replication" }} - - name: MARIADB_REPLICATION_MODE - value: "master" - - name: MARIADB_REPLICATION_USER - value: {{ .Values.auth.replicationUser | quote }} - {{- if .Values.auth.usePasswordFiles }} - - name: MARIADB_REPLICATION_PASSWORD_FILE - value: {{ default "/opt/bitnami/mariadb/secrets/mariadb-replication-password" .Values.auth.customPasswordFiles.replicator }} - {{- else }} - - name: MARIADB_REPLICATION_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "mariadb.secretName" . }} - key: mariadb-replication-password - {{- end }} - {{- end }} - {{- if .Values.primary.extraFlags }} - - name: MARIADB_EXTRA_FLAGS - value: "{{ .Values.primary.extraFlags }}" - {{- end }} - {{- if .Values.primary.startupWaitOptions }} - - name: MARIADB_STARTUP_WAIT_RETRIES - value: "{{ .Values.primary.startupWaitOptions.retries | default 300 }}" - - name: MARIADB_STARTUP_WAIT_SLEEP_TIME - value: "{{ .Values.primary.startupWaitOptions.sleepTime | default 2 }}" - {{- end }} - {{- if .Values.primary.extraEnvVars }} - {{- include "common.tplvalues.render" (dict "value" .Values.primary.extraEnvVars "context" $) | nindent 12 }} - {{- end }} - {{- if or .Values.primary.extraEnvVarsCM .Values.primary.extraEnvVarsSecret }} - envFrom: - {{- if .Values.primary.extraEnvVarsCM }} - - configMapRef: - name: {{ .Values.primary.extraEnvVarsCM }} - {{- end }} - {{- if .Values.primary.extraEnvVarsSecret }} - - secretRef: - name: {{ .Values.primary.extraEnvVarsSecret }} - {{- end }} - {{- end }} - ports: - - name: mysql - containerPort: {{ .Values.primary.containerPorts.mysql }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.primary.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.primary.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.primary.startupProbe.enabled }} - startupProbe: {{- omit .Values.primary.startupProbe "enabled" | toYaml | nindent 12 }} - exec: - command: - - /bin/bash - - -ec - - | - password_aux="${MARIADB_ROOT_PASSWORD:-}" - if [[ -f "${MARIADB_ROOT_PASSWORD_FILE:-}" ]]; then - password_aux=$(cat "$MARIADB_ROOT_PASSWORD_FILE") - fi - mariadb-admin ping -uroot -p"${password_aux}" - {{- end }} - {{- if .Values.primary.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.primary.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.primary.livenessProbe.enabled }} - livenessProbe: {{- omit .Values.primary.livenessProbe "enabled" | toYaml | nindent 12 }} - exec: - command: - - /bin/bash - - -ec - - | - password_aux="${MARIADB_ROOT_PASSWORD:-}" - if [[ -f "${MARIADB_ROOT_PASSWORD_FILE:-}" ]]; then - password_aux=$(cat "$MARIADB_ROOT_PASSWORD_FILE") - fi - mariadb-admin status -uroot -p"${password_aux}" - {{- end }} - {{- if .Values.primary.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.primary.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.primary.readinessProbe.enabled }} - readinessProbe: {{- omit .Values.primary.readinessProbe "enabled" | toYaml | nindent 12 }} - exec: - command: - - /bin/bash - - -ec - - | - password_aux="${MARIADB_ROOT_PASSWORD:-}" - if [[ -f "${MARIADB_ROOT_PASSWORD_FILE:-}" ]]; then - password_aux=$(cat "$MARIADB_ROOT_PASSWORD_FILE") - fi - mariadb-admin ping -uroot -p"${password_aux}" - {{- end }} - {{- end }} - {{- if .Values.primary.resources }} - resources: {{ toYaml .Values.primary.resources | nindent 12 }} - {{- else if ne .Values.primary.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.primary.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: data - mountPath: /bitnami/mariadb - {{- if .Values.primary.persistence.subPath }} - subPath: {{ .Values.primary.persistence.subPath }} - {{- end }} - {{- if or .Values.initdbScriptsConfigMap .Values.initdbScripts }} - - name: custom-init-scripts - mountPath: /docker-entrypoint-initdb.d - {{- end }} - {{- if or .Values.primary.configuration .Values.primary.existingConfigmap }} - - name: config - mountPath: /opt/bitnami/mariadb/conf/my.cnf - subPath: my.cnf - {{- end }} - {{- if and .Values.auth.usePasswordFiles (not .Values.auth.customPasswordFiles) }} - - name: mariadb-credentials - mountPath: /opt/bitnami/mariadb/secrets/ - {{- end }} - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: empty-dir - mountPath: /opt/bitnami/mariadb/conf - subPath: app-conf-dir - - name: empty-dir - mountPath: /opt/bitnami/mariadb/tmp - subPath: app-tmp-dir - - name: empty-dir - mountPath: /opt/bitnami/mariadb/logs - subPath: app-logs-dir - {{- if .Values.primary.extraVolumeMounts }} - {{- include "common.tplvalues.render" (dict "value" .Values.primary.extraVolumeMounts "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.metrics.enabled }} - - name: metrics - image: {{ include "mariadb.metrics.image" . }} - imagePullPolicy: {{ .Values.metrics.image.pullPolicy | quote }} - {{- if .Values.metrics.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.metrics.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - env: - {{- if .Values.auth.usePasswordFiles }} - - name: MARIADB_ROOT_PASSWORD_FILE - value: {{ default "/opt/bitnami/mysqld-exporter/secrets/mariadb-root-password" .Values.auth.customPasswordFiles.root }} - {{- else }} - - name: MARIADB_ROOT_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "mariadb.secretName" . }} - key: mariadb-root-password - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- else }} - command: - - /bin/bash - - -ec - - | - password_aux="${MARIADB_ROOT_PASSWORD:-}" - if [[ -f "${MARIADB_ROOT_PASSWORD_FILE:-}" ]]; then - password_aux=$(cat "$MARIADB_ROOT_PASSWORD_FILE") - fi - MYSQLD_EXPORTER_PASSWORD=${password_aux} /bin/mysqld_exporter --mysqld.address=localhost:{{ .Values.primary.containerPorts.mysql }} --mysqld.username=root --web.listen-address=:{{ .Values.metrics.containerPorts.http }} {{- range .Values.metrics.extraArgs.primary }} {{ . }} {{- end }} - {{- end }} - ports: - - name: metrics - containerPort: {{ .Values.metrics.containerPorts.http }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.metrics.livenessProbe.enabled }} - livenessProbe: {{- omit .Values.metrics.livenessProbe "enabled" | toYaml | nindent 12 }} - httpGet: - path: /metrics - port: metrics - {{- end }} - {{- if .Values.metrics.readinessProbe.enabled }} - readinessProbe: {{- omit .Values.metrics.readinessProbe "enabled" | toYaml | nindent 12 }} - httpGet: - path: /metrics - port: metrics - {{- end }} - {{- end }} - {{- if .Values.metrics.resources }} - resources: {{- toYaml .Values.metrics.resources | nindent 12 }} - {{- else if ne .Values.metrics.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.metrics.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - {{- if and .Values.auth.usePasswordFiles (not .Values.auth.customPasswordFiles) }} - - name: mariadb-credentials - mountPath: /opt/bitnami/mysqld-exporter/secrets/ - {{- end }} - {{- if .Values.metrics.extraVolumeMounts.primary }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.extraVolumeMounts.primary "context" $) | nindent 12 }} - {{- end }} - {{- end }} - {{- if .Values.primary.sidecars }} - {{- include "common.tplvalues.render" (dict "value" .Values.primary.sidecars "context" $) | nindent 8 }} - {{- end }} - volumes: - - name: empty-dir - emptyDir: {} - {{- if or .Values.primary.configuration .Values.primary.existingConfigmap }} - - name: config - configMap: - name: {{ include "mariadb.primary.configmapName" . }} - {{- end }} - {{- if or .Values.initdbScriptsConfigMap .Values.initdbScripts }} - - name: custom-init-scripts - configMap: - name: {{ template "mariadb.initdbScriptsCM" . }} - {{- end }} - {{- if and .Values.auth.usePasswordFiles (not .Values.auth.customPasswordFiles) }} - - name: mariadb-credentials - secret: - secretName: {{ template "mariadb.secretName" . }} - items: - - key: mariadb-root-password - path: mariadb-root-password - - key: mariadb-password - path: mariadb-password - {{- if eq .Values.architecture "replication" }} - - key: mariadb-replication-password - path: mariadb-replication-password - {{- end }} - {{- end }} - {{- if .Values.primary.extraVolumes }} - {{- include "common.tplvalues.render" (dict "value" .Values.primary.extraVolumes "context" $) | nindent 8 }} - {{- end }} - {{- if and .Values.primary.persistence.enabled .Values.primary.persistence.existingClaim }} - - name: data - persistentVolumeClaim: - claimName: {{ tpl .Values.primary.persistence.existingClaim . }} - {{- else if not .Values.primary.persistence.enabled }} - - name: data - emptyDir: {} - {{- else if and .Values.primary.persistence.enabled (not .Values.primary.persistence.existingClaim) }} - volumeClaimTemplates: - - metadata: - name: data - {{- $claimLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.persistence.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.matchLabels" ( dict "customLabels" $claimLabels "context" $ ) | nindent 10 }} - app.kubernetes.io/component: primary - {{- if .Values.primary.persistence.annotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.primary.persistence.annotations "context" $ ) | nindent 10 }} - {{- end }} - spec: - accessModes: - {{- range .Values.primary.persistence.accessModes }} - - {{ . | quote }} - {{- end }} - resources: - requests: - storage: {{ .Values.primary.persistence.size | quote }} - {{ include "common.storage.class" (dict "persistence" .Values.primary.persistence "global" .Values.global) }} - {{- if .Values.primary.persistence.selector }} - selector: {{- include "common.tplvalues.render" (dict "value" .Values.primary.persistence.selector "context" $) | nindent 10 }} - {{- end -}} - {{- end }} diff --git a/charts/mariadb/mariadb/templates/primary/svc.yaml b/charts/mariadb/mariadb/templates/primary/svc.yaml deleted file mode 100644 index b0fc974..0000000 --- a/charts/mariadb/mariadb/templates/primary/svc.yaml +++ /dev/null @@ -1,62 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -apiVersion: v1 -kind: Service -metadata: - name: {{ include "mariadb.primary.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary - annotations: - {{- if or .Values.primary.service.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.service.annotations .Values.commonAnnotations ) "context" . ) }} - {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} - {{- if and .Values.metrics.enabled .Values.metrics.annotations }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.annotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - type: {{ .Values.primary.service.type }} - {{- if and .Values.primary.service.clusterIP (eq .Values.primary.service.type "ClusterIP") }} - clusterIP: {{ .Values.primary.service.clusterIP }} - {{- end }} - {{- if and .Values.primary.service.externalTrafficPolicy (or (eq .Values.primary.service.type "LoadBalancer") (eq .Values.primary.service.type "NodePort")) }} - externalTrafficPolicy: {{ .Values.primary.service.externalTrafficPolicy | quote }} - {{- end }} - {{- if and (eq .Values.primary.service.type "LoadBalancer") .Values.primary.service.loadBalancerSourceRanges }} - loadBalancerSourceRanges: {{- toYaml .Values.primary.service.loadBalancerSourceRanges | nindent 4 }} - {{ end }} - {{- if (and (eq .Values.primary.service.type "LoadBalancer") (not (empty .Values.primary.service.loadBalancerIP))) }} - loadBalancerIP: {{ .Values.primary.service.loadBalancerIP }} - {{- end }} - {{- if .Values.primary.service.sessionAffinity }} - sessionAffinity: {{ .Values.primary.service.sessionAffinity }} - {{- end }} - {{- if .Values.primary.service.sessionAffinityConfig }} - sessionAffinityConfig: {{- include "common.tplvalues.render" (dict "value" .Values.primary.service.sessionAffinityConfig "context" $) | nindent 4 }} - {{- end }} - ports: - - name: mysql - port: {{ coalesce .Values.primary.service.ports.mysql .Values.primary.service.port }} - protocol: TCP - targetPort: mysql - {{- if (and (or (eq .Values.primary.service.type "NodePort") (eq .Values.primary.service.type "LoadBalancer")) (coalesce .Values.primary.service.nodePorts.mysql .Values.primary.service.nodePort)) }} - nodePort: {{ coalesce .Values.primary.service.nodePorts.mysql .Values.primary.service.nodePort }} - {{- else if eq .Values.primary.service.type "ClusterIP" }} - nodePort: null - {{- end }} - {{- if and .Values.metrics.enabled (gt (.Values.primary.service.ports.metrics | int) 0) }} - - name: metrics - port: {{ .Values.primary.service.ports.metrics }} - protocol: TCP - targetPort: metrics - {{- end }} - {{- if .Values.primary.service.extraPorts }} - {{- include "common.tplvalues.render" (dict "value" .Values.primary.service.extraPorts "context" $) | nindent 4 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary diff --git a/charts/mariadb/mariadb/templates/prometheusrules.yaml b/charts/mariadb/mariadb/templates/prometheusrules.yaml deleted file mode 100644 index d24f0e2..0000000 --- a/charts/mariadb/mariadb/templates/prometheusrules.yaml +++ /dev/null @@ -1,24 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled .Values.metrics.prometheusRule.enabled }} -apiVersion: monitoring.coreos.com/v1 -kind: PrometheusRule -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ default .Release.Namespace .Values.metrics.prometheusRule.namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: metrics - {{- if .Values.metrics.prometheusRule.additionalLabels }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.prometheusRule.additionalLabels "context" $ ) | nindent 4 }} - {{- end }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - groups: - - name: {{ include "common.names.fullname" . }} - rules: {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.prometheusRule.rules "context" $ ) | nindent 6 }} -{{- end }} diff --git a/charts/mariadb/mariadb/templates/role.yaml b/charts/mariadb/mariadb/templates/role.yaml deleted file mode 100644 index 4102e66..0000000 --- a/charts/mariadb/mariadb/templates/role.yaml +++ /dev/null @@ -1,23 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.serviceAccount.create .Values.rbac.create }} -apiVersion: {{ include "common.capabilities.rbac.apiVersion" . }} -kind: Role -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -rules: - - apiGroups: - - "" - resources: - - endpoints - verbs: - - get -{{- end }} diff --git a/charts/mariadb/mariadb/templates/rolebinding.yaml b/charts/mariadb/mariadb/templates/rolebinding.yaml deleted file mode 100644 index 08d8b3d..0000000 --- a/charts/mariadb/mariadb/templates/rolebinding.yaml +++ /dev/null @@ -1,23 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.serviceAccount.create .Values.rbac.create }} -kind: RoleBinding -apiVersion: {{ include "common.capabilities.rbac.apiVersion" . }} -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -subjects: - - kind: ServiceAccount - name: {{ include "mariadb.serviceAccountName" . }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: {{ include "common.names.fullname" . -}} -{{- end }} diff --git a/charts/mariadb/mariadb/templates/secondary/configmap.yaml b/charts/mariadb/mariadb/templates/secondary/configmap.yaml deleted file mode 100644 index 770173f..0000000 --- a/charts/mariadb/mariadb/templates/secondary/configmap.yaml +++ /dev/null @@ -1,20 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if (include "mariadb.secondary.createConfigmap" .) }} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ include "mariadb.secondary.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: secondary - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: - my.cnf: |- - {{- include "common.tplvalues.render" ( dict "value" .Values.secondary.configuration "context" $ ) | nindent 4 }} -{{- end -}} diff --git a/charts/mariadb/mariadb/templates/secondary/pdb.yaml b/charts/mariadb/mariadb/templates/secondary/pdb.yaml deleted file mode 100644 index 1e6b2c6..0000000 --- a/charts/mariadb/mariadb/templates/secondary/pdb.yaml +++ /dev/null @@ -1,28 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and (eq .Values.architecture "replication") .Values.secondary.pdb.create }} -apiVersion: {{ include "common.capabilities.policy.apiVersion" . }} -kind: PodDisruptionBudget -metadata: - name: {{ include "mariadb.secondary.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: secondary - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- if .Values.secondary.pdb.minAvailable }} - minAvailable: {{ .Values.secondary.pdb.minAvailable }} - {{- end }} - {{- if or .Values.secondary.pdb.maxUnavailable (not .Values.secondary.pdb.minAvailable) }} - maxUnavailable: {{ .Values.secondary.pdb.maxUnavailable | default 1 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.secondary.podLabels .Values.commonLabels ) "context" . ) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: secondary -{{- end }} diff --git a/charts/mariadb/mariadb/templates/secondary/statefulset.yaml b/charts/mariadb/mariadb/templates/secondary/statefulset.yaml deleted file mode 100644 index b8560c4..0000000 --- a/charts/mariadb/mariadb/templates/secondary/statefulset.yaml +++ /dev/null @@ -1,414 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if eq .Values.architecture "replication" }} -apiVersion: {{ include "common.capabilities.statefulset.apiVersion" . }} -kind: StatefulSet -metadata: - name: {{ include "mariadb.secondary.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: secondary - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - replicas: {{ .Values.secondary.replicaCount }} - revisionHistoryLimit: {{ .Values.secondary.revisionHistoryLimit }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.secondary.podLabels .Values.commonLabels ) "context" . ) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: secondary - serviceName: {{ include "mariadb.secondary.fullname" . }} - podManagementPolicy: {{ .Values.secondary.podManagementPolicy }} - {{- if .Values.secondary.updateStrategy }} - updateStrategy: {{- toYaml .Values.secondary.updateStrategy | nindent 4 }} - {{- end }} - template: - metadata: - annotations: - {{- if (include "mariadb.secondary.createConfigmap" .) }} - checksum/configuration: {{ include (print $.Template.BasePath "/secondary/configmap.yaml") . | sha256sum }} - {{- end }} - {{- if .Values.secondary.podAnnotations }} - {{- include "common.tplvalues.render" (dict "value" .Values.secondary.podAnnotations "context" $) | nindent 8 }} - {{- end }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $podLabels "context" $ ) | nindent 8 }} - app.kubernetes.io/component: secondary - spec: - {{- include "mariadb.imagePullSecrets" . | nindent 6 }} - {{- if or .Values.secondary.schedulerName .Values.schedulerName }} - schedulerName: {{ (coalesce .Values.secondary.schedulerName .Values.schedulerName) | quote }} - {{- end }} - serviceAccountName: {{ template "mariadb.serviceAccountName" . }} - automountServiceAccountToken: {{ .Values.secondary.automountServiceAccountToken }} - {{- if .Values.secondary.hostAliases }} - hostAliases: {{- include "common.tplvalues.render" (dict "value" .Values.secondary.hostAliases "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.secondary.affinity }} - affinity: {{- include "common.tplvalues.render" (dict "value" .Values.secondary.affinity "context" $) | nindent 8 }} - {{- else }} - affinity: - podAffinity: {{- include "common.affinities.pods" (dict "type" .Values.secondary.podAffinityPreset "component" "secondary" "customLabels" $podLabels "context" $) | nindent 10 }} - podAntiAffinity: {{- include "common.affinities.pods" (dict "type" .Values.secondary.podAntiAffinityPreset "component" "secondary" "customLabels" $podLabels "context" $) | nindent 10 }} - nodeAffinity: {{- include "common.affinities.nodes" (dict "type" .Values.secondary.nodeAffinityPreset.type "key" .Values.secondary.nodeAffinityPreset.key "values" .Values.secondary.nodeAffinityPreset.values) | nindent 10 }} - {{- end }} - {{- if .Values.secondary.nodeSelector }} - nodeSelector: {{- include "common.tplvalues.render" (dict "value" .Values.secondary.nodeSelector "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.secondary.tolerations }} - tolerations: {{- include "common.tplvalues.render" (dict "value" .Values.secondary.tolerations "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.secondary.topologySpreadConstraints }} - topologySpreadConstraints: {{- include "common.tplvalues.render" (dict "value" .Values.secondary.topologySpreadConstraints "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.secondary.priorityClassName }} - priorityClassName: {{ .Values.secondary.priorityClassName | quote }} - {{- else if .Values.priorityClassName }} - priorityClassName: {{ .Values.priorityClassName | quote }} - {{- end }} - {{- if .Values.secondary.runtimeClassName }} - runtimeClassName: {{ .Values.secondary.runtimeClassName | quote }} - {{- else if .Values.runtimeClassName }} - runtimeClassName: {{ .Values.runtimeClassName | quote }} - {{- end }} - {{- if .Values.secondary.podSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.secondary.podSecurityContext "context" $) | nindent 8 }} - {{- end }} - initContainers: - - name: preserve-logs-symlinks - image: {{ include "mariadb.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy | quote }} - {{- if .Values.secondary.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.primary.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.secondary.resources }} - resources: {{ toYaml .Values.secondary.resources | nindent 12 }} - {{- else if ne .Values.secondary.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.secondary.resourcesPreset) | nindent 12 }} - {{- end }} - command: - - /bin/bash - args: - - -ec - - | - #!/bin/bash - - . /opt/bitnami/scripts/libfs.sh - # We copy the logs folder because it has symlinks to stdout and stderr - if ! is_dir_empty /opt/bitnami/mariadb/logs; then - cp -r /opt/bitnami/mariadb/logs /emptydir/app-logs-dir - fi - volumeMounts: - - name: empty-dir - mountPath: /emptydir - {{- if and .Values.secondary.podSecurityContext.enabled .Values.volumePermissions.enabled .Values.secondary.persistence.enabled }} - - name: volume-permissions - image: {{ include "mariadb.volumePermissions.image" . }} - imagePullPolicy: {{ .Values.volumePermissions.image.pullPolicy | quote }} - command: - - /bin/bash - - -ec - - | - chown -R {{ .Values.secondary.containerSecurityContext.runAsUser }}:{{ .Values.secondary.podSecurityContext.fsGroup }} /bitnami/mariadb - securityContext: - runAsUser: 0 - {{- if .Values.volumePermissions.resources }} - resources: {{- toYaml .Values.volumePermissions.resources | nindent 12 }} - {{- else if ne .Values.volumePermissions.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.volumePermissions.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: data - mountPath: /bitnami/mariadb - {{- if .Values.secondary.persistence.subPath }} - subPath: {{ .Values.secondary.persistence.subPath }} - {{- end }} - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - {{- end }} - {{- if .Values.secondary.initContainers }} - {{- include "common.tplvalues.render" (dict "value" .Values.secondary.initContainers "context" $) | nindent 8 }} - {{- end }} - containers: - - name: mariadb - image: {{ include "mariadb.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy | quote }} - {{- if .Values.secondary.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.secondary.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - {{- else if .Values.secondary.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.secondary.command "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- else if .Values.secondary.args }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.secondary.args "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.secondary.lifecycleHooks }} - lifecycle: {{- include "common.tplvalues.render" (dict "value" .Values.secondary.lifecycleHooks "context" $) | nindent 12 }} - {{- end }} - env: - - name: BITNAMI_DEBUG - value: {{ ternary "true" "false" (or .Values.image.debug .Values.diagnosticMode.enabled) | quote }} - - name: MARIADB_REPLICATION_MODE - value: "slave" - - name: MARIADB_MASTER_HOST - value: {{ include "mariadb.primary.fullname" . }} - - name: MARIADB_MASTER_PORT_NUMBER - value: {{ coalesce .Values.primary.service.ports.mysql .Values.primary.service.port | quote }} - - name: MARIADB_MASTER_ROOT_USER - value: "root" - {{- if .Values.auth.usePasswordFiles }} - - name: MARIADB_MASTER_ROOT_PASSWORD_FILE - value: {{ default "/opt/bitnami/mariadb/secrets/mariadb-root-password" .Values.auth.customPasswordFiles.root }} - {{- else }} - - name: MARIADB_MASTER_ROOT_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "mariadb.secretName" . }} - key: mariadb-root-password - {{- end }} - - name: MARIADB_REPLICATION_USER - value: {{ .Values.auth.replicationUser | quote }} - {{- if .Values.auth.usePasswordFiles }} - - name: MARIADB_REPLICATION_PASSWORD_FILE - value: {{ default "/opt/bitnami/mariadb/secrets/mariadb-replication-password" .Values.auth.customPasswordFiles.replicator }} - {{- else }} - - name: MARIADB_REPLICATION_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "mariadb.secretName" . }} - key: mariadb-replication-password - {{- end }} - {{- if .Values.secondary.extraFlags }} - - name: MARIADB_EXTRA_FLAGS - value: "{{ .Values.secondary.extraFlags }}" - {{- end }} - {{- if .Values.secondary.startupWaitOptions }} - - name: MARIADB_STARTUP_WAIT_RETRIES - value: "{{ .Values.secondary.startupWaitOptions.retries | default 300 }}" - - name: MARIADB_STARTUP_WAIT_SLEEP_TIME - value: "{{ .Values.secondary.startupWaitOptions.sleepTime | default 2 }}" - {{- end }} - {{- if .Values.secondary.extraEnvVars }} - {{- include "common.tplvalues.render" (dict "value" .Values.secondary.extraEnvVars "context" $) | nindent 12 }} - {{- end }} - {{- if or .Values.secondary.extraEnvVarsCM .Values.secondary.extraEnvVarsSecret }} - envFrom: - {{- if .Values.secondary.extraEnvVarsCM }} - - configMapRef: - name: {{ .Values.secondary.extraEnvVarsCM }} - {{- end }} - {{- if .Values.secondary.extraEnvVarsSecret }} - - secretRef: - name: {{ .Values.secondary.extraEnvVarsSecret }} - {{- end }} - {{- end }} - ports: - - name: mysql - containerPort: {{ .Values.secondary.containerPorts.mysql }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.secondary.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.secondary.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.secondary.startupProbe.enabled }} - startupProbe: {{- omit .Values.secondary.startupProbe "enabled" | toYaml | nindent 12 }} - exec: - command: - - /bin/bash - - -ec - - | - password_aux="${MARIADB_MASTER_ROOT_PASSWORD:-}" - if [[ -f "${MARIADB_MASTER_ROOT_PASSWORD_FILE:-}" ]]; then - password_aux=$(cat "$MARIADB_MASTER_ROOT_PASSWORD_FILE") - fi - mariadb-admin ping -uroot -p"${password_aux}" - {{- end }} - {{- if .Values.secondary.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.secondary.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.secondary.livenessProbe.enabled }} - livenessProbe: {{- omit .Values.secondary.livenessProbe "enabled" | toYaml | nindent 12 }} - exec: - command: - - /bin/bash - - -ec - - | - password_aux="${MARIADB_MASTER_ROOT_PASSWORD:-}" - if [[ -f "${MARIADB_MASTER_ROOT_PASSWORD_FILE:-}" ]]; then - password_aux=$(cat "$MARIADB_MASTER_ROOT_PASSWORD_FILE") - fi - mariadb-admin status -uroot -p"${password_aux}" - {{- end }} - {{- if .Values.secondary.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.secondary.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.secondary.readinessProbe.enabled }} - readinessProbe: {{- omit .Values.secondary.readinessProbe "enabled" | toYaml | nindent 12 }} - exec: - command: - - /bin/bash - - -ec - - | - password_aux="${MARIADB_MASTER_ROOT_PASSWORD:-}" - if [[ -f "${MARIADB_MASTER_ROOT_PASSWORD_FILE:-}" ]]; then - password_aux=$(cat "$MARIADB_MASTER_ROOT_PASSWORD_FILE") - fi - mariadb-admin ping -uroot -p"${password_aux}" - {{- end }} - {{- end }} - {{- if .Values.secondary.resources }} - resources: {{ toYaml .Values.secondary.resources | nindent 12 }} - {{- else if ne .Values.secondary.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.secondary.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: data - mountPath: /bitnami/mariadb - {{- if .Values.secondary.persistence.subPath }} - subPath: {{ .Values.secondary.persistence.subPath }} - {{- end }} - {{- if or .Values.secondary.configuration .Values.secondary.existingConfigmap }} - - name: config - mountPath: /opt/bitnami/mariadb/conf/my.cnf - subPath: my.cnf - {{- end }} - {{- if and .Values.auth.usePasswordFiles (not .Values.auth.customPasswordFiles) }} - - name: mariadb-credentials - mountPath: /opt/bitnami/mariadb/secrets/ - {{- end }} - {{- if .Values.secondary.extraVolumeMounts }} - {{- include "common.tplvalues.render" (dict "value" .Values.secondary.extraVolumeMounts "context" $) | nindent 12 }} - {{- end }} - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: empty-dir - mountPath: /opt/bitnami/mariadb/conf - subPath: app-conf-dir - - name: empty-dir - mountPath: /opt/bitnami/mariadb/tmp - subPath: app-tmp-dir - - name: empty-dir - mountPath: /opt/bitnami/mariadb/logs - subPath: app-logs-dir - {{- if .Values.metrics.enabled }} - - name: metrics - image: {{ include "mariadb.metrics.image" . }} - imagePullPolicy: {{ .Values.metrics.image.pullPolicy | quote }} - {{- if .Values.metrics.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.metrics.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - env: - {{- if .Values.auth.usePasswordFiles }} - - name: MARIADB_ROOT_PASSWORD_FILE - value: {{ default "/opt/bitnami/mysqld-exporter/secrets/mariadb-root-password" .Values.auth.customPasswordFiles.root }} - {{- else }} - - name: MARIADB_ROOT_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "mariadb.secretName" . }} - key: mariadb-root-password - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- else }} - command: - - /bin/bash - - -ec - - | - password_aux="${MARIADB_ROOT_PASSWORD:-}" - if [[ -f "${MARIADB_ROOT_PASSWORD_FILE:-}" ]]; then - password_aux=$(cat "$MARIADB_ROOT_PASSWORD_FILE") - fi - MYSQLD_EXPORTER_PASSWORD=${password_aux} /bin/mysqld_exporter --mysqld.address=localhost:{{ .Values.secondary.containerPorts.mysql }} --mysqld.username=root --web.listen-address=:{{ .Values.metrics.containerPorts.http }} {{- range .Values.metrics.extraArgs.primary }} {{ . }} {{- end }} - {{- end }} - ports: - - name: metrics - containerPort: {{ .Values.metrics.containerPorts.http }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.metrics.livenessProbe.enabled }} - livenessProbe: {{- omit .Values.metrics.livenessProbe "enabled" | toYaml | nindent 12 }} - httpGet: - path: /metrics - port: metrics - {{- end }} - {{- if .Values.metrics.readinessProbe.enabled }} - readinessProbe: {{- omit .Values.metrics.readinessProbe "enabled" | toYaml | nindent 12 }} - httpGet: - path: /metrics - port: metrics - {{- end }} - {{- end }} - {{- if .Values.metrics.resources }} - resources: {{- toYaml .Values.metrics.resources | nindent 12 }} - {{- else if ne .Values.metrics.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.metrics.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - {{- if and .Values.auth.usePasswordFiles (not .Values.auth.customPasswordFiles) }} - - name: mariadb-credentials - mountPath: /opt/bitnami/mysqld-exporter/secrets/ - {{- end }} - - name: empty-dir - mountPath: /tmp - subPath: app-tmp-dir - {{- if .Values.metrics.extraVolumeMounts.secondary }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.extraVolumeMounts.secondary "context" $) | nindent 12 }} - {{- end }} - {{- end }} - {{- if .Values.secondary.sidecars }} - {{- include "common.tplvalues.render" (dict "value" .Values.secondary.sidecars "context" $) | nindent 8 }} - {{- end }} - volumes: - {{- if or .Values.secondary.configuration .Values.secondary.existingConfigmap }} - - name: config - configMap: - name: {{ include "mariadb.secondary.configmapName" . }} - {{- end }} - {{- if and .Values.auth.usePasswordFiles (not .Values.auth.customPasswordFiles) }} - - name: mariadb-credentials - secret: - secretName: {{ template "mariadb.secretName" . }} - items: - - key: mariadb-root-password - path: mariadb-root-password - - key: mariadb-replication-password - path: mariadb-replication-password - {{- end }} - - name: empty-dir - emptyDir: {} - {{- if .Values.secondary.extraVolumes }} - {{- include "common.tplvalues.render" (dict "value" .Values.secondary.extraVolumes "context" $) | nindent 8 }} - {{- end }} - {{- if not .Values.secondary.persistence.enabled }} - - name: data - emptyDir: {} - {{- else }} - volumeClaimTemplates: - - metadata: - name: data - {{- $claimLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.secondary.persistence.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.matchLabels" ( dict "customLabels" $claimLabels "context" $ ) | nindent 10 }} - app.kubernetes.io/component: secondary - {{- if .Values.secondary.persistence.annotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.secondary.persistence.annotations "context" $ ) | nindent 10 }} - {{- end }} - spec: - accessModes: - {{- range .Values.secondary.persistence.accessModes }} - - {{ . | quote }} - {{- end }} - resources: - requests: - storage: {{ .Values.secondary.persistence.size | quote }} - {{ include "common.storage.class" (dict "persistence" .Values.secondary.persistence "global" .Values.global) }} - {{- if .Values.secondary.persistence.selector }} - selector: {{- include "common.tplvalues.render" (dict "value" .Values.secondary.persistence.selector "context" $) | nindent 10 }} - {{- end -}} - {{- end }} -{{- end }} diff --git a/charts/mariadb/mariadb/templates/secondary/svc.yaml b/charts/mariadb/mariadb/templates/secondary/svc.yaml deleted file mode 100644 index cd6f1d1..0000000 --- a/charts/mariadb/mariadb/templates/secondary/svc.yaml +++ /dev/null @@ -1,64 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if eq .Values.architecture "replication" }} -apiVersion: v1 -kind: Service -metadata: - name: {{ include "mariadb.secondary.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: secondary - annotations: - {{- if or .Values.secondary.service.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.secondary.service.annotations .Values.commonAnnotations ) "context" . ) }} - {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} - {{- if and .Values.metrics.enabled .Values.metrics.annotations }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.annotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - type: {{ .Values.secondary.service.type }} - {{- if and .Values.secondary.service.clusterIP (eq .Values.secondary.service.type "ClusterIP") }} - clusterIP: {{ .Values.secondary.service.clusterIP }} - {{- end }} - {{- if and .Values.secondary.service.externalTrafficPolicy (or (eq .Values.secondary.service.type "LoadBalancer") (eq .Values.secondary.service.type "NodePort")) }} - externalTrafficPolicy: {{ .Values.secondary.service.externalTrafficPolicy | quote }} - {{- end }} - {{- if and (eq .Values.secondary.service.type "LoadBalancer") .Values.secondary.service.loadBalancerSourceRanges }} - loadBalancerSourceRanges: {{- toYaml .Values.secondary.service.loadBalancerSourceRanges | nindent 4 }} - {{ end }} - {{- if and (eq .Values.secondary.service.type "LoadBalancer") (not (empty .Values.secondary.service.loadBalancerIP)) }} - loadBalancerIP: {{ .Values.secondary.service.loadBalancerIP }} - {{- end }} - {{- if .Values.secondary.service.sessionAffinity }} - sessionAffinity: {{ .Values.secondary.service.sessionAffinity }} - {{- end }} - {{- if .Values.secondary.service.sessionAffinityConfig }} - sessionAffinityConfig: {{- include "common.tplvalues.render" (dict "value" .Values.secondary.service.sessionAffinityConfig "context" $) | nindent 4 }} - {{- end }} - ports: - - name: mysql - port: {{ coalesce .Values.secondary.service.ports.mysql .Values.secondary.service.port }} - protocol: TCP - targetPort: mysql - {{- if (and (or (eq .Values.secondary.service.type "NodePort") (eq .Values.secondary.service.type "LoadBalancer")) (coalesce .Values.secondary.service.nodePorts.mysql .Values.secondary.service.nodePort)) }} - nodePort: {{ coalesce .Values.secondary.service.nodePorts.mysql .Values.secondary.service.nodePort }} - {{- else if eq .Values.secondary.service.type "ClusterIP" }} - nodePort: null - {{- end }} - {{- if and .Values.metrics.enabled (gt (.Values.secondary.service.ports.metrics | int) 0) }} - - name: metrics - port: {{ .Values.secondary.service.ports.metrics }} - protocol: TCP - targetPort: metrics - {{- end }} - {{- if .Values.secondary.service.extraPorts }} - {{- include "common.tplvalues.render" (dict "value" .Values.secondary.service.extraPorts "context" $) | nindent 4 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.secondary.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: secondary -{{- end }} diff --git a/charts/mariadb/mariadb/templates/secrets.yaml b/charts/mariadb/mariadb/templates/secrets.yaml deleted file mode 100644 index 9551ee2..0000000 --- a/charts/mariadb/mariadb/templates/secrets.yaml +++ /dev/null @@ -1,98 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- $host := include "mariadb.primary.fullname" . }} -{{- $port := print .Values.primary.service.ports.mysql }} -{{- $rootPassword := include "common.secrets.passwords.manage" ( dict "secret" ( include "mariadb.secretName" . ) "key" "mariadb-root-password" "providedValues" ( list "auth.rootPassword" ) "context" $ ) | trimAll "\"" | b64dec }} -{{- $password := .Values.auth.password }} -{{- if and .Values.auth.username ( include "mariadb.secret.existPassword" . ) }} -{{- $password = include "common.secrets.passwords.manage" ( dict "secret" ( include "mariadb.secretName" . ) "key" "mariadb-password" "providedValues" ( list "auth.password" ) "context" $ ) | trimAll "\"" | b64dec }} -{{- else if ( and .Values.auth.username ( not .Values.auth.forcePassword ) ( empty .Values.auth.password ) ) }} -{{- $password = randAlphaNum 10 }} -{{- end }} -{{- if eq ( include "mariadb.createSecret" . ) "true" }} -apiVersion: v1 -kind: Secret -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: Opaque -data: - {{- if ( not .Values.auth.forcePassword ) }} - mariadb-root-password: {{ print $rootPassword | b64enc | quote }} - {{- else }} - mariadb-root-password: {{ required "A MariaDB Root Password is required!" .Values.auth.rootPassword | b64enc | quote }} - {{- end }} - {{- if .Values.auth.username }} - {{- if ( not .Values.auth.forcePassword ) }} - mariadb-password: {{ print $password | b64enc | quote }} - {{- else }} - mariadb-password: {{ required "A MariaDB Database Password is required!" $password | b64enc | quote }} - {{- end }} - {{- end }} - {{- if eq .Values.architecture "replication" }} - {{- if ( not .Values.auth.forcePassword ) }} - mariadb-replication-password: {{ include "common.secrets.passwords.manage" ( dict "secret" ( include "common.names.fullname" . ) "key" "mariadb-replication-password" "providedValues" ( list "auth.replicationPassword" ) "context" $ ) }} - {{- else }} - mariadb-replication-password: {{ required "A MariaDB Replication Password is required!" .Values.auth.replicationPassword | b64enc | quote }} - {{- end }} - {{- end }} -{{- end }} - -{{- if .Values.serviceBindings.enabled }} ---- -apiVersion: v1 -kind: Secret -metadata: - name: {{ include "common.names.fullname" . }}-svcbind-root - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: servicebinding.io/mysql -data: - provider: {{ print "bitnami" | b64enc | quote }} - type: {{ print "mysql" | b64enc | quote }} - host: {{ print $host | b64enc | quote }} - port: {{ print $port | b64enc | quote }} - username: {{ print "root" | b64enc | quote }} - password: {{ print $rootPassword | b64enc | quote }} - uri: {{ printf "mysql://root:%s@%s:%s" $rootPassword $host $port | b64enc | quote }} - -{{- if .Values.auth.username }} -{{- $database := .Values.auth.database }} ---- -apiVersion: v1 -kind: Secret -metadata: - name: {{ include "common.names.fullname" . }}-svcbind-custom-user - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: servicebinding.io/mysql -data: - provider: {{ print "bitnami" | b64enc | quote }} - type: {{ print "mysql" | b64enc | quote }} - host: {{ print $host | b64enc | quote }} - port: {{ print $port | b64enc | quote }} - username: {{ print .Values.auth.username | b64enc | quote }} - {{- if $database }} - database: {{ print $database | b64enc | quote }} - {{- end }} - {{- if and .Values.auth.forcePassword ( empty $password ) }} - password: {{ required "A MariaDB Database Password is required!" $password | b64enc | quote }} - {{- else }} - password: {{ print $password | b64enc | quote }} - {{- end }} - uri: {{ printf "mysql://%s:%s@%s:%s/%s" .Values.auth.username $password $host $port $database | b64enc | quote }} -{{- end }} -{{- end }} diff --git a/charts/mariadb/mariadb/templates/serviceaccount.yaml b/charts/mariadb/mariadb/templates/serviceaccount.yaml deleted file mode 100644 index 411decf..0000000 --- a/charts/mariadb/mariadb/templates/serviceaccount.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.serviceAccount.create }} -apiVersion: v1 -kind: ServiceAccount -metadata: - name: {{ include "mariadb.serviceAccountName" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if or .Values.serviceAccount.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.serviceAccount.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -automountServiceAccountToken: {{ .Values.serviceAccount.automountServiceAccountToken }} -{{- end }} diff --git a/charts/mariadb/mariadb/templates/servicemonitor.yaml b/charts/mariadb/mariadb/templates/servicemonitor.yaml deleted file mode 100644 index 046189d..0000000 --- a/charts/mariadb/mariadb/templates/servicemonitor.yaml +++ /dev/null @@ -1,44 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled .Values.metrics.serviceMonitor.enabled }} -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ default .Release.Namespace .Values.metrics.serviceMonitor.namespace | quote }} - {{- $labels := include "common.tplvalues.merge" ( dict "values" ( list .Values.metrics.serviceMonitor.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $labels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - jobLabel: {{ .Values.metrics.serviceMonitor.jobLabel | quote }} - endpoints: - - port: metrics - {{- if .Values.metrics.serviceMonitor.interval }} - interval: {{ .Values.metrics.serviceMonitor.interval }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.scrapeTimeout }} - scrapeTimeout: {{ .Values.metrics.serviceMonitor.scrapeTimeout }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.honorLabels }} - honorLabels: {{ .Values.metrics.serviceMonitor.honorLabels }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.metricRelabelings }} - metricRelabelings: {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.serviceMonitor.metricRelabelings "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.relabelings }} - relabelings: {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.serviceMonitor.relabelings "context" $) | nindent 8 }} - {{- end }} - namespaceSelector: - matchNames: - - {{ .Release.Namespace | quote }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 6 }} - {{- if .Values.metrics.serviceMonitor.selector }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.serviceMonitor.selector "context" $) | nindent 6 }} - {{- end }} -{{- end }} diff --git a/charts/mariadb/mariadb/values.schema.json b/charts/mariadb/mariadb/values.schema.json deleted file mode 100644 index 500c4eb..0000000 --- a/charts/mariadb/mariadb/values.schema.json +++ /dev/null @@ -1,176 +0,0 @@ -{ - "$schema": "http://json-schema.org/schema#", - "type": "object", - "properties": { - "architecture": { - "type": "string", - "title": "MariaDB architecture", - "form": true, - "description": "Allowed values: `standalone` or `replication`" - }, - "auth": { - "type": "object", - "title": "Authentication configuration", - "form": true, - "properties": { - "rootPassword": { - "type": "string", - "title": "MariaDB root password", - "form": true, - "description": "Defaults to a random 10-character alphanumeric string if not set" - }, - "database": { - "type": "string", - "title": "MariaDB custom database", - "description": "Name of the custom database to be created during the 1st initialization of MariaDB", - "form": true - }, - "username": { - "type": "string", - "title": "MariaDB custom user", - "description": "Name of the custom user to be created during the 1st initialization of MariaDB. This user only has permissions on the MariaDB custom database", - "form": true - }, - "password": { - "type": "string", - "title": "Password for MariaDB custom user", - "description": "Defaults to a random 10-character alphanumeric string if not set", - "form": true, - "hidden": { - "value": false, - "path": "usePassword" - } - }, - "replicationUser": { - "type": "string", - "title": "MariaDB replication user", - "description": "Name of user used to manage replication.", - "form": true, - "hidden": { - "value": "standalone", - "path": "architecture" - } - }, - "replicationPassword": { - "type": "string", - "title": "Password for MariaDB replication user", - "description": "Defaults to a random 10-character alphanumeric string if not set", - "form": true, - "hidden": { - "value": "standalone", - "path": "architecture" - } - } - } - }, - "primary": { - "type": "object", - "title": "Primary replicas settings", - "form": true, - "properties": { - "persistence": { - "type": "object", - "title": "Persistence for primary replicas", - "form": true, - "properties": { - "enabled": { - "type": "boolean", - "form": true, - "title": "Enable persistence", - "description": "Enable persistence using Persistent Volume Claims" - }, - "size": { - "type": "string", - "title": "Persistent Volume Size", - "form": true, - "render": "slider", - "sliderMin": 1, - "sliderMax": 100, - "sliderUnit": "Gi", - "hidden": { - "value": false, - "path": "persistence/enabled" - } - } - } - } - } - }, - "secondary": { - "type": "object", - "title": "Secondary replicas settings", - "form": true, - "hidden": { - "value": false, - "path": "replication/enabled" - }, - "properties": { - "persistence": { - "type": "object", - "title": "Persistence for secondary replicas", - "form": true, - "properties": { - "enabled": { - "type": "boolean", - "form": true, - "title": "Enable persistence", - "description": "Enable persistence using Persistent Volume Claims" - }, - "size": { - "type": "string", - "title": "Persistent Volume Size", - "form": true, - "render": "slider", - "sliderMin": 1, - "sliderMax": 100, - "sliderUnit": "Gi", - "hidden": { - "value": false, - "path": "persistence/enabled" - } - } - } - } - } - }, - "volumePermissions": { - "type": "object", - "properties": { - "enabled": { - "type": "boolean", - "form": true, - "title": "Enable Init Containers", - "description": "Use an init container to set required folder permissions on the data volume before mounting it in the final destination" - } - } - }, - "metrics": { - "type": "object", - "form": true, - "title": "Prometheus metrics details", - "properties": { - "enabled": { - "type": "boolean", - "title": "Create Prometheus metrics exporter", - "description": "Create a side-car container to expose Prometheus metrics", - "form": true - }, - "serviceMonitor": { - "type": "object", - "properties": { - "enabled": { - "type": "boolean", - "title": "Create Prometheus Operator ServiceMonitor", - "description": "Create a ServiceMonitor to track metrics using Prometheus Operator", - "form": true, - "hidden": { - "value": false, - "path": "metrics/enabled" - } - } - } - } - } - } - } -} diff --git a/charts/mariadb/mariadb/values.yaml b/charts/mariadb/mariadb/values.yaml deleted file mode 100644 index c6c7e31..0000000 --- a/charts/mariadb/mariadb/values.yaml +++ /dev/null @@ -1,1373 +0,0 @@ -# Copyright Broadcom, Inc. All Rights Reserved. -# SPDX-License-Identifier: APACHE-2.0 - -## @section Global parameters -## Global Docker image parameters -## Please, note that this will override the image parameters, including dependencies, configured to use the global value -## Current available global Docker image parameters: imageRegistry, imagePullSecrets and storageClass -## - -## @param global.imageRegistry Global Docker Image registry -## @param global.imagePullSecrets Global Docker registry secret names as an array -## @param global.defaultStorageClass Global default StorageClass for Persistent Volume(s) -## @param global.storageClass DEPRECATED: use global.defaultStorageClass instead -## -global: - imageRegistry: "" - ## E.g. - ## imagePullSecrets: - ## - myRegistryKeySecretName - ## - imagePullSecrets: [] - defaultStorageClass: "" - storageClass: "" - ## Compatibility adaptations for Kubernetes platforms - ## - compatibility: - ## Compatibility adaptations for Openshift - ## - openshift: - ## @param global.compatibility.openshift.adaptSecurityContext Adapt the securityContext sections of the deployment to make them compatible with Openshift restricted-v2 SCC: remove runAsUser, runAsGroup and fsGroup and let the platform use their allowed default IDs. Possible values: auto (apply if the detected running cluster is Openshift), force (perform the adaptation always), disabled (do not perform adaptation) - ## - adaptSecurityContext: auto -## @section Common parameters -## - -## @param kubeVersion Force target Kubernetes version (using Helm capabilities if not set) -## -kubeVersion: "" -## @param nameOverride String to partially override mariadb.fullname -## -nameOverride: "" -## @param fullnameOverride String to fully override mariadb.fullname -## -fullnameOverride: "" -## @param clusterDomain Default Kubernetes cluster domain -## -clusterDomain: cluster.local -## @param commonAnnotations Common annotations to add to all MariaDB resources (sub-charts are not considered) -## -commonAnnotations: {} -## @param commonLabels Common labels to add to all MariaDB resources (sub-charts are not considered) -## -commonLabels: {} -## @param schedulerName Name of the scheduler (other than default) to dispatch pods -## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ -## -schedulerName: "" -## @param runtimeClassName Name of the Runtime Class for all MariaDB pods -## ref: https://kubernetes.io/docs/concepts/containers/runtime-class/ -## -runtimeClassName: "" -## @param extraDeploy Array of extra objects to deploy with the release (evaluated as a template) -## -extraDeploy: [] -## Enable diagnostic mode in the deployment -## -diagnosticMode: - ## @param diagnosticMode.enabled Enable diagnostic mode (all probes will be disabled and the command will be overridden) - ## - enabled: false - ## @param diagnosticMode.command Command to override all containers in the deployment - ## - command: - - sleep - ## @param diagnosticMode.args Args to override all containers in the deployment - ## - args: - - infinity -## @param serviceBindings.enabled Create secret for service binding (Experimental) -## Ref: https://servicebinding.io/service-provider/ -## -serviceBindings: - enabled: false -## @section MariaDB common parameters -## - -## Bitnami MariaDB image -## ref: https://hub.docker.com/r/bitnami/mariadb/tags/ -## @param image.registry [default: REGISTRY_NAME] MariaDB image registry -## @param image.repository [default: REPOSITORY_NAME/mariadb] MariaDB image repository -## @skip image.tag MariaDB image tag (immutable tags are recommended) -## @param image.digest MariaDB image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag -## @param image.pullPolicy MariaDB image pull policy -## @param image.pullSecrets Specify docker-registry secret names as an array -## @param image.debug Specify if debug logs should be enabled -## -image: - registry: docker.io - repository: bitnami/mariadb - tag: 11.4.3-debian-12-r1 - digest: "" - ## Specify a imagePullPolicy - ## Defaults to 'Always' if image tag is 'latest', else set to 'IfNotPresent' - ## ref: https://kubernetes.io/docs/concepts/containers/images/#pre-pulled-images - ## - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets (secrets must be manually created in the namespace) - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## Example: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## Set to true if you would like to see extra information on logs - ## It turns BASH and/or NAMI debugging in the image - ## - debug: false -## @param architecture MariaDB architecture (`standalone` or `replication`) -## -architecture: standalone -## MariaDB Authentication parameters -## -auth: - ## @param auth.rootPassword Password for the `root` user. Ignored if existing secret is provided. - ## ref: https://github.com/bitnami/containers/tree/main/bitnami/mariadb#setting-the-root-password-on-first-run - ## - rootPassword: "" - ## @param auth.database Name for a custom database to create - ## ref: https://github.com/bitnami/containers/blob/main/bitnami/mariadb/README.md#creating-a-database-on-first-run - ## - database: my_database - ## @param auth.username Name for a custom user to create - ## ref: https://github.com/bitnami/containers/blob/main/bitnami/mariadb/README.md#creating-a-database-user-on-first-run - ## - username: "" - ## @param auth.password Password for the new user. Ignored if existing secret is provided - ## - password: "" - ## @param auth.replicationUser MariaDB replication user - ## ref: https://github.com/bitnami/containers/tree/main/bitnami/mariadb#setting-up-a-replication-cluster - ## - replicationUser: replicator - ## @param auth.replicationPassword MariaDB replication user password. Ignored if existing secret is provided - ## ref: https://github.com/bitnami/containers/tree/main/bitnami/mariadb#setting-up-a-replication-cluster - ## - replicationPassword: "" - ## @param auth.existingSecret Use existing secret for password details (`auth.rootPassword`, `auth.password`, `auth.replicationPassword` will be ignored and picked up from this secret). The secret has to contain the keys `mariadb-root-password`, `mariadb-replication-password` and `mariadb-password` - ## - existingSecret: "" - ## @param auth.forcePassword Force users to specify required passwords - ## - forcePassword: false - ## @param auth.usePasswordFiles Mount credentials as files instead of using environment variables - ## - usePasswordFiles: false - ## @param auth.customPasswordFiles Use custom password files when `auth.usePasswordFiles` is set to `true`. Define path for keys `root` and `user`, also define `replicator` if `architecture` is set to `replication` - ## Example: - ## customPasswordFiles: - ## root: /vault/secrets/mariadb-root - ## user: /vault/secrets/mariadb-user - ## replicator: /vault/secrets/mariadb-replicator - ## - customPasswordFiles: {} -## @param initdbScripts Dictionary of initdb scripts -## Specify dictionary of scripts to be run at first boot -## Example: -## initdbScripts: -## my_init_script.sh: | -## #!/bin/bash -## echo "Do something." -## -initdbScripts: {} -## @param initdbScriptsConfigMap ConfigMap with the initdb scripts (Note: Overrides `initdbScripts`) -## -initdbScriptsConfigMap: "" -## @section MariaDB Primary parameters -## - -## Mariadb Primary parameters -## -primary: - ## @param primary.name Name of the primary database (eg primary, master, leader, ...) - ## - name: primary - ## @param primary.command Override default container command on MariaDB Primary container(s) (useful when using custom images) - ## - command: [] - ## @param primary.args Override default container args on MariaDB Primary container(s) (useful when using custom images) - ## - args: [] - ## @param primary.lifecycleHooks for the MariaDB Primary container(s) to automate configuration before or after startup - ## - lifecycleHooks: {} - ## @param primary.automountServiceAccountToken Mount Service Account token in pod - ## - automountServiceAccountToken: false - ## @param primary.hostAliases Add deployment host aliases - ## https://kubernetes.io/docs/concepts/services-networking/add-entries-to-pod-etc-hosts-with-host-aliases/ - ## - hostAliases: [] - ## @param primary.containerPorts.mysql Container port for mysql - ## - containerPorts: - mysql: 3306 - ## @param primary.configuration [string] MariaDB Primary configuration to be injected as ConfigMap - ## ref: https://mysql.com/kb/en/mysql/configuring-mysql-with-mycnf/#example-of-configuration-file - ## - configuration: |- - [mysqld] - skip-name-resolve - explicit_defaults_for_timestamp - basedir=/opt/bitnami/mariadb - datadir=/bitnami/mariadb/data - plugin_dir=/opt/bitnami/mariadb/plugin - port={{ .Values.primary.containerPorts.mysql }} - socket=/opt/bitnami/mariadb/tmp/mysql.sock - tmpdir=/opt/bitnami/mariadb/tmp - max_allowed_packet=16M - bind-address=* - pid-file=/opt/bitnami/mariadb/tmp/mysqld.pid - log-error=/opt/bitnami/mariadb/logs/mysqld.log - character-set-server=UTF8 - collation-server=utf8_general_ci - slow_query_log=0 - long_query_time=10.0 - binlog_expire_logs_seconds=2592000 - - [client] - port=3306 - socket=/opt/bitnami/mariadb/tmp/mysql.sock - default-character-set=UTF8 - plugin_dir=/opt/bitnami/mariadb/plugin - - [manager] - port=3306 - socket=/opt/bitnami/mariadb/tmp/mysql.sock - pid-file=/opt/bitnami/mariadb/tmp/mysqld.pid - ## @param primary.existingConfigmap Name of existing ConfigMap with MariaDB Primary configuration. - ## NOTE: When it's set the 'configuration' parameter is ignored - ## - existingConfigmap: "" - ## @param primary.updateStrategy.type MariaDB primary statefulset strategy type - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#update-strategies - ## - updateStrategy: - ## StrategyType - ## Can be set to RollingUpdate or OnDelete - ## - type: RollingUpdate - ## @param primary.rollingUpdatePartition Partition update strategy for Mariadb Primary statefulset - ## https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#partitions - ## - rollingUpdatePartition: "" - ## @param primary.podAnnotations Additional pod annotations for MariaDB primary pods - ## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ - ## - podAnnotations: {} - ## @param primary.podLabels Extra labels for MariaDB primary pods - ## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/ - ## - podLabels: {} - ## @param primary.podAffinityPreset MariaDB primary pod affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity - ## - podAffinityPreset: "" - ## @param primary.podAntiAffinityPreset MariaDB primary pod anti-affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` - ## Ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity - ## - podAntiAffinityPreset: soft - ## Mariadb Primary node affinity preset - ## Ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity - ## - nodeAffinityPreset: - ## @param primary.nodeAffinityPreset.type MariaDB primary node affinity preset type. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` - ## - type: "" - ## @param primary.nodeAffinityPreset.key MariaDB primary node label key to match Ignored if `primary.affinity` is set. - ## E.g. - ## key: "kubernetes.io/e2e-az-name" - ## - key: "" - ## @param primary.nodeAffinityPreset.values MariaDB primary node label values to match. Ignored if `primary.affinity` is set. - ## E.g. - ## values: - ## - e2e-az1 - ## - e2e-az2 - ## - values: [] - ## @param primary.affinity Affinity for MariaDB primary pods assignment - ## Ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity - ## Note: podAffinityPreset, podAntiAffinityPreset, and nodeAffinityPreset will be ignored when it's set - ## - affinity: {} - ## @param primary.nodeSelector Node labels for MariaDB primary pods assignment - ## Ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/ - ## - nodeSelector: {} - ## @param primary.tolerations Tolerations for MariaDB primary pods assignment - ## Ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ - ## - tolerations: [] - ## @param primary.schedulerName Name of the k8s scheduler (other than default) - ## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ - ## - schedulerName: "" - ## @param primary.podManagementPolicy podManagementPolicy to manage scaling operation of MariaDB primary pods - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#pod-management-policies - ## - podManagementPolicy: "" - ## @param primary.topologySpreadConstraints Topology Spread Constraints for MariaDB primary pods assignment - ## ref: https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/ - ## E.g. - ## topologySpreadConstraints: - ## - maxSkew: 1 - ## topologyKey: topology.kubernetes.io/zone - ## whenUnsatisfiable: DoNotSchedule - ## - topologySpreadConstraints: [] - ## @param primary.priorityClassName Priority class for MariaDB primary pods assignment - ## Ref: https://kubernetes.io/docs/concepts/configuration/pod-priority-preemption/ - ## - priorityClassName: "" - ## @param primary.runtimeClassName Runtime Class for MariaDB primary pods - ## Ref: https://kubernetes.io/docs/concepts/containers/runtime-class/ - ## - runtimeClassName: "" - ## MariaDB primary Pod security context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-pod - ## @param primary.podSecurityContext.enabled Enable security context for MariaDB primary pods - ## @param primary.podSecurityContext.fsGroupChangePolicy Set filesystem group change policy - ## @param primary.podSecurityContext.sysctls Set kernel settings using the sysctl interface - ## @param primary.podSecurityContext.supplementalGroups Set filesystem extra groups - ## @param primary.podSecurityContext.fsGroup Group ID for the mounted volumes' filesystem - ## - podSecurityContext: - enabled: true - fsGroupChangePolicy: Always - sysctls: [] - supplementalGroups: [] - fsGroup: 1001 - ## MariaDB primary container security context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container - ## @param primary.containerSecurityContext.enabled MariaDB primary container securityContext - ## @param primary.containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container - ## @param primary.containerSecurityContext.runAsUser User ID for the MariaDB primary container - ## @param primary.containerSecurityContext.runAsGroup Group ID for the MariaDB primary container - ## @param primary.containerSecurityContext.runAsNonRoot Set primary container's Security Context runAsNonRoot - ## @param primary.containerSecurityContext.privileged Set primary container's Security Context privileged - ## @param primary.containerSecurityContext.allowPrivilegeEscalation Set primary container's Security Context allowPrivilegeEscalation - ## @param primary.containerSecurityContext.readOnlyRootFilesystem Set container's Security Context readOnlyRootFilesystem - ## @param primary.containerSecurityContext.capabilities.drop List of capabilities to be dropped - ## @param primary.containerSecurityContext.seccompProfile.type Set container's Security Context seccomp profile - ## - containerSecurityContext: - enabled: true - seLinuxOptions: {} - runAsUser: 1001 - runAsGroup: 1001 - runAsNonRoot: true - privileged: false - allowPrivilegeEscalation: false - readOnlyRootFilesystem: true - capabilities: - drop: ["ALL"] - seccompProfile: - type: "RuntimeDefault" - ## MariaDB primary container's resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## We usually recommend not to specify default resources and to leave this as a conscious - ## choice for the user. This also increases chances charts run on environments with little - ## resources, such as Minikube. If you do want to specify resources, uncomment the following - ## lines, adjust them as necessary, and remove the curly braces after 'resources:'. - ## @param primary.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if primary.resources is set (primary.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "micro" - ## @param primary.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} - ## Configure extra options for MariaDB primary containers' liveness, readiness and startup probes - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes) - ## @param primary.startupProbe.enabled Enable startupProbe - ## @param primary.startupProbe.initialDelaySeconds Initial delay seconds for startupProbe - ## @param primary.startupProbe.periodSeconds Period seconds for startupProbe - ## @param primary.startupProbe.timeoutSeconds Timeout seconds for startupProbe - ## @param primary.startupProbe.failureThreshold Failure threshold for startupProbe - ## @param primary.startupProbe.successThreshold Success threshold for startupProbe - ## - startupProbe: - enabled: false - initialDelaySeconds: 120 - periodSeconds: 15 - timeoutSeconds: 5 - failureThreshold: 10 - successThreshold: 1 - ## Configure extra options for liveness probe - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes - ## @param primary.livenessProbe.enabled Enable livenessProbe - ## @param primary.livenessProbe.initialDelaySeconds Initial delay seconds for livenessProbe - ## @param primary.livenessProbe.periodSeconds Period seconds for livenessProbe - ## @param primary.livenessProbe.timeoutSeconds Timeout seconds for livenessProbe - ## @param primary.livenessProbe.failureThreshold Failure threshold for livenessProbe - ## @param primary.livenessProbe.successThreshold Success threshold for livenessProbe - ## - livenessProbe: - enabled: true - initialDelaySeconds: 120 - periodSeconds: 10 - timeoutSeconds: 1 - failureThreshold: 3 - successThreshold: 1 - ## @param primary.readinessProbe.enabled Enable readinessProbe - ## @param primary.readinessProbe.initialDelaySeconds Initial delay seconds for readinessProbe - ## @param primary.readinessProbe.periodSeconds Period seconds for readinessProbe - ## @param primary.readinessProbe.timeoutSeconds Timeout seconds for readinessProbe - ## @param primary.readinessProbe.failureThreshold Failure threshold for readinessProbe - ## @param primary.readinessProbe.successThreshold Success threshold for readinessProbe - ## - readinessProbe: - enabled: true - initialDelaySeconds: 30 - periodSeconds: 10 - timeoutSeconds: 1 - failureThreshold: 3 - successThreshold: 1 - ## @param primary.customStartupProbe Override default startup probe for MariaDB primary containers - ## - customStartupProbe: {} - ## @param primary.customLivenessProbe Override default liveness probe for MariaDB primary containers - ## - customLivenessProbe: {} - ## @param primary.customReadinessProbe Override default readiness probe for MariaDB primary containers - ## - customReadinessProbe: {} - ## @param primary.startupWaitOptions Override default builtin startup wait check options for MariaDB primary containers - ## `bitnami/mariadb` Docker image has built-in startup check mechanism, - ## which periodically checks if MariaDB service has started up and stops it - ## if all checks have failed after X tries. Use these to control these checks. - ## ref: https://github.com/bitnami/containers/tree/main/bitnami/mariadb/pull/240 - ## Example (with default options): - ## startupWaitOptions: - ## retries: 300 - ## waitTime: 2 - ## - startupWaitOptions: {} - ## @param primary.extraFlags MariaDB primary additional command line flags - ## Can be used to specify command line flags, for example: - ## E.g. - ## extraFlags: "--max-connect-errors=1000 --max_connections=155" - ## - extraFlags: "" - ## @param primary.extraEnvVars Extra environment variables to be set on MariaDB primary containers - ## E.g. - ## extraEnvVars: - ## - name: TZ - ## value: "Europe/Paris" - ## - extraEnvVars: [] - ## @param primary.extraEnvVarsCM Name of existing ConfigMap containing extra env vars for MariaDB primary containers - ## - extraEnvVarsCM: "" - ## @param primary.extraEnvVarsSecret Name of existing Secret containing extra env vars for MariaDB primary containers - ## - extraEnvVarsSecret: "" - ## Enable persistence using Persistent Volume Claims - ## ref: https://kubernetes.io/docs/concepts/storage/persistent-volumes/ - ## - persistence: - ## @param primary.persistence.enabled Enable persistence on MariaDB primary replicas using a `PersistentVolumeClaim`. If false, use emptyDir - ## - enabled: true - ## @param primary.persistence.existingClaim Name of an existing `PersistentVolumeClaim` for MariaDB primary replicas - ## NOTE: When it's set the rest of persistence parameters are ignored - ## - existingClaim: "" - ## @param primary.persistence.subPath Subdirectory of the volume to mount at - ## - subPath: "" - ## @param primary.persistence.storageClass MariaDB primary persistent volume storage Class - ## If defined, storageClassName: - ## If set to "-", storageClassName: "", which disables dynamic provisioning - ## If undefined (the default) or set to null, no storageClassName spec is - ## set, choosing the default provisioner. (gp2 on AWS, standard on - ## GKE, AWS & OpenStack) - ## - storageClass: "" - ## @param primary.persistence.labels Labels for the PVC - ## - labels: {} - ## @param primary.persistence.annotations MariaDB primary persistent volume claim annotations - ## - annotations: {} - ## @param primary.persistence.accessModes MariaDB primary persistent volume access Modes - ## - accessModes: - - ReadWriteOnce - ## @param primary.persistence.size MariaDB primary persistent volume size - ## - size: 8Gi - ## @param primary.persistence.selector Selector to match an existing Persistent Volume - ## selector: - ## matchLabels: - ## app: my-app - ## - selector: {} - ## @param primary.extraVolumes Optionally specify extra list of additional volumes to the MariaDB Primary pod(s) - ## - extraVolumes: [] - ## @param primary.extraVolumeMounts Optionally specify extra list of additional volumeMounts for the MariaDB Primary container(s) - ## - extraVolumeMounts: [] - ## @param primary.initContainers Add additional init containers for the MariaDB Primary pod(s) - ## - initContainers: [] - ## @param primary.sidecars Add additional sidecar containers for the MariaDB Primary pod(s) - ## - sidecars: [] - ## MariaDB Primary Service parameters - ## - service: - ## @param primary.service.type MariaDB Primary Kubernetes service type - ## - type: ClusterIP - ports: - ## @param primary.service.ports.mysql MariaDB Primary Kubernetes service port for MariaDB - ## - mysql: 3306 - ## @param primary.service.ports.metrics MariaDB Primary Kubernetes service port for metrics - ## - metrics: 9104 - ## @param primary.service.nodePorts.mysql MariaDB Primary Kubernetes service node port - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport - ## - nodePorts: - mysql: "" - ## @param primary.service.clusterIP MariaDB Primary Kubernetes service clusterIP IP - ## - clusterIP: "" - ## @param primary.service.loadBalancerIP MariaDB Primary loadBalancerIP if service type is `LoadBalancer` - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#internal-load-balancer - ## - loadBalancerIP: "" - ## @param primary.service.externalTrafficPolicy Enable client source IP preservation - ## ref https://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer/#preserving-the-client-source-ip - ## - externalTrafficPolicy: Cluster - ## @param primary.service.loadBalancerSourceRanges Address that are allowed when MariaDB Primary service is LoadBalancer - ## https://kubernetes.io/docs/tasks/access-application-cluster/configure-cloud-provider-firewall/#restrict-access-for-loadbalancer-service - ## E.g. - ## loadBalancerSourceRanges: - ## - 10.10.10.0/24 - ## - loadBalancerSourceRanges: [] - ## @param primary.service.extraPorts Extra ports to expose (normally used with the `sidecar` value) - ## - extraPorts: [] - ## @param primary.service.annotations Provide any additional annotations which may be required - ## - annotations: {} - ## @param primary.service.sessionAffinity Session Affinity for Kubernetes service, can be "None" or "ClientIP" - ## If "ClientIP", consecutive client requests will be directed to the same Pod - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies - ## - sessionAffinity: None - ## @param primary.service.sessionAffinityConfig Additional settings for the sessionAffinity - ## sessionAffinityConfig: - ## clientIP: - ## timeoutSeconds: 300 - ## - sessionAffinityConfig: {} - ## MariaDB primary Pod Disruption Budget configuration - ## ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb/ - ## - pdb: - ## @param primary.pdb.create Enable/disable a Pod Disruption Budget creation for MariaDB primary pods - ## - create: true - ## @param primary.pdb.minAvailable Minimum number/percentage of MariaDB primary pods that must still be available after the eviction - ## - minAvailable: "" - ## @param primary.pdb.maxUnavailable Maximum number/percentage of MariaDB primary pods that can be unavailable after the eviction. Defaults to `1` if both `primary.pdb.minAvailable` and `primary.pdb.maxUnavailable` are empty. - ## - maxUnavailable: "" - ## @param primary.revisionHistoryLimit Maximum number of revisions that will be maintained in the StatefulSet - ## - revisionHistoryLimit: 10 -## @section MariaDB Secondary parameters -## - -## Mariadb Secondary parameters -## -secondary: - ## @param secondary.name Name of the secondary database (eg secondary, slave, ...) - ## - name: secondary - ## @param secondary.replicaCount Number of MariaDB secondary replicas - ## - replicaCount: 1 - ## @param secondary.command Override default container command on MariaDB Secondary container(s) (useful when using custom images) - ## - command: [] - ## @param secondary.args Override default container args on MariaDB Secondary container(s) (useful when using custom images) - ## - args: [] - ## @param secondary.lifecycleHooks for the MariaDB Secondary container(s) to automate configuration before or after startup - ## - lifecycleHooks: {} - ## @param secondary.automountServiceAccountToken Mount Service Account token in pod - ## - automountServiceAccountToken: false - ## @param secondary.hostAliases Add deployment host aliases - ## https://kubernetes.io/docs/concepts/services-networking/add-entries-to-pod-etc-hosts-with-host-aliases/ - ## - hostAliases: [] - ## @param secondary.containerPorts.mysql Container port for mysql - ## - containerPorts: - mysql: 3306 - ## @param secondary.configuration [string] MariaDB Secondary configuration to be injected as ConfigMap - ## ref: https://mysql.com/kb/en/mysql/configuring-mysql-with-mycnf/#example-of-configuration-file - ## - configuration: |- - [mysqld] - skip-name-resolve - explicit_defaults_for_timestamp - basedir=/opt/bitnami/mariadb - datadir=/bitnami/mariadb/data - port={{ .Values.secondary.containerPorts.mysql }} - socket=/opt/bitnami/mariadb/tmp/mysql.sock - tmpdir=/opt/bitnami/mariadb/tmp - max_allowed_packet=16M - bind-address=0.0.0.0 - pid-file=/opt/bitnami/mariadb/tmp/mysqld.pid - log-error=/opt/bitnami/mariadb/logs/mysqld.log - character-set-server=UTF8 - collation-server=utf8_general_ci - slow_query_log=0 - long_query_time=10.0 - binlog_expire_logs_seconds=2592000 - - [client] - port=3306 - socket=/opt/bitnami/mariadb/tmp/mysql.sock - default-character-set=UTF8 - - [manager] - port=3306 - socket=/opt/bitnami/mariadb/tmp/mysql.sock - pid-file=/opt/bitnami/mariadb/tmp/mysqld.pid - ## @param secondary.existingConfigmap Name of existing ConfigMap with MariaDB Secondary configuration. - ## NOTE: When it's set the 'configuration' parameter is ignored - ## - existingConfigmap: "" - ## @param secondary.updateStrategy.type MariaDB secondary statefulset strategy type - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#update-strategies - ## - updateStrategy: - ## StrategyType - ## Can be set to RollingUpdate or OnDelete - ## - type: RollingUpdate - ## @param secondary.rollingUpdatePartition Partition update strategy for Mariadb Secondary statefulset - ## https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#partitions - ## - rollingUpdatePartition: "" - ## @param secondary.podAnnotations Additional pod annotations for MariaDB secondary pods - ## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ - ## - podAnnotations: {} - ## @param secondary.podLabels Extra labels for MariaDB secondary pods - ## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/ - ## - podLabels: {} - ## @param secondary.podAffinityPreset MariaDB secondary pod affinity preset. Ignored if `secondary.affinity` is set. Allowed values: `soft` or `hard` - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity - ## - podAffinityPreset: "" - ## @param secondary.podAntiAffinityPreset MariaDB secondary pod anti-affinity preset. Ignored if `secondary.affinity` is set. Allowed values: `soft` or `hard` - ## Ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity - ## - podAntiAffinityPreset: soft - ## Mariadb Secondary node affinity preset - ## Ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity - ## - nodeAffinityPreset: - ## @param secondary.nodeAffinityPreset.type MariaDB secondary node affinity preset type. Ignored if `secondary.affinity` is set. Allowed values: `soft` or `hard` - ## - type: "" - ## @param secondary.nodeAffinityPreset.key MariaDB secondary node label key to match Ignored if `secondary.affinity` is set. - ## E.g. - ## key: "kubernetes.io/e2e-az-name" - ## - key: "" - ## @param secondary.nodeAffinityPreset.values MariaDB secondary node label values to match. Ignored if `secondary.affinity` is set. - ## E.g. - ## values: - ## - e2e-az1 - ## - e2e-az2 - ## - values: [] - ## @param secondary.affinity Affinity for MariaDB secondary pods assignment - ## Ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity - ## Note: podAffinityPreset, podAntiAffinityPreset, and nodeAffinityPreset will be ignored when it's set - ## - affinity: {} - ## @param secondary.nodeSelector Node labels for MariaDB secondary pods assignment - ## Ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/ - ## - nodeSelector: {} - ## @param secondary.tolerations Tolerations for MariaDB secondary pods assignment - ## Ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ - ## - tolerations: [] - ## @param secondary.topologySpreadConstraints Topology Spread Constraints for MariaDB secondary pods assignment - ## ref: https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/ - ## E.g. - ## topologySpreadConstraints: - ## - maxSkew: 1 - ## topologyKey: topology.kubernetes.io/zone - ## whenUnsatisfiable: DoNotSchedule - ## - topologySpreadConstraints: [] - ## @param secondary.priorityClassName Priority class for MariaDB secondary pods assignment - ## Ref: https://kubernetes.io/docs/concepts/configuration/pod-priority-preemption/ - ## - priorityClassName: "" - ## @param secondary.runtimeClassName Runtime Class for MariaDB secondary pods - ## Ref: https://kubernetes.io/docs/concepts/containers/runtime-class/ - ## - runtimeClassName: "" - ## @param secondary.schedulerName Name of the k8s scheduler (other than default) - ## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ - ## - schedulerName: "" - ## @param secondary.podManagementPolicy podManagementPolicy to manage scaling operation of MariaDB secondary pods - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#pod-management-policies - ## - podManagementPolicy: "" - ## MariaDB secondary Pod security context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-pod - ## @param secondary.podSecurityContext.enabled Enable security context for MariaDB secondary pods - ## @param secondary.podSecurityContext.fsGroupChangePolicy Set filesystem group change policy - ## @param secondary.podSecurityContext.sysctls Set kernel settings using the sysctl interface - ## @param secondary.podSecurityContext.supplementalGroups Set filesystem extra groups - ## @param secondary.podSecurityContext.fsGroup Group ID for the mounted volumes' filesystem - ## - podSecurityContext: - enabled: true - fsGroupChangePolicy: Always - sysctls: [] - supplementalGroups: [] - fsGroup: 1001 - ## MariaDB secondary container security context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container - ## @param secondary.containerSecurityContext.enabled MariaDB secondary container securityContext - ## @param secondary.containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container - ## @param secondary.containerSecurityContext.runAsUser User ID for the MariaDB secondary container - ## @param secondary.containerSecurityContext.runAsGroup Group ID for the MariaDB secondary container - ## @param secondary.containerSecurityContext.runAsNonRoot Set secondary container's Security Context runAsNonRoot - ## @param secondary.containerSecurityContext.privileged Set secondary container's Security Context privileged - ## @param secondary.containerSecurityContext.allowPrivilegeEscalation Set secondary container's Security Context allowPrivilegeEscalation - ## @param secondary.containerSecurityContext.readOnlyRootFilesystem Set container's Security Context readOnlyRootFilesystem - ## @param secondary.containerSecurityContext.capabilities.drop List of capabilities to be dropped - ## @param secondary.containerSecurityContext.seccompProfile.type Set container's Security Context seccomp profile - ## - containerSecurityContext: - enabled: true - seLinuxOptions: {} - runAsUser: 1001 - runAsGroup: 1001 - runAsNonRoot: true - privileged: false - allowPrivilegeEscalation: false - readOnlyRootFilesystem: true - capabilities: - drop: ["ALL"] - seccompProfile: - type: "RuntimeDefault" - ## MariaDB secondary container's resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## We usually recommend not to specify default resources and to leave this as a conscious - ## choice for the user. This also increases chances charts run on environments with little - ## resources, such as Minikube. If you do want to specify resources, uncomment the following - ## lines, adjust them as necessary, and remove the curly braces after 'resources:'. - ## @param secondary.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if secondary.resources is set (secondary.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "micro" - ## @param secondary.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} - ## Configure extra options for MariaDB Secondary containers' liveness, readiness and startup probes - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes) - ## @param secondary.startupProbe.enabled Enable startupProbe - ## @param secondary.startupProbe.initialDelaySeconds Initial delay seconds for startupProbe - ## @param secondary.startupProbe.periodSeconds Period seconds for startupProbe - ## @param secondary.startupProbe.timeoutSeconds Timeout seconds for startupProbe - ## @param secondary.startupProbe.failureThreshold Failure threshold for startupProbe - ## @param secondary.startupProbe.successThreshold Success threshold for startupProbe - ## - startupProbe: - enabled: false - initialDelaySeconds: 120 - periodSeconds: 15 - timeoutSeconds: 5 - failureThreshold: 10 - successThreshold: 1 - ## Configure extra options for liveness probe - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes - ## @param secondary.livenessProbe.enabled Enable livenessProbe - ## @param secondary.livenessProbe.initialDelaySeconds Initial delay seconds for livenessProbe - ## @param secondary.livenessProbe.periodSeconds Period seconds for livenessProbe - ## @param secondary.livenessProbe.timeoutSeconds Timeout seconds for livenessProbe - ## @param secondary.livenessProbe.failureThreshold Failure threshold for livenessProbe - ## @param secondary.livenessProbe.successThreshold Success threshold for livenessProbe - ## - livenessProbe: - enabled: true - initialDelaySeconds: 120 - periodSeconds: 10 - timeoutSeconds: 1 - failureThreshold: 3 - successThreshold: 1 - ## @param secondary.readinessProbe.enabled Enable readinessProbe - ## @param secondary.readinessProbe.initialDelaySeconds Initial delay seconds for readinessProbe - ## @param secondary.readinessProbe.periodSeconds Period seconds for readinessProbe - ## @param secondary.readinessProbe.timeoutSeconds Timeout seconds for readinessProbe - ## @param secondary.readinessProbe.failureThreshold Failure threshold for readinessProbe - ## @param secondary.readinessProbe.successThreshold Success threshold for readinessProbe - ## - readinessProbe: - enabled: true - initialDelaySeconds: 30 - periodSeconds: 10 - timeoutSeconds: 1 - failureThreshold: 3 - successThreshold: 1 - ## @param secondary.customStartupProbe Override default startup probe for MariaDB secondary containers - ## - customStartupProbe: {} - ## @param secondary.customLivenessProbe Override default liveness probe for MariaDB secondary containers - ## - customLivenessProbe: {} - ## @param secondary.customReadinessProbe Override default readiness probe for MariaDB secondary containers - ## - customReadinessProbe: {} - ## @param secondary.startupWaitOptions Override default builtin startup wait check options for MariaDB secondary containers - ## `bitnami/mariadb` Docker image has built-in startup check mechanism, - ## which periodically checks if MariaDB service has started up and stops it - ## if all checks have failed after X tries. Use these to control these checks. - ## ref: https://github.com/bitnami/containers/tree/main/bitnami/mariadb/pull/240 - ## Example (with default options): - ## startupWaitOptions: - ## retries: 300 - ## waitTime: 2 - ## - startupWaitOptions: {} - ## @param secondary.extraFlags MariaDB secondary additional command line flags - ## Can be used to specify command line flags, for example: - ## E.g. - ## extraFlags: "--max-connect-errors=1000 --max_connections=155" - ## - extraFlags: "" - ## @param secondary.extraEnvVars Extra environment variables to be set on MariaDB secondary containers - ## E.g. - ## extraEnvVars: - ## - name: TZ - ## value: "Europe/Paris" - ## - extraEnvVars: [] - ## @param secondary.extraEnvVarsCM Name of existing ConfigMap containing extra env vars for MariaDB secondary containers - ## - extraEnvVarsCM: "" - ## @param secondary.extraEnvVarsSecret Name of existing Secret containing extra env vars for MariaDB secondary containers - ## - extraEnvVarsSecret: "" - ## Enable persistence using Persistent Volume Claims - ## ref: https://kubernetes.io/docs/concepts/storage/persistent-volumes/ - ## - persistence: - ## @param secondary.persistence.enabled Enable persistence on MariaDB secondary replicas using a `PersistentVolumeClaim` - ## - enabled: true - ## @param secondary.persistence.subPath Subdirectory of the volume to mount at - ## - subPath: "" - ## @param secondary.persistence.storageClass MariaDB secondary persistent volume storage Class - ## If defined, storageClassName: - ## If set to "-", storageClassName: "", which disables dynamic provisioning - ## If undefined (the default) or set to null, no storageClassName spec is - ## set, choosing the default provisioner. (gp2 on AWS, standard on - ## GKE, AWS & OpenStack) - ## - storageClass: "" - ## @param secondary.persistence.labels Labels for the PVC - ## - labels: {} - ## @param secondary.persistence.annotations MariaDB secondary persistent volume claim annotations - ## - annotations: {} - ## @param secondary.persistence.accessModes MariaDB secondary persistent volume access Modes - ## - accessModes: - - ReadWriteOnce - ## @param secondary.persistence.size MariaDB secondary persistent volume size - ## - size: 8Gi - ## @param secondary.persistence.selector Selector to match an existing Persistent Volume - ## selector: - ## matchLabels: - ## app: my-app - ## - selector: {} - ## @param secondary.extraVolumes Optionally specify extra list of additional volumes to the MariaDB secondary pod(s) - ## - extraVolumes: [] - ## @param secondary.extraVolumeMounts Optionally specify extra list of additional volumeMounts for the MariaDB secondary container(s) - ## - extraVolumeMounts: [] - ## @param secondary.initContainers Add additional init containers for the MariaDB secondary pod(s) - ## - initContainers: [] - ## @param secondary.sidecars Add additional sidecar containers for the MariaDB secondary pod(s) - ## - sidecars: [] - ## MariaDB Secondary Service parameters - ## - service: - ## @param secondary.service.type MariaDB secondary Kubernetes service type - ## - type: ClusterIP - ports: - ## @param secondary.service.ports.mysql MariaDB secondary Kubernetes service port for MariaDB - ## - mysql: 3306 - ## @param secondary.service.ports.metrics MariaDB secondary Kubernetes service port for metrics - ## - metrics: 9104 - ## @param secondary.service.nodePorts.mysql MariaDB secondary Kubernetes service node port - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport - ## - nodePorts: - mysql: "" - ## @param secondary.service.clusterIP MariaDB secondary Kubernetes service clusterIP IP - ## e.g: - ## clusterIP: None - ## - clusterIP: "" - ## @param secondary.service.loadBalancerIP MariaDB secondary loadBalancerIP if service type is `LoadBalancer` - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#internal-load-balancer - ## - loadBalancerIP: "" - ## @param secondary.service.externalTrafficPolicy Enable client source IP preservation - ## ref https://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer/#preserving-the-client-source-ip - ## - externalTrafficPolicy: Cluster - ## @param secondary.service.loadBalancerSourceRanges Address that are allowed when MariaDB secondary service is LoadBalancer - ## https://kubernetes.io/docs/tasks/access-application-cluster/configure-cloud-provider-firewall/#restrict-access-for-loadbalancer-service - ## E.g. - ## loadBalancerSourceRanges: - ## - 10.10.10.0/24 - ## - loadBalancerSourceRanges: [] - ## @param secondary.service.extraPorts Extra ports to expose (normally used with the `sidecar` value) - ## - extraPorts: [] - ## @param secondary.service.annotations Provide any additional annotations which may be required - ## - annotations: {} - ## @param secondary.service.sessionAffinity Session Affinity for Kubernetes service, can be "None" or "ClientIP" - ## If "ClientIP", consecutive client requests will be directed to the same Pod - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies - ## - sessionAffinity: None - ## @param secondary.service.sessionAffinityConfig Additional settings for the sessionAffinity - ## sessionAffinityConfig: - ## clientIP: - ## timeoutSeconds: 300 - ## - sessionAffinityConfig: {} - ## MariaDB secondary Pod Disruption Budget configuration - ## ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb/ - ## - pdb: - ## @param secondary.pdb.create Enable/disable a Pod Disruption Budget creation for MariaDB secondary pods - ## - create: true - ## @param secondary.pdb.minAvailable Minimum number/percentage of MariaDB secondary pods that should remain scheduled - ## - minAvailable: "" - ## @param secondary.pdb.maxUnavailable Maximum number/percentage of MariaDB secondary pods that may be made unavailable. Defaults to `1` if both `secondary.pdb.minAvailable` and `secondary.pdb.maxUnavailable` are empty. - ## - maxUnavailable: "" - ## @param secondary.revisionHistoryLimit Maximum number of revisions that will be maintained in the StatefulSet - ## - revisionHistoryLimit: 10 -## @section RBAC parameters -## - -## MariaDB pods ServiceAccount -## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/ -## -serviceAccount: - ## @param serviceAccount.create Enable the creation of a ServiceAccount for MariaDB pods - ## - create: true - ## @param serviceAccount.name Name of the created ServiceAccount - ## If not set and create is true, a name is generated using the mariadb.fullname template - ## - name: "" - ## @param serviceAccount.annotations Annotations for MariaDB Service Account - ## - annotations: {} - ## @param serviceAccount.automountServiceAccountToken Automount service account token for the server service account - ## - automountServiceAccountToken: false -## Role Based Access -## ref: https://kubernetes.io/docs/admin/authorization/rbac/ -## -rbac: - ## @param rbac.create Whether to create and use RBAC resources or not - ## - create: false -## @section Volume Permissions parameters -## - -## Init containers parameters: -## volumePermissions: Change the owner and group of the persistent volume mountpoint to runAsUser:fsGroup values from the securityContext section. -## -volumePermissions: - ## @param volumePermissions.enabled Enable init container that changes the owner and group of the persistent volume(s) mountpoint to `runAsUser:fsGroup` - ## - enabled: false - ## @param volumePermissions.image.registry [default: REGISTRY_NAME] Init container volume-permissions image registry - ## @param volumePermissions.image.repository [default: REPOSITORY_NAME/os-shell] Init container volume-permissions image repository - ## @skip volumePermissions.image.tag Init container volume-permissions image tag (immutable tags are recommended) - ## @param volumePermissions.image.digest Init container volume-permissions image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag - ## @param volumePermissions.image.pullPolicy Init container volume-permissions image pull policy - ## @param volumePermissions.image.pullSecrets Specify docker-registry secret names as an array - ## - image: - registry: docker.io - repository: bitnami/os-shell - tag: 12-debian-12-r30 - digest: "" - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets (secrets must be manually created in the namespace) - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## Example: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## @param volumePermissions.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if volumePermissions.resources is set (volumePermissions.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "nano" - ## @param volumePermissions.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} -## @section Metrics parameters -## - -## Mysqld Prometheus exporter parameters -## -metrics: - ## @param metrics.enabled Start a side-car prometheus exporter - ## - enabled: false - ## @param metrics.image.registry [default: REGISTRY_NAME] Exporter image registry - ## @param metrics.image.repository [default: REPOSITORY_NAME/mysqld-exporter] Exporter image repository - ## @skip metrics.image.tag Exporter image tag (immutable tags are recommended) - ## @param metrics.image.digest Exporter image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag - ## @param metrics.image.pullPolicy Exporter image pull policy - ## @param metrics.image.pullSecrets Specify docker-registry secret names as an array - ## - image: - registry: docker.io - repository: bitnami/mysqld-exporter - tag: 0.15.1-debian-12-r34 - digest: "" - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets (secrets must be manually created in the namespace) - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## Example: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## @param metrics.annotations [object] Annotations for the Exporter pod - ## - annotations: - prometheus.io/scrape: "true" - prometheus.io/port: "9104" - ## @param metrics.extraArgs [object] Extra args to be passed to mysqld_exporter - ## ref: https://github.com/prometheus/mysqld_exporter/ - ## E.g. - ## - --collect.auto_increment.columns - ## - --collect.binlog_size - ## - --collect.engine_innodb_status - ## - --collect.engine_tokudb_status - ## - --collect.global_status - ## - --collect.global_variables - ## - --collect.info_schema.clientstats - ## - --collect.info_schema.innodb_metrics - ## - --collect.info_schema.innodb_tablespaces - ## - --collect.info_schema.innodb_cmp - ## - --collect.info_schema.innodb_cmpmem - ## - --collect.info_schema.processlist - ## - --collect.info_schema.processlist.min_time - ## - --collect.info_schema.query_response_time - ## - --collect.info_schema.tables - ## - --collect.info_schema.tables.databases - ## - --collect.info_schema.tablestats - ## - --collect.info_schema.userstats - ## - --collect.perf_schema.eventsstatements - ## - --collect.perf_schema.eventsstatements.digest_text_limit - ## - --collect.perf_schema.eventsstatements.limit - ## - --collect.perf_schema.eventsstatements.timelimit - ## - --collect.perf_schema.eventswaits - ## - --collect.perf_schema.file_events - ## - --collect.perf_schema.file_instances - ## - --collect.perf_schema.indexiowaits - ## - --collect.perf_schema.tableiowaits - ## - --collect.perf_schema.tablelocks - ## - --collect.perf_schema.replication_group_member_stats - ## - --collect.slave_status - ## - --collect.slave_hosts - ## - --collect.heartbeat - ## - --collect.heartbeat.database - ## - --collect.heartbeat.table - ## - extraArgs: - primary: [] - secondary: [] - ## @param metrics.extraVolumeMounts [object] Optionally specify extra list of additional volumeMounts for the MariaDB metrics container(s) - ## - extraVolumeMounts: - primary: [] - secondary: [] - ## @param metrics.containerPorts.http Container port for http - ## - containerPorts: - http: 9104 - ## MariaDB metrics container Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container - ## @param metrics.containerSecurityContext.enabled Enable security context for MariaDB metrics container - ## @param metrics.containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container - ## @param metrics.containerSecurityContext.runAsUser User ID for the MariaDB metrics container - ## @param metrics.containerSecurityContext.runAsGroup Group ID for the MariaDB metrics container - ## @param metrics.containerSecurityContext.runAsNonRoot Set metrics container's Security Context runAsNonRoot - ## @param metrics.containerSecurityContext.privileged Set metrics container's Security Context privileged - ## @param metrics.containerSecurityContext.allowPrivilegeEscalation Set metrics container's Security Context allowPrivilegeEscalation - ## @param metrics.containerSecurityContext.readOnlyRootFilesystem Set container's Security Context readOnlyRootFilesystem - ## @param metrics.containerSecurityContext.capabilities.drop List of capabilities to be dropped - ## @param metrics.containerSecurityContext.seccompProfile.type Set container's Security Context seccomp profile - ## Example: - ## containerSecurityContext: - ## enabled: true - ## capabilities: - ## drop: ["NET_RAW"] - ## readOnlyRootFilesystem: true - ## - containerSecurityContext: - enabled: false - privileged: false - runAsNonRoot: true - seLinuxOptions: {} - runAsUser: 1001 - runAsGroup: 1001 - readOnlyRootFilesystem: true - allowPrivilegeEscalation: false - capabilities: - drop: ["ALL"] - seccompProfile: - type: "RuntimeDefault" - ## Mysqld Prometheus exporter resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## We usually recommend not to specify default resources and to leave this as a conscious - ## choice for the user. This also increases chances charts run on environments with little - ## resources, such as Minikube. If you do want to specify resources, uncomment the following - ## lines, adjust them as necessary, and remove the curly braces after 'resources:'. - ## @param metrics.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if metrics.resources is set (metrics.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "nano" - ## @param metrics.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} - ## Configure extra options for liveness probe - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes - ## @param metrics.livenessProbe.enabled Enable livenessProbe - ## @param metrics.livenessProbe.initialDelaySeconds Initial delay seconds for livenessProbe - ## @param metrics.livenessProbe.periodSeconds Period seconds for livenessProbe - ## @param metrics.livenessProbe.timeoutSeconds Timeout seconds for livenessProbe - ## @param metrics.livenessProbe.failureThreshold Failure threshold for livenessProbe - ## @param metrics.livenessProbe.successThreshold Success threshold for livenessProbe - ## - livenessProbe: - enabled: true - initialDelaySeconds: 120 - periodSeconds: 10 - timeoutSeconds: 1 - successThreshold: 1 - failureThreshold: 3 - ## Configure extra options for readiness probe - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes - ## @param metrics.readinessProbe.enabled Enable readinessProbe - ## @param metrics.readinessProbe.initialDelaySeconds Initial delay seconds for readinessProbe - ## @param metrics.readinessProbe.periodSeconds Period seconds for readinessProbe - ## @param metrics.readinessProbe.timeoutSeconds Timeout seconds for readinessProbe - ## @param metrics.readinessProbe.failureThreshold Failure threshold for readinessProbe - ## @param metrics.readinessProbe.successThreshold Success threshold for readinessProbe - ## - readinessProbe: - enabled: true - initialDelaySeconds: 30 - periodSeconds: 10 - timeoutSeconds: 1 - successThreshold: 1 - failureThreshold: 3 - ## Prometheus Service Monitor - ## ref: https://github.com/coreos/prometheus-operator - ## - serviceMonitor: - ## @param metrics.serviceMonitor.enabled Create ServiceMonitor Resource for scraping metrics using PrometheusOperator - ## - enabled: false - ## @param metrics.serviceMonitor.namespace Namespace which Prometheus is running in - ## - namespace: "" - ## @param metrics.serviceMonitor.jobLabel The name of the label on the target service to use as the job name in prometheus. - ## - jobLabel: "" - ## @param metrics.serviceMonitor.interval Interval at which metrics should be scraped - ## - interval: 30s - ## @param metrics.serviceMonitor.scrapeTimeout Specify the timeout after which the scrape is ended - ## e.g: - ## scrapeTimeout: 30s - ## - scrapeTimeout: "" - ## @param metrics.serviceMonitor.relabelings RelabelConfigs to apply to samples before scraping - ## ref: https://github.com/coreos/prometheus-operator/blob/master/Documentation/api.md#relabelconfig - ## - relabelings: [] - ## @param metrics.serviceMonitor.metricRelabelings MetricRelabelConfigs to apply to samples before ingestion - ## ref: https://github.com/coreos/prometheus-operator/blob/master/Documentation/api.md#relabelconfig - ## - metricRelabelings: [] - ## @param metrics.serviceMonitor.honorLabels honorLabels chooses the metric's labels on collisions with target labels - ## - honorLabels: false - ## @param metrics.serviceMonitor.selector ServiceMonitor selector labels - ## ref: https://github.com/bitnami/charts/tree/main/bitnami/prometheus-operator#prometheus-configuration - ## - ## selector: - ## prometheus: my-prometheus - ## - selector: {} - ## @param metrics.serviceMonitor.labels Extra labels for the ServiceMonitor - ## - labels: {} - ## Prometheus Operator PrometheusRule configuration - ## - prometheusRule: - ## @param metrics.prometheusRule.enabled if `true`, creates a Prometheus Operator PrometheusRule (also requires `metrics.enabled` to be `true` and `metrics.prometheusRule.rules`) - ## - enabled: false - ## @param metrics.prometheusRule.namespace Namespace for the PrometheusRule Resource (defaults to the Release Namespace) - ## - namespace: "" - ## @param metrics.prometheusRule.additionalLabels Additional labels that can be used so PrometheusRule will be discovered by Prometheus - ## - additionalLabels: {} - ## @param metrics.prometheusRule.rules Prometheus Rule definitions - ## - alert: MariaDB-Down - ## expr: absent(up{job="mariadb"} == 1) - ## for: 5m - ## labels: - ## severity: warning - ## service: mariadb - ## annotations: - ## message: 'MariaDB instance {{ `{{` }} $labels.instance {{ `}}` }} is down' - ## summary: MariaDB instance is down - ## - rules: [] -## @section NetworkPolicy parameters -## ref: https://kubernetes.io/docs/concepts/services-networking/network-policies/ -## -networkPolicy: - ## @param networkPolicy.enabled Enable creation of NetworkPolicy resources - ## - enabled: true - ## @param networkPolicy.allowExternal The Policy model to apply - ## When set to false, only pods with the correct client label will have network access to the ports MariaDB is - ## listening on. When true, MariaDB will accept connections from any source (with the correct destination port). - ## - allowExternal: true - ## @param networkPolicy.allowExternalEgress Allow the pod to access any range of port and all destinations. - ## - allowExternalEgress: true - ## @param networkPolicy.extraIngress [array] Add extra ingress rules to the NetworkPolicy - ## e.g: - ## extraIngress: - ## - ports: - ## - port: 1234 - ## from: - ## - podSelector: - ## - matchLabels: - ## - role: frontend - ## - podSelector: - ## - matchExpressions: - ## - key: role - ## operator: In - ## values: - ## - frontend - ## - extraIngress: [] - ## @param networkPolicy.extraEgress [array] Add extra ingress rules to the NetworkPolicy - ## e.g: - ## extraEgress: - ## - ports: - ## - port: 1234 - ## to: - ## - podSelector: - ## - matchLabels: - ## - role: frontend - ## - podSelector: - ## - matchExpressions: - ## - key: role - ## operator: In - ## values: - ## - frontend - ## - extraEgress: [] - ## @param networkPolicy.ingressNSMatchLabels [object] Labels to match to allow traffic from other namespaces - ## @param networkPolicy.ingressNSPodMatchLabels [object] Pod labels to match to allow traffic from other namespaces - ## - ingressNSMatchLabels: {} - ingressNSPodMatchLabels: {} diff --git a/charts/mariadb/values-overrides.yaml b/charts/mariadb/values-overrides.yaml deleted file mode 100644 index 8bc0914..0000000 --- a/charts/mariadb/values-overrides.yaml +++ /dev/null @@ -1,8 +0,0 @@ -# @see https://artifacthub.io/packages/helm/bitnami/mariadb?modal=values -fullnameOverride: mariadb -# architecture: replication -architecture: standalone -auth: - username: drupal - database: drupal - existingSecret: mariadb \ No newline at end of file diff --git a/charts/phpmyadmin/phpmyadmin/.helmignore b/charts/phpmyadmin/phpmyadmin/.helmignore deleted file mode 100644 index 207983f..0000000 --- a/charts/phpmyadmin/phpmyadmin/.helmignore +++ /dev/null @@ -1,25 +0,0 @@ -# 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 -# img folder -img/ -# Changelog -CHANGELOG.md diff --git a/charts/phpmyadmin/phpmyadmin/Chart.lock b/charts/phpmyadmin/phpmyadmin/Chart.lock deleted file mode 100644 index 109d91c..0000000 --- a/charts/phpmyadmin/phpmyadmin/Chart.lock +++ /dev/null @@ -1,9 +0,0 @@ -dependencies: -- name: mariadb - repository: oci://registry-1.docker.io/bitnamicharts - version: 19.0.6 -- name: common - repository: oci://registry-1.docker.io/bitnamicharts - version: 2.23.0 -digest: sha256:8626b452b34cfdd615a5e97ad13de70344e5fd14ee39db107795e20d5006b3c8 -generated: "2024-09-19T09:40:39.92295583Z" diff --git a/charts/phpmyadmin/phpmyadmin/Chart.yaml b/charts/phpmyadmin/phpmyadmin/Chart.yaml deleted file mode 100644 index e5933f4..0000000 --- a/charts/phpmyadmin/phpmyadmin/Chart.yaml +++ /dev/null @@ -1,38 +0,0 @@ -annotations: - category: Infrastructure - images: | - - name: apache-exporter - image: docker.io/bitnami/apache-exporter:1.0.8-debian-12-r9 - - name: phpmyadmin - image: docker.io/bitnami/phpmyadmin:5.2.1-debian-12-r42 - licenses: Apache-2.0 -apiVersion: v2 -appVersion: 5.2.1 -dependencies: -- condition: db.bundleTestDB - name: mariadb - repository: oci://registry-1.docker.io/bitnamicharts - tags: - - phpmyadmin-database - version: 19.x.x -- name: common - repository: oci://registry-1.docker.io/bitnamicharts - tags: - - bitnami-common - version: 2.x.x -description: phpMyAdmin is a free software tool written in PHP, intended to handle - the administration of MySQL over the Web. phpMyAdmin supports a wide range of operations - on MySQL and MariaDB. -home: https://bitnami.com -icon: https://bitnami.com/assets/stacks/phpmyadmin/img/phpmyadmin-stack-220x234.png -keywords: -- mariadb -- mysql -- phpmyadmin -maintainers: -- name: Broadcom, Inc. All Rights Reserved. - url: https://github.com/bitnami/charts -name: phpmyadmin -sources: -- https://github.com/bitnami/charts/tree/main/bitnami/phpmyadmin -version: 17.0.7 diff --git a/charts/phpmyadmin/phpmyadmin/README.md b/charts/phpmyadmin/phpmyadmin/README.md deleted file mode 100644 index 97bc819..0000000 --- a/charts/phpmyadmin/phpmyadmin/README.md +++ /dev/null @@ -1,588 +0,0 @@ - - -# Bitnami package for phpMyAdmin - -phpMyAdmin is a free software tool written in PHP, intended to handle the administration of MySQL over the Web. phpMyAdmin supports a wide range of operations on MySQL and MariaDB. - -[Overview of phpMyAdmin](https://www.phpmyadmin.net/) - -Trademarks: This software listing is packaged by Bitnami. The respective trademarks mentioned in the offering are owned by the respective companies, and use of them does not imply any affiliation or endorsement. - -## TL;DR - -```console -helm install my-release oci://registry-1.docker.io/bitnamicharts/phpmyadmin -``` - -Looking to use phpMyAdmin in production? Try [VMware Tanzu Application Catalog](https://bitnami.com/enterprise), the commercial edition of the Bitnami catalog. - -## Introduction - -This chart bootstraps a [phpMyAdmin](https://github.com/bitnami/containers/tree/main/bitnami/phpmyadmin) deployment on a [Kubernetes](https://kubernetes.io) cluster using the [Helm](https://helm.sh) package manager. - -As a portable web application written primarily in PHP, phpMyAdmin has become one of the most popular MySQL administration tools, especially for web hosting services. - -Bitnami charts can be used with [Kubeapps](https://kubeapps.dev/) for deployment and management of Helm Charts in clusters. - -## Prerequisites - -- Kubernetes 1.23+ -- Helm 3.8.0+ - -## Installing the Chart - -To install the chart with the release name `my-release`: - -```console -helm install my-release oci://REGISTRY_NAME/REPOSITORY_NAME/phpmyadmin -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -The command deploys phpMyAdmin on the Kubernetes cluster in the default configuration. The [Parameters](#parameters) section lists the parameters that can be configured during installation. - -> **Tip**: List all releases using `helm list` - -## Configuration and installation details - -### Resource requests and limits - -Bitnami charts allow setting resource requests and limits for all containers inside the chart deployment. These are inside the `resources` value (check parameter table). Setting requests is essential for production workloads and these should be adapted to your specific use case. - -To make this process easier, the chart contains the `resourcesPreset` values, which automatically sets the `resources` section according to different presets. Check these presets in [the bitnami/common chart](https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15). However, in production workloads using `resourcePreset` is discouraged as it may not fully adapt to your specific needs. Find more information on container resource management in the [official Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/). - -### [Rolling VS Immutable tags](https://docs.vmware.com/en/VMware-Tanzu-Application-Catalog/services/tutorials/GUID-understand-rolling-tags-containers-index.html) - -It is strongly recommended to use immutable tags in a production environment. This ensures your deployment does not change automatically if the same tag is updated with a different image. - -Bitnami will release a new chart updating its containers if a new version of the main container, significant changes, or critical vulnerabilities exist. - -### Ingress - -This chart provides support for Ingress resources. If you have an ingress controller installed on your cluster, such as [nginx-ingress-controller](https://github.com/bitnami/charts/tree/main/bitnami/nginx-ingress-controller) or [contour](https://github.com/bitnami/charts/tree/main/bitnami/contour) you can utilize the ingress controller to serve your application. - -To enable ingress integration, please set `ingress.enabled` to `true`. - -#### Hosts - -Most likely you will only want to have one hostname that maps to this phpMyAdmin installation. If that's your case, the property `ingress.hostname` will set it. However, it is possible to have more than one host. To facilitate this, the `ingress.extraHosts` object can be specified as an array. You can also use `ingress.extraTLS` to add the TLS configuration for extra hosts. - -For each host indicated at `ingress.extraHosts`, please indicate a `name`, `path`, and any `annotations` that you may want the ingress controller to know about. - -For annotations, please see [this document](https://github.com/kubernetes/ingress-nginx/blob/main/docs/user-guide/nginx-configuration/annotations.md). Not all annotations are supported by all ingress controllers, but this document does a good job of indicating which annotation is supported by many popular ingress controllers. - -### TLS Secrets - -This chart will facilitate the creation of TLS secrets for use with the ingress controller, however, this is not required. There are some common use cases: - -- Helm generates and manages certificate secrets (default). -- User generates certificates and helm manages secrets. -- User generates and manages certificates separately. -- An additional tool (like [cert-manager](https://github.com/jetstack/cert-manager/)) manages the secrets for the application. - -In the second case, a certificate and a key are needed. We would expect them to look like this: - -- certificate files should look like (and there can be more than one certificate if there is a certificate chain) - - ```console - -----BEGIN CERTIFICATE----- - MIID6TCCAtGgAwIBAgIJAIaCwivkeB5EMA0GCSqGSIb3DQEBCwUAMFYxCzAJBgNV - ... - jScrvkiBO65F46KioCL9h5tDvomdU1aqpI/CBzhvZn1c0ZTf87tGQR8NK7v7 - -----END CERTIFICATE----- - ``` - -- keys should look like: - - ```console - -----BEGIN RSA PRIVATE KEY----- - MIIEogIBAAKCAQEAvLYcyu8f3skuRyUgeeNpeDvYBCDcgq+LsWap6zbX5f8oLqp4 - ... - wrj2wDbCDCFmfqnSJ+dKI3vFLlEz44sAV8jX/kd4Y6ZTQhlLbYc= - -----END RSA PRIVATE KEY----- - ``` - -If you are going to generate certificates yourself and want helm to manage the secret, please copy these values into the `certificate` and `key` values for a given `ingress.secrets` entry. - -If you want to manage TLS secrets outside of Helm, please know that you can create a TLS secret and pass its name via the parameter `ingress.existingSecretName`. - -To make use of cert-manager, you need to add the the `cert-manager.io/cluster-issuer:` annotation to the ingress object via `ingress.annotations`. - -### Adding extra environment variables - -In case you want to add extra environment variables (useful for advanced operations like custom init scripts), you can use the `extraEnvVars` property. - -```yaml -extraEnvVars: - - name: LOG_LEVEL - value: DEBUG -``` - -Alternatively, you can use a ConfigMap or a Secret with the environment variables. To do so, use the `extraEnvVarsCM` or the `extraEnvVarsSecret` values. - -### Sidecars and Init Containers - -If you have a need for additional containers to run within the same pod as the PhpMyAdmin app (e.g. an additional metrics or logging exporter), you can do so via the `sidecars` config parameter. Simply define your container according to the Kubernetes container spec. - -```yaml -sidecars: - - name: your-image-name - image: your-image - imagePullPolicy: Always - ports: - - name: portname - containerPort: 1234 -``` - -Similarly, you can add extra init containers using the `initContainers` parameter. - -```yaml -initContainers: - - name: your-image-name - image: your-image - imagePullPolicy: Always - ports: - - name: portname - containerPort: 1234 -``` - -### Deploying extra resources - -There are cases where you may want to deploy extra objects, such a ConfigMap containing your app's configuration or some extra deployment with a micro service used by your app. For covering this case, the chart allows adding the full specification of other objects using the `extraDeploy` parameter. - -### Setting Pod's affinity - -This chart allows you to set your custom affinity using the `XXX.affinity` parameter(s). Find more information about Pod's affinity in the [kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity). - -As an alternative, you can use of the preset configurations for pod affinity, pod anti-affinity, and node affinity available at the [bitnami/common](https://github.com/bitnami/charts/tree/main/bitnami/common#affinities) chart. To do so, set the `podAffinityPreset`, `XpodAntiAffinityPreset`, or `nodeAffinityPreset` parameters. - -## Parameters - -### Global parameters - -| Name | Description | Value | -| ----------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ---------- | -| `global.imageRegistry` | Global Docker image registry | `""` | -| `global.imagePullSecrets` | Global Docker registry secret names as an array | `[]` | -| `global.defaultStorageClass` | Global default StorageClass for Persistent Volume(s) | `""` | -| `global.compatibility.openshift.adaptSecurityContext` | Adapt the securityContext sections of the deployment to make them compatible with Openshift restricted-v2 SCC: remove runAsUser, runAsGroup and fsGroup and let the platform use their allowed default IDs. Possible values: auto (apply if the detected running cluster is Openshift), force (perform the adaptation always), disabled (do not perform adaptation) | `disabled` | - -### Common parameters - -| Name | Description | Value | -| ------------------- | -------------------------------------------------------------------------------------------- | --------------- | -| `kubeVersion` | Force target Kubernetes version (using Helm capabilities if not set) | `""` | -| `nameOverride` | String to partially override common.names.fullname template (will maintain the release name) | `""` | -| `fullnameOverride` | String to fully override common.names.fullname template | `""` | -| `commonLabels` | Add labels to all the deployed resources | `{}` | -| `commonAnnotations` | Add annotations to all the deployed resources | `{}` | -| `clusterDomain` | Kubernetes Cluster Domain | `cluster.local` | -| `extraDeploy` | Array of extra objects to deploy with the release | `[]` | - -### phpMyAdmin parameters - -| Name | Description | Value | -| -------------------- | ---------------------------------------------------------------------------------------------------------- | ---------------------------- | -| `image.registry` | phpMyAdmin image registry | `REGISTRY_NAME` | -| `image.repository` | phpMyAdmin image repository | `REPOSITORY_NAME/phpmyadmin` | -| `image.digest` | phpMyAdmin image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `image.pullPolicy` | Image pull policy | `IfNotPresent` | -| `image.pullSecrets` | Specify docker-registry secret names as an array | `[]` | -| `image.debug` | Enable phpmyadmin image debug mode | `false` | -| `command` | Override default container command (useful when using custom images) | `[]` | -| `args` | Override default container args (useful when using custom images) | `[]` | -| `lifecycleHooks` | for the phpmyadmin container(s) to automate configuration before or after startup | `{}` | -| `extraEnvVars` | Extra environment variables to be set on PhpMyAdmin container | `[]` | -| `extraEnvVarsCM` | Name of a existing ConfigMap containing extra env vars | `""` | -| `extraEnvVarsSecret` | Name of a existing Secret containing extra env vars | `""` | - -### phpMyAdmin deployment parameters - -| Name | Description | Value | -| --------------------------------------------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ---------------- | -| `automountServiceAccountToken` | Mount Service Account token in pod | `false` | -| `hostAliases` | Deployment pod host aliases | `[]` | -| `containerPorts.http` | HTTP port to expose at container level | `8080` | -| `containerPorts.https` | HTTPS port to expose at container level | `8443` | -| `extraContainerPorts` | Optionally specify extra list of additional ports for phpMyAdmin container(s) | `[]` | -| `updateStrategy.type` | Strategy to use to update Pods | `RollingUpdate` | -| `podSecurityContext.enabled` | Enable phpMyAdmin pods' Security Context | `true` | -| `podSecurityContext.fsGroupChangePolicy` | Set filesystem group change policy | `Always` | -| `podSecurityContext.sysctls` | Set kernel settings using the sysctl interface | `[]` | -| `podSecurityContext.supplementalGroups` | Set filesystem extra groups | `[]` | -| `podSecurityContext.fsGroup` | User ID for the container | `1001` | -| `containerSecurityContext.enabled` | Enabled containers' Security Context | `true` | -| `containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `nil` | -| `containerSecurityContext.runAsUser` | Set containers' Security Context runAsUser | `1001` | -| `containerSecurityContext.runAsGroup` | Set containers' Security Context runAsGroup | `1001` | -| `containerSecurityContext.runAsNonRoot` | Set container's Security Context runAsNonRoot | `true` | -| `containerSecurityContext.privileged` | Set container's Security Context privileged | `false` | -| `containerSecurityContext.readOnlyRootFilesystem` | Set container's Security Context readOnlyRootFilesystem | `true` | -| `containerSecurityContext.allowPrivilegeEscalation` | Set container's Security Context allowPrivilegeEscalation | `false` | -| `containerSecurityContext.capabilities.drop` | List of capabilities to be dropped | `["ALL"]` | -| `containerSecurityContext.seccompProfile.type` | Set container's Security Context seccomp profile | `RuntimeDefault` | -| `replicas` | Number of replicas | `1` | -| `resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if resources is set (resources is recommended for production). | `micro` | -| `resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `startupProbe.enabled` | Enable startupProbe | `false` | -| `startupProbe.httpGet.path` | Request path for startupProbe | `/` | -| `startupProbe.httpGet.port` | Port for startupProbe | `http` | -| `startupProbe.initialDelaySeconds` | Initial delay seconds for startupProbe | `30` | -| `startupProbe.periodSeconds` | Period seconds for startupProbe | `10` | -| `startupProbe.timeoutSeconds` | Timeout seconds for startupProbe | `30` | -| `startupProbe.failureThreshold` | Failure threshold for startupProbe | `6` | -| `startupProbe.successThreshold` | Success threshold for startupProbe | `1` | -| `livenessProbe.enabled` | Enable livenessProbe | `true` | -| `livenessProbe.tcpSocket.port` | Port for livenessProbe | `http` | -| `livenessProbe.initialDelaySeconds` | Initial delay seconds for livenessProbe | `30` | -| `livenessProbe.periodSeconds` | Period seconds for livenessProbe | `10` | -| `livenessProbe.timeoutSeconds` | Timeout seconds for livenessProbe | `30` | -| `livenessProbe.failureThreshold` | Failure threshold for livenessProbe | `6` | -| `livenessProbe.successThreshold` | Success threshold for livenessProbe | `1` | -| `readinessProbe.enabled` | Enable readinessProbe | `true` | -| `readinessProbe.httpGet.path` | Request path for readinessProbe | `/` | -| `readinessProbe.httpGet.port` | Port for readinessProbe | `http` | -| `readinessProbe.initialDelaySeconds` | Initial delay seconds for readinessProbe | `30` | -| `readinessProbe.periodSeconds` | Period seconds for readinessProbe | `10` | -| `readinessProbe.timeoutSeconds` | Timeout seconds for readinessProbe | `30` | -| `readinessProbe.failureThreshold` | Failure threshold for readinessProbe | `6` | -| `readinessProbe.successThreshold` | Success threshold for readinessProbe | `1` | -| `customStartupProbe` | Override default startup probe | `{}` | -| `customLivenessProbe` | Override default liveness probe | `{}` | -| `customReadinessProbe` | Override default readiness probe | `{}` | -| `podLabels` | Extra labels for PhpMyAdmin pods | `{}` | -| `podAnnotations` | Annotations for PhpMyAdmin pods | `{}` | -| `podAffinityPreset` | Pod affinity preset. Ignored if `affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `podAntiAffinityPreset` | Pod anti-affinity preset. Ignored if `affinity` is set. Allowed values: `soft` or `hard` | `soft` | -| `nodeAffinityPreset.type` | Node affinity preset type. Ignored if `affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `nodeAffinityPreset.key` | Node label key to match. Ignored if `affinity` is set. | `""` | -| `nodeAffinityPreset.values` | Node label values to match. Ignored if `affinity` is set. | `[]` | -| `affinity` | Affinity for pod assignment. Evaluated as a template. | `{}` | -| `nodeSelector` | Node labels for pod assignment. Evaluated as a template. | `{}` | -| `tolerations` | Tolerations for pod assignment. Evaluated as a template. | `[]` | -| `priorityClassName` | phpmyadmin pods' priorityClassName | `""` | -| `schedulerName` | Name of the k8s scheduler (other than default) | `""` | -| `topologySpreadConstraints` | Topology Spread Constraints for pod assignment | `[]` | -| `extraVolumes` | Optionally specify extra list of additional volumes for PhpMyAdmin pods | `[]` | -| `extraVolumeMounts` | Optionally specify extra list of additional volumeMounts for PhpMyAdmin container(s) | `[]` | -| `initContainers` | Add init containers to the PhpMyAdmin pods | `[]` | -| `pdb.create` | Enable/disable a Pod Disruption Budget creation | `true` | -| `pdb.minAvailable` | Minimum number/percentage of pods that should remain scheduled | `""` | -| `pdb.maxUnavailable` | Maximum number/percentage of pods that may be made unavailable. Defaults to `1` if both `pdb.minAvailable` and `pdb.maxUnavailable` are empty. | `""` | -| `sidecars` | Add sidecar containers to the PhpMyAdmin pods | `[]` | - -### Traffic Exposure parameters - -| Name | Description | Value | -| ---------------------------------- | -------------------------------------------------------------------------------------------------------------------------------- | ------------------------ | -| `service.type` | Kubernetes Service type | `ClusterIP` | -| `service.ports.http` | Service HTTP port | `80` | -| `service.ports.https` | Service HTTPS port | `443` | -| `service.nodePorts.http` | Kubernetes http node port | `""` | -| `service.nodePorts.https` | Kubernetes https node port | `""` | -| `service.clusterIP` | PhpMyAdmin service clusterIP IP | `""` | -| `service.loadBalancerIP` | Load balancer IP for the phpMyAdmin Service (optional, cloud specific) | `""` | -| `service.loadBalancerSourceRanges` | Addresses that are allowed when service is LoadBalancer | `[]` | -| `service.externalTrafficPolicy` | Enable client source IP preservation | `Cluster` | -| `service.extraPorts` | Extra ports to expose (normally used with the `sidecar` value) | `[]` | -| `service.annotations` | Provide any additional annotations that may be required for the PhpMyAdmin service | `{}` | -| `service.sessionAffinity` | Session Affinity for Kubernetes service, can be "None" or "ClientIP" | `None` | -| `service.sessionAffinityConfig` | Additional settings for the sessionAffinity | `{}` | -| `ingress.enabled` | Set to true to enable ingress record generation | `false` | -| `ingress.apiVersion` | Force Ingress API version (automatically detected if not set) | `""` | -| `ingress.hostname` | When the ingress is enabled, a host pointing to this will be created | `phpmyadmin.local` | -| `ingress.pathType` | Ingress path type | `ImplementationSpecific` | -| `ingress.path` | Default path for the ingress record | `/` | -| `ingress.extraPaths` | An array with additional arbitrary paths that may need to be added to the ingress under the main host | `[]` | -| `ingress.annotations` | Additional annotations for the Ingress resource. To enable certificate autogeneration, place here your cert-manager annotations. | `{}` | -| `ingress.tls` | Enable TLS configuration for the hostname defined at `ingress.hostname` parameter | `false` | -| `ingress.extraHosts` | The list of additional hostnames to be covered with this ingress record. | `[]` | -| `ingress.extraTls` | The tls configuration for additional hostnames to be covered with this ingress record. | `[]` | -| `ingress.secrets` | If you're providing your own certificates and want to manage the secret via helm, | `[]` | -| `ingress.existingSecretName` | If you're providing your own certificate and want to manage the secret yourself, | `""` | -| `ingress.ingressClassName` | IngressClass that will be be used to implement the Ingress (Kubernetes 1.18+) | `""` | -| `ingress.extraRules` | Additional rules to be covered with this ingress record | `[]` | - -### Database parameters - -| Name | Description | Value | -| -------------------------- | ----------------------------------------------------------------- | ------- | -| `db.allowArbitraryServer` | Enable connection to arbitrary MySQL server | `true` | -| `db.port` | Database port to use to connect | `3306` | -| `db.chartName` | Database suffix if included in the same release | `""` | -| `db.host` | Database Hostname. Ignored when `db.chartName` is set. | `""` | -| `db.bundleTestDB` | Deploy a MariaDB instance for testing purposes | `false` | -| `db.enableSsl` | Enable SSL for the connection between phpMyAdmin and the database | `false` | -| `db.ssl.clientKey` | Client key file when using SSL | `""` | -| `db.ssl.clientCertificate` | Client certificate file when using SSL | `""` | -| `db.ssl.caCertificate` | CA file when using SSL | `""` | -| `db.ssl.ciphers` | List of allowable ciphers for connections when using SSL | `[]` | -| `db.ssl.verify` | Enable SSL certificate validation | `true` | -| `mariadb` | MariaDB chart configuration | `{}` | - -### Other Parameters - -| Name | Description | Value | -| --------------------------------------------- | ---------------------------------------------------------------------- | ------- | -| `serviceAccount.create` | Enable creation of ServiceAccount for PhpMyAdmin pod | `true` | -| `serviceAccount.name` | The name of the ServiceAccount to use. | `""` | -| `serviceAccount.automountServiceAccountToken` | Allows auto mount of ServiceAccountToken on the serviceAccount created | `false` | -| `serviceAccount.annotations` | Additional custom annotations for the ServiceAccount | `{}` | - -### Metrics parameters - -| Name | Description | Value | -| ------------------------------------------ | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | --------------------------------- | -| `metrics.enabled` | Start a side-car prometheus exporter | `false` | -| `metrics.image.registry` | Apache exporter image registry | `REGISTRY_NAME` | -| `metrics.image.repository` | Apache exporter image repository | `REPOSITORY_NAME/apache-exporter` | -| `metrics.image.digest` | Apache exporter image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `metrics.image.pullPolicy` | Image pull policy | `IfNotPresent` | -| `metrics.image.pullSecrets` | Specify docker-registry secret names as an array | `[]` | -| `metrics.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if metrics.resources is set (metrics.resources is recommended for production). | `nano` | -| `metrics.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `metrics.service.type` | Prometheus metrics service type | `ClusterIP` | -| `metrics.service.port` | Prometheus metrics service port | `9117` | -| `metrics.service.annotations` | Annotations for Prometheus metrics service | `{}` | -| `metrics.service.clusterIP` | phpmyadmin service Cluster IP | `""` | -| `metrics.service.loadBalancerIP` | Load Balancer IP if the Prometheus metrics server type is `LoadBalancer` | `""` | -| `metrics.service.loadBalancerSourceRanges` | phpmyadmin service Load Balancer sources | `[]` | -| `metrics.service.externalTrafficPolicy` | phpmyadmin service external traffic policy | `Cluster` | -| `metrics.service.sessionAffinity` | Session Affinity for Kubernetes service, can be "None" or "ClientIP" | `None` | -| `metrics.service.sessionAffinityConfig` | Additional settings for the sessionAffinity | `{}` | -| `metrics.serviceMonitor.enabled` | Create ServiceMonitor Resource for scraping metrics using PrometheusOperator | `false` | -| `metrics.serviceMonitor.namespace` | Specify the namespace in which the serviceMonitor resource will be created | `""` | -| `metrics.serviceMonitor.jobLabel` | The name of the label on the target service to use as the job name in prometheus. | `""` | -| `metrics.serviceMonitor.interval` | Specify the interval at which metrics should be scraped | `30s` | -| `metrics.serviceMonitor.scrapeTimeout` | Specify the timeout after which the scrape is ended | `""` | -| `metrics.serviceMonitor.relabelings` | RelabelConfigs to apply to samples before scraping | `[]` | -| `metrics.serviceMonitor.metricRelabelings` | Specify Metric Relabelings to add to the scrape endpoint | `[]` | -| `metrics.serviceMonitor.labels` | Extra labels for the ServiceMonitor | `{}` | -| `metrics.serviceMonitor.honorLabels` | Specify honorLabels parameter to add the scrape endpoint | `false` | -| `metrics.serviceMonitor.selector` | ServiceMonitor selector labels | `{}` | - -### NetworkPolicy parameters - -| Name | Description | Value | -| --------------------------------------- | --------------------------------------------------------------- | ------ | -| `networkPolicy.enabled` | Specifies whether a NetworkPolicy should be created | `true` | -| `networkPolicy.allowExternal` | Don't require server label for connections | `true` | -| `networkPolicy.allowExternalEgress` | Allow the pod to access any range of port and all destinations. | `true` | -| `networkPolicy.extraIngress` | Add extra ingress rules to the NetworkPolicy | `[]` | -| `networkPolicy.extraEgress` | Add extra ingress rules to the NetworkPolicy | `[]` | -| `networkPolicy.ingressNSMatchLabels` | Labels to match to allow traffic from other namespaces | `{}` | -| `networkPolicy.ingressNSPodMatchLabels` | Pod labels to match to allow traffic from other namespaces | `{}` | - -For more information please refer to the [bitnami/phpmyadmin](https://github.com/bitnami/containers/tree/main/bitnami/phpmyadmin) image documentation. - -Specify each parameter using the `--set key=value[,key=value]` argument to `helm install`. For example, - -```console -helm install my-release \ - --set db.host=mymariadb,db.port=3306 oci://REGISTRY_NAME/REPOSITORY_NAME/phpmyadmin -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -The above command sets the phpMyAdmin to connect to a database in `mymariadb` host and `3306` port respectively. - -Alternatively, a YAML file that specifies the values for the above parameters can be provided while installing the chart. For example, - -```console -helm install my-release -f values.yaml oci://REGISTRY_NAME/REPOSITORY_NAME/phpmyadmin -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. -> **Tip**: You can use the default [values.yaml](https://github.com/bitnami/charts/tree/main/bitnami/phpmyadmin/values.yaml) - -## Troubleshooting - -Find more information about how to deal with common errors related to Bitnami's Helm charts in [this troubleshooting guide](https://docs.bitnami.com/general/how-to/troubleshoot-helm-chart-issues). - -## Upgrading - -### To 17.0.0 - -This major release bumps the MariaDB version to 11.4. Follow the [upstream instructions](https://mariadb.com/kb/en/upgrading-from-mariadb-11-3-to-mariadb-11-4/) for upgrading from MariaDB 11.3 to 11.4. No major issues are expected during the upgrade. - -### To 16.0.0 - -This major bump changes the following security defaults: - -- `runAsGroup` is changed from `0` to `1001` -- `readOnlyRootFilesystem` is set to `true` -- `resourcesPreset` is changed from `none` to the minimum size working in our test suites (NOTE: `resourcesPreset` is not meant for production usage, but `resources` adapted to your use case). -- `global.compatibility.openshift.adaptSecurityContext` is changed from `disabled` to `auto`. -- The `networkPolicy` section has been normalized amongst all Bitnami charts. Compared to the previous approach, the values section has been simplified (check the Parameters section) and now it set to `enabled=true` by default. Egress traffic is allowed by default and ingress traffic is allowed by all pods but only to the ports set in `containerPorts` and `extraContainerPorts`. - -This could potentially break any customization or init scripts used in your deployment. If this is the case, change the default values to the previous ones. - -### To 15.0.0 - -This major release bumps the MariaDB chart version to [18.x.x](https://github.com/bitnami/charts/pull/24804); no major issues are expected during the upgrade. - -### To 14.0.0 - -This major release bumps the MariaDB version to 11.2. No major issues are expected during the upgrade. - -### To 13.0.0 - -This major release bumps the MariaDB version to 11.1. No major issues are expected during the upgrade. - -### To 12.0.0 - -This major release bumps the MariaDB version to 11.0. Follow the [upstream instructions](https://mariadb.com/kb/en/upgrading-from-mariadb-10-11-to-mariadb-11-0/) for upgrading from MariaDB 10.11 to 11.0. No major issues are expected during the upgrade. - -### To 11.0.0 - -This major release bumps the MariaDB version to 10.11. Follow the [upstream instructions](https://mariadb.com/kb/en/upgrading-from-mariadb-10-6-to-mariadb-10-11/) for upgrading from MariaDB 10.6 to 10.11. No major issues are expected during the upgrade. - -### To 10.0.0 - -This major release bumps the MariaDB version to 10.6. Follow the [upstream instructions](https://mariadb.com/kb/en/upgrading-from-mariadb-105-to-mariadb-106/) for upgrading from MariaDB 10.5 to 10.6. No major issues are expected during the upgrade. - -### To 9.0.0 - -This major release renames several values in this chart and adds missing features, in order to be inline with the rest of assets in the Bitnami charts repository. - -Affected values: - -- `service.port` was deprecated. We recommend using `service.ports.http` instead. -- `service.httpsPort` was deprecated. We recommend using `service.ports.https` instead. -- `metrics.serviceMonitor.additionalLabels` renamed as `metrics.serviceMonitor.labels` - -Additionally updates the MariaDB subchart to it newest major, 10.0.0, which contains similar changes. Check [MariaDB Upgrading Notes](https://github.com/bitnami/charts/tree/main/bitnami/mariadb#to-1000) for more information. - -### To 8.0.0 - -- Chart labels were adapted to follow the [Helm charts standard labels](https://helm.sh/docs/chart_best_practices/labels/#standard-labels). - -Consequences: - -- Backwards compatibility is not guaranteed. However, you can easily workaround this issue by removing PhpMyAdmin deployment before upgrading (the following example assumes that the release name is `phpmyadmin`): - -```console -export MARIADB_ROOT_PASSWORD=$(kubectl get secret --namespace default phpmyadmin-mariadb -o jsonpath="{.data.mariadb-root-password}" | base64 -d) -export MARIADB_PASSWORD=$(kubectl get secret --namespace default phpmyadmin-mariadb -o jsonpath="{.data.mariadb-password}" | base64 -d) -kubectl delete deployments.apps phpmyadmin -helm upgrade phpmyadmin oci://REGISTRY_NAME/REPOSITORY_NAME/phpmyadmin --set mariadb.auth.rootPassword=$MARIADB_ROOT_PASSWORD,mariadb.auth.password=$MARIADB_PASSWORD -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -### To 7.0.0 - -In this major there were two main changes introduced: - -1. Adaptation to Helm v2 EOL -2. Updated MariaDB dependency version - -Please read the update notes carefully. - -#### 1. Adaptation to Helm v2 EOL - -[On November 13, 2020, Helm v2 support was formally finished](https://github.com/helm/charts#status-of-the-project), this major version is the result of the required changes applied to the Helm Chart to be able to incorporate the different features added in Helm v3 and to be consistent with the Helm project itself regarding the Helm v2 EOL. - -##### What changes were introduced in this major version? - -- Previous versions of this Helm Chart use `apiVersion: v1` (installable by both Helm 2 and 3), this Helm Chart was updated to `apiVersion: v2` (installable by Helm 3 only). [Here](https://helm.sh/docs/topics/charts/#the-apiversion-field) you can find more information about the `apiVersion` field. -- Move dependency information from the *requirements.yaml* to the *Chart.yaml* -- After running `helm dependency update`, a *Chart.lock* file is generated containing the same structure used in the previous *requirements.lock* -- The different fields present in the *Chart.yaml* file has been ordered alphabetically in a homogeneous way for all the Bitnami Helm Charts - -##### Considerations when upgrading to this version - -- If you want to upgrade to this version from a previous one installed with Helm v3, you shouldn't face any issues -- If you want to upgrade to this version using Helm v2, this scenario is not supported as this version doesn't support Helm v2 anymore -- If you installed the previous version with Helm v2 and wants to upgrade to this version with Helm v3, please refer to the [official Helm documentation](https://helm.sh/docs/topics/v2_v3_migration/#migration-use-cases) about migrating from Helm v2 to v3 - -##### Useful links - -- -- -- - -#### 2. Updated MariaDB dependency version - -In this major the MariaDB dependency version was also bumped to a new major version that introduces several incompatilibites. Therefore, backwards compatibility is not guaranteed unless an external database is used. Check [MariaDB Upgrading Notes](https://github.com/bitnami/charts/tree/main/bitnami/mariadb#to-800) for more information. - -To upgrade to `7.0.0`, it should be done reusing the PVCs used to hold both the MariaDB and phpMyAdmin data on your previous release. To do so, follow the instructions below (the following example assumes that the release name is `phpmyadmin` and that `db.bundleTestDB=true` when the chart was first installed): - -> NOTE: Please, create a backup of your database before running any of those actions. The steps below would be only valid if your application (e.g. any plugins or custom code) is compatible with MariaDB 10.5.x - -Obtain the credentials and the names of the PVCs used to hold both the MariaDB and phpMyAdmin data on your current release: - -```console -export MARIADB_ROOT_PASSWORD=$(kubectl get secret --namespace default phpmyadmin-mariadb -o jsonpath="{.data.mariadb-root-password}" | base64 -d) -export MARIADB_PASSWORD=$(kubectl get secret --namespace default phpmyadmin-mariadb -o jsonpath="{.data.mariadb-password}" | base64 -d) -export MARIADB_PVC=$(kubectl get pvc -l app=mariadb,component=master,release=phpmyadmin -o jsonpath="{.items[0].metadata.name}") -``` - -Delete the phpMyAdmin deployment and delete the MariaDB statefulsets: - -```console - kubectl delete deployments.apps phpmyadmin - - kubectl delete statefulsets.apps phpmyadmin-mariadb-master - - kubectl delete statefulsets.apps phpmyadmin-mariadb-slave - -``` - -Now the upgrade works: - -```console -helm upgrade phpmyadmin oci://REGISTRY_NAME/REPOSITORY_NAME/phpmyadmin --set mariadb.primary.persistence.existingClaim=$MARIADB_PVC --set mariadb.auth.rootPassword=$MARIADB_ROOT_PASSWORD --set mariadb.auth.password=$MARIADB_PASSWORD --set db.bundleTestDB=true -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -Finally, you should see the lines below in MariaDB container logs: - -```console -$ kubectl logs $(kubectl get pods -l app.kubernetes.io/instance=phpmyadmin,app.kubernetes.io/name=mariadb,app.kubernetes.io/component=primary -o jsonpath="{.items[0].metadata.name}") -... -mariadb 12:13:24.98 INFO ==> Using persisted data -mariadb 12:13:25.01 INFO ==> Running mysql_upgrade -... -``` - -### To 6.0.0 - -The [Bitnami phpMyAdmin](https://github.com/bitnami/containers/tree/main/bitnami/phpmyadmin) image was migrated to a "non-root" user approach. Previously the container ran as the `root` user and the Apache daemon was started as the `daemon` user. From now on, both the container and the Apache daemon run as user `1001`. You can revert this behavior by setting the parameters `containerSecurityContext.runAsUser` to `root`. -Chart labels and Ingress configuration were also adapted to follow the Helm charts best practices. - -Consequences: - -- The HTTP/HTTPS ports exposed by the container are now `8080/8443` instead of `80/443`. -- No writing permissions will be granted on `config.inc.php` by default. -- Backwards compatibility is not guaranteed. - -To upgrade to `6.0.0`, backup your previous MariaDB databases, install a new phpMyAdmin chart and import the MariaDB backups. - -This version also introduces `bitnami/common`, a [library chart](https://helm.sh/docs/topics/library_charts/#helm) as a dependency. More documentation about this new utility could be found [here](https://github.com/bitnami/charts/tree/main/bitnami/common#bitnami-common-library-chart). Please, make sure that you have updated the chart dependencies before executing any upgrade. - -### To 1.0.0 - -Backwards compatibility is not guaranteed unless you modify the labels used on the chart's deployments. -Use the workaround below to upgrade from versions previous to `1.0.0`. The following example assumes that the release name is `phpmyadmin`: - -```console -kubectl patch deployment phpmyadmin-phpmyadmin --type=json -p='[{"op": "remove", "path": "/spec/selector/matchLabels/chart"}]' -``` - -## License - -Copyright © 2024 Broadcom. The term "Broadcom" refers to Broadcom Inc. and/or its subsidiaries. - -Licensed under the Apache License, Version 2.0 (the "License"); -you may not use this file except in compliance with the License. -You may obtain a copy of the License at - - - -Unless required by applicable law or agreed to in writing, software -distributed under the License is distributed on an "AS IS" BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -See the License for the specific language governing permissions and -limitations under the License. \ No newline at end of file diff --git a/charts/phpmyadmin/phpmyadmin/templates/NOTES.txt b/charts/phpmyadmin/phpmyadmin/templates/NOTES.txt deleted file mode 100644 index 5a42dd9..0000000 --- a/charts/phpmyadmin/phpmyadmin/templates/NOTES.txt +++ /dev/null @@ -1,69 +0,0 @@ -CHART NAME: {{ .Chart.Name }} -CHART VERSION: {{ .Chart.Version }} -APP VERSION: {{ .Chart.AppVersion }} - -** Please be patient while the chart is being deployed ** - -1. Get the application URL by running these commands: - -{{- if .Values.ingress.enabled }} - - You should be able to access your new phpMyAdmin installation through - - http{{ if $.Values.ingress.tls }}s{{ end }}://{{ .Values.ingress.hostname }}/ - -{{- else if eq .Values.service.type "LoadBalancer" }} - - NOTE: It may take a few minutes for the LoadBalancer IP to be available. - Watch the status with: 'kubectl get svc --namespace {{ .Release.Namespace }} -w {{ include "common.names.fullname" . }}' - - export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ include "common.names.fullname" . }} --template "{{ "{{ range (index .status.loadBalancer.ingress 0) }}{{ . }}{{ end }}" }}") - {{- $port:=( coalesce .Values.service.ports.http .Values.service.port) | toString }} - echo "phpMyAdmin URL: http://$SERVICE_IP{{- if ne $port "80" }}:{{coalesce .Values.service.ports.http .Values.service.port }}{{ end }}/" - -{{- else if contains "NodePort" .Values.service.type }} - - export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ include "common.names.fullname" . }}' - export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}") - echo "phpMyAdmin URL: http://$NODE_IP:$NODE_PORT" - -{{- else if contains "ClusterIP" .Values.service.type }} - - echo "phpMyAdmin URL: http://127.0.0.1:{{coalesce .Values.service.ports.http .Values.service.port }}" - kubectl port-forward --namespace {{ .Release.Namespace }} svc/{{ template "common.names.fullname" . }} {{coalesce .Values.service.ports.http .Values.service.port }}:{{coalesce .Values.service.ports.http .Values.service.port }} - -{{- end }} - -2. How to log in -{{- if or (.Values.db.chartName) (.Values.db.host) }} - -phpMyAdmin has been configured to connect to a database in {{ if .Values.db.chartName }}{{ template "phpmyadmin.dbfullname" . }}{{ else }}{{ .Values.db.host }}{{ end }} -{{- if .Values.db.port }}with port {{ .Values.db.port }} {{ end }} -Please login using a database username and password. -{{- else if .Values.db.bundleTestDB }} -For testing purposes, phpMyAdmin has been configured to point to a test MariaDB -instance. Please login using the following credentials: - - Username: root - Password : $(kubectl get secret --namespace {{ .Release.Namespace }} {{ template "phpmyadmin.mariadb.fullname" . }} -o jsonpath="{.data.mariadb-root-password}" | base64 -d) - -{{- else }} -phpMyAdmin has not been configure to point to a specific database. Please provide the db host, -username and password at log in or upgrade the release with a specific database: - -$ helm upgrade {{ .Release.Name }} oci://registry-1.docker.io/bitnamicharts/phpmyadmin --set db.host=mydb - -{{- end }} - -{{- include "phpmyadmin.validateValues" . -}} -{{- include "phpmyadmin.checkRollingTags" . -}} - -{{- $passwordValidationErrors := list -}} -{{- if .Values.mariadb.enabled }} - {{- $mariadbSecretName := include "magento.databaseSecretName" . -}} - {{- $mariadbPasswordValidationErrors := include "common.validations.values.mariadb.passwords" (dict "secret" $mariadbSecretName "subchart" true "context" $) -}} - {{- $passwordValidationErrors = append $passwordValidationErrors $mariadbPasswordValidationErrors -}} -{{- end }} -{{- include "common.errors.upgrade.passwords.empty" (dict "validationErrors" $passwordValidationErrors "context" $) -}} -{{- include "common.warnings.resources" (dict "sections" (list "metrics" "") "context" $) }} -{{- include "common.warnings.modifiedImages" (dict "images" (list .Values.image .Values.metrics.image) "context" $) }} \ No newline at end of file diff --git a/charts/phpmyadmin/phpmyadmin/templates/_helpers.tpl b/charts/phpmyadmin/phpmyadmin/templates/_helpers.tpl deleted file mode 100644 index 064a7bc..0000000 --- a/charts/phpmyadmin/phpmyadmin/templates/_helpers.tpl +++ /dev/null @@ -1,86 +0,0 @@ -{{/* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{/* vim: set filetype=mustache: */}} - -{{/* -Return the proper PHPMyAdmin image name -*/}} -{{- define "phpmyadmin.image" -}} -{{ include "common.images.image" (dict "imageRoot" .Values.image "global" .Values.global) }} -{{- end -}} - -{{/* -Return the proper image name (for the metrics image) -*/}} -{{- define "phpmyadmin.metrics.image" -}} -{{ include "common.images.image" (dict "imageRoot" .Values.metrics.image "global" .Values.global) }} -{{- end -}} - -{{/* -Return the proper Docker Image Registry Secret Names -*/}} -{{- define "phpmyadmin.imagePullSecrets" -}} -{{- include "common.images.pullSecrets" (dict "images" (list .Values.image .Values.metrics.image) "global" .Values.global) -}} -{{- end -}} - -{{/* - Create the name of the service account to use - */}} -{{- define "phpmyadmin.serviceAccountName" -}} -{{- if .Values.serviceAccount.create -}} - {{ default (include "common.names.fullname" .) .Values.serviceAccount.name }} -{{- else -}} - {{ default "default" .Values.serviceAccount.name }} -{{- end -}} -{{- end -}} - -{{/* -Create a default fully qualified app name. -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -*/}} -{{- define "phpmyadmin.mariadb.fullname" -}} -{{- printf "%s-%s" .Release.Name "mariadb" | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Create a fully qualified database name if the database is part of the same release than phpmyadmin. -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -*/}} -{{- define "phpmyadmin.dbfullname" -}} -{{- printf "%s-%s" .Release.Name .Values.db.chartName | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Compile all warnings into a single message, and call fail. -*/}} -{{- define "phpmyadmin.validateValues" -}} -{{- $messages := list -}} -{{- $messages := append $messages (include "phpmyadmin.validateValues.db.ssl" .) -}} -{{- $messages := without $messages "" -}} -{{- $message := join "\n" $messages -}} - -{{- if $message -}} -{{- printf "\nVALUES VALIDATION:\n%s" $message | fail -}} -{{- end -}} -{{- end -}} - -{{/* Validate values of phpMyAdmin - must provide a valid database ssl configuration */}} -{{- define "phpmyadmin.validateValues.db.ssl" -}} -{{- if and .Values.db.enableSsl (empty .Values.db.ssl.clientKey) (empty .Values.db.ssl.clientCertificate) (empty .Values.db.ssl.caCertificate) -}} -phpMyAdmin: db.ssl - Invalid database ssl configuration. You enabled SSL for the connection - between phpMyAdmin and the database but no key/certificates were provided - (--set db.ssl.clientKey="xxxx", --set db.ssl.clientCertificate="yyyy") -{{- end -}} -{{- end -}} - -{{/* -Check if there are rolling tags in the images -*/}} -{{- define "phpmyadmin.checkRollingTags" -}} -{{- include "common.warnings.rollingTag" .Values.image }} -{{- include "common.warnings.rollingTag" .Values.metrics.image }} -{{- end -}} diff --git a/charts/phpmyadmin/phpmyadmin/templates/certs.yaml b/charts/phpmyadmin/phpmyadmin/templates/certs.yaml deleted file mode 100644 index 6fb29a5..0000000 --- a/charts/phpmyadmin/phpmyadmin/templates/certs.yaml +++ /dev/null @@ -1,27 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.db.enableSsl }} -apiVersion: v1 -kind: Secret -metadata: - name: {{ printf "%s-certs" (include "common.names.fullname" .) }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: Opaque -data: - {{- if not (empty .Values.db.ssl.clientKey) }} - server_key.pem: {{ .Values.db.ssl.clientKey | b64enc | quote }} - {{- end }} - {{- if not (empty .Values.db.ssl.clientCertificate) }} - server_certificate.pem: {{ .Values.db.ssl.clientCertificate | b64enc | quote }} - {{- end }} - {{- if not (empty .Values.db.ssl.caCertificate) }} - ca_certificate.pem: {{ .Values.db.ssl.caCertificate | b64enc | quote }} - {{- end }} -{{- end }} diff --git a/charts/phpmyadmin/phpmyadmin/templates/deployment.yaml b/charts/phpmyadmin/phpmyadmin/templates/deployment.yaml deleted file mode 100644 index ba88865..0000000 --- a/charts/phpmyadmin/phpmyadmin/templates/deployment.yaml +++ /dev/null @@ -1,306 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -apiVersion: {{ include "common.capabilities.deployment.apiVersion" . }} -kind: Deployment -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.podLabels .Values.commonLabels ) "context" . ) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - replicas: {{ .Values.replicas }} - strategy: {{- include "common.tplvalues.render" (dict "value" .Values.updateStrategy "context" $ ) | nindent 4 }} - template: - metadata: - labels: {{- include "common.labels.standard" ( dict "customLabels" $podLabels "context" $ ) | nindent 8 }} - {{- if or .Values.podAnnotations (and .Values.metrics.enabled .Values.metrics.podAnnotations) }} - annotations: - {{- if .Values.podAnnotations }} - {{- include "common.tplvalues.render" ( dict "value" .Values.podAnnotations "context" $) | nindent 8 }} - {{- end }} - {{- if and .Values.metrics.enabled .Values.metrics.podAnnotations }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.podAnnotations "context" $) | nindent 8 }} - {{- end }} - {{- end }} - spec: - {{- include "phpmyadmin.imagePullSecrets" . | nindent 6 }} - automountServiceAccountToken: {{ .Values.automountServiceAccountToken }} - {{- if .Values.hostAliases }} - # yamllint disable rule:indentation - hostAliases: {{- include "common.tplvalues.render" (dict "value" .Values.hostAliases "context" $) | nindent 8 }} - # yamllint enable rule:indentation - {{- end }} - {{- if .Values.affinity }} - affinity: {{- include "common.tplvalues.render" ( dict "value" .Values.affinity "context" $) | nindent 8 }} - {{- else }} - affinity: - podAffinity: {{- include "common.affinities.pods" (dict "type" .Values.podAffinityPreset "customLabels" $podLabels "context" $) | nindent 10 }} - podAntiAffinity: {{- include "common.affinities.pods" (dict "type" .Values.podAntiAffinityPreset "customLabels" $podLabels "context" $) | nindent 10 }} - nodeAffinity: {{- include "common.affinities.nodes" (dict "type" .Values.nodeAffinityPreset.type "key" .Values.nodeAffinityPreset.key "values" .Values.nodeAffinityPreset.values) | nindent 10 }} - {{- end }} - {{- if .Values.schedulerName }} - schedulerName: {{ .Values.schedulerName }} - {{- end }} - {{- if .Values.topologySpreadConstraints }} - topologySpreadConstraints: {{- include "common.tplvalues.render" (dict "value" .Values.topologySpreadConstraints "context" .) | nindent 8 }} - {{- end }} - priorityClassName: {{ .Values.priorityClassName | quote }} - {{- if .Values.nodeSelector }} - nodeSelector: {{- include "common.tplvalues.render" ( dict "value" .Values.nodeSelector "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.tolerations }} - tolerations: {{- include "common.tplvalues.render" (dict "value" .Values.tolerations "context" .) | nindent 8 }} - {{- end }} - {{- if .Values.podSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.podSecurityContext "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.serviceAccount.create }} - serviceAccountName: {{ template "phpmyadmin.serviceAccountName" . }} - {{- end }} - initContainers: - - name: prepare-base-dir - image: {{ include "phpmyadmin.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy | quote }} - {{- if .Values.resources }} - resources: {{- toYaml .Values.resources | nindent 12 }} - {{- else if ne .Values.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.resourcesPreset) | nindent 12 }} - {{- end }} - {{- if .Values.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - command: - - /bin/bash - args: - - -ec - - | - #!/bin/bash - - . /opt/bitnami/scripts/liblog.sh - . /opt/bitnami/scripts/libfs.sh - - info "Copying base dir to empty dir" - # In order to not break the application functionality (such as upgrades or plugins) we need - # to make the base directory writable, so we need to copy it to an empty dir volume - cp -r --preserve=mode /opt/bitnami/phpmyadmin /emptydir/app-base-dir - - info "Copying symlinks to stdout/stderr" - # We copy the logs folder because it has symlinks to stdout and stderr - if ! is_dir_empty /opt/bitnami/apache/logs; then - cp -r /opt/bitnami/apache/logs /emptydir/apache-logs-dir - fi - info "Copying php var directory" - # PhpMyAdmin will fail to start if the php var folder is not populated - if ! is_dir_empty /opt/bitnami/php/var; then - cp -r /opt/bitnami/php/var /emptydir/php-var-dir - fi - info "Copy operation completed" - volumeMounts: - - name: empty-dir - mountPath: /emptydir - {{- if .Values.initContainers }} - {{- include "common.tplvalues.render" (dict "value" .Values.initContainers "context" $) | nindent 8 }} - {{- end }} - containers: - - name: {{ .Chart.Name }} - image: {{ template "phpmyadmin.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy }} - {{- if .Values.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.command "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.args }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.args "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.lifecycleHooks }} - lifecycle: {{- include "common.tplvalues.render" (dict "value" .Values.lifecycleHooks "context" $) | nindent 12 }} - {{- end }} - env: - - name: BITNAMI_DEBUG - value: {{ ternary "true" "false" .Values.image.debug | quote }} - - name: DATABASE_PORT_NUMBER - value: {{ .Values.db.port | quote }} - {{- if .Values.db.chartName }} - - name: DATABASE_HOST - value: {{ (include "phpmyadmin.dbfullname" .) | quote }} - {{- else if .Values.db.bundleTestDB }} - - name: DATABASE_HOST - value: {{ (include "phpmyadmin.mariadb.fullname" .) | quote }} - {{- else }} - - name: DATABASE_HOST - value: {{ .Values.db.host | quote }} - {{- end }} - {{- if and (not .Values.db.chartName) (not .Values.db.host) }} - - name: PHPMYADMIN_ALLOW_NO_PASSWORD - value: "true" - - name: PHPMYADMIN_ALLOW_ARBITRARY_SERVER - value: {{ .Values.db.allowArbitraryServer | quote }} - {{- else }} - - name: PHPMYADMIN_ALLOW_NO_PASSWORD - value: "false" - {{- end }} - - name: DATABASE_ENABLE_SSL - value: {{ ternary "yes" "no" .Values.db.enableSsl | quote }} - {{- if .Values.db.enableSsl }} - {{- if not (empty .Values.db.ssl.clientKey) }} - - name: DATABASE_SSL_KEY - value: "/db_certs/server_key.pem" - {{- end }} - {{- if not (empty .Values.db.ssl.clientCertificate) }} - - name: DATABASE_SSL_CERT - value: "/db_certs/server_certificate.pem" - {{- end }} - {{- if not (empty .Values.db.ssl.caCertificate) }} - - name: DATABASE_SSL_CA - value: "/db_certs/ca_certificate.pem" - {{- end }} - {{- if .Values.db.ssl.ciphers }} - - name: DATABASE_SSL_CIPHERS - values: {{ .Values.db.ssl.ciphers | quote }} - {{- end }} - - name: DATABASE_SSL_VERIFY - value: {{ ternary "yes" "no" .Values.db.ssl.verify | quote }} - {{- end }} - {{- if .Values.extraEnvVars }} - {{- include "common.tplvalues.render" (dict "value" .Values.extraEnvVars "context" $) | nindent 12 }} - {{- end }} - {{- if or .Values.extraEnvVarsCM .Values.extraEnvVarsSecret }} - envFrom: - {{- if .Values.extraEnvVarsCM }} - - configMapRef: - name: {{ include "common.tplvalues.render" (dict "value" .Values.extraEnvVarsCM "context" $) }} - {{- end }} - {{- if .Values.extraEnvVarsSecret }} - - secretRef: - name: {{ include "common.tplvalues.render" (dict "value" .Values.extraEnvVarsSecret "context" $) }} - {{- end }} - {{- end }} - ports: - - name: http - containerPort: {{ .Values.containerPorts.http }} - protocol: TCP - - name: https - containerPort: {{ .Values.containerPorts.https }} - protocol: TCP - {{- if .Values.extraContainerPorts }} - {{- include "common.tplvalues.render" (dict "value" .Values.extraContainerPorts "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.startupProbe.enabled }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.startupProbe "enabled") "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.livenessProbe.enabled }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.livenessProbe "enabled") "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.readinessProbe.enabled }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.readinessProbe "enabled") "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.resources }} - resources: {{- toYaml .Values.resources | nindent 12 }} - {{- else if ne .Values.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /opt/bitnami/apache/conf - subPath: apache-conf-dir - - name: empty-dir - mountPath: /opt/bitnami/apache/logs - subPath: apache-logs-dir - - name: empty-dir - mountPath: /opt/bitnami/apache/var/run - subPath: apache-tmp-dir - - name: empty-dir - mountPath: /opt/bitnami/php/etc - subPath: php-conf-dir - - name: empty-dir - mountPath: /opt/bitnami/php/tmp - subPath: php-tmp-dir - - name: empty-dir - mountPath: /opt/bitnami/php/var - subPath: php-var-dir - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: empty-dir - mountPath: /opt/bitnami/phpmyadmin - subPath: app-base-dir - {{- if .Values.db.enableSsl }} - - name: ssl-certs - mountPath: /db_certs - {{- end }} - {{- if .Values.extraVolumeMounts }} - {{- include "common.tplvalues.render" (dict "value" .Values.extraVolumeMounts "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.metrics.enabled }} - - name: metrics - image: {{ template "phpmyadmin.metrics.image" . }} - imagePullPolicy: {{ .Values.metrics.image.pullPolicy | quote }} - command: ['/bin/apache_exporter', '--scrape_uri', 'http://status.localhost:8080/server-status/?auto'] - ports: - - name: metrics - containerPort: 9117 - livenessProbe: - httpGet: - path: /metrics - port: metrics - initialDelaySeconds: 15 - timeoutSeconds: 5 - readinessProbe: - httpGet: - path: /metrics - port: metrics - initialDelaySeconds: 5 - timeoutSeconds: 1 - {{- if .Values.metrics.resources }} - resources: {{- toYaml .Values.metrics.resources | nindent 12 }} - {{- else if ne .Values.metrics.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.metrics.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - {{- end }} - {{- if .Values.sidecars }} - {{- include "common.tplvalues.render" ( dict "value" .Values.sidecars "context" $) | nindent 8 }} - {{- end }} - volumes: - - name: empty-dir - emptyDir: {} - {{- if .Values.db.enableSsl }} - - name: ssl-certs - secret: - secretName: {{ printf "%s-certs" (include "common.names.fullname" .) }} - items: - {{- if not (empty .Values.db.ssl.clientKey) }} - - key: server_key.pem - path: server_key.pem - {{- end }} - {{- if not (empty .Values.db.ssl.clientCertificate) }} - - key: server_certificate.pem - path: server_certificate.pem - {{- end }} - {{- if not (empty .Values.db.ssl.caCertificate) }} - - key: ca_certificate.pem - path: ca_certificate.pem - {{- end }} - {{- end }} - {{- if .Values.extraVolumes }} - {{- include "common.tplvalues.render" (dict "value" .Values.extraVolumes "context" $) | nindent 8 }} - {{- end }} diff --git a/charts/phpmyadmin/phpmyadmin/templates/extra-list.yaml b/charts/phpmyadmin/phpmyadmin/templates/extra-list.yaml deleted file mode 100644 index 329f5c6..0000000 --- a/charts/phpmyadmin/phpmyadmin/templates/extra-list.yaml +++ /dev/null @@ -1,9 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- range .Values.extraDeploy }} ---- -{{ include "common.tplvalues.render" (dict "value" . "context" $) }} -{{- end }} diff --git a/charts/phpmyadmin/phpmyadmin/templates/ingress.yaml b/charts/phpmyadmin/phpmyadmin/templates/ingress.yaml deleted file mode 100644 index 278b7e9..0000000 --- a/charts/phpmyadmin/phpmyadmin/templates/ingress.yaml +++ /dev/null @@ -1,65 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.ingress.enabled }} -apiVersion: {{ include "common.capabilities.ingress.apiVersion" . }} -kind: Ingress -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if or .Values.ingress.annotations .Values.commonAnnotations .Values.ingress.certManager }} - annotations: - {{- if .Values.ingress.certManager }} - kubernetes.io/tls-acme: "true" - {{- end }} - {{- if or .Values.ingress.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.ingress.annotations .Values.commonAnnotations ) "context" . ) }} - {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} - {{- end }} -spec: - {{- if and .Values.ingress.ingressClassName (eq "true" (include "common.ingress.supportsIngressClassname" .)) }} - ingressClassName: {{ .Values.ingress.ingressClassName | quote }} - {{- end }} - rules: - {{- if .Values.ingress.hostname }} - - host: {{ .Values.ingress.hostname }} - http: - paths: - {{- if .Values.ingress.extraPaths }} - {{- toYaml .Values.ingress.extraPaths | nindent 10 }} - {{- end }} - - path: {{ .Values.ingress.path }} - {{- if eq "true" (include "common.ingress.supportsPathType" .) }} - pathType: {{ .Values.ingress.pathType }} - {{- end }} - backend: {{- include "common.ingress.backend" (dict "serviceName" (include "common.names.fullname" .) "servicePort" "http" "context" $) | nindent 14 }} - {{- end }} - {{- range .Values.ingress.extraHosts }} - - host: {{ .name | quote }} - http: - paths: - - path: {{ default "/" .path }} - {{- if eq "true" (include "common.ingress.supportsPathType" $) }} - pathType: {{ default "ImplementationSpecific" .pathType }} - {{- end }} - backend: {{- include "common.ingress.backend" (dict "serviceName" (include "common.names.fullname" $) "servicePort" "http" "context" $) | nindent 14 }} - {{- end }} - {{- if .Values.ingress.extraRules }} - {{- include "common.tplvalues.render" (dict "value" .Values.ingress.extraRules "context" $) | nindent 4 }} - {{- end }} - {{- if or .Values.ingress.tls .Values.ingress.extraTls }} - tls: - {{- if .Values.ingress.tls }} - - hosts: - - {{ .Values.ingress.hostname }} - secretName: {{ .Values.ingress.existingSecretName | default (printf "%s-tls" .Values.ingress.hostname) }} - {{- end }} - {{- if .Values.ingress.extraTls }} - {{- include "common.tplvalues.render" ( dict "value" .Values.ingress.extraTls "context" $ ) | nindent 4 }} - {{- end }} - {{- end }} -{{- end }} diff --git a/charts/phpmyadmin/phpmyadmin/templates/metrics-svc.yaml b/charts/phpmyadmin/phpmyadmin/templates/metrics-svc.yaml deleted file mode 100644 index dbd012c..0000000 --- a/charts/phpmyadmin/phpmyadmin/templates/metrics-svc.yaml +++ /dev/null @@ -1,45 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.metrics.enabled }} -apiVersion: v1 -kind: Service -metadata: - name: {{ printf "%s-metrics" (include "common.names.fullname" .) }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: metrics - {{- if or .Values.metrics.service.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.metrics.service.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: {{ .Values.metrics.service.type }} - {{- if and .Values.metrics.service.clusterIP (eq .Values.metrics.service.type "ClusterIP") }} - clusterIP: {{ .Values.metrics.service.clusterIP }} - {{- end }} - {{- if (or (eq .Values.metrics.service.type "LoadBalancer") (eq .Values.metrics.service.type "NodePort")) }} - externalTrafficPolicy: {{ .Values.metrics.service.externalTrafficPolicy | quote }} - {{- end }} - {{- if eq .Values.metrics.service.type "LoadBalancer" }} - loadBalancerSourceRanges: {{ .Values.metrics.service.loadBalancerSourceRanges }} - {{- end }} - {{- if (and (eq .Values.metrics.service.type "LoadBalancer") (not (empty .Values.metrics.service.loadBalancerIP))) }} - loadBalancerIP: {{ .Values.metrics.service.loadBalancerIP }} - {{- end }} - {{- if .Values.metrics.service.sessionAffinity }} - sessionAffinity: {{ .Values.metrics.service.sessionAffinity }} - {{- end }} - {{- if .Values.metrics.service.sessionAffinityConfig }} - sessionAffinityConfig: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.service.sessionAffinityConfig "context" $) | nindent 4 }} - {{- end }} - ports: - - port: {{ .Values.metrics.service.port }} - targetPort: metrics - protocol: TCP - name: metrics - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} -{{- end }} diff --git a/charts/phpmyadmin/phpmyadmin/templates/networkpolicy.yaml b/charts/phpmyadmin/phpmyadmin/templates/networkpolicy.yaml deleted file mode 100644 index cc38767..0000000 --- a/charts/phpmyadmin/phpmyadmin/templates/networkpolicy.yaml +++ /dev/null @@ -1,80 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.networkPolicy.enabled }} -kind: NetworkPolicy -apiVersion: {{ include "common.capabilities.networkPolicy.apiVersion" . }} -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.podLabels .Values.commonLabels ) "context" . ) }} - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - policyTypes: - - Ingress - - Egress - {{- if .Values.networkPolicy.allowExternalEgress }} - egress: - - {} - {{- else }} - egress: - # Allow dns resolution - - ports: - - port: 53 - protocol: UDP - - port: 53 - protocol: TCP - # Allow outbound connections to MariaDB - - ports: - - port: {{ .Values.db.port }} - {{- if .Values.db.bundleTestDB }} - to: - - podSelector: - matchLabels: - app.kubernetes.io/name: mariadb - app.kubernetes.io/instance: {{ .Release.Name }} - {{- end }} - {{- if .Values.networkPolicy.extraEgress }} - {{- include "common.tplvalues.render" ( dict "value" .Values.networkPolicy.extraEgress "context" $ ) | nindent 4 }} - {{- end }} - {{- end }} - ingress: - - ports: - - port: {{ .Values.containerPorts.http }} - - port: {{ .Values.containerPorts.https }} - {{- range .Values.extraContainerPorts }} - - port: {{ . }} - {{- end }} - {{- if not .Values.networkPolicy.allowExternal }} - from: - - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 14 }} - - podSelector: - matchLabels: - {{ template "common.names.fullname" . }}-client: "true" - {{- if .Values.networkPolicy.ingressNSMatchLabels }} - - namespaceSelector: - matchLabels: - {{- range $key, $value := .Values.networkPolicy.ingressNSMatchLabels }} - {{ $key | quote }}: {{ $value | quote }} - {{- end }} - {{- if .Values.networkPolicy.ingressNSPodMatchLabels }} - podSelector: - matchLabels: - {{- range $key, $value := .Values.networkPolicy.ingressNSPodMatchLabels }} - {{ $key | quote }}: {{ $value | quote }} - {{- end }} - {{- end }} - {{- end }} - {{- end }} - {{- if .Values.networkPolicy.extraIngress }} - {{- include "common.tplvalues.render" ( dict "value" .Values.networkPolicy.extraIngress "context" $ ) | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/phpmyadmin/phpmyadmin/templates/pdb.yaml b/charts/phpmyadmin/phpmyadmin/templates/pdb.yaml deleted file mode 100644 index b17ff44..0000000 --- a/charts/phpmyadmin/phpmyadmin/templates/pdb.yaml +++ /dev/null @@ -1,26 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.pdb.create }} -apiVersion: {{ include "common.capabilities.policy.apiVersion" . }} -kind: PodDisruptionBudget -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- if .Values.pdb.minAvailable }} - minAvailable: {{ .Values.pdb.minAvailable }} - {{- end }} - {{- if or .Values.pdb.maxUnavailable ( not .Values.pdb.minAvailable ) }} - maxUnavailable: {{ .Values.pdb.maxUnavailable | default 1 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.podLabels .Values.commonLabels ) "context" . ) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} -{{- end }} diff --git a/charts/phpmyadmin/phpmyadmin/templates/service.yaml b/charts/phpmyadmin/phpmyadmin/templates/service.yaml deleted file mode 100644 index a447a80..0000000 --- a/charts/phpmyadmin/phpmyadmin/templates/service.yaml +++ /dev/null @@ -1,59 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -apiVersion: v1 -kind: Service -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if or .Values.service.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.service.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: {{ .Values.service.type }} - {{- if and .Values.service.clusterIP (eq .Values.service.type "ClusterIP") }} - clusterIP: {{ .Values.service.clusterIP }} - {{- end }} - {{- if or (eq .Values.service.type "LoadBalancer") (eq .Values.service.type "NodePort") }} - externalTrafficPolicy: {{ .Values.service.externalTrafficPolicy | quote }} - {{- end }} - {{- if and (eq .Values.service.type "LoadBalancer") .Values.service.loadBalancerSourceRanges }} - loadBalancerSourceRanges: {{- toYaml .Values.service.loadBalancerSourceRanges | nindent 4 }} - {{- end }} - {{- if (and (eq .Values.service.type "LoadBalancer") (not (empty .Values.service.loadBalancerIP))) }} - loadBalancerIP: {{ .Values.service.loadBalancerIP }} - {{- end }} - {{- if .Values.service.sessionAffinity }} - sessionAffinity: {{ .Values.service.sessionAffinity }} - {{- end }} - {{- if .Values.service.sessionAffinityConfig }} - sessionAffinityConfig: {{- include "common.tplvalues.render" (dict "value" .Values.service.sessionAffinityConfig "context" $) | nindent 4 }} - {{- end }} - ports: - - name: http - port: {{ coalesce .Values.service.ports.http .Values.service.port }} - protocol: TCP - targetPort: http - {{- if (and (or (eq .Values.service.type "NodePort") (eq .Values.service.type "LoadBalancer")) (not (empty .Values.service.nodePorts.http))) }} - nodePort: {{ .Values.service.nodePorts.http }} - {{- else if eq .Values.service.type "ClusterIP" }} - nodePort: null - {{- end }} - - name: https - port: {{ coalesce .Values.service.ports.https .Values.service.httpsPort }} - protocol: TCP - targetPort: https - {{- if (and (or (eq .Values.service.type "NodePort") (eq .Values.service.type "LoadBalancer")) (not (empty .Values.service.nodePorts.https))) }} - nodePort: {{ .Values.service.nodePorts.https }} - {{- else if eq .Values.service.type "ClusterIP" }} - nodePort: null - {{- end }} - {{- if .Values.service.extraPorts }} - {{- include "common.tplvalues.render" (dict "value" .Values.service.extraPorts "context" $) | nindent 4 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} diff --git a/charts/phpmyadmin/phpmyadmin/templates/serviceaccount.yaml b/charts/phpmyadmin/phpmyadmin/templates/serviceaccount.yaml deleted file mode 100644 index 65ee3fc..0000000 --- a/charts/phpmyadmin/phpmyadmin/templates/serviceaccount.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.serviceAccount.create }} -apiVersion: v1 -kind: ServiceAccount -automountServiceAccountToken: {{ .Values.serviceAccount.automountServiceAccountToken }} -metadata: - name: {{ template "phpmyadmin.serviceAccountName" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if or .Values.serviceAccount.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.serviceAccount.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/phpmyadmin/phpmyadmin/templates/servicemonitor.yaml b/charts/phpmyadmin/phpmyadmin/templates/servicemonitor.yaml deleted file mode 100644 index f287d34..0000000 --- a/charts/phpmyadmin/phpmyadmin/templates/servicemonitor.yaml +++ /dev/null @@ -1,46 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled .Values.metrics.serviceMonitor.enabled }} -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ default .Release.Namespace .Values.metrics.serviceMonitor.namespace }} - {{- $labels := include "common.tplvalues.merge" ( dict "values" ( list .Values.metrics.serviceMonitor.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $labels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: metrics - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - jobLabel: {{ .Values.metrics.serviceMonitor.jobLabel | quote }} - endpoints: - - port: metrics - {{- if .Values.metrics.serviceMonitor.interval }} - interval: {{ .Values.metrics.serviceMonitor.interval }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.scrapeTimeout }} - scrapeTimeout: {{ .Values.metrics.serviceMonitor.scrapeTimeout }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.honorLabels }} - honorLabels: {{ .Values.metrics.serviceMonitor.honorLabels }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.metricRelabelings }} - metricRelabelings: {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.serviceMonitor.metricRelabelings "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.relabelings }} - relabelings: {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.serviceMonitor.relabelings "context" $) | nindent 8 }} - {{- end }} - namespaceSelector: - matchNames: - - {{ .Release.Namespace | quote }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: metrics - {{- if .Values.metrics.serviceMonitor.selector }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.serviceMonitor.selector "context" $) | nindent 6 }} - {{- end }} -{{- end }} diff --git a/charts/phpmyadmin/phpmyadmin/templates/tls-secrets.yaml b/charts/phpmyadmin/phpmyadmin/templates/tls-secrets.yaml deleted file mode 100644 index bb0a81e..0000000 --- a/charts/phpmyadmin/phpmyadmin/templates/tls-secrets.yaml +++ /dev/null @@ -1,43 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.ingress.enabled (not .Values.ingress.existingSecretName) }} -{{- if .Values.ingress.secrets }} -{{- range .Values.ingress.secrets }} -apiVersion: v1 -kind: Secret -metadata: - name: {{ .name }} - namespace: {{ $.Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $.Values.commonLabels "context" $ ) | nindent 4 }} - {{- if $.Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $.Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: kubernetes.io/tls -data: - tls.crt: {{ .certificate | b64enc }} - tls.key: {{ .key | b64enc }} ---- -{{- end }} -{{- else if and .Values.ingress.tls (and (not (hasKey .Values.ingress.annotations "cert-manager.io/cluster-issuer")) (not .Values.ingress.certManager)) }} -{{- $secretName := printf "%s-tls" .Values.ingress.hostname }} -{{- $ca := genCA "phpmyadmin-ca" 365 }} -{{- $cert := genSignedCert .Values.ingress.hostname nil (list .Values.ingress.hostname) 365 $ca }} -apiVersion: v1 -kind: Secret -metadata: - name: {{ $secretName }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: kubernetes.io/tls -data: - tls.crt: {{ include "common.secrets.lookup" (dict "secret" $secretName "key" "tls.crt" "defaultValue" $cert.Cert "context" $) }} - tls.key: {{ include "common.secrets.lookup" (dict "secret" $secretName "key" "tls.key" "defaultValue" $cert.Key "context" $) }} - ca.crt: {{ include "common.secrets.lookup" (dict "secret" $secretName "key" "ca.crt" "defaultValue" $ca.Cert "context" $) }} -{{- end }} -{{- end }} diff --git a/charts/phpmyadmin/phpmyadmin/values.yaml b/charts/phpmyadmin/phpmyadmin/values.yaml deleted file mode 100644 index 883192e..0000000 --- a/charts/phpmyadmin/phpmyadmin/values.yaml +++ /dev/null @@ -1,803 +0,0 @@ -# Copyright Broadcom, Inc. All Rights Reserved. -# SPDX-License-Identifier: APACHE-2.0 - -## @section Global parameters -## Global Docker image parameters -## Please, note that this will override the image parameters, including dependencies, configured to use the global value -## Current available global Docker image parameters: imageRegistry, imagePullSecrets and storageClass - -## @param global.imageRegistry Global Docker image registry -## @param global.imagePullSecrets Global Docker registry secret names as an array -## @param global.defaultStorageClass Global default StorageClass for Persistent Volume(s) -## -global: - imageRegistry: "" - ## E.g. - ## imagePullSecrets: - ## - myRegistryKeySecretName - ## - imagePullSecrets: [] - defaultStorageClass: "" - ## Compatibility adaptations for Kubernetes platforms - ## - compatibility: - ## Compatibility adaptations for Openshift - ## - openshift: - ## @param global.compatibility.openshift.adaptSecurityContext Adapt the securityContext sections of the deployment to make them compatible with Openshift restricted-v2 SCC: remove runAsUser, runAsGroup and fsGroup and let the platform use their allowed default IDs. Possible values: auto (apply if the detected running cluster is Openshift), force (perform the adaptation always), disabled (do not perform adaptation) - ## - adaptSecurityContext: disabled -## @section Common parameters - -## @param kubeVersion Force target Kubernetes version (using Helm capabilities if not set) -## -kubeVersion: "" -## @param nameOverride String to partially override common.names.fullname template (will maintain the release name) -## -nameOverride: "" -## @param fullnameOverride String to fully override common.names.fullname template -## -fullnameOverride: "" -## @param commonLabels Add labels to all the deployed resources -## -commonLabels: {} -## @param commonAnnotations Add annotations to all the deployed resources -## -commonAnnotations: {} -## @param clusterDomain Kubernetes Cluster Domain -## -clusterDomain: cluster.local -## @param extraDeploy Array of extra objects to deploy with the release -## -extraDeploy: [] -## @section phpMyAdmin parameters - -## Bitnami PhpMyAdmin image version -## ref: https://hub.docker.com/r/bitnami/phpmyadmin/tags/ -## @param image.registry [default: REGISTRY_NAME] phpMyAdmin image registry -## @param image.repository [default: REPOSITORY_NAME/phpmyadmin] phpMyAdmin image repository -## @skip image.tag phpMyAdmin image tag (immutable tags are recommended) -## @param image.digest phpMyAdmin image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag -## @param image.pullPolicy Image pull policy -## @param image.pullSecrets Specify docker-registry secret names as an array -## @param image.debug Enable phpmyadmin image debug mode -## -image: - registry: docker.io - repository: bitnami/phpmyadmin - tag: 5.2.1-debian-12-r42 - digest: "" - ## Specify a imagePullPolicy - ## - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets. - ## Secrets must be manually created in the namespace. - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## Example: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## Enable debug mode - ## - debug: false -## @param command Override default container command (useful when using custom images) -## -command: [] -## @param args Override default container args (useful when using custom images) -## -args: [] -## @param lifecycleHooks for the phpmyadmin container(s) to automate configuration before or after startup -## -lifecycleHooks: {} -## @param extraEnvVars Extra environment variables to be set on PhpMyAdmin container -## For example: -## extraEnvVars: -## - name: PHP_UPLOAD_MAX_FILESIZE -## value: "80M" -## -extraEnvVars: [] -## @param extraEnvVarsCM Name of a existing ConfigMap containing extra env vars -## -extraEnvVarsCM: "" -## @param extraEnvVarsSecret Name of a existing Secret containing extra env vars -## -extraEnvVarsSecret: "" -## @section phpMyAdmin deployment parameters - -## @param automountServiceAccountToken Mount Service Account token in pod -## -automountServiceAccountToken: false -## @param hostAliases [array] Deployment pod host aliases -## https://kubernetes.io/docs/concepts/services-networking/add-entries-to-pod-etc-hosts-with-host-aliases/ -## -hostAliases: - ## Necessary for apache-exporter to work - ## - - ip: "127.0.0.1" - hostnames: - - "status.localhost" -## phpMyAdmin container ports to open -## @param containerPorts.http HTTP port to expose at container level -## @param containerPorts.https HTTPS port to expose at container level -## -containerPorts: - http: 8080 - https: 8443 -## @param extraContainerPorts Optionally specify extra list of additional ports for phpMyAdmin container(s) -## e.g: -## extraContainerPorts: -## - name: myservice -## containerPort: 9090 -## -extraContainerPorts: [] -## @param updateStrategy.type Strategy to use to update Pods -## -updateStrategy: - ## StrategyType - ## Can be set to RollingUpdate or OnDelete - ## - type: RollingUpdate -## phpMyAdmin pods' Security Context -## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-pod -## @param podSecurityContext.enabled Enable phpMyAdmin pods' Security Context -## @param podSecurityContext.fsGroupChangePolicy Set filesystem group change policy -## @param podSecurityContext.sysctls Set kernel settings using the sysctl interface -## @param podSecurityContext.supplementalGroups Set filesystem extra groups -## @param podSecurityContext.fsGroup User ID for the container -## -podSecurityContext: - enabled: true - fsGroupChangePolicy: Always - sysctls: [] - supplementalGroups: [] - fsGroup: 1001 -## phpMyAdmin containers' Security Context (only main container) -## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container -## @param containerSecurityContext.enabled Enabled containers' Security Context -## @param containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container -## @param containerSecurityContext.runAsUser Set containers' Security Context runAsUser -## @param containerSecurityContext.runAsGroup Set containers' Security Context runAsGroup -## @param containerSecurityContext.runAsNonRoot Set container's Security Context runAsNonRoot -## @param containerSecurityContext.privileged Set container's Security Context privileged -## @param containerSecurityContext.readOnlyRootFilesystem Set container's Security Context readOnlyRootFilesystem -## @param containerSecurityContext.allowPrivilegeEscalation Set container's Security Context allowPrivilegeEscalation -## @param containerSecurityContext.capabilities.drop List of capabilities to be dropped -## @param containerSecurityContext.seccompProfile.type Set container's Security Context seccomp profile -## -containerSecurityContext: - enabled: true - seLinuxOptions: null - runAsUser: 1001 - runAsGroup: 1001 - runAsNonRoot: true - privileged: false - readOnlyRootFilesystem: true - allowPrivilegeEscalation: false - capabilities: - drop: ["ALL"] - seccompProfile: - type: "RuntimeDefault" -## phpMyAdmin number of pod replicas -## @param replicas Number of replicas -replicas: 1 -## phpMyAdmin containers' resource requests and limits -## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ -## We usually recommend not to specify default resources and to leave this as a conscious -## choice for the user. This also increases chances charts run on environments with little -## resources, such as Minikube. If you do want to specify resources, uncomment the following -## lines, adjust them as necessary, and remove the curly braces after 'resources:'. -## @param resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if resources is set (resources is recommended for production). -## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 -## -resourcesPreset: "micro" -## @param resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) -## Example: -## resources: -## requests: -## cpu: 2 -## memory: 512Mi -## limits: -## cpu: 3 -## memory: 1024Mi -## -resources: {} -## phpMyAdmin containers' startup probe. Evaluated as a template. -## ref: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#container-probes -## @param startupProbe.enabled Enable startupProbe -## @param startupProbe.httpGet.path Request path for startupProbe -## @param startupProbe.httpGet.port Port for startupProbe -## @param startupProbe.initialDelaySeconds Initial delay seconds for startupProbe -## @param startupProbe.periodSeconds Period seconds for startupProbe -## @param startupProbe.timeoutSeconds Timeout seconds for startupProbe -## @param startupProbe.failureThreshold Failure threshold for startupProbe -## @param startupProbe.successThreshold Success threshold for startupProbe -## -startupProbe: - enabled: false - initialDelaySeconds: 30 - timeoutSeconds: 30 - periodSeconds: 10 - successThreshold: 1 - failureThreshold: 6 - httpGet: - path: / - port: http -## phpMyAdmin containers' liveness probe. Evaluated as a template. -## ref: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#container-probes -## @param livenessProbe.enabled Enable livenessProbe -## @param livenessProbe.tcpSocket.port Port for livenessProbe -## @param livenessProbe.initialDelaySeconds Initial delay seconds for livenessProbe -## @param livenessProbe.periodSeconds Period seconds for livenessProbe -## @param livenessProbe.timeoutSeconds Timeout seconds for livenessProbe -## @param livenessProbe.failureThreshold Failure threshold for livenessProbe -## @param livenessProbe.successThreshold Success threshold for livenessProbe -## -livenessProbe: - enabled: true - initialDelaySeconds: 30 - timeoutSeconds: 30 - periodSeconds: 10 - successThreshold: 1 - failureThreshold: 6 - tcpSocket: - port: http -## phpMyAdmin containers' readiness probes. Evaluated as a template. -## ref: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#container-probes -## @param readinessProbe.enabled Enable readinessProbe -## @param readinessProbe.httpGet.path Request path for readinessProbe -## @param readinessProbe.httpGet.port Port for readinessProbe -## @param readinessProbe.initialDelaySeconds Initial delay seconds for readinessProbe -## @param readinessProbe.periodSeconds Period seconds for readinessProbe -## @param readinessProbe.timeoutSeconds Timeout seconds for readinessProbe -## @param readinessProbe.failureThreshold Failure threshold for readinessProbe -## @param readinessProbe.successThreshold Success threshold for readinessProbe -## -readinessProbe: - enabled: true - initialDelaySeconds: 30 - timeoutSeconds: 30 - periodSeconds: 10 - successThreshold: 1 - failureThreshold: 6 - httpGet: - path: / - port: http -## @param customStartupProbe Override default startup probe -## -customStartupProbe: {} -## @param customLivenessProbe Override default liveness probe -## -customLivenessProbe: {} -## @param customReadinessProbe Override default readiness probe -## -customReadinessProbe: {} -## @param podLabels Extra labels for PhpMyAdmin pods -## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/ -## -podLabels: {} -## @param podAnnotations Annotations for PhpMyAdmin pods -## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ -## -podAnnotations: {} -## @param podAffinityPreset Pod affinity preset. Ignored if `affinity` is set. Allowed values: `soft` or `hard` -## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity -## -podAffinityPreset: "" -## @param podAntiAffinityPreset Pod anti-affinity preset. Ignored if `affinity` is set. Allowed values: `soft` or `hard` -## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity -## -podAntiAffinityPreset: soft -## Node affinity preset -## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity -## -nodeAffinityPreset: - ## @param nodeAffinityPreset.type Node affinity preset type. Ignored if `affinity` is set. Allowed values: `soft` or `hard` - ## - type: "" - ## @param nodeAffinityPreset.key Node label key to match. Ignored if `affinity` is set. - ## E.g. - ## key: "kubernetes.io/e2e-az-name" - ## - key: "" - ## @param nodeAffinityPreset.values Node label values to match. Ignored if `affinity` is set. - ## E.g. - ## values: - ## - e2e-az1 - ## - e2e-az2 - ## - values: [] -## @param affinity Affinity for pod assignment. Evaluated as a template. -## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity -## Note: podAffinityPreset, podAntiAffinityPreset, and nodeAffinityPreset will be ignored when it's set -## -affinity: {} -## @param nodeSelector Node labels for pod assignment. Evaluated as a template. -## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/ -## -nodeSelector: {} -## @param tolerations Tolerations for pod assignment. Evaluated as a template. -## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ -## -tolerations: [] -## @param priorityClassName phpmyadmin pods' priorityClassName -## -priorityClassName: "" -## @param schedulerName Name of the k8s scheduler (other than default) -## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ -## -schedulerName: "" -## @param topologySpreadConstraints Topology Spread Constraints for pod assignment -## https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/ -## The value is evaluated as a template -## -topologySpreadConstraints: [] -## @param extraVolumes Optionally specify extra list of additional volumes for PhpMyAdmin pods -## -extraVolumes: [] -## @param extraVolumeMounts Optionally specify extra list of additional volumeMounts for PhpMyAdmin container(s) -## -extraVolumeMounts: [] -## @param initContainers Add init containers to the PhpMyAdmin pods -## Example: -## initContainers: -## - name: your-image-name -## image: your-image -## imagePullPolicy: Always -## ports: -## - name: portname -## containerPort: 1234 -## -initContainers: [] -## Pod Disruption Budget configuration -## ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb -## @param pdb.create Enable/disable a Pod Disruption Budget creation -## @param pdb.minAvailable Minimum number/percentage of pods that should remain scheduled -## @param pdb.maxUnavailable Maximum number/percentage of pods that may be made unavailable. Defaults to `1` if both `pdb.minAvailable` and `pdb.maxUnavailable` are empty. -## -pdb: - create: true - minAvailable: "" - maxUnavailable: "" -## @param sidecars Add sidecar containers to the PhpMyAdmin pods -## Example: -## sidecars: -## - name: your-image-name -## image: your-image -## imagePullPolicy: Always -## ports: -## - name: portname -## containerPort: 1234 -## -sidecars: [] -## @section Traffic Exposure parameters - -## Service configuration -## -service: - ## @param service.type Kubernetes Service type - ## - type: ClusterIP - ## @param service.ports.http Service HTTP port - ## @param service.ports.https Service HTTPS port - ## - ports: - http: 80 - https: 443 - ## Specify the nodePort values for the LoadBalancer and NodePort service types - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport - ## @param service.nodePorts.http Kubernetes http node port - ## @param service.nodePorts.https Kubernetes https node port - ## - nodePorts: - http: "" - https: "" - ## @param service.clusterIP PhpMyAdmin service clusterIP IP - ## e.g: - ## clusterIP: None - ## - clusterIP: "" - ## @param service.loadBalancerIP Load balancer IP for the phpMyAdmin Service (optional, cloud specific) - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-loadbalancer - ## - loadBalancerIP: "" - ## @param service.loadBalancerSourceRanges Addresses that are allowed when service is LoadBalancer - ## https://kubernetes.io/docs/tasks/access-application-cluster/configure-cloud-provider-firewall/#restrict-access-for-loadbalancer-service - ## Example: - ## loadBalancerSourceRanges: - ## - 10.10.10.0/24 - ## - loadBalancerSourceRanges: [] - ## @param service.externalTrafficPolicy Enable client source IP preservation - ## ref https://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer/#preserving-the-client-source-ip - ## - externalTrafficPolicy: Cluster - ## @param service.extraPorts Extra ports to expose (normally used with the `sidecar` value) - ## - extraPorts: [] - ## @param service.annotations Provide any additional annotations that may be required for the PhpMyAdmin service - ## - annotations: {} - ## @param service.sessionAffinity Session Affinity for Kubernetes service, can be "None" or "ClientIP" - ## If "ClientIP", consecutive client requests will be directed to the same Pod - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies - ## - sessionAffinity: None - ## @param service.sessionAffinityConfig Additional settings for the sessionAffinity - ## sessionAffinityConfig: - ## clientIP: - ## timeoutSeconds: 300 - ## - sessionAffinityConfig: {} -## Ingress configuration -## -ingress: - ## @param ingress.enabled Set to true to enable ingress record generation - ## - enabled: false - ## @param ingress.apiVersion Force Ingress API version (automatically detected if not set) - ## - apiVersion: "" - ## DEPRECATED: Use ingress.annotations instead of ingress.certManager - ## certManager: false - ## - - ## @param ingress.hostname When the ingress is enabled, a host pointing to this will be created - ## - hostname: phpmyadmin.local - ## @param ingress.pathType Ingress path type - ## - pathType: ImplementationSpecific - ## @param ingress.path Default path for the ingress record - ## NOTE: You may need to set this to '/*' in order to use this with ALB ingress controllers - ## - path: / - ## @param ingress.extraPaths An array with additional arbitrary paths that may need to be added to the ingress under the main host - ## e.g: - ## extraPaths: - ## - path: /* - ## backend: - ## serviceName: ssl-redirect - ## servicePort: use-annotation - ## - extraPaths: [] - ## @param ingress.annotations Additional annotations for the Ingress resource. To enable certificate autogeneration, place here your cert-manager annotations. - ## For a full list of possible ingress annotations, please see - ## ref: https://github.com/kubernetes/ingress-nginx/blob/main/docs/user-guide/nginx-configuration/annotations.md - ## Use this parameter to set the required annotations for cert-manager, see - ## ref: https://cert-manager.io/docs/usage/ingress/#supported-annotations - ## - ## e.g: - ## annotations: - ## kubernetes.io/ingress.class: nginx - ## cert-manager.io/cluster-issuer: cluster-issuer-name - ## - annotations: {} - ## @param ingress.tls Enable TLS configuration for the hostname defined at `ingress.hostname` parameter - ## TLS certificates will be retrieved from a TLS secret with name: {{- printf "%s-tls" .Values.ingress.hostname }} - ## You can use the ingress.secrets parameter to create this TLS secret, relay on cert-manager to create it, or - ## let the chart create self-signed certificates for you - ## - tls: false - ## @param ingress.extraHosts The list of additional hostnames to be covered with this ingress record. - ## Most likely the hostname above will be enough, but in the event more hosts are needed, this is an array - ## Example: - ## extraHosts: - ## - name: phpmyadmin.local - ## path: / - ## - extraHosts: [] - ## @param ingress.extraTls The tls configuration for additional hostnames to be covered with this ingress record. - ## see: https://kubernetes.io/docs/concepts/services-networking/ingress/#tls - ## Example: - ## extraTls: - ## - hosts: - ## - phpmyadmin.local - ## secretName: phpmyadmin.local-tls - ## - extraTls: [] - ## @param ingress.secrets If you're providing your own certificates and want to manage the secret via helm, - ## please use this to add the certificates as secrets key and certificate should start with - ## -----BEGIN CERTIFICATE----- or -----BEGIN RSA PRIVATE KEY----- - ## name should line up with a secretName set further up - ## - ## If it is not set and you're using cert-manager, this is unneeded, as it will create the secret for you - ## If it is not set and you're NOT using cert-manager either, self-signed certificates will be created - ## It is also possible to create and manage the certificates outside of this helm chart - ## Please see README.md for more information - ## - ## Example - ## secrets: - ## - name: phpmyadmin.local-tls - ## key: "" - ## certificate: "" - ## - secrets: [] - ## @param ingress.existingSecretName If you're providing your own certificate and want to manage the secret yourself, - ## please provide the name of the secret with this parameter. This secret will then be used for tls termination. - ## It has higher priority than the cert-manager or the generation of the certificate from the chart. - ## - ## Example: - ## existingSecretName: "byo-phpmyadmin-tls" - ## - existingSecretName: "" - ## @param ingress.ingressClassName IngressClass that will be be used to implement the Ingress (Kubernetes 1.18+) - ## This is supported in Kubernetes 1.18+ and required if you have more than one IngressClass marked as the default for your cluster . - ## ref: https://kubernetes.io/blog/2020/04/02/improvements-to-the-ingress-api-in-kubernetes-1.18/ - ## - ingressClassName: "" - ## @param ingress.extraRules Additional rules to be covered with this ingress record - ## ref: https://kubernetes.io/docs/concepts/services-networking/ingress/#ingress-rules - ## e.g: - ## extraRules: - ## - host: phpmyadmin.local - ## http: - ## path: / - ## backend: - ## service: - ## name: phpmyadmin-svc - ## port: - ## name: http - ## - extraRules: [] -## @section Database parameters - -## Database configuration -## -db: - ## @param db.allowArbitraryServer Enable connection to arbitrary MySQL server - ## If you do not want the user to be able to specify an arbitrary MySQL server at login time, set this to false - ## - allowArbitraryServer: true - ## @param db.port Database port to use to connect - ## - port: 3306 - ## @param db.chartName Database suffix if included in the same release - ## If you are deploying phpMyAdmin as part of a release and the database is part - ## of the release, you can pass a suffix that will be used to find the database - ## in releasename-dbSuffix. Please note that this setting precedes db.host - ## e.g: - ## chartName: mariadb - ## - chartName: "" - ## @param db.host Database Hostname. Ignored when `db.chartName` is set. - ## e.g: - ## host: foo - ## - host: "" - ## @param db.bundleTestDB Deploy a MariaDB instance for testing purposes - ## - bundleTestDB: false - ## @param db.enableSsl Enable SSL for the connection between phpMyAdmin and the database - ## - enableSsl: false - ssl: - ## @param db.ssl.clientKey Client key file when using SSL - ## - clientKey: "" - ## @param db.ssl.clientCertificate Client certificate file when using SSL - ## - clientCertificate: "" - ## @param db.ssl.caCertificate CA file when using SSL - ## - caCertificate: "" - ## @param db.ssl.ciphers List of allowable ciphers for connections when using SSL - ## - ciphers: [] - ## @param db.ssl.verify Enable SSL certificate validation - ## - verify: true -## @param mariadb MariaDB chart configuration -## https://github.com/bitnami/charts/blob/main/bitnami/mariadb/values.yaml -## -mariadb: {} -## @section Other Parameters - -## Service account for PhpMyAdmin to use. -## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/ -## -serviceAccount: - ## @param serviceAccount.create Enable creation of ServiceAccount for PhpMyAdmin pod - ## - create: true - ## @param serviceAccount.name The name of the ServiceAccount to use. - ## If not set and create is true, a name is generated using the common.names.fullname template - ## - name: "" - ## @param serviceAccount.automountServiceAccountToken Allows auto mount of ServiceAccountToken on the serviceAccount created - ## Can be set to false if pods using this serviceAccount do not need to use K8s API - ## - automountServiceAccountToken: false - ## @param serviceAccount.annotations Additional custom annotations for the ServiceAccount - ## - annotations: {} -## @section Metrics parameters - -## Prometheus Exporter / Metrics -## -metrics: - ## @param metrics.enabled Start a side-car prometheus exporter - ## - enabled: false - ## @param metrics.image.registry [default: REGISTRY_NAME] Apache exporter image registry - ## @param metrics.image.repository [default: REPOSITORY_NAME/apache-exporter] Apache exporter image repository - ## @skip metrics.image.tag Apache exporter image tag (immutable tags are recommended) - ## @param metrics.image.digest Apache exporter image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag - ## @param metrics.image.pullPolicy Image pull policy - ## @param metrics.image.pullSecrets Specify docker-registry secret names as an array - ## - image: - registry: docker.io - repository: bitnami/apache-exporter - tag: 1.0.8-debian-12-r9 - digest: "" - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets. - ## Secrets must be manually created in the namespace. - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## Example: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## @param metrics.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if metrics.resources is set (metrics.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "nano" - ## @param metrics.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## - resources: {} - ## Prometheus Exporter service configuration - ## - service: - ## @param metrics.service.type Prometheus metrics service type - ## - type: ClusterIP - ## @param metrics.service.port Prometheus metrics service port - ## - port: 9117 - ## @param metrics.service.annotations [object] Annotations for Prometheus metrics service - ## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ - annotations: - prometheus.io/scrape: "true" - prometheus.io/port: "{{ .Values.metrics.service.port }}" - ## @param metrics.service.clusterIP phpmyadmin service Cluster IP - ## e.g.: - ## clusterIP: None - ## - clusterIP: "" - ## @param metrics.service.loadBalancerIP Load Balancer IP if the Prometheus metrics server type is `LoadBalancer` - ## Use serviceLoadBalancerIP to request a specific static IP, otherwise leave blank - ## - loadBalancerIP: "" - ## @param metrics.service.loadBalancerSourceRanges phpmyadmin service Load Balancer sources - ## ref: https://kubernetes.io/docs/tasks/access-application-cluster/configure-cloud-provider-firewall/#restrict-access-for-loadbalancer-service - ## e.g: - ## loadBalancerSourceRanges: - ## - 10.10.10.0/24 - ## - loadBalancerSourceRanges: [] - ## @param metrics.service.externalTrafficPolicy phpmyadmin service external traffic policy - ## ref http://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer/#preserving-the-client-source-ip - ## - externalTrafficPolicy: Cluster - ## @param metrics.service.sessionAffinity Session Affinity for Kubernetes service, can be "None" or "ClientIP" - ## If "ClientIP", consecutive client requests will be directed to the same Pod - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies - ## - sessionAffinity: None - ## @param metrics.service.sessionAffinityConfig Additional settings for the sessionAffinity - ## sessionAffinityConfig: - ## clientIP: - ## timeoutSeconds: 300 - ## - sessionAffinityConfig: {} - ## Prometheus Service Monitor - ## ref: https://github.com/coreos/prometheus-operator - ## https://github.com/coreos/prometheus-operator/blob/master/Documentation/api.md#endpoint - ## - serviceMonitor: - ## @param metrics.serviceMonitor.enabled Create ServiceMonitor Resource for scraping metrics using PrometheusOperator - ## - enabled: false - ## @param metrics.serviceMonitor.namespace Specify the namespace in which the serviceMonitor resource will be created - ## - namespace: "" - ## @param metrics.serviceMonitor.jobLabel The name of the label on the target service to use as the job name in prometheus. - ## - jobLabel: "" - ## @param metrics.serviceMonitor.interval Specify the interval at which metrics should be scraped - ## - interval: 30s - ## @param metrics.serviceMonitor.scrapeTimeout Specify the timeout after which the scrape is ended - ## e.g: - ## scrapeTimeout: 30s - ## - scrapeTimeout: "" - ## @param metrics.serviceMonitor.relabelings RelabelConfigs to apply to samples before scraping - ## ref: https://github.com/coreos/prometheus-operator/blob/master/Documentation/api.md#relabelconfig - ## - relabelings: [] - ## @param metrics.serviceMonitor.metricRelabelings Specify Metric Relabelings to add to the scrape endpoint - ## ref: https://github.com/coreos/prometheus-operator/blob/master/Documentation/api.md#relabelconfig - ## - metricRelabelings: [] - ## @param metrics.serviceMonitor.labels Extra labels for the ServiceMonitor - ## - labels: {} - ## @param metrics.serviceMonitor.honorLabels Specify honorLabels parameter to add the scrape endpoint - ## - honorLabels: false - ## @param metrics.serviceMonitor.selector ServiceMonitor selector labels - ## ref: https://github.com/bitnami/charts/tree/main/bitnami/prometheus-operator#prometheus-configuration - ## - ## selector: - ## prometheus: my-prometheus - ## - selector: {} -## @section NetworkPolicy parameters - -## Network Policy configuration -## ref: https://kubernetes.io/docs/concepts/services-networking/network-policies/ -## -networkPolicy: - ## @param networkPolicy.enabled Specifies whether a NetworkPolicy should be created - ## - enabled: true - ## @param networkPolicy.allowExternal Don't require server label for connections - ## The Policy model to apply. When set to false, only pods with the correct - ## server label will have network access to the ports server is listening - ## on. When true, server will accept connections from any source - ## (with the correct destination port). - ## - allowExternal: true - ## @param networkPolicy.allowExternalEgress Allow the pod to access any range of port and all destinations. - ## - allowExternalEgress: true - ## @param networkPolicy.extraIngress [array] Add extra ingress rules to the NetworkPolicy - ## e.g: - ## extraIngress: - ## - ports: - ## - port: 1234 - ## from: - ## - podSelector: - ## - matchLabels: - ## - role: frontend - ## - podSelector: - ## - matchExpressions: - ## - key: role - ## operator: In - ## values: - ## - frontend - extraIngress: [] - ## @param networkPolicy.extraEgress [array] Add extra ingress rules to the NetworkPolicy - ## e.g: - ## extraEgress: - ## - ports: - ## - port: 1234 - ## to: - ## - podSelector: - ## - matchLabels: - ## - role: frontend - ## - podSelector: - ## - matchExpressions: - ## - key: role - ## operator: In - ## values: - ## - frontend - ## - extraEgress: [] - ## @param networkPolicy.ingressNSMatchLabels [object] Labels to match to allow traffic from other namespaces - ## @param networkPolicy.ingressNSPodMatchLabels [object] Pod labels to match to allow traffic from other namespaces - ## - ingressNSMatchLabels: {} - ingressNSPodMatchLabels: {} diff --git a/charts/phpmyadmin/values-overrides.yaml b/charts/phpmyadmin/values-overrides.yaml deleted file mode 100644 index f6f6b59..0000000 --- a/charts/phpmyadmin/values-overrides.yaml +++ /dev/null @@ -1,6 +0,0 @@ -fullnameOverride: phpmyadmin -db: - allowArbitraryServer: false - port: 3306 - host: "mariadb.futureporn.svc.cluster.local" - bundleTestDB: false \ No newline at end of file diff --git a/charts/postgresql/postgresql/.helmignore b/charts/postgresql/postgresql/.helmignore deleted file mode 100644 index 207983f..0000000 --- a/charts/postgresql/postgresql/.helmignore +++ /dev/null @@ -1,25 +0,0 @@ -# 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 -# img folder -img/ -# Changelog -CHANGELOG.md diff --git a/charts/postgresql/postgresql/Chart.lock b/charts/postgresql/postgresql/Chart.lock deleted file mode 100644 index ab93b70..0000000 --- a/charts/postgresql/postgresql/Chart.lock +++ /dev/null @@ -1,6 +0,0 @@ -dependencies: -- name: common - repository: oci://registry-1.docker.io/bitnamicharts - version: 2.20.5 -digest: sha256:5b98791747a148b9d4956b81bb8635f49a0ae831869d700d52e514b8fd1a2445 -generated: "2024-07-16T12:16:35.737563+02:00" diff --git a/charts/postgresql/postgresql/Chart.yaml b/charts/postgresql/postgresql/Chart.yaml deleted file mode 100644 index f0686fd..0000000 --- a/charts/postgresql/postgresql/Chart.yaml +++ /dev/null @@ -1,37 +0,0 @@ -annotations: - category: Database - images: | - - name: os-shell - image: docker.io/bitnami/os-shell:12-debian-12-r24 - - name: postgres-exporter - image: docker.io/bitnami/postgres-exporter:0.15.0-debian-12-r36 - - name: postgresql - image: docker.io/bitnami/postgresql:16.3.0-debian-12-r19 - licenses: Apache-2.0 -apiVersion: v2 -appVersion: 16.3.0 -dependencies: -- name: common - repository: oci://registry-1.docker.io/bitnamicharts - tags: - - bitnami-common - version: 2.x.x -description: PostgreSQL (Postgres) is an open source object-relational database known - for reliability and data integrity. ACID-compliant, it supports foreign keys, joins, - views, triggers and stored procedures. -home: https://bitnami.com -icon: https://bitnami.com/assets/stacks/postgresql/img/postgresql-stack-220x234.png -keywords: -- postgresql -- postgres -- database -- sql -- replication -- cluster -maintainers: -- name: Broadcom, Inc. All Rights Reserved. - url: https://github.com/bitnami/charts -name: postgresql -sources: -- https://github.com/bitnami/charts/tree/main/bitnami/postgresql -version: 15.5.17 diff --git a/charts/postgresql/postgresql/README.md b/charts/postgresql/postgresql/README.md deleted file mode 100644 index 56352af..0000000 --- a/charts/postgresql/postgresql/README.md +++ /dev/null @@ -1,1021 +0,0 @@ - - -# Bitnami package for PostgreSQL - -PostgreSQL (Postgres) is an open source object-relational database known for reliability and data integrity. ACID-compliant, it supports foreign keys, joins, views, triggers and stored procedures. - -[Overview of PostgreSQL](http://www.postgresql.org) - -Trademarks: This software listing is packaged by Bitnami. The respective trademarks mentioned in the offering are owned by the respective companies, and use of them does not imply any affiliation or endorsement. - -## TL;DR - -```console -helm install my-release oci://registry-1.docker.io/bitnamicharts/postgresql -``` - -Looking to use PostgreSQL in production? Try [VMware Tanzu Application Catalog](https://bitnami.com/enterprise), the commercial edition of the Bitnami catalog. - -## Introduction - -This chart bootstraps a [PostgreSQL](https://github.com/bitnami/containers/tree/main/bitnami/postgresql) deployment on a [Kubernetes](https://kubernetes.io) cluster using the [Helm](https://helm.sh) package manager. - -For HA, please see [this repo](https://github.com/bitnami/charts/tree/main/bitnami/postgresql-ha) - -Bitnami charts can be used with [Kubeapps](https://kubeapps.dev/) for deployment and management of Helm Charts in clusters. - -## Prerequisites - -- Kubernetes 1.23+ -- Helm 3.8.0+ -- PV provisioner support in the underlying infrastructure - -## Installing the Chart - -To install the chart with the release name `my-release`: - -```console -helm install my-release oci://REGISTRY_NAME/REPOSITORY_NAME/postgresql -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -The command deploys PostgreSQL on the Kubernetes cluster in the default configuration. The [Parameters](#parameters) section lists the parameters that can be configured during installation. - -> **Tip**: List all releases using `helm list` - -## Configuration and installation details - -### Resource requests and limits - -Bitnami charts allow setting resource requests and limits for all containers inside the chart deployment. These are inside the `resources` value (check parameter table). Setting requests is essential for production workloads and these should be adapted to your specific use case. - -To make this process easier, the chart contains the `resourcesPreset` values, which automatically sets the `resources` section according to different presets. Check these presets in [the bitnami/common chart](https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15). However, in production workloads using `resourcePreset` is discouraged as it may not fully adapt to your specific needs. Find more information on container resource management in the [official Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/). - -### [Rolling VS Immutable tags](https://docs.vmware.com/en/VMware-Tanzu-Application-Catalog/services/tutorials/GUID-understand-rolling-tags-containers-index.html) - -It is strongly recommended to use immutable tags in a production environment. This ensures your deployment does not change automatically if the same tag is updated with a different image. - -Bitnami will release a new chart updating its containers if a new version of the main container, significant changes, or critical vulnerabilities exist. - -### Customizing primary and read replica services in a replicated configuration - -At the top level, there is a service object which defines the services for both primary and readReplicas. For deeper customization, there are service objects for both the primary and read types individually. This allows you to override the values in the top level service object so that the primary and read can be of different service types and with different clusterIPs / nodePorts. Also in the case you want the primary and read to be of type nodePort, you will need to set the nodePorts to different values to prevent a collision. The values that are deeper in the primary.service or readReplicas.service objects will take precedence over the top level service object. - -### Use a different PostgreSQL version - -To modify the application version used in this chart, specify a different version of the image using the `image.tag` parameter and/or a different repository using the `image.repository` parameter. - -### LDAP - -LDAP support can be enabled in the chart by specifying the `ldap.` parameters while creating a release. The following parameters should be configured to properly enable the LDAP support in the chart. - -- **ldap.enabled**: Enable LDAP support. Defaults to `false`. -- **ldap.uri**: LDAP URL beginning in the form `ldap[s]://:`. No defaults. -- **ldap.base**: LDAP base DN. No defaults. -- **ldap.binddn**: LDAP bind DN. No defaults. -- **ldap.bindpw**: LDAP bind password. No defaults. -- **ldap.bslookup**: LDAP base lookup. No defaults. -- **ldap.nss_initgroups_ignoreusers**: LDAP ignored users. `root,nslcd`. -- **ldap.scope**: LDAP search scope. No defaults. -- **ldap.tls_reqcert**: LDAP TLS check on server certificates. No defaults. - -For example: - -```text -ldap.enabled="true" -ldap.uri="ldap://my_ldap_server" -ldap.base="dc=example\,dc=org" -ldap.binddn="cn=admin\,dc=example\,dc=org" -ldap.bindpw="admin" -ldap.bslookup="ou=group-ok\,dc=example\,dc=org" -ldap.nss_initgroups_ignoreusers="root\,nslcd" -ldap.scope="sub" -ldap.tls_reqcert="demand" -``` - -Next, login to the PostgreSQL server using the `psql` client and add the PAM authenticated LDAP users. - -> Note: Parameters including commas must be escaped as shown in the above example. - -### postgresql.conf / pg_hba.conf files as configMap - -This helm chart also supports to customize the PostgreSQL configuration file. You can add additional PostgreSQL configuration parameters using the `primary.extendedConfiguration`/`readReplicas.extendedConfiguration` parameters as a string. Alternatively, to replace the entire default configuration use `primary.configuration`. - -You can also add a custom pg_hba.conf using the `primary.pgHbaConfiguration` parameter. - -In addition to these options, you can also set an external ConfigMap with all the configuration files. This is done by setting the `primary.existingConfigmap` parameter. Note that this will override the two previous options. - -### Initialize a fresh instance - -The [Bitnami PostgreSQL](https://github.com/bitnami/containers/tree/main/bitnami/postgresql) image allows you to use your custom scripts to initialize a fresh instance. In order to execute the scripts, you can specify custom scripts using the `primary.initdb.scripts` parameter as a string. - -In addition, you can also set an external ConfigMap with all the initialization scripts. This is done by setting the `primary.initdb.scriptsConfigMap` parameter. Note that this will override the two previous options. If your initialization scripts contain sensitive information such as credentials or passwords, you can use the `primary.initdb.scriptsSecret` parameter. - -The allowed extensions are `.sh`, `.sql` and `.sql.gz`. - -### Securing traffic using TLS - -TLS support can be enabled in the chart by specifying the `tls.` parameters while creating a release. The following parameters should be configured to properly enable the TLS support in the chart: - -- `tls.enabled`: Enable TLS support. Defaults to `false` -- `tls.certificatesSecret`: Name of an existing secret that contains the certificates. No defaults. -- `tls.certFilename`: Certificate filename. No defaults. -- `tls.certKeyFilename`: Certificate key filename. No defaults. - -For example: - -- First, create the secret with the cetificates files: - - ```console - kubectl create secret generic certificates-tls-secret --from-file=./cert.crt --from-file=./cert.key --from-file=./ca.crt - ``` - -- Then, use the following parameters: - - ```console - volumePermissions.enabled=true - tls.enabled=true - tls.certificatesSecret="certificates-tls-secret" - tls.certFilename="cert.crt" - tls.certKeyFilename="cert.key" - ``` - - > Note TLS and VolumePermissions: PostgreSQL requires certain permissions on sensitive files (such as certificate keys) to start up. Due to an on-going [issue](https://github.com/kubernetes/kubernetes/issues/57923) regarding kubernetes permissions and the use of `containerSecurityContext.runAsUser`, you must enable `volumePermissions` to ensure everything works as expected. - -### Sidecars - -If you need additional containers to run within the same pod as PostgreSQL (e.g. an additional metrics or logging exporter), you can do so via the `sidecars` config parameter. Simply define your container according to the Kubernetes container spec. - -```yaml -# For the PostgreSQL primary -primary: - sidecars: - - name: your-image-name - image: your-image - imagePullPolicy: Always - ports: - - name: portname - containerPort: 1234 -# For the PostgreSQL replicas -readReplicas: - sidecars: - - name: your-image-name - image: your-image - imagePullPolicy: Always - ports: - - name: portname - containerPort: 1234 -``` - -### Metrics - -The chart optionally can start a metrics exporter for [prometheus](https://prometheus.io). The metrics endpoint (port 9187) is not exposed and it is expected that the metrics are collected from inside the k8s cluster using something similar as the described in the [example Prometheus scrape configuration](https://github.com/prometheus/prometheus/blob/master/documentation/examples/prometheus-kubernetes.yml). - -The exporter allows to create custom metrics from additional SQL queries. See the Chart's `values.yaml` for an example and consult the [exporters documentation](https://github.com/wrouesnel/postgres_exporter#adding-new-metrics-via-a-config-file) for more details. - -### Use of global variables - -In more complex scenarios, we may have the following tree of dependencies - -```text - +--------------+ - | | - +------------+ Chart 1 +-----------+ - | | | | - | --------+------+ | - | | | - | | | - | | | - | | | - v v v -+-------+------+ +--------+------+ +--------+------+ -| | | | | | -| PostgreSQL | | Sub-chart 1 | | Sub-chart 2 | -| | | | | | -+--------------+ +---------------+ +---------------+ -``` - -The three charts below depend on the parent chart Chart 1. However, subcharts 1 and 2 may need to connect to PostgreSQL as well. In order to do so, subcharts 1 and 2 need to know the PostgreSQL credentials, so one option for deploying could be deploy Chart 1 with the following parameters: - -```text -postgresql.auth.username=testuser -subchart1.postgresql.auth.username=testuser -subchart2.postgresql.auth.username=testuser -postgresql.auth.password=testpass -subchart1.postgresql.auth.password=testpass -subchart2.postgresql.auth.password=testpass -postgresql.auth.database=testdb -subchart1.postgresql.auth.database=testdb -subchart2.postgresql.auth.database=testdb -``` - -If the number of dependent sub-charts increases, installing the chart with parameters can become increasingly difficult. An alternative would be to set the credentials using global variables as follows: - -```text -global.postgresql.auth.username=testuser -global.postgresql.auth.password=testpass -global.postgresql.auth.database=testdb -``` - -This way, the credentials will be available in all of the subcharts. - -### Backup and restore PostgreSQL deployments - -To back up and restore Bitnami PostgreSQL Helm chart deployments on Kubernetes, you need to back up the persistent volumes from the source deployment and attach them to a new deployment using [Velero](https://velero.io/), a Kubernetes backup/restore tool. - -These are the steps you will usually follow to back up and restore your PostgreSQL cluster data: - -- Install Velero on the source and destination clusters. -- Use Velero to back up the PersistentVolumes (PVs) used by the deployment on the source cluster. -- Use Velero to restore the backed-up PVs on the destination cluster. -- Create a new deployment on the destination cluster with the same chart, deployment name, credentials and other parameters as the original. This new deployment will use the restored PVs and hence the original data. - -Refer to our detailed [tutorial on backing up and restoring PostgreSQL deployments on Kubernetes](https://docs.vmware.com/en/VMware-Tanzu-Application-Catalog/services/tutorials/GUID-migrate-data-tac-velero-index.html) for more information. - -### NetworkPolicy - -To enable network policy for PostgreSQL, install [a networking plugin that implements the Kubernetes NetworkPolicy spec](https://kubernetes.io/docs/tasks/administer-cluster/declare-network-policy#before-you-begin), and set `networkPolicy.enabled` to `true`. - -For Kubernetes v1.5 & v1.6, you must also turn on NetworkPolicy by setting the DefaultDeny namespace annotation. Note: this will enforce policy for _all_ pods in the namespace: - -```console -kubectl annotate namespace default "net.beta.kubernetes.io/network-policy={\"ingress\":{\"isolation\":\"DefaultDeny\"}}" -``` - -With NetworkPolicy enabled, traffic will be limited to just port 5432. - -For more precise policy, set `networkPolicy.allowExternal=false`. This will only allow pods with the generated client label to connect to PostgreSQL. -This label will be displayed in the output of a successful install. - -### Differences between Bitnami PostgreSQL image and [Docker Official](https://hub.docker.com/_/postgres) image - -- The Docker Official PostgreSQL image does not support replication. If you pass any replication environment variable, this would be ignored. The only environment variables supported by the Docker Official image are POSTGRES_USER, POSTGRES_DB, POSTGRES_PASSWORD, POSTGRES_INITDB_ARGS, POSTGRES_INITDB_WALDIR and PGDATA. All the remaining environment variables are specific to the Bitnami PostgreSQL image. -- The Bitnami PostgreSQL image is non-root by default. This requires that you run the pod with `securityContext` and updates the permissions of the volume with an `initContainer`. A key benefit of this configuration is that the pod follows security best practices and is prepared to run on Kubernetes distributions with hard security constraints like OpenShift. -- For OpenShift up to 4.10, let set the volume permissions, security context, runAsUser and fsGroup automatically by OpenShift and disable the predefined settings of the helm chart: primary.securityContext.enabled=false,primary.containerSecurityContext.enabled=false,volumePermissions.enabled=false,shmVolume.enabled=false -- For OpenShift 4.11 and higher, let set OpenShift the runAsUser and fsGroup automatically. Configure the pod and container security context to restrictive defaults and disable the volume permissions setup: primary. - podSecurityContext.fsGroup=null,primary.podSecurityContext.seccompProfile.type=RuntimeDefault,primary.containerSecurityContext.runAsUser=null,primary.containerSecurityContext.allowPrivilegeEscalation=false,primary.containerSecurityContext.runAsNonRoot=true,primary.containerSecurityContext.seccompProfile.type=RuntimeDefault,primary.containerSecurityContext.capabilities.drop=['ALL'],volumePermissions.enabled=false,shmVolume.enabled=false - -### Setting Pod's affinity - -This chart allows you to set your custom affinity using the `XXX.affinity` parameter(s). Find more information about Pod's affinity in the [kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity). - -As an alternative, you can use of the preset configurations for pod affinity, pod anti-affinity, and node affinity available at the [bitnami/common](https://github.com/bitnami/charts/tree/main/bitnami/common#affinities) chart. To do so, set the `XXX.podAffinityPreset`, `XXX.podAntiAffinityPreset`, or `XXX.nodeAffinityPreset` parameters. - -## Persistence - -The [Bitnami PostgreSQL](https://github.com/bitnami/containers/tree/main/bitnami/postgresql) image stores the PostgreSQL data and configurations at the `/bitnami/postgresql` path of the container. - -Persistent Volume Claims are used to keep the data across deployments. This is known to work in GCE, AWS, and minikube. -See the [Parameters](#parameters) section to configure the PVC or to disable persistence. - -If you already have data in it, you will fail to sync to standby nodes for all commits, details can refer to the [code present in the container repository](https://github.com/bitnami/containers/tree/main/bitnami/postgresql). If you need to use those data, please covert them to sql and import after `helm install` finished. - -## Parameters - -### Global parameters - -| Name | Description | Value | -| ---------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------ | -| `global.imageRegistry` | Global Docker image registry | `""` | -| `global.imagePullSecrets` | Global Docker registry secret names as an array | `[]` | -| `global.defaultStorageClass` | Global default StorageClass for Persistent Volume(s) | `""` | -| `global.storageClass` | DEPRECATED: use global.defaultStorageClass instead | `""` | -| `global.postgresql.auth.postgresPassword` | Password for the "postgres" admin user (overrides `auth.postgresPassword`) | `""` | -| `global.postgresql.auth.username` | Name for a custom user to create (overrides `auth.username`) | `""` | -| `global.postgresql.auth.password` | Password for the custom user to create (overrides `auth.password`) | `""` | -| `global.postgresql.auth.database` | Name for a custom database to create (overrides `auth.database`) | `""` | -| `global.postgresql.auth.existingSecret` | Name of existing secret to use for PostgreSQL credentials (overrides `auth.existingSecret`). | `""` | -| `global.postgresql.auth.secretKeys.adminPasswordKey` | Name of key in existing secret to use for PostgreSQL credentials (overrides `auth.secretKeys.adminPasswordKey`). Only used when `global.postgresql.auth.existingSecret` is set. | `""` | -| `global.postgresql.auth.secretKeys.userPasswordKey` | Name of key in existing secret to use for PostgreSQL credentials (overrides `auth.secretKeys.userPasswordKey`). Only used when `global.postgresql.auth.existingSecret` is set. | `""` | -| `global.postgresql.auth.secretKeys.replicationPasswordKey` | Name of key in existing secret to use for PostgreSQL credentials (overrides `auth.secretKeys.replicationPasswordKey`). Only used when `global.postgresql.auth.existingSecret` is set. | `""` | -| `global.postgresql.service.ports.postgresql` | PostgreSQL service port (overrides `service.ports.postgresql`) | `""` | -| `global.compatibility.openshift.adaptSecurityContext` | Adapt the securityContext sections of the deployment to make them compatible with Openshift restricted-v2 SCC: remove runAsUser, runAsGroup and fsGroup and let the platform use their allowed default IDs. Possible values: auto (apply if the detected running cluster is Openshift), force (perform the adaptation always), disabled (do not perform adaptation) | `auto` | - -### Common parameters - -| Name | Description | Value | -| ------------------------ | -------------------------------------------------------------------------------------------- | --------------- | -| `kubeVersion` | Override Kubernetes version | `""` | -| `nameOverride` | String to partially override common.names.fullname template (will maintain the release name) | `""` | -| `fullnameOverride` | String to fully override common.names.fullname template | `""` | -| `clusterDomain` | Kubernetes Cluster Domain | `cluster.local` | -| `extraDeploy` | Array of extra objects to deploy with the release (evaluated as a template) | `[]` | -| `commonLabels` | Add labels to all the deployed resources | `{}` | -| `commonAnnotations` | Add annotations to all the deployed resources | `{}` | -| `diagnosticMode.enabled` | Enable diagnostic mode (all probes will be disabled and the command will be overridden) | `false` | -| `diagnosticMode.command` | Command to override all containers in the statefulset | `["sleep"]` | -| `diagnosticMode.args` | Args to override all containers in the statefulset | `["infinity"]` | - -### PostgreSQL common parameters - -| Name | Description | Value | -| ---------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ---------------------------- | -| `image.registry` | PostgreSQL image registry | `REGISTRY_NAME` | -| `image.repository` | PostgreSQL image repository | `REPOSITORY_NAME/postgresql` | -| `image.digest` | PostgreSQL image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `image.pullPolicy` | PostgreSQL image pull policy | `IfNotPresent` | -| `image.pullSecrets` | Specify image pull secrets | `[]` | -| `image.debug` | Specify if debug values should be set | `false` | -| `auth.enablePostgresUser` | Assign a password to the "postgres" admin user. Otherwise, remote access will be blocked for this user | `true` | -| `auth.postgresPassword` | Password for the "postgres" admin user. Ignored if `auth.existingSecret` is provided | `""` | -| `auth.username` | Name for a custom user to create | `""` | -| `auth.password` | Password for the custom user to create. Ignored if `auth.existingSecret` is provided | `""` | -| `auth.database` | Name for a custom database to create | `""` | -| `auth.replicationUsername` | Name of the replication user | `repl_user` | -| `auth.replicationPassword` | Password for the replication user. Ignored if `auth.existingSecret` is provided | `""` | -| `auth.existingSecret` | Name of existing secret to use for PostgreSQL credentials. `auth.postgresPassword`, `auth.password`, and `auth.replicationPassword` will be ignored and picked up from this secret. The secret might also contains the key `ldap-password` if LDAP is enabled. `ldap.bind_password` will be ignored and picked from this secret in this case. | `""` | -| `auth.secretKeys.adminPasswordKey` | Name of key in existing secret to use for PostgreSQL credentials. Only used when `auth.existingSecret` is set. | `postgres-password` | -| `auth.secretKeys.userPasswordKey` | Name of key in existing secret to use for PostgreSQL credentials. Only used when `auth.existingSecret` is set. | `password` | -| `auth.secretKeys.replicationPasswordKey` | Name of key in existing secret to use for PostgreSQL credentials. Only used when `auth.existingSecret` is set. | `replication-password` | -| `auth.usePasswordFiles` | Mount credentials as a files instead of using an environment variable | `false` | -| `architecture` | PostgreSQL architecture (`standalone` or `replication`) | `standalone` | -| `replication.synchronousCommit` | Set synchronous commit mode. Allowed values: `on`, `remote_apply`, `remote_write`, `local` and `off` | `off` | -| `replication.numSynchronousReplicas` | Number of replicas that will have synchronous replication. Note: Cannot be greater than `readReplicas.replicaCount`. | `0` | -| `replication.applicationName` | Cluster application name. Useful for advanced replication settings | `my_application` | -| `containerPorts.postgresql` | PostgreSQL container port | `5432` | -| `audit.logHostname` | Log client hostnames | `false` | -| `audit.logConnections` | Add client log-in operations to the log file | `false` | -| `audit.logDisconnections` | Add client log-outs operations to the log file | `false` | -| `audit.pgAuditLog` | Add operations to log using the pgAudit extension | `""` | -| `audit.pgAuditLogCatalog` | Log catalog using pgAudit | `off` | -| `audit.clientMinMessages` | Message log level to share with the user | `error` | -| `audit.logLinePrefix` | Template for log line prefix (default if not set) | `""` | -| `audit.logTimezone` | Timezone for the log timestamps | `""` | -| `ldap.enabled` | Enable LDAP support | `false` | -| `ldap.server` | IP address or name of the LDAP server. | `""` | -| `ldap.port` | Port number on the LDAP server to connect to | `""` | -| `ldap.prefix` | String to prepend to the user name when forming the DN to bind | `""` | -| `ldap.suffix` | String to append to the user name when forming the DN to bind | `""` | -| `ldap.basedn` | Root DN to begin the search for the user in | `""` | -| `ldap.binddn` | DN of user to bind to LDAP | `""` | -| `ldap.bindpw` | Password for the user to bind to LDAP | `""` | -| `ldap.searchAttribute` | Attribute to match against the user name in the search | `""` | -| `ldap.searchFilter` | The search filter to use when doing search+bind authentication | `""` | -| `ldap.scheme` | Set to `ldaps` to use LDAPS | `""` | -| `ldap.tls.enabled` | Se to true to enable TLS encryption | `false` | -| `ldap.uri` | LDAP URL beginning in the form `ldap[s]://host[:port]/basedn`. If provided, all the other LDAP parameters will be ignored. | `""` | -| `postgresqlDataDir` | PostgreSQL data dir folder | `/bitnami/postgresql/data` | -| `postgresqlSharedPreloadLibraries` | Shared preload libraries (comma-separated list) | `pgaudit` | -| `shmVolume.enabled` | Enable emptyDir volume for /dev/shm for PostgreSQL pod(s) | `true` | -| `shmVolume.sizeLimit` | Set this to enable a size limit on the shm tmpfs | `""` | -| `tls.enabled` | Enable TLS traffic support | `false` | -| `tls.autoGenerated` | Generate automatically self-signed TLS certificates | `false` | -| `tls.preferServerCiphers` | Whether to use the server's TLS cipher preferences rather than the client's | `true` | -| `tls.certificatesSecret` | Name of an existing secret that contains the certificates | `""` | -| `tls.certFilename` | Certificate filename | `""` | -| `tls.certKeyFilename` | Certificate key filename | `""` | -| `tls.certCAFilename` | CA Certificate filename | `""` | -| `tls.crlFilename` | File containing a Certificate Revocation List | `""` | - -### PostgreSQL Primary parameters - -| Name | Description | Value | -| ----------------------------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | --------------------- | -| `primary.name` | Name of the primary database (eg primary, master, leader, ...) | `primary` | -| `primary.configuration` | PostgreSQL Primary main configuration to be injected as ConfigMap | `""` | -| `primary.pgHbaConfiguration` | PostgreSQL Primary client authentication configuration | `""` | -| `primary.existingConfigmap` | Name of an existing ConfigMap with PostgreSQL Primary configuration | `""` | -| `primary.extendedConfiguration` | Extended PostgreSQL Primary configuration (appended to main or default configuration) | `""` | -| `primary.existingExtendedConfigmap` | Name of an existing ConfigMap with PostgreSQL Primary extended configuration | `""` | -| `primary.initdb.args` | PostgreSQL initdb extra arguments | `""` | -| `primary.initdb.postgresqlWalDir` | Specify a custom location for the PostgreSQL transaction log | `""` | -| `primary.initdb.scripts` | Dictionary of initdb scripts | `{}` | -| `primary.initdb.scriptsConfigMap` | ConfigMap with scripts to be run at first boot | `""` | -| `primary.initdb.scriptsSecret` | Secret with scripts to be run at first boot (in case it contains sensitive information) | `""` | -| `primary.initdb.user` | Specify the PostgreSQL username to execute the initdb scripts | `""` | -| `primary.initdb.password` | Specify the PostgreSQL password to execute the initdb scripts | `""` | -| `primary.preInitDb.scripts` | Dictionary of pre-init scripts | `{}` | -| `primary.preInitDb.scriptsConfigMap` | ConfigMap with pre-init scripts to be run | `""` | -| `primary.preInitDb.scriptsSecret` | Secret with pre-init scripts to be run | `""` | -| `primary.standby.enabled` | Whether to enable current cluster's primary as standby server of another cluster or not | `false` | -| `primary.standby.primaryHost` | The Host of replication primary in the other cluster | `""` | -| `primary.standby.primaryPort` | The Port of replication primary in the other cluster | `""` | -| `primary.extraEnvVars` | Array with extra environment variables to add to PostgreSQL Primary nodes | `[]` | -| `primary.extraEnvVarsCM` | Name of existing ConfigMap containing extra env vars for PostgreSQL Primary nodes | `""` | -| `primary.extraEnvVarsSecret` | Name of existing Secret containing extra env vars for PostgreSQL Primary nodes | `""` | -| `primary.command` | Override default container command (useful when using custom images) | `[]` | -| `primary.args` | Override default container args (useful when using custom images) | `[]` | -| `primary.livenessProbe.enabled` | Enable livenessProbe on PostgreSQL Primary containers | `true` | -| `primary.livenessProbe.initialDelaySeconds` | Initial delay seconds for livenessProbe | `30` | -| `primary.livenessProbe.periodSeconds` | Period seconds for livenessProbe | `10` | -| `primary.livenessProbe.timeoutSeconds` | Timeout seconds for livenessProbe | `5` | -| `primary.livenessProbe.failureThreshold` | Failure threshold for livenessProbe | `6` | -| `primary.livenessProbe.successThreshold` | Success threshold for livenessProbe | `1` | -| `primary.readinessProbe.enabled` | Enable readinessProbe on PostgreSQL Primary containers | `true` | -| `primary.readinessProbe.initialDelaySeconds` | Initial delay seconds for readinessProbe | `5` | -| `primary.readinessProbe.periodSeconds` | Period seconds for readinessProbe | `10` | -| `primary.readinessProbe.timeoutSeconds` | Timeout seconds for readinessProbe | `5` | -| `primary.readinessProbe.failureThreshold` | Failure threshold for readinessProbe | `6` | -| `primary.readinessProbe.successThreshold` | Success threshold for readinessProbe | `1` | -| `primary.startupProbe.enabled` | Enable startupProbe on PostgreSQL Primary containers | `false` | -| `primary.startupProbe.initialDelaySeconds` | Initial delay seconds for startupProbe | `30` | -| `primary.startupProbe.periodSeconds` | Period seconds for startupProbe | `10` | -| `primary.startupProbe.timeoutSeconds` | Timeout seconds for startupProbe | `1` | -| `primary.startupProbe.failureThreshold` | Failure threshold for startupProbe | `15` | -| `primary.startupProbe.successThreshold` | Success threshold for startupProbe | `1` | -| `primary.customLivenessProbe` | Custom livenessProbe that overrides the default one | `{}` | -| `primary.customReadinessProbe` | Custom readinessProbe that overrides the default one | `{}` | -| `primary.customStartupProbe` | Custom startupProbe that overrides the default one | `{}` | -| `primary.lifecycleHooks` | for the PostgreSQL Primary container to automate configuration before or after startup | `{}` | -| `primary.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if primary.resources is set (primary.resources is recommended for production). | `nano` | -| `primary.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `primary.podSecurityContext.enabled` | Enable security context | `true` | -| `primary.podSecurityContext.fsGroupChangePolicy` | Set filesystem group change policy | `Always` | -| `primary.podSecurityContext.sysctls` | Set kernel settings using the sysctl interface | `[]` | -| `primary.podSecurityContext.supplementalGroups` | Set filesystem extra groups | `[]` | -| `primary.podSecurityContext.fsGroup` | Group ID for the pod | `1001` | -| `primary.containerSecurityContext.enabled` | Enabled containers' Security Context | `true` | -| `primary.containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `primary.containerSecurityContext.runAsUser` | Set containers' Security Context runAsUser | `1001` | -| `primary.containerSecurityContext.runAsGroup` | Set containers' Security Context runAsGroup | `1001` | -| `primary.containerSecurityContext.runAsNonRoot` | Set container's Security Context runAsNonRoot | `true` | -| `primary.containerSecurityContext.privileged` | Set container's Security Context privileged | `false` | -| `primary.containerSecurityContext.readOnlyRootFilesystem` | Set container's Security Context readOnlyRootFilesystem | `true` | -| `primary.containerSecurityContext.allowPrivilegeEscalation` | Set container's Security Context allowPrivilegeEscalation | `false` | -| `primary.containerSecurityContext.capabilities.drop` | List of capabilities to be dropped | `["ALL"]` | -| `primary.containerSecurityContext.seccompProfile.type` | Set container's Security Context seccomp profile | `RuntimeDefault` | -| `primary.automountServiceAccountToken` | Mount Service Account token in pod | `false` | -| `primary.hostAliases` | PostgreSQL primary pods host aliases | `[]` | -| `primary.hostNetwork` | Specify if host network should be enabled for PostgreSQL pod (postgresql primary) | `false` | -| `primary.hostIPC` | Specify if host IPC should be enabled for PostgreSQL pod (postgresql primary) | `false` | -| `primary.labels` | Map of labels to add to the statefulset (postgresql primary) | `{}` | -| `primary.annotations` | Annotations for PostgreSQL primary pods | `{}` | -| `primary.podLabels` | Map of labels to add to the pods (postgresql primary) | `{}` | -| `primary.podAnnotations` | Map of annotations to add to the pods (postgresql primary) | `{}` | -| `primary.podAffinityPreset` | PostgreSQL primary pod affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `primary.podAntiAffinityPreset` | PostgreSQL primary pod anti-affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `soft` | -| `primary.nodeAffinityPreset.type` | PostgreSQL primary node affinity preset type. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `primary.nodeAffinityPreset.key` | PostgreSQL primary node label key to match Ignored if `primary.affinity` is set. | `""` | -| `primary.nodeAffinityPreset.values` | PostgreSQL primary node label values to match. Ignored if `primary.affinity` is set. | `[]` | -| `primary.affinity` | Affinity for PostgreSQL primary pods assignment | `{}` | -| `primary.nodeSelector` | Node labels for PostgreSQL primary pods assignment | `{}` | -| `primary.tolerations` | Tolerations for PostgreSQL primary pods assignment | `[]` | -| `primary.topologySpreadConstraints` | Topology Spread Constraints for pod assignment spread across your cluster among failure-domains. Evaluated as a template | `[]` | -| `primary.priorityClassName` | Priority Class to use for each pod (postgresql primary) | `""` | -| `primary.schedulerName` | Use an alternate scheduler, e.g. "stork". | `""` | -| `primary.terminationGracePeriodSeconds` | Seconds PostgreSQL primary pod needs to terminate gracefully | `""` | -| `primary.updateStrategy.type` | PostgreSQL Primary statefulset strategy type | `RollingUpdate` | -| `primary.updateStrategy.rollingUpdate` | PostgreSQL Primary statefulset rolling update configuration parameters | `{}` | -| `primary.extraVolumeMounts` | Optionally specify extra list of additional volumeMounts for the PostgreSQL Primary container(s) | `[]` | -| `primary.extraVolumes` | Optionally specify extra list of additional volumes for the PostgreSQL Primary pod(s) | `[]` | -| `primary.sidecars` | Add additional sidecar containers to the PostgreSQL Primary pod(s) | `[]` | -| `primary.initContainers` | Add additional init containers to the PostgreSQL Primary pod(s) | `[]` | -| `primary.pdb.create` | Enable/disable a Pod Disruption Budget creation | `true` | -| `primary.pdb.minAvailable` | Minimum number/percentage of pods that should remain scheduled | `""` | -| `primary.pdb.maxUnavailable` | Maximum number/percentage of pods that may be made unavailable. Defaults to `1` if both `primary.pdb.minAvailable` and `primary.pdb.maxUnavailable` are empty. | `""` | -| `primary.extraPodSpec` | Optionally specify extra PodSpec for the PostgreSQL Primary pod(s) | `{}` | -| `primary.networkPolicy.enabled` | Specifies whether a NetworkPolicy should be created | `true` | -| `primary.networkPolicy.allowExternal` | Don't require server label for connections | `true` | -| `primary.networkPolicy.allowExternalEgress` | Allow the pod to access any range of port and all destinations. | `true` | -| `primary.networkPolicy.extraIngress` | Add extra ingress rules to the NetworkPolicy | `[]` | -| `primary.networkPolicy.extraEgress` | Add extra ingress rules to the NetworkPolicy | `[]` | -| `primary.networkPolicy.ingressNSMatchLabels` | Labels to match to allow traffic from other namespaces | `{}` | -| `primary.networkPolicy.ingressNSPodMatchLabels` | Pod labels to match to allow traffic from other namespaces | `{}` | -| `primary.service.type` | Kubernetes Service type | `ClusterIP` | -| `primary.service.ports.postgresql` | PostgreSQL service port | `5432` | -| `primary.service.nodePorts.postgresql` | Node port for PostgreSQL | `""` | -| `primary.service.clusterIP` | Static clusterIP or None for headless services | `""` | -| `primary.service.annotations` | Annotations for PostgreSQL primary service | `{}` | -| `primary.service.loadBalancerClass` | Load balancer class if service type is `LoadBalancer` | `""` | -| `primary.service.loadBalancerIP` | Load balancer IP if service type is `LoadBalancer` | `""` | -| `primary.service.externalTrafficPolicy` | Enable client source IP preservation | `Cluster` | -| `primary.service.loadBalancerSourceRanges` | Addresses that are allowed when service is LoadBalancer | `[]` | -| `primary.service.extraPorts` | Extra ports to expose in the PostgreSQL primary service | `[]` | -| `primary.service.sessionAffinity` | Session Affinity for Kubernetes service, can be "None" or "ClientIP" | `None` | -| `primary.service.sessionAffinityConfig` | Additional settings for the sessionAffinity | `{}` | -| `primary.service.headless.annotations` | Additional custom annotations for headless PostgreSQL primary service | `{}` | -| `primary.persistence.enabled` | Enable PostgreSQL Primary data persistence using PVC | `true` | -| `primary.persistence.volumeName` | Name to assign the volume | `data` | -| `primary.persistence.existingClaim` | Name of an existing PVC to use | `""` | -| `primary.persistence.mountPath` | The path the volume will be mounted at | `/bitnami/postgresql` | -| `primary.persistence.subPath` | The subdirectory of the volume to mount to | `""` | -| `primary.persistence.storageClass` | PVC Storage Class for PostgreSQL Primary data volume | `""` | -| `primary.persistence.accessModes` | PVC Access Mode for PostgreSQL volume | `["ReadWriteOnce"]` | -| `primary.persistence.size` | PVC Storage Request for PostgreSQL volume | `8Gi` | -| `primary.persistence.annotations` | Annotations for the PVC | `{}` | -| `primary.persistence.labels` | Labels for the PVC | `{}` | -| `primary.persistence.selector` | Selector to match an existing Persistent Volume (this value is evaluated as a template) | `{}` | -| `primary.persistence.dataSource` | Custom PVC data source | `{}` | -| `primary.persistentVolumeClaimRetentionPolicy.enabled` | Enable Persistent volume retention policy for Primary Statefulset | `false` | -| `primary.persistentVolumeClaimRetentionPolicy.whenScaled` | Volume retention behavior when the replica count of the StatefulSet is reduced | `Retain` | -| `primary.persistentVolumeClaimRetentionPolicy.whenDeleted` | Volume retention behavior that applies when the StatefulSet is deleted | `Retain` | - -### PostgreSQL read only replica parameters (only used when `architecture` is set to `replication`) - -| Name | Description | Value | -| ---------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | --------------------- | -| `readReplicas.name` | Name of the read replicas database (eg secondary, slave, ...) | `read` | -| `readReplicas.replicaCount` | Number of PostgreSQL read only replicas | `1` | -| `readReplicas.extendedConfiguration` | Extended PostgreSQL read only replicas configuration (appended to main or default configuration) | `""` | -| `readReplicas.extraEnvVars` | Array with extra environment variables to add to PostgreSQL read only nodes | `[]` | -| `readReplicas.extraEnvVarsCM` | Name of existing ConfigMap containing extra env vars for PostgreSQL read only nodes | `""` | -| `readReplicas.extraEnvVarsSecret` | Name of existing Secret containing extra env vars for PostgreSQL read only nodes | `""` | -| `readReplicas.command` | Override default container command (useful when using custom images) | `[]` | -| `readReplicas.args` | Override default container args (useful when using custom images) | `[]` | -| `readReplicas.livenessProbe.enabled` | Enable livenessProbe on PostgreSQL read only containers | `true` | -| `readReplicas.livenessProbe.initialDelaySeconds` | Initial delay seconds for livenessProbe | `30` | -| `readReplicas.livenessProbe.periodSeconds` | Period seconds for livenessProbe | `10` | -| `readReplicas.livenessProbe.timeoutSeconds` | Timeout seconds for livenessProbe | `5` | -| `readReplicas.livenessProbe.failureThreshold` | Failure threshold for livenessProbe | `6` | -| `readReplicas.livenessProbe.successThreshold` | Success threshold for livenessProbe | `1` | -| `readReplicas.readinessProbe.enabled` | Enable readinessProbe on PostgreSQL read only containers | `true` | -| `readReplicas.readinessProbe.initialDelaySeconds` | Initial delay seconds for readinessProbe | `5` | -| `readReplicas.readinessProbe.periodSeconds` | Period seconds for readinessProbe | `10` | -| `readReplicas.readinessProbe.timeoutSeconds` | Timeout seconds for readinessProbe | `5` | -| `readReplicas.readinessProbe.failureThreshold` | Failure threshold for readinessProbe | `6` | -| `readReplicas.readinessProbe.successThreshold` | Success threshold for readinessProbe | `1` | -| `readReplicas.startupProbe.enabled` | Enable startupProbe on PostgreSQL read only containers | `false` | -| `readReplicas.startupProbe.initialDelaySeconds` | Initial delay seconds for startupProbe | `30` | -| `readReplicas.startupProbe.periodSeconds` | Period seconds for startupProbe | `10` | -| `readReplicas.startupProbe.timeoutSeconds` | Timeout seconds for startupProbe | `1` | -| `readReplicas.startupProbe.failureThreshold` | Failure threshold for startupProbe | `15` | -| `readReplicas.startupProbe.successThreshold` | Success threshold for startupProbe | `1` | -| `readReplicas.customLivenessProbe` | Custom livenessProbe that overrides the default one | `{}` | -| `readReplicas.customReadinessProbe` | Custom readinessProbe that overrides the default one | `{}` | -| `readReplicas.customStartupProbe` | Custom startupProbe that overrides the default one | `{}` | -| `readReplicas.lifecycleHooks` | for the PostgreSQL read only container to automate configuration before or after startup | `{}` | -| `readReplicas.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if readReplicas.resources is set (readReplicas.resources is recommended for production). | `nano` | -| `readReplicas.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `readReplicas.podSecurityContext.enabled` | Enable security context | `true` | -| `readReplicas.podSecurityContext.fsGroupChangePolicy` | Set filesystem group change policy | `Always` | -| `readReplicas.podSecurityContext.sysctls` | Set kernel settings using the sysctl interface | `[]` | -| `readReplicas.podSecurityContext.supplementalGroups` | Set filesystem extra groups | `[]` | -| `readReplicas.podSecurityContext.fsGroup` | Group ID for the pod | `1001` | -| `readReplicas.containerSecurityContext.enabled` | Enabled containers' Security Context | `true` | -| `readReplicas.containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `readReplicas.containerSecurityContext.runAsUser` | Set containers' Security Context runAsUser | `1001` | -| `readReplicas.containerSecurityContext.runAsGroup` | Set containers' Security Context runAsGroup | `1001` | -| `readReplicas.containerSecurityContext.runAsNonRoot` | Set container's Security Context runAsNonRoot | `true` | -| `readReplicas.containerSecurityContext.privileged` | Set container's Security Context privileged | `false` | -| `readReplicas.containerSecurityContext.readOnlyRootFilesystem` | Set container's Security Context readOnlyRootFilesystem | `true` | -| `readReplicas.containerSecurityContext.allowPrivilegeEscalation` | Set container's Security Context allowPrivilegeEscalation | `false` | -| `readReplicas.containerSecurityContext.capabilities.drop` | List of capabilities to be dropped | `["ALL"]` | -| `readReplicas.containerSecurityContext.seccompProfile.type` | Set container's Security Context seccomp profile | `RuntimeDefault` | -| `readReplicas.automountServiceAccountToken` | Mount Service Account token in pod | `false` | -| `readReplicas.hostAliases` | PostgreSQL read only pods host aliases | `[]` | -| `readReplicas.hostNetwork` | Specify if host network should be enabled for PostgreSQL pod (PostgreSQL read only) | `false` | -| `readReplicas.hostIPC` | Specify if host IPC should be enabled for PostgreSQL pod (postgresql primary) | `false` | -| `readReplicas.labels` | Map of labels to add to the statefulset (PostgreSQL read only) | `{}` | -| `readReplicas.annotations` | Annotations for PostgreSQL read only pods | `{}` | -| `readReplicas.podLabels` | Map of labels to add to the pods (PostgreSQL read only) | `{}` | -| `readReplicas.podAnnotations` | Map of annotations to add to the pods (PostgreSQL read only) | `{}` | -| `readReplicas.podAffinityPreset` | PostgreSQL read only pod affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `readReplicas.podAntiAffinityPreset` | PostgreSQL read only pod anti-affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `soft` | -| `readReplicas.nodeAffinityPreset.type` | PostgreSQL read only node affinity preset type. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `readReplicas.nodeAffinityPreset.key` | PostgreSQL read only node label key to match Ignored if `primary.affinity` is set. | `""` | -| `readReplicas.nodeAffinityPreset.values` | PostgreSQL read only node label values to match. Ignored if `primary.affinity` is set. | `[]` | -| `readReplicas.affinity` | Affinity for PostgreSQL read only pods assignment | `{}` | -| `readReplicas.nodeSelector` | Node labels for PostgreSQL read only pods assignment | `{}` | -| `readReplicas.tolerations` | Tolerations for PostgreSQL read only pods assignment | `[]` | -| `readReplicas.topologySpreadConstraints` | Topology Spread Constraints for pod assignment spread across your cluster among failure-domains. Evaluated as a template | `[]` | -| `readReplicas.priorityClassName` | Priority Class to use for each pod (PostgreSQL read only) | `""` | -| `readReplicas.schedulerName` | Use an alternate scheduler, e.g. "stork". | `""` | -| `readReplicas.terminationGracePeriodSeconds` | Seconds PostgreSQL read only pod needs to terminate gracefully | `""` | -| `readReplicas.updateStrategy.type` | PostgreSQL read only statefulset strategy type | `RollingUpdate` | -| `readReplicas.updateStrategy.rollingUpdate` | PostgreSQL read only statefulset rolling update configuration parameters | `{}` | -| `readReplicas.extraVolumeMounts` | Optionally specify extra list of additional volumeMounts for the PostgreSQL read only container(s) | `[]` | -| `readReplicas.extraVolumes` | Optionally specify extra list of additional volumes for the PostgreSQL read only pod(s) | `[]` | -| `readReplicas.sidecars` | Add additional sidecar containers to the PostgreSQL read only pod(s) | `[]` | -| `readReplicas.initContainers` | Add additional init containers to the PostgreSQL read only pod(s) | `[]` | -| `readReplicas.pdb.create` | Enable/disable a Pod Disruption Budget creation | `true` | -| `readReplicas.pdb.minAvailable` | Minimum number/percentage of pods that should remain scheduled | `""` | -| `readReplicas.pdb.maxUnavailable` | Maximum number/percentage of pods that may be made unavailable. Defaults to `1` if both `readReplicas.pdb.minAvailable` and `readReplicas.pdb.maxUnavailable` are empty. | `""` | -| `readReplicas.extraPodSpec` | Optionally specify extra PodSpec for the PostgreSQL read only pod(s) | `{}` | -| `readReplicas.networkPolicy.enabled` | Specifies whether a NetworkPolicy should be created | `true` | -| `readReplicas.networkPolicy.allowExternal` | Don't require server label for connections | `true` | -| `readReplicas.networkPolicy.allowExternalEgress` | Allow the pod to access any range of port and all destinations. | `true` | -| `readReplicas.networkPolicy.extraIngress` | Add extra ingress rules to the NetworkPolicy | `[]` | -| `readReplicas.networkPolicy.extraEgress` | Add extra ingress rules to the NetworkPolicy | `[]` | -| `readReplicas.networkPolicy.ingressNSMatchLabels` | Labels to match to allow traffic from other namespaces | `{}` | -| `readReplicas.networkPolicy.ingressNSPodMatchLabels` | Pod labels to match to allow traffic from other namespaces | `{}` | -| `readReplicas.service.type` | Kubernetes Service type | `ClusterIP` | -| `readReplicas.service.ports.postgresql` | PostgreSQL service port | `5432` | -| `readReplicas.service.nodePorts.postgresql` | Node port for PostgreSQL | `""` | -| `readReplicas.service.clusterIP` | Static clusterIP or None for headless services | `""` | -| `readReplicas.service.annotations` | Annotations for PostgreSQL read only service | `{}` | -| `readReplicas.service.loadBalancerClass` | Load balancer class if service type is `LoadBalancer` | `""` | -| `readReplicas.service.loadBalancerIP` | Load balancer IP if service type is `LoadBalancer` | `""` | -| `readReplicas.service.externalTrafficPolicy` | Enable client source IP preservation | `Cluster` | -| `readReplicas.service.loadBalancerSourceRanges` | Addresses that are allowed when service is LoadBalancer | `[]` | -| `readReplicas.service.extraPorts` | Extra ports to expose in the PostgreSQL read only service | `[]` | -| `readReplicas.service.sessionAffinity` | Session Affinity for Kubernetes service, can be "None" or "ClientIP" | `None` | -| `readReplicas.service.sessionAffinityConfig` | Additional settings for the sessionAffinity | `{}` | -| `readReplicas.service.headless.annotations` | Additional custom annotations for headless PostgreSQL read only service | `{}` | -| `readReplicas.persistence.enabled` | Enable PostgreSQL read only data persistence using PVC | `true` | -| `readReplicas.persistence.existingClaim` | Name of an existing PVC to use | `""` | -| `readReplicas.persistence.mountPath` | The path the volume will be mounted at | `/bitnami/postgresql` | -| `readReplicas.persistence.subPath` | The subdirectory of the volume to mount to | `""` | -| `readReplicas.persistence.storageClass` | PVC Storage Class for PostgreSQL read only data volume | `""` | -| `readReplicas.persistence.accessModes` | PVC Access Mode for PostgreSQL volume | `["ReadWriteOnce"]` | -| `readReplicas.persistence.size` | PVC Storage Request for PostgreSQL volume | `8Gi` | -| `readReplicas.persistence.annotations` | Annotations for the PVC | `{}` | -| `readReplicas.persistence.labels` | Labels for the PVC | `{}` | -| `readReplicas.persistence.selector` | Selector to match an existing Persistent Volume (this value is evaluated as a template) | `{}` | -| `readReplicas.persistence.dataSource` | Custom PVC data source | `{}` | -| `readReplicas.persistentVolumeClaimRetentionPolicy.enabled` | Enable Persistent volume retention policy for read only Statefulset | `false` | -| `readReplicas.persistentVolumeClaimRetentionPolicy.whenScaled` | Volume retention behavior when the replica count of the StatefulSet is reduced | `Retain` | -| `readReplicas.persistentVolumeClaimRetentionPolicy.whenDeleted` | Volume retention behavior that applies when the StatefulSet is deleted | `Retain` | - -### Backup parameters - -| Name | Description | Value | -| ------------------------------------------------------------------ | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | -| `backup.enabled` | Enable the logical dump of the database "regularly" | `false` | -| `backup.cronjob.schedule` | Set the cronjob parameter schedule | `@daily` | -| `backup.cronjob.timeZone` | Set the cronjob parameter timeZone | `""` | -| `backup.cronjob.concurrencyPolicy` | Set the cronjob parameter concurrencyPolicy | `Allow` | -| `backup.cronjob.failedJobsHistoryLimit` | Set the cronjob parameter failedJobsHistoryLimit | `1` | -| `backup.cronjob.successfulJobsHistoryLimit` | Set the cronjob parameter successfulJobsHistoryLimit | `3` | -| `backup.cronjob.startingDeadlineSeconds` | Set the cronjob parameter startingDeadlineSeconds | `""` | -| `backup.cronjob.ttlSecondsAfterFinished` | Set the cronjob parameter ttlSecondsAfterFinished | `""` | -| `backup.cronjob.restartPolicy` | Set the cronjob parameter restartPolicy | `OnFailure` | -| `backup.cronjob.podSecurityContext.enabled` | Enable PodSecurityContext for CronJob/Backup | `true` | -| `backup.cronjob.podSecurityContext.fsGroupChangePolicy` | Set filesystem group change policy | `Always` | -| `backup.cronjob.podSecurityContext.sysctls` | Set kernel settings using the sysctl interface | `[]` | -| `backup.cronjob.podSecurityContext.supplementalGroups` | Set filesystem extra groups | `[]` | -| `backup.cronjob.podSecurityContext.fsGroup` | Group ID for the CronJob | `1001` | -| `backup.cronjob.containerSecurityContext.enabled` | Enabled containers' Security Context | `true` | -| `backup.cronjob.containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `backup.cronjob.containerSecurityContext.runAsUser` | Set containers' Security Context runAsUser | `1001` | -| `backup.cronjob.containerSecurityContext.runAsGroup` | Set containers' Security Context runAsGroup | `1001` | -| `backup.cronjob.containerSecurityContext.runAsNonRoot` | Set container's Security Context runAsNonRoot | `true` | -| `backup.cronjob.containerSecurityContext.privileged` | Set container's Security Context privileged | `false` | -| `backup.cronjob.containerSecurityContext.readOnlyRootFilesystem` | Set container's Security Context readOnlyRootFilesystem | `true` | -| `backup.cronjob.containerSecurityContext.allowPrivilegeEscalation` | Set container's Security Context allowPrivilegeEscalation | `false` | -| `backup.cronjob.containerSecurityContext.capabilities.drop` | List of capabilities to be dropped | `["ALL"]` | -| `backup.cronjob.containerSecurityContext.seccompProfile.type` | Set container's Security Context seccomp profile | `RuntimeDefault` | -| `backup.cronjob.command` | Set backup container's command to run | `["/bin/sh","-c","pg_dumpall --clean --if-exists --load-via-partition-root --quote-all-identifiers --no-password --file=${PGDUMP_DIR}/pg_dumpall-$(date '+%Y-%m-%d-%H-%M').pgdump"]` | -| `backup.cronjob.labels` | Set the cronjob labels | `{}` | -| `backup.cronjob.annotations` | Set the cronjob annotations | `{}` | -| `backup.cronjob.nodeSelector` | Node labels for PostgreSQL backup CronJob pod assignment | `{}` | -| `backup.cronjob.tolerations` | Tolerations for PostgreSQL backup CronJob pods assignment | `[]` | -| `backup.cronjob.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if backup.cronjob.resources is set (backup.cronjob.resources is recommended for production). | `nano` | -| `backup.cronjob.resources` | Set container requests and limits for different resources like CPU or memory | `{}` | -| `backup.cronjob.networkPolicy.enabled` | Specifies whether a NetworkPolicy should be created | `true` | -| `backup.cronjob.storage.enabled` | Enable using a `PersistentVolumeClaim` as backup data volume | `true` | -| `backup.cronjob.storage.existingClaim` | Provide an existing `PersistentVolumeClaim` (only when `architecture=standalone`) | `""` | -| `backup.cronjob.storage.resourcePolicy` | Setting it to "keep" to avoid removing PVCs during a helm delete operation. Leaving it empty will delete PVCs after the chart deleted | `""` | -| `backup.cronjob.storage.storageClass` | PVC Storage Class for the backup data volume | `""` | -| `backup.cronjob.storage.accessModes` | PV Access Mode | `["ReadWriteOnce"]` | -| `backup.cronjob.storage.size` | PVC Storage Request for the backup data volume | `8Gi` | -| `backup.cronjob.storage.annotations` | PVC annotations | `{}` | -| `backup.cronjob.storage.mountPath` | Path to mount the volume at | `/backup/pgdump` | -| `backup.cronjob.storage.subPath` | Subdirectory of the volume to mount at | `""` | -| `backup.cronjob.storage.volumeClaimTemplates.selector` | A label query over volumes to consider for binding (e.g. when using local volumes) | `{}` | -| `backup.cronjob.extraVolumeMounts` | Optionally specify extra list of additional volumeMounts for the backup container | `[]` | -| `backup.cronjob.extraVolumes` | Optionally specify extra list of additional volumes for the backup container | `[]` | - -### Volume Permissions parameters - -| Name | Description | Value | -| ---------------------------------------------------------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | -------------------------- | -| `volumePermissions.enabled` | Enable init container that changes the owner and group of the persistent volume | `false` | -| `volumePermissions.image.registry` | Init container volume-permissions image registry | `REGISTRY_NAME` | -| `volumePermissions.image.repository` | Init container volume-permissions image repository | `REPOSITORY_NAME/os-shell` | -| `volumePermissions.image.digest` | Init container volume-permissions image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `volumePermissions.image.pullPolicy` | Init container volume-permissions image pull policy | `IfNotPresent` | -| `volumePermissions.image.pullSecrets` | Init container volume-permissions image pull secrets | `[]` | -| `volumePermissions.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if volumePermissions.resources is set (volumePermissions.resources is recommended for production). | `nano` | -| `volumePermissions.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `volumePermissions.containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `volumePermissions.containerSecurityContext.runAsUser` | User ID for the init container | `0` | -| `volumePermissions.containerSecurityContext.runAsGroup` | Group ID for the init container | `0` | -| `volumePermissions.containerSecurityContext.runAsNonRoot` | runAsNonRoot for the init container | `false` | -| `volumePermissions.containerSecurityContext.seccompProfile.type` | seccompProfile.type for the init container | `RuntimeDefault` | - -### Other Parameters - -| Name | Description | Value | -| --------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------- | ------- | -| `serviceBindings.enabled` | Create secret for service binding (Experimental) | `false` | -| `serviceAccount.create` | Enable creation of ServiceAccount for PostgreSQL pod | `true` | -| `serviceAccount.name` | The name of the ServiceAccount to use. | `""` | -| `serviceAccount.automountServiceAccountToken` | Allows auto mount of ServiceAccountToken on the serviceAccount created | `false` | -| `serviceAccount.annotations` | Additional custom annotations for the ServiceAccount | `{}` | -| `rbac.create` | Create Role and RoleBinding (required for PSP to work) | `false` | -| `rbac.rules` | Custom RBAC rules to set | `[]` | -| `psp.create` | Whether to create a PodSecurityPolicy. WARNING: PodSecurityPolicy is deprecated in Kubernetes v1.21 or later, unavailable in v1.25 or later | `false` | - -### Metrics Parameters - -| Name | Description | Value | -| ----------------------------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ----------------------------------- | -| `metrics.enabled` | Start a prometheus exporter | `false` | -| `metrics.image.registry` | PostgreSQL Prometheus Exporter image registry | `REGISTRY_NAME` | -| `metrics.image.repository` | PostgreSQL Prometheus Exporter image repository | `REPOSITORY_NAME/postgres-exporter` | -| `metrics.image.digest` | PostgreSQL image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `metrics.image.pullPolicy` | PostgreSQL Prometheus Exporter image pull policy | `IfNotPresent` | -| `metrics.image.pullSecrets` | Specify image pull secrets | `[]` | -| `metrics.collectors` | Control enabled collectors | `{}` | -| `metrics.customMetrics` | Define additional custom metrics | `{}` | -| `metrics.extraEnvVars` | Extra environment variables to add to PostgreSQL Prometheus exporter | `[]` | -| `metrics.containerSecurityContext.enabled` | Enabled containers' Security Context | `true` | -| `metrics.containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `metrics.containerSecurityContext.runAsUser` | Set containers' Security Context runAsUser | `1001` | -| `metrics.containerSecurityContext.runAsGroup` | Set containers' Security Context runAsGroup | `1001` | -| `metrics.containerSecurityContext.runAsNonRoot` | Set container's Security Context runAsNonRoot | `true` | -| `metrics.containerSecurityContext.privileged` | Set container's Security Context privileged | `false` | -| `metrics.containerSecurityContext.readOnlyRootFilesystem` | Set container's Security Context readOnlyRootFilesystem | `true` | -| `metrics.containerSecurityContext.allowPrivilegeEscalation` | Set container's Security Context allowPrivilegeEscalation | `false` | -| `metrics.containerSecurityContext.capabilities.drop` | List of capabilities to be dropped | `["ALL"]` | -| `metrics.containerSecurityContext.seccompProfile.type` | Set container's Security Context seccomp profile | `RuntimeDefault` | -| `metrics.livenessProbe.enabled` | Enable livenessProbe on PostgreSQL Prometheus exporter containers | `true` | -| `metrics.livenessProbe.initialDelaySeconds` | Initial delay seconds for livenessProbe | `5` | -| `metrics.livenessProbe.periodSeconds` | Period seconds for livenessProbe | `10` | -| `metrics.livenessProbe.timeoutSeconds` | Timeout seconds for livenessProbe | `5` | -| `metrics.livenessProbe.failureThreshold` | Failure threshold for livenessProbe | `6` | -| `metrics.livenessProbe.successThreshold` | Success threshold for livenessProbe | `1` | -| `metrics.readinessProbe.enabled` | Enable readinessProbe on PostgreSQL Prometheus exporter containers | `true` | -| `metrics.readinessProbe.initialDelaySeconds` | Initial delay seconds for readinessProbe | `5` | -| `metrics.readinessProbe.periodSeconds` | Period seconds for readinessProbe | `10` | -| `metrics.readinessProbe.timeoutSeconds` | Timeout seconds for readinessProbe | `5` | -| `metrics.readinessProbe.failureThreshold` | Failure threshold for readinessProbe | `6` | -| `metrics.readinessProbe.successThreshold` | Success threshold for readinessProbe | `1` | -| `metrics.startupProbe.enabled` | Enable startupProbe on PostgreSQL Prometheus exporter containers | `false` | -| `metrics.startupProbe.initialDelaySeconds` | Initial delay seconds for startupProbe | `10` | -| `metrics.startupProbe.periodSeconds` | Period seconds for startupProbe | `10` | -| `metrics.startupProbe.timeoutSeconds` | Timeout seconds for startupProbe | `1` | -| `metrics.startupProbe.failureThreshold` | Failure threshold for startupProbe | `15` | -| `metrics.startupProbe.successThreshold` | Success threshold for startupProbe | `1` | -| `metrics.customLivenessProbe` | Custom livenessProbe that overrides the default one | `{}` | -| `metrics.customReadinessProbe` | Custom readinessProbe that overrides the default one | `{}` | -| `metrics.customStartupProbe` | Custom startupProbe that overrides the default one | `{}` | -| `metrics.containerPorts.metrics` | PostgreSQL Prometheus exporter metrics container port | `9187` | -| `metrics.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if metrics.resources is set (metrics.resources is recommended for production). | `nano` | -| `metrics.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `metrics.service.ports.metrics` | PostgreSQL Prometheus Exporter service port | `9187` | -| `metrics.service.clusterIP` | Static clusterIP or None for headless services | `""` | -| `metrics.service.sessionAffinity` | Control where client requests go, to the same pod or round-robin | `None` | -| `metrics.service.annotations` | Annotations for Prometheus to auto-discover the metrics endpoint | `{}` | -| `metrics.serviceMonitor.enabled` | Create ServiceMonitor Resource for scraping metrics using Prometheus Operator | `false` | -| `metrics.serviceMonitor.namespace` | Namespace for the ServiceMonitor Resource (defaults to the Release Namespace) | `""` | -| `metrics.serviceMonitor.interval` | Interval at which metrics should be scraped. | `""` | -| `metrics.serviceMonitor.scrapeTimeout` | Timeout after which the scrape is ended | `""` | -| `metrics.serviceMonitor.labels` | Additional labels that can be used so ServiceMonitor will be discovered by Prometheus | `{}` | -| `metrics.serviceMonitor.selector` | Prometheus instance selector labels | `{}` | -| `metrics.serviceMonitor.relabelings` | RelabelConfigs to apply to samples before scraping | `[]` | -| `metrics.serviceMonitor.metricRelabelings` | MetricRelabelConfigs to apply to samples before ingestion | `[]` | -| `metrics.serviceMonitor.honorLabels` | Specify honorLabels parameter to add the scrape endpoint | `false` | -| `metrics.serviceMonitor.jobLabel` | The name of the label on the target service to use as the job name in prometheus. | `""` | -| `metrics.prometheusRule.enabled` | Create a PrometheusRule for Prometheus Operator | `false` | -| `metrics.prometheusRule.namespace` | Namespace for the PrometheusRule Resource (defaults to the Release Namespace) | `""` | -| `metrics.prometheusRule.labels` | Additional labels that can be used so PrometheusRule will be discovered by Prometheus | `{}` | -| `metrics.prometheusRule.rules` | PrometheusRule definitions | `[]` | - -Specify each parameter using the `--set key=value[,key=value]` argument to `helm install`. For example, - -```console -helm install my-release \ - --set auth.postgresPassword=secretpassword - oci://REGISTRY_NAME/REPOSITORY_NAME/postgresql -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -The above command sets the PostgreSQL `postgres` account password to `secretpassword`. - -> NOTE: Once this chart is deployed, it is not possible to change the application's access credentials, such as usernames or passwords, using Helm. To change these application credentials after deployment, delete any persistent volumes (PVs) used by the chart and re-deploy it, or use the application's built-in administrative tools if available. -> **Warning** Setting a password will be ignored on new installation in case when previous PostgreSQL release was deleted through the helm command. In that case, old PVC will have an old password, and setting it through helm won't take effect. Deleting persistent volumes (PVs) will solve the issue. Refer to [issue 2061](https://github.com/bitnami/charts/issues/2061) for more details - -Alternatively, a YAML file that specifies the values for the parameters can be provided while installing the chart. For example, - -```console -helm install my-release -f values.yaml oci://REGISTRY_NAME/REPOSITORY_NAME/postgresql -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. -> **Tip**: You can use the default [values.yaml](https://github.com/bitnami/charts/tree/main/bitnami/postgresql/values.yaml) - -## Troubleshooting - -Find more information about how to deal with common errors related to Bitnami's Helm charts in [this troubleshooting guide](https://docs.bitnami.com/general/how-to/troubleshoot-helm-chart-issues). - -## Upgrading - -### To 15.0.0 - -This major bump changes the following security defaults: - -- `runAsGroup` is changed from `0` to `1001` -- `readOnlyRootFilesystem` is set to `true` -- `resourcesPreset` is changed from `none` to the minimum size working in our test suites (NOTE: `resourcesPreset` is not meant for production usage, but `resources` adapted to your use case). -- `global.compatibility.openshift.adaptSecurityContext` is changed from `disabled` to `auto`. - -This could potentially break any customization or init scripts used in your deployment. If this is the case, change the default values to the previous ones. - -### To 14.0.0 - -This major version adapts the NetworkPolicy objects to the most recent Bitnami standards. Now there is a separate object for `primary` and for `readReplicas`, being located in their corresponding sections. It is also enabled by default in other to comply with the best security standards. - -Check the parameter section for the new value structure. - -### To 13.0.0 - -This major version changes the default PostgreSQL image from 15.x to 16.x. Follow the [official instructions](https://www.postgresql.org/docs/16/upgrading.html) to upgrade to 16.x. - -### To 12.0.0 - -This major version changes the default PostgreSQL image from 14.x to 15.x. Follow the [official instructions](https://www.postgresql.org/docs/15/upgrading.html) to upgrade to 15.x. - -### To 11.0.0 - -In this version the application version was bumped to _14.x_ series. Also, this major release renames several values in this chart and adds missing features, in order to be inline with the rest of assets in the Bitnami charts repository. - -- _replication.enabled_ parameter is deprecated in favor of _architecture_ parameter that accepts two values: _standalone_ and _replication_. -- _replication.singleService_ and _replication.uniqueServices_ parameters are deprecated. When using replication, each statefulset (primary and read-only) has its own headless service & service allowing to connect to read-only replicas through the service (round-robin) or individually. -- _postgresqlPostgresPassword_, _postgresqlUsername_, _postgresqlPassword_, _postgresqlDatabase_, _replication.user_, _replication.password_, and _existingSecret_ parameters have been regrouped under the _auth_ map. The _auth_ map uses a new perspective to configure authentication, so please read carefully each sub-parameter description. -- _extraEnv_ has been deprecated in favor of _primary.extraEnvVars_ and _readReplicas.extraEnvVars_. -- _postgresqlConfiguration_, _pgHbaConfiguration_, _configurationConfigMap_, _postgresqlExtendedConf_, and _extendedConfConfigMap_ have been deprecated in favor of _primary.configuration_, _primary.pgHbaConfiguration_, _primary.existingConfigmap_, _primary.extendedConfiguration_, and _primary.existingExtendedConfigmap_. -- _postgresqlInitdbArgs_, _postgresqlInitdbWalDir_, _initdbScripts_, _initdbScriptsConfigMap_, _initdbScriptsSecret_, _initdbUser_ and _initdbPassword_ have been regrouped under the _primary.initdb_ map. -- _postgresqlMaxConnections_, _postgresqlPostgresConnectionLimit_, _postgresqlDbUserConnectionLimit_, _postgresqlTcpKeepalivesInterval_, _postgresqlTcpKeepalivesIdle_, _postgresqlTcpKeepalivesCount_, _postgresqlStatementTimeout_ and _postgresqlPghbaRemoveFilters_ parameters are deprecated. Use _XXX.extraEnvVars_ instead. -- _primaryAsStandBy_ has been deprecated in favor of _primary.standby_. -- _securityContext_ and _containerSecurityContext_ have been deprecated in favor of _primary.podSecurityContext_, _primary.containerSecurityContext_, _readReplicas.podSecurityContext_, and _readReplicas.containerSecurityContext_. -- _livenessProbe_ and _readinessProbe_ maps have been deprecated in favor of _primary.livenessProbe_, _primary.readinessProbe_, _readReplicas.livenessProbe_ and _readReplicas.readinessProbe_ maps. -- _persistence_ map has been deprecated in favor of _primary.persistence_ and _readReplicas.persistence_ maps. -- _networkPolicy_ map has been completely refactored. -- _service_ map has been deprecated in favor of _primary.service_ and _readReplicas.service_ maps. -- _metrics.service.port_ has been regrouped under the _metrics.service.ports_ map. -- _serviceAccount.enabled_ and _serviceAccount.autoMount_ have been deprecated in favor of _serviceAccount.create_ and _serviceAccount.automountServiceAccountToken_. - -#### How to upgrade to version 11.0.0 - -To upgrade to _11.0.0_ from _10.x_, it should be done reusing the PVC(s) used to hold the PostgreSQL data on your previous release. To do so, follow the instructions below (the following example assumes that the release name is _postgresql_): - -> NOTE: Please, create a backup of your database before running any of these actions. - -1. Obtain the credentials and the names of the PVCs used to hold the PostgreSQL data on your current release: - -```console -export POSTGRESQL_PASSWORD=$(kubectl get secret --namespace default postgresql -o jsonpath="{.data.postgresql-password}" | base64 --decode) -export POSTGRESQL_PVC=$(kubectl get pvc -l app.kubernetes.io/instance=postgresql,role=primary -o jsonpath="{.items[0].metadata.name}") -``` - -1. Delete the PostgreSQL statefulset (notice the option _--cascade=false_) and secret: - -```console -kubectl delete statefulsets.apps postgresql-postgresql --namespace default --cascade=false -kubectl delete secret postgresql --namespace default -``` - -1. Upgrade your release using the same PostgreSQL version: - -```console -CURRENT_VERSION=$(kubectl exec postgresql-postgresql-0 -- bash -c 'echo $BITNAMI_IMAGE_VERSION') -helm upgrade postgresql bitnami/postgresql \ - --set auth.postgresPassword=$POSTGRESQL_PASSWORD \ - --set primary.persistence.existingClaim=$POSTGRESQL_PVC \ - --set image.tag=$CURRENT_VERSION -``` - -1. You will have to delete the existing PostgreSQL pod and the new statefulset is going to create a new one - -```console -kubectl delete pod postgresql-postgresql-0 -``` - -1. Finally, you should see the lines below in PostgreSQL container logs: - -```text -$ kubectl logs $(kubectl get pods -l app.kubernetes.io/instance=postgresql,app.kubernetes.io/name=postgresql,app.kubernetes.io/component=primary -o jsonpath="{.items[0].metadata.name}") -... -postgresql 08:05:12.59 INFO ==> Deploying PostgreSQL with persisted data... -... -``` - -> NOTE: the instructions above reuse the same PostgreSQL version you were using in your chart release. Otherwise, you will find an error such as the one below when upgrading since the new chart major version also bumps the application version. To workaround this issue you need to upgrade database, please refer to the [official PostgreSQL documentation](https://www.postgresql.org/docs/current/upgrading.html) for more information about this. - -```console -$ kubectl logs $(kubectl get pods -l app.kubernetes.io/instance=postgresql,app.kubernetes.io/name=postgresql,app.kubernetes.io/component=primary -o jsonpath="{.items[0].metadata.name}") - ... -postgresql 08:10:14.72 INFO ==> ** Starting PostgreSQL ** -2022-02-01 08:10:14.734 GMT [1] FATAL: database files are incompatible with server -2022-02-01 08:10:14.734 GMT [1] DETAIL: The data directory was initialized by PostgreSQL version 11, which is not compatible with this version 14.1. -``` - -### To 10.0.0 - -[On November 13, 2020, Helm v2 support was formally finished](https://github.com/helm/charts#status-of-the-project), this major version is the result of the required changes applied to the Helm Chart to be able to incorporate the different features added in Helm v3 and to be consistent with the Helm project itself regarding the Helm v2 EOL. - -- Previous versions of this Helm Chart use `apiVersion: v1` (installable by both Helm 2 and 3), this Helm Chart was updated to `apiVersion: v2` (installable by Helm 3 only). [Here](https://helm.sh/docs/topics/charts/#the-apiversion-field) you can find more information about the `apiVersion` field. -- Move dependency information from the _requirements.yaml_ to the _Chart.yaml_ -- After running _helm dependency update_, a _Chart.lock_ file is generated containing the same structure used in the previous _requirements.lock_ -- The different fields present in the _Chart.yaml_ file has been ordered alphabetically in a homogeneous way for all the Bitnami Helm Chart. -- The term _master_ has been replaced with _primary_ and _slave_ with _readReplicas_ throughout the chart. Role names have changed from _master_ and _slave_ to _primary_ and _read_. - -#### Considerations when upgrading to this version - -- If you want to upgrade to this version using Helm v2, this scenario is not supported as this version does not support Helm v2 anymore. -- If you installed the previous version with Helm v2 and wants to upgrade to this version with Helm v3, please refer to the [official Helm documentation](https://helm.sh/docs/topics/v2_v3_migration/#migration-use-cases) about migrating from Helm v2 to v3. - -#### Useful links - -- [Bitnami Tutorial](https://docs.vmware.com/en/VMware-Tanzu-Application-Catalog/services/tutorials/GUID-resolve-helm2-helm3-post-migration-issues-index.html) -- [Helm docs](https://helm.sh/docs/topics/v2_v3_migration) -- [Helm Blog](https://helm.sh/blog/migrate-from-helm-v2-to-helm-v3) - -#### How to upgrade to version 10.0.0 - -To upgrade to _10.0.0_ from _9.x_, it should be done reusing the PVC(s) used to hold the PostgreSQL data on your previous release. To do so, follow the instructions below (the following example assumes that the release name is _postgresql_): - -> NOTE: Please, create a backup of your database before running any of those actions. - -1. Obtain the credentials and the names of the PVCs used to hold the PostgreSQL data on your current release: - -```console -export POSTGRESQL_PASSWORD=$(kubectl get secret --namespace default postgresql -o jsonpath="{.data.postgresql-password}" | base64 --decode) -export POSTGRESQL_PVC=$(kubectl get pvc -l app.kubernetes.io/instance=postgresql,role=primary -o jsonpath="{.items[0].metadata.name}") -``` - -1. Delete the PostgreSQL statefulset (notice the option _--cascade=false_): - -```console -kubectl delete statefulsets.apps postgresql-postgresql --namespace default --cascade=false -``` - -1. Upgrade your release using the same PostgreSQL version: - -```console -helm upgrade postgresql bitnami/postgresql \ - --set postgresqlPassword=$POSTGRESQL_PASSWORD \ - --set persistence.existingClaim=$POSTGRESQL_PVC -``` - -1. Delete the existing PostgreSQL pod and the new statefulset will create a new one: - -```console -kubectl delete pod postgresql-postgresql-0 -``` - -1. Finally, you should see the lines below in PostgreSQL container logs: - -```text -$ kubectl logs $(kubectl get pods -l app.kubernetes.io/instance=postgresql,app.kubernetes.io/name=postgresql,role=primary -o jsonpath="{.items[0].metadata.name}") -... -postgresql 08:05:12.59 INFO ==> Deploying PostgreSQL with persisted data... -... -``` - -### To 9.0.0 - -In this version the chart was adapted to follow the [Helm standard labels](https://helm.sh/docs/chart_best_practices/labels/#standard-labels). - -- Some inmutable objects were modified to adopt Helm standard labels introducing backward incompatibilities. - -#### How to upgrade to version 9.0.0 - -To upgrade to _9.0.0_ from _8.x_, it should be done reusing the PVC(s) used to hold the PostgreSQL data on your previous release. To do so, follow the instructions below (the following example assumes that the release name is _postgresql_): - -> NOTE: Please, create a backup of your database before running any of those actions. - -1. Obtain the credentials and the names of the PVCs used to hold the PostgreSQL data on your current release: - -```console -export POSTGRESQL_PASSWORD=$(kubectl get secret --namespace default postgresql -o jsonpath="{.data.postgresql-password}" | base64 --decode) -export POSTGRESQL_PVC=$(kubectl get pvc -l app=postgresql,role=master -o jsonpath="{.items[0].metadata.name}") -``` - -1. Delete the PostgreSQL statefulset (notice the option _--cascade=false_): - -```console -kubectl delete statefulsets.apps postgresql-postgresql --namespace default --cascade=false -``` - -1. Upgrade your release using the same PostgreSQL version: - -```console -helm upgrade postgresql bitnami/postgresql \ - --set postgresqlPassword=$POSTGRESQL_PASSWORD \ - --set persistence.existingClaim=$POSTGRESQL_PVC -``` - -1. Delete the existing PostgreSQL pod and the new statefulset will create a new one: - -```console -kubectl delete pod postgresql-postgresql-0 -``` - -1. Finally, you should see the lines below in PostgreSQL container logs: - -```text -$ kubectl logs $(kubectl get pods -l app.kubernetes.io/instance=postgresql,app.kubernetes.io/name=postgresql,role=master -o jsonpath="{.items[0].metadata.name}") -... -postgresql 08:05:12.59 INFO ==> Deploying PostgreSQL with persisted data... -... -``` - -## License - -Copyright © 2024 Broadcom. The term "Broadcom" refers to Broadcom Inc. and/or its subsidiaries. - -Licensed under the Apache License, Version 2.0 (the "License"); -you may not use this file except in compliance with the License. -You may obtain a copy of the License at - - - -Unless required by applicable law or agreed to in writing, software -distributed under the License is distributed on an "AS IS" BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -See the License for the specific language governing permissions and -limitations under the License. \ No newline at end of file diff --git a/charts/postgresql/postgresql/templates/NOTES.txt b/charts/postgresql/postgresql/templates/NOTES.txt deleted file mode 100644 index fd7a751..0000000 --- a/charts/postgresql/postgresql/templates/NOTES.txt +++ /dev/null @@ -1,117 +0,0 @@ -CHART NAME: {{ .Chart.Name }} -CHART VERSION: {{ .Chart.Version }} -APP VERSION: {{ .Chart.AppVersion }} - -** Please be patient while the chart is being deployed ** - -{{- if .Values.diagnosticMode.enabled }} -The chart has been deployed in diagnostic mode. All probes have been disabled and the command has been overwritten with: - - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 4 }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 4 }} - -Get the list of pods by executing: - - kubectl get pods --namespace {{ .Release.Namespace }} -l app.kubernetes.io/instance={{ .Release.Name }} - -Access the pod you want to debug by executing - - kubectl exec --namespace {{ .Release.Namespace }} -ti -- /opt/bitnami/scripts/postgresql/entrypoint.sh /bin/bash - -In order to replicate the container startup scripts execute this command: - - /opt/bitnami/scripts/postgresql/entrypoint.sh /opt/bitnami/scripts/postgresql/run.sh - -{{- else }} - -{{- $customUser := include "postgresql.v1.username" . }} -{{- $postgresPassword := include "common.secrets.lookup" (dict "secret" (include "common.names.fullname" .) "key" .Values.auth.secretKeys.adminPasswordKey "defaultValue" (ternary .Values.auth.postgresPassword .Values.auth.password (eq $customUser "postgres")) "context" $) -}} -{{- $authEnabled := and (not (or .Values.global.postgresql.auth.existingSecret .Values.auth.existingSecret)) (or $postgresPassword .Values.auth.enablePostgresUser (and (not (empty $customUser)) (ne $customUser "postgres"))) }} -{{- if not $authEnabled }} - -WARNING: PostgreSQL has been configured without authentication, this is not recommended for production environments. -{{- end }} - -PostgreSQL can be accessed via port {{ include "postgresql.v1.service.port" . }} on the following DNS names from within your cluster: - - {{ include "postgresql.v1.primary.fullname" . }}.{{ .Release.Namespace }}.svc.cluster.local - Read/Write connection - -{{- if eq .Values.architecture "replication" }} - - {{ include "postgresql.v1.readReplica.fullname" . }}.{{ .Release.Namespace }}.svc.cluster.local - Read only connection - -{{- end }} - -{{- if and (not (empty $customUser)) (ne $customUser "postgres") }} -{{- if .Values.auth.enablePostgresUser }} - -To get the password for "postgres" run: - - export POSTGRES_ADMIN_PASSWORD=$(kubectl get secret --namespace {{ .Release.Namespace }} {{ include "postgresql.v1.secretName" . }} -o jsonpath="{.data.{{include "postgresql.v1.adminPasswordKey" .}}}" | base64 -d) -{{- end }} - -To get the password for "{{ $customUser }}" run: - - export POSTGRES_PASSWORD=$(kubectl get secret --namespace {{ .Release.Namespace }} {{ include "postgresql.v1.secretName" . }} -o jsonpath="{.data.{{include "postgresql.v1.userPasswordKey" .}}}" | base64 -d) -{{- else }} -{{- if .Values.auth.enablePostgresUser }} - -To get the password for "{{ default "postgres" $customUser }}" run: - - export POSTGRES_PASSWORD=$(kubectl get secret --namespace {{ .Release.Namespace }} {{ include "postgresql.v1.secretName" . }} -o jsonpath="{.data.{{ ternary "password" (include "postgresql.v1.adminPasswordKey" .) (and (not (empty $customUser)) (ne $customUser "postgres")) }}}" | base64 -d) -{{- end }} -{{- end }} - -To connect to your database run the following command: - {{- if $authEnabled }} - - kubectl run {{ include "common.names.fullname" . }}-client --rm --tty -i --restart='Never' --namespace {{ .Release.Namespace }} --image {{ include "postgresql.v1.image" . }} --env="PGPASSWORD=$POSTGRES_PASSWORD" \ - --command -- psql --host {{ include "postgresql.v1.primary.fullname" . }} -U {{ default "postgres" $customUser }} -d {{- if include "postgresql.v1.database" . }} {{ include "postgresql.v1.database" . }}{{- else }} postgres{{- end }} -p {{ include "postgresql.v1.service.port" . }} - {{- else }} - - kubectl run {{ include "common.names.fullname" . }}-client --rm --tty -i --restart='Never' --namespace {{ .Release.Namespace }} --image {{ include "postgresql.v1.image" . }} \ - --command -- psql --host {{ include "postgresql.v1.primary.fullname" . }} -d {{- if include "postgresql.v1.database" . }} {{ include "postgresql.v1.database" . }}{{- else }} postgres{{- end }} -p {{ include "postgresql.v1.service.port" . }} - {{- end }} - - > NOTE: If you access the container using bash, make sure that you execute "/opt/bitnami/scripts/postgresql/entrypoint.sh /bin/bash" in order to avoid the error "psql: local user with ID {{ .Values.primary.containerSecurityContext.runAsUser }}} does not exist" - -To connect to your database from outside the cluster execute the following commands: - -{{- if contains "NodePort" .Values.primary.service.type }} - - export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}") - export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ include "postgresql.v1.primary.fullname" . }}) - {{- if $authEnabled }} - PGPASSWORD="$POSTGRES_PASSWORD" psql --host $NODE_IP --port $NODE_PORT -U {{ default "postgres" $customUser }} -d {{- if include "postgresql.v1.database" . }} {{ include "postgresql.v1.database" . }}{{- else }} postgres{{- end }} - {{- else }} - psql --host $NODE_IP --port $NODE_PORT -d {{- if include "postgresql.v1.database" . }} {{ include "postgresql.v1.database" . }}{{- else }} postgres{{- end }} - {{- end }} -{{- else if contains "LoadBalancer" .Values.primary.service.type }} - - NOTE: It may take a few minutes for the LoadBalancer IP to be available. - Watch the status with: 'kubectl get svc --namespace {{ .Release.Namespace }} -w {{ include "postgresql.v1.primary.fullname" . }}' - - export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ include "postgresql.v1.primary.fullname" . }} --template "{{ "{{ range (index .status.loadBalancer.ingress 0) }}{{ . }}{{ end }}" }}") - {{- if $authEnabled }} - PGPASSWORD="$POSTGRES_PASSWORD" psql --host $SERVICE_IP --port {{ include "postgresql.v1.service.port" . }} -U {{ default "postgres" $customUser }} -d {{- if include "postgresql.v1.database" . }} {{ include "postgresql.v1.database" . }}{{- else }} postgres{{- end }} - {{- else }} - psql --host $SERVICE_IP --port {{ include "postgresql.v1.service.port" . }} -d {{- if include "postgresql.v1.database" . }} {{ include "postgresql.v1.database" . }}{{- else }} postgres{{- end }} - {{- end }} -{{- else if contains "ClusterIP" .Values.primary.service.type }} - - kubectl port-forward --namespace {{ .Release.Namespace }} svc/{{ include "postgresql.v1.primary.fullname" . }} {{ include "postgresql.v1.service.port" . }}:{{ include "postgresql.v1.service.port" . }} & - {{- if $authEnabled }} - PGPASSWORD="$POSTGRES_PASSWORD" psql --host 127.0.0.1 -U {{ default "postgres" $customUser }} -d {{- if include "postgresql.v1.database" . }} {{ include "postgresql.v1.database" . }}{{- else }} postgres{{- end }} -p {{ include "postgresql.v1.service.port" . }} - {{- else }} - psql --host 127.0.0.1 -d {{- if include "postgresql.v1.database" . }} {{ include "postgresql.v1.database" . }}{{- else }} postgres{{- end }} -p {{ include "postgresql.v1.service.port" . }} - {{- end }} -{{- end }} -{{- end }} - -WARNING: The configured password will be ignored on new installation in case when previous PostgreSQL release was deleted through the helm command. In that case, old PVC will have an old password, and setting it through helm won't take effect. Deleting persistent volumes (PVs) will solve the issue. - -{{- include "postgresql.v1.validateValues" . -}} -{{- include "common.warnings.rollingTag" .Values.image -}} -{{- include "common.warnings.rollingTag" .Values.volumePermissions.image }} -{{- include "common.warnings.resources" (dict "sections" (list "metrics" "primary" "readReplicas" "volumePermissions") "context" $) }} -{{- include "common.warnings.modifiedImages" (dict "images" (list .Values.image .Values.volumePermissions.image .Values.metrics.image) "context" $) }} \ No newline at end of file diff --git a/charts/postgresql/postgresql/templates/_helpers.tpl b/charts/postgresql/postgresql/templates/_helpers.tpl deleted file mode 100644 index d78e869..0000000 --- a/charts/postgresql/postgresql/templates/_helpers.tpl +++ /dev/null @@ -1,417 +0,0 @@ -{{/* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{/* vim: set filetype=mustache: */}} - -{{/* -Create a default fully qualified app name for PostgreSQL Primary objects -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -*/}} -{{- define "postgresql.v1.primary.fullname" -}} -{{- if eq .Values.architecture "replication" -}} - {{- printf "%s-%s" (include "common.names.fullname" .) .Values.primary.name | trunc 63 | trimSuffix "-" -}} -{{- else -}} - {{- include "common.names.fullname" . -}} -{{- end -}} -{{- end -}} - -{{/* -Create a default fully qualified app name for PostgreSQL read-only replicas objects -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -*/}} -{{- define "postgresql.v1.readReplica.fullname" -}} -{{- printf "%s-%s" (include "common.names.fullname" .) .Values.readReplicas.name | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Create the default FQDN for PostgreSQL primary headless service -We truncate at 63 chars because of the DNS naming spec. -*/}} -{{- define "postgresql.v1.primary.svc.headless" -}} -{{- printf "%s-hl" (include "postgresql.v1.primary.fullname" .) | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Create the default FQDN for PostgreSQL read-only replicas headless service -We truncate at 63 chars because of the DNS naming spec. -*/}} -{{- define "postgresql.v1.readReplica.svc.headless" -}} -{{- printf "%s-hl" (include "postgresql.v1.readReplica.fullname" .) | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Return the proper PostgreSQL image name -*/}} -{{- define "postgresql.v1.image" -}} -{{ include "common.images.image" (dict "imageRoot" .Values.image "global" .Values.global) }} -{{- end -}} - -{{/* -Return the proper PostgreSQL metrics image name -*/}} -{{- define "postgresql.v1.metrics.image" -}} -{{ include "common.images.image" (dict "imageRoot" .Values.metrics.image "global" .Values.global) }} -{{- end -}} - -{{/* -Return the proper image name (for the init container volume-permissions image) -*/}} -{{- define "postgresql.v1.volumePermissions.image" -}} -{{ include "common.images.image" (dict "imageRoot" .Values.volumePermissions.image "global" .Values.global) }} -{{- end -}} - -{{/* -Return the proper Docker Image Registry Secret Names -*/}} -{{- define "postgresql.v1.imagePullSecrets" -}} -{{ include "common.images.renderPullSecrets" (dict "images" (list .Values.image .Values.metrics.image .Values.volumePermissions.image) "context" $) }} -{{- end -}} - -{{/* -Return the name for a custom user to create -*/}} -{{- define "postgresql.v1.username" -}} -{{- if .Values.global.postgresql.auth.username -}} - {{- .Values.global.postgresql.auth.username -}} -{{- else -}} - {{- .Values.auth.username -}} -{{- end -}} -{{- end -}} - -{{/* -Return the name for a custom database to create -*/}} -{{- define "postgresql.v1.database" -}} -{{- if .Values.global.postgresql.auth.database -}} - {{- printf "%s" (tpl .Values.global.postgresql.auth.database $) -}} -{{- else if .Values.auth.database -}} - {{- printf "%s" (tpl .Values.auth.database $) -}} -{{- end -}} -{{- end -}} - -{{/* -Get the password secret. -*/}} -{{- define "postgresql.v1.secretName" -}} -{{- if .Values.global.postgresql.auth.existingSecret -}} - {{- printf "%s" (tpl .Values.global.postgresql.auth.existingSecret $) -}} -{{- else if .Values.auth.existingSecret -}} - {{- printf "%s" (tpl .Values.auth.existingSecret $) -}} -{{- else -}} - {{- printf "%s" (include "common.names.fullname" .) -}} -{{- end -}} -{{- end -}} - -{{/* -Get the replication-password key. -*/}} -{{- define "postgresql.v1.replicationPasswordKey" -}} -{{- if or .Values.global.postgresql.auth.existingSecret .Values.auth.existingSecret -}} - {{- if .Values.global.postgresql.auth.secretKeys.replicationPasswordKey -}} - {{- printf "%s" (tpl .Values.global.postgresql.auth.secretKeys.replicationPasswordKey $) -}} - {{- else if .Values.auth.secretKeys.replicationPasswordKey -}} - {{- printf "%s" (tpl .Values.auth.secretKeys.replicationPasswordKey $) -}} - {{- else -}} - {{- "replication-password" -}} - {{- end -}} -{{- else -}} - {{- "replication-password" -}} -{{- end -}} -{{- end -}} - -{{/* -Get the admin-password key. -*/}} -{{- define "postgresql.v1.adminPasswordKey" -}} -{{- if or .Values.global.postgresql.auth.existingSecret .Values.auth.existingSecret -}} - {{- if .Values.global.postgresql.auth.secretKeys.adminPasswordKey -}} - {{- printf "%s" (tpl .Values.global.postgresql.auth.secretKeys.adminPasswordKey $) -}} - {{- else if .Values.auth.secretKeys.adminPasswordKey -}} - {{- printf "%s" (tpl .Values.auth.secretKeys.adminPasswordKey $) -}} - {{- end -}} -{{- else -}} - {{- "postgres-password" -}} -{{- end -}} -{{- end -}} - -{{/* -Get the user-password key. -*/}} -{{- define "postgresql.v1.userPasswordKey" -}} -{{- if or .Values.global.postgresql.auth.existingSecret .Values.auth.existingSecret -}} - {{- if or (empty (include "postgresql.v1.username" .)) (eq (include "postgresql.v1.username" .) "postgres") -}} - {{- printf "%s" (include "postgresql.v1.adminPasswordKey" .) -}} - {{- else -}} - {{- if .Values.global.postgresql.auth.secretKeys.userPasswordKey -}} - {{- printf "%s" (tpl .Values.global.postgresql.auth.secretKeys.userPasswordKey $) -}} - {{- else if .Values.auth.secretKeys.userPasswordKey -}} - {{- printf "%s" (tpl .Values.auth.secretKeys.userPasswordKey $) -}} - {{- end -}} - {{- end -}} -{{- else -}} - {{- "password" -}} -{{- end -}} -{{- end -}} - -{{/* -Return true if a secret object should be created -*/}} -{{- define "postgresql.v1.createSecret" -}} -{{- $customUser := include "postgresql.v1.username" . -}} -{{- $postgresPassword := include "common.secrets.lookup" (dict "secret" (include "common.names.fullname" .) "key" .Values.auth.secretKeys.adminPasswordKey "defaultValue" (ternary (coalesce .Values.global.postgresql.auth.postgresPassword .Values.auth.postgresPassword .Values.global.postgresql.auth.password .Values.auth.password) (coalesce .Values.global.postgresql.auth.postgresPassword .Values.auth.postgresPassword) (or (empty $customUser) (eq $customUser "postgres"))) "context" $) -}} -{{- if and (not (or .Values.global.postgresql.auth.existingSecret .Values.auth.existingSecret)) (or $postgresPassword .Values.auth.enablePostgresUser (and (not (empty $customUser)) (ne $customUser "postgres")) (eq .Values.architecture "replication") (and .Values.ldap.enabled (or .Values.ldap.bind_password .Values.ldap.bindpw))) -}} - {{- true -}} -{{- end -}} -{{- end -}} - -{{/* -Return PostgreSQL service port -*/}} -{{- define "postgresql.v1.service.port" -}} -{{- if .Values.global.postgresql.service.ports.postgresql -}} - {{- .Values.global.postgresql.service.ports.postgresql -}} -{{- else -}} - {{- .Values.primary.service.ports.postgresql -}} -{{- end -}} -{{- end -}} - -{{/* -Return PostgreSQL service port -*/}} -{{- define "postgresql.v1.readReplica.service.port" -}} -{{- if .Values.global.postgresql.service.ports.postgresql -}} - {{- .Values.global.postgresql.service.ports.postgresql -}} -{{- else -}} - {{- .Values.readReplicas.service.ports.postgresql -}} -{{- end -}} -{{- end -}} - -{{/* -Get the PostgreSQL primary configuration ConfigMap name. -*/}} -{{- define "postgresql.v1.primary.configmapName" -}} -{{- if .Values.primary.existingConfigmap -}} - {{- printf "%s" (tpl .Values.primary.existingConfigmap $) -}} -{{- else -}} - {{- printf "%s-configuration" (include "postgresql.v1.primary.fullname" .) -}} -{{- end -}} -{{- end -}} - -{{/* -Return true if a configmap object should be created for PostgreSQL primary with the configuration -*/}} -{{- define "postgresql.v1.primary.createConfigmap" -}} -{{- if and (or .Values.primary.configuration .Values.primary.pgHbaConfiguration) (not .Values.primary.existingConfigmap) -}} - {{- true -}} -{{- else -}} -{{- end -}} -{{- end -}} - -{{/* -Get the PostgreSQL primary extended configuration ConfigMap name. -*/}} -{{- define "postgresql.v1.primary.extendedConfigmapName" -}} -{{- if .Values.primary.existingExtendedConfigmap -}} - {{- printf "%s" (tpl .Values.primary.existingExtendedConfigmap $) -}} -{{- else -}} - {{- printf "%s-extended-configuration" (include "postgresql.v1.primary.fullname" .) -}} -{{- end -}} -{{- end -}} - -{{/* -Get the PostgreSQL read replica extended configuration ConfigMap name. -*/}} -{{- define "postgresql.v1.readReplicas.extendedConfigmapName" -}} - {{- printf "%s-extended-configuration" (include "postgresql.v1.readReplica.fullname" .) -}} -{{- end -}} - -{{/* -Return true if a configmap object should be created for PostgreSQL primary with the extended configuration -*/}} -{{- define "postgresql.v1.primary.createExtendedConfigmap" -}} -{{- if and .Values.primary.extendedConfiguration (not .Values.primary.existingExtendedConfigmap) -}} - {{- true -}} -{{- else -}} -{{- end -}} -{{- end -}} - -{{/* -Return true if a configmap object should be created for PostgreSQL read replica with the extended configuration -*/}} -{{- define "postgresql.v1.readReplicas.createExtendedConfigmap" -}} -{{- if .Values.readReplicas.extendedConfiguration -}} - {{- true -}} -{{- else -}} -{{- end -}} -{{- end -}} - -{{/* - Create the name of the service account to use - */}} -{{- define "postgresql.v1.serviceAccountName" -}} -{{- if .Values.serviceAccount.create -}} - {{ default (include "common.names.fullname" .) .Values.serviceAccount.name }} -{{- else -}} - {{ default "default" .Values.serviceAccount.name }} -{{- end -}} -{{- end -}} - -{{/* -Return true if a configmap should be mounted with PostgreSQL configuration -*/}} -{{- define "postgresql.v1.mountConfigurationCM" -}} -{{- if or .Values.primary.configuration .Values.primary.pgHbaConfiguration .Values.primary.existingConfigmap -}} - {{- true -}} -{{- end -}} -{{- end -}} - -{{/* -Get the pre-initialization scripts ConfigMap name. -*/}} -{{- define "postgresql.v1.preInitDb.scriptsCM" -}} -{{- if .Values.primary.preInitDb.scriptsConfigMap -}} - {{- printf "%s" (tpl .Values.primary.preInitDb.scriptsConfigMap $) -}} -{{- else -}} - {{- printf "%s-preinit-scripts" (include "postgresql.v1.primary.fullname" .) -}} -{{- end -}} -{{- end -}} - -{{/* -Get the initialization scripts ConfigMap name. -*/}} -{{- define "postgresql.v1.initdb.scriptsCM" -}} -{{- if .Values.primary.initdb.scriptsConfigMap -}} - {{- printf "%s" (tpl .Values.primary.initdb.scriptsConfigMap $) -}} -{{- else -}} - {{- printf "%s-init-scripts" (include "postgresql.v1.primary.fullname" .) -}} -{{- end -}} -{{- end -}} - -{{/* -Return true if TLS is enabled for LDAP connection -*/}} -{{- define "postgresql.v1.ldap.tls.enabled" -}} -{{- if and (kindIs "string" .Values.ldap.tls) (not (empty .Values.ldap.tls)) -}} - {{- true -}} -{{- else if and (kindIs "map" .Values.ldap.tls) .Values.ldap.tls.enabled -}} - {{- true -}} -{{- end -}} -{{- end -}} - -{{/* -Get the readiness probe command -*/}} -{{- define "postgresql.v1.readinessProbeCommand" -}} -{{- $customUser := include "postgresql.v1.username" . -}} -- | -{{- if (include "postgresql.v1.database" .) }} - exec pg_isready -U {{ default "postgres" $customUser | quote }} -d "dbname={{ include "postgresql.v1.database" . }} {{- if .Values.tls.enabled }} sslcert={{ include "postgresql.v1.tlsCert" . }} sslkey={{ include "postgresql.v1.tlsCertKey" . }}{{- end }}" -h 127.0.0.1 -p {{ .Values.containerPorts.postgresql }} -{{- else }} - exec pg_isready -U {{ default "postgres" $customUser | quote }} {{- if .Values.tls.enabled }} -d "sslcert={{ include "postgresql.v1.tlsCert" . }} sslkey={{ include "postgresql.v1.tlsCertKey" . }}"{{- end }} -h 127.0.0.1 -p {{ .Values.containerPorts.postgresql }} -{{- end }} -{{- if contains "bitnami/" .Values.image.repository }} - [ -f /opt/bitnami/postgresql/tmp/.initialized ] || [ -f /bitnami/postgresql/.initialized ] -{{- end }} -{{- end -}} - -{{/* -Compile all warnings into a single message, and call fail. -*/}} -{{- define "postgresql.v1.validateValues" -}} -{{- $messages := list -}} -{{- $messages := append $messages (include "postgresql.v1.validateValues.ldapConfigurationMethod" .) -}} -{{- $messages := append $messages (include "postgresql.v1.validateValues.psp" .) -}} -{{- $messages := without $messages "" -}} -{{- $message := join "\n" $messages -}} - -{{- if $message -}} -{{- printf "\nVALUES VALIDATION:\n%s" $message | fail -}} -{{- end -}} -{{- end -}} - -{{/* -Validate values of Postgresql - If ldap.url is used then you don't need the other settings for ldap -*/}} -{{- define "postgresql.v1.validateValues.ldapConfigurationMethod" -}} -{{- if and .Values.ldap.enabled (and (not (empty .Values.ldap.url)) (not (empty .Values.ldap.server))) -}} -postgresql: ldap.url, ldap.server - You cannot set both `ldap.url` and `ldap.server` at the same time. - Please provide a unique way to configure LDAP. - More info at https://www.postgresql.org/docs/current/auth-ldap.html -{{- end -}} -{{- end -}} - -{{/* -Validate values of Postgresql - If PSP is enabled RBAC should be enabled too -*/}} -{{- define "postgresql.v1.validateValues.psp" -}} -{{- if and .Values.psp.create (not .Values.rbac.create) -}} -postgresql: psp.create, rbac.create - RBAC should be enabled if PSP is enabled in order for PSP to work. - More info at https://kubernetes.io/docs/concepts/policy/pod-security-policy/#authorizing-policies -{{- end -}} -{{- end -}} - -{{/* -Return the path to the cert file. -*/}} -{{- define "postgresql.v1.tlsCert" -}} -{{- if .Values.tls.autoGenerated -}} - {{- printf "/opt/bitnami/postgresql/certs/tls.crt" -}} -{{- else -}} - {{- required "Certificate filename is required when TLS in enabled" .Values.tls.certFilename | printf "/opt/bitnami/postgresql/certs/%s" -}} -{{- end -}} -{{- end -}} - -{{/* -Return the path to the cert key file. -*/}} -{{- define "postgresql.v1.tlsCertKey" -}} -{{- if .Values.tls.autoGenerated -}} - {{- printf "/opt/bitnami/postgresql/certs/tls.key" -}} -{{- else -}} -{{- required "Certificate Key filename is required when TLS in enabled" .Values.tls.certKeyFilename | printf "/opt/bitnami/postgresql/certs/%s" -}} -{{- end -}} -{{- end -}} - -{{/* -Return the path to the CA cert file. -*/}} -{{- define "postgresql.v1.tlsCACert" -}} -{{- if .Values.tls.autoGenerated -}} - {{- printf "/opt/bitnami/postgresql/certs/ca.crt" -}} -{{- else -}} - {{- printf "/opt/bitnami/postgresql/certs/%s" .Values.tls.certCAFilename -}} -{{- end -}} -{{- end -}} - -{{/* -Return the path to the CRL file. -*/}} -{{- define "postgresql.v1.tlsCRL" -}} -{{- if .Values.tls.crlFilename -}} -{{- printf "/opt/bitnami/postgresql/certs/%s" .Values.tls.crlFilename -}} -{{- end -}} -{{- end -}} - -{{/* -Return true if a TLS credentials secret object should be created -*/}} -{{- define "postgresql.v1.createTlsSecret" -}} -{{- if and .Values.tls.autoGenerated (not .Values.tls.certificatesSecret) -}} - {{- true -}} -{{- end -}} -{{- end -}} - -{{/* -Return the path to the CA cert file. -*/}} -{{- define "postgresql.v1.tlsSecretName" -}} -{{- if .Values.tls.autoGenerated -}} - {{- printf "%s-crt" (include "common.names.fullname" .) -}} -{{- else -}} - {{ required "A secret containing TLS certificates is required when TLS is enabled" .Values.tls.certificatesSecret }} -{{- end -}} -{{- end -}} diff --git a/charts/postgresql/postgresql/templates/backup/cronjob.yaml b/charts/postgresql/postgresql/templates/backup/cronjob.yaml deleted file mode 100644 index 9689ac0..0000000 --- a/charts/postgresql/postgresql/templates/backup/cronjob.yaml +++ /dev/null @@ -1,138 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.backup.enabled }} -{{- $customUser := include "postgresql.v1.username" . }} -apiVersion: batch/v1 -kind: CronJob -metadata: - name: {{ include "postgresql.v1.primary.fullname" . }}-pgdumpall - namespace: {{ .Release.Namespace | quote }} - {{- $labels := include "common.tplvalues.merge" ( dict "values" ( list .Values.backup.cronjob.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $labels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: pg_dumpall - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.backup.cronjob.annotations .Values.commonAnnotations ) "context" . ) }} - {{- if $annotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - schedule: {{ quote .Values.backup.cronjob.schedule }} - {{- if .Values.backup.cronjob.timezone }} - timeZone: {{ .Values.backup.cronjob.timezone | quote }} - {{- end }} - concurrencyPolicy: {{ .Values.backup.cronjob.concurrencyPolicy }} - failedJobsHistoryLimit: {{ .Values.backup.cronjob.failedJobsHistoryLimit }} - successfulJobsHistoryLimit: {{ .Values.backup.cronjob.successfulJobsHistoryLimit }} - {{- if .Values.backup.cronjob.startingDeadlineSeconds }} - startingDeadlineSeconds: {{ .Values.backup.cronjob.startingDeadlineSeconds }} - {{- end }} - jobTemplate: - spec: - {{- if .Values.backup.cronjob.ttlSecondsAfterFinished }} - ttlSecondsAfterFinished: {{ .Values.backup.cronjob.ttlSecondsAfterFinished }} - {{- end }} - template: - metadata: - labels: {{- include "common.labels.standard" ( dict "customLabels" $labels "context" $ ) | nindent 12 }} - app.kubernetes.io/component: pg_dumpall - {{- if $annotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 12 }} - {{- end }} - spec: - {{- include "postgresql.v1.imagePullSecrets" . | nindent 10 }} - {{- if .Values.backup.cronjob.nodeSelector }} - nodeSelector: {{- include "common.tplvalues.render" (dict "value" .Values.backup.cronjob.nodeSelector "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.backup.cronjob.tolerations }} - tolerations: {{- include "common.tplvalues.render" (dict "value" .Values.backup.cronjob.tolerations "context" $) | nindent 12 }} - {{- end }} - containers: - - name: {{ include "postgresql.v1.primary.fullname" . }}-pgdumpall - image: {{ include "postgresql.v1.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy | quote }} - env: - - name: PGUSER - {{- if .Values.auth.enablePostgresUser }} - value: postgres - {{- else }} - value: {{ $customUser | quote }} - {{- end }} - {{- if .Values.auth.usePasswordFiles }} - - name: PGPASSFILE - value: {{ printf "/opt/bitnami/postgresql/secrets/%s" (include "postgresql.v1.adminPasswordKey" .) }} - {{- else }} - - name: PGPASSWORD - valueFrom: - secretKeyRef: - name: {{ include "postgresql.v1.secretName" . }} - key: {{ include "postgresql.v1.adminPasswordKey" . }} - {{- end }} - - name: PGHOST - value: {{ include "postgresql.v1.primary.fullname" . }} - - name: PGPORT - value: {{ include "postgresql.v1.service.port" . | quote }} - - name: PGDUMP_DIR - value: {{ .Values.backup.cronjob.storage.mountPath }} - {{- if .Values.tls.enabled }} - - name: PGSSLROOTCERT - {{- if .Values.tls.autoGenerated }} - value: /tmp/certs/ca.crt - {{- else }} - value: {{ printf "/tmp/certs/%s" .Values.tls.certCAFilename }} - {{- end }} - {{- end }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.backup.cronjob.command "context" $) | nindent 14 }} - volumeMounts: - {{- if .Values.tls.enabled }} - - name: raw-certificates - mountPath: /tmp/certs - {{- end }} - {{- if .Values.backup.cronjob.storage.enabled }} - - name: datadir - mountPath: {{ .Values.backup.cronjob.storage.mountPath }} - subPath: {{ .Values.backup.cronjob.storage.subPath }} - {{- end }} - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - {{- if .Values.backup.cronjob.extraVolumeMounts }} - {{- include "common.tplvalues.render" (dict "value" .Values.backup.cronjob.extraVolumeMounts "context" $) | nindent 14 }} - {{- end }} - {{- if .Values.backup.cronjob.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.backup.cronjob.containerSecurityContext "context" $) | nindent 14 }} - {{- end }} - {{- if .Values.backup.cronjob.resources }} - resources: {{- toYaml .Values.backup.cronjob.resources | nindent 14 }} - {{- else if ne .Values.backup.cronjob.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.backup.cronjob.resourcesPreset) | nindent 14 }} - {{- end }} - restartPolicy: {{ .Values.backup.cronjob.restartPolicy }} - {{- if .Values.backup.cronjob.podSecurityContext.enabled }} - securityContext: - fsGroup: {{ .Values.backup.cronjob.podSecurityContext.fsGroup }} - {{- end }} - volumes: - {{- if .Values.tls.enabled }} - - name: raw-certificates - secret: - secretName: {{ include "postgresql.v1.tlsSecretName" . }} - {{- end }} - {{- if .Values.backup.cronjob.storage.enabled }} - {{- if .Values.backup.cronjob.storage.existingClaim }} - - name: datadir - persistentVolumeClaim: - claimName: {{ printf "%s" (tpl .Values.backup.cronjob.storage.existingClaim .) }} - {{- else }} - - name: datadir - persistentVolumeClaim: - claimName: {{ include "postgresql.v1.primary.fullname" . }}-pgdumpall - {{- end }} - {{- end }} - - name: empty-dir - emptyDir: {} - {{- if .Values.backup.cronjob.extraVolumes }} - {{- include "common.tplvalues.render" ( dict "value" .Values.backup.cronjob.extraVolumes "context" $ ) | nindent 12 }} - {{- end }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/backup/networkpolicy.yaml b/charts/postgresql/postgresql/templates/backup/networkpolicy.yaml deleted file mode 100644 index 21e1add..0000000 --- a/charts/postgresql/postgresql/templates/backup/networkpolicy.yaml +++ /dev/null @@ -1,32 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.backup.enabled .Values.backup.cronjob.networkPolicy.enabled }} -kind: NetworkPolicy -apiVersion: {{ include "common.capabilities.networkPolicy.apiVersion" . }} -metadata: - name: {{ include "postgresql.v1.primary.fullname" . }}-pgdumpall - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: pg_dumpall - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.backup.cronjob.podLabels .Values.commonLabels ) "context" . ) }} - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: pg_dumpall - policyTypes: - - Egress - egress: - - ports: - - port: 5432 - protocol: TCP - - port: 53 - protocol: TCP - - port: 53 - protocol: UDP -{{- end }} diff --git a/charts/postgresql/postgresql/templates/backup/pvc.yaml b/charts/postgresql/postgresql/templates/backup/pvc.yaml deleted file mode 100644 index 84a7c60..0000000 --- a/charts/postgresql/postgresql/templates/backup/pvc.yaml +++ /dev/null @@ -1,34 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.backup.enabled .Values.backup.cronjob.storage.enabled (not .Values.backup.cronjob.storage.existingClaim) -}} -apiVersion: v1 -kind: PersistentVolumeClaim -metadata: - name: {{ include "postgresql.v1.primary.fullname" . }}-pgdumpall - namespace: {{ .Release.Namespace | quote }} - {{- $labels := include "common.tplvalues.merge" ( dict "values" ( list .Values.backup.cronjob.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $labels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: pg_dumpall - {{- if or .Values.backup.cronjob.annotations .Values.commonAnnotations .Values.backup.cronjob.storage.resourcePolicy }} - annotations: - {{- if or .Values.backup.cronjob.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.backup.cronjob.annotations .Values.commonAnnotations ) "context" . ) }} - {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} - {{- if .Values.backup.cronjob.storage.resourcePolicy }} - helm.sh/resource-policy: {{ .Values.backup.cronjob.storage.resourcePolicy | quote }} - {{- end }} - {{- end }} -spec: - accessModes: - {{- range .Values.backup.cronjob.storage.accessModes }} - - {{ . | quote }} - {{- end }} - resources: - requests: - storage: {{ .Values.backup.cronjob.storage.size | quote }} - {{ include "common.storage.class" (dict "persistence" .Values.backup.cronjob.storage "global" .Values.global) }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/extra-list.yaml b/charts/postgresql/postgresql/templates/extra-list.yaml deleted file mode 100644 index 329f5c6..0000000 --- a/charts/postgresql/postgresql/templates/extra-list.yaml +++ /dev/null @@ -1,9 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- range .Values.extraDeploy }} ---- -{{ include "common.tplvalues.render" (dict "value" . "context" $) }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/primary/configmap.yaml b/charts/postgresql/postgresql/templates/primary/configmap.yaml deleted file mode 100644 index 48376a1..0000000 --- a/charts/postgresql/postgresql/templates/primary/configmap.yaml +++ /dev/null @@ -1,26 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if (include "postgresql.v1.primary.createConfigmap" .) }} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ printf "%s-configuration" (include "postgresql.v1.primary.fullname" .) }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: - {{- if .Values.primary.configuration }} - postgresql.conf: | - {{- include "common.tplvalues.render" ( dict "value" .Values.primary.configuration "context" $ ) | nindent 4 }} - {{- end }} - {{- if .Values.primary.pgHbaConfiguration }} - pg_hba.conf: | - {{- include "common.tplvalues.render" ( dict "value" .Values.primary.pgHbaConfiguration "context" $ ) | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/primary/extended-configmap.yaml b/charts/postgresql/postgresql/templates/primary/extended-configmap.yaml deleted file mode 100644 index d9644fc..0000000 --- a/charts/postgresql/postgresql/templates/primary/extended-configmap.yaml +++ /dev/null @@ -1,20 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if (include "postgresql.v1.primary.createExtendedConfigmap" .) }} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ printf "%s-extended-configuration" (include "postgresql.v1.primary.fullname" .) }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: - override.conf: |- - {{- include "common.tplvalues.render" ( dict "value" .Values.primary.extendedConfiguration "context" $ ) | nindent 4 }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/primary/initialization-configmap.yaml b/charts/postgresql/postgresql/templates/primary/initialization-configmap.yaml deleted file mode 100644 index 739fe47..0000000 --- a/charts/postgresql/postgresql/templates/primary/initialization-configmap.yaml +++ /dev/null @@ -1,17 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.primary.initdb.scripts (not .Values.primary.initdb.scriptsConfigMap) }} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ printf "%s-init-scripts" (include "postgresql.v1.primary.fullname" .) }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: {{- include "common.tplvalues.render" (dict "value" .Values.primary.initdb.scripts "context" .) | nindent 2 }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/primary/metrics-configmap.yaml b/charts/postgresql/postgresql/templates/primary/metrics-configmap.yaml deleted file mode 100644 index 9e9e644..0000000 --- a/charts/postgresql/postgresql/templates/primary/metrics-configmap.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled .Values.metrics.customMetrics }} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ printf "%s-metrics" (include "postgresql.v1.primary.fullname" .) }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: - custom-metrics.yaml: {{ toYaml .Values.metrics.customMetrics | quote }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/primary/metrics-svc.yaml b/charts/postgresql/postgresql/templates/primary/metrics-svc.yaml deleted file mode 100644 index 581d5fe..0000000 --- a/charts/postgresql/postgresql/templates/primary/metrics-svc.yaml +++ /dev/null @@ -1,31 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.metrics.enabled }} -apiVersion: v1 -kind: Service -metadata: - name: {{ printf "%s-metrics" (include "postgresql.v1.primary.fullname" .) }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: metrics - {{- if or .Values.commonAnnotations .Values.metrics.service.annotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.metrics.service.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: ClusterIP - sessionAffinity: {{ .Values.metrics.service.sessionAffinity }} - {{- if .Values.metrics.service.clusterIP }} - clusterIP: {{ .Values.metrics.service.clusterIP }} - {{- end }} - ports: - - name: http-metrics - port: {{ .Values.metrics.service.ports.metrics }} - targetPort: http-metrics - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary -{{- end }} diff --git a/charts/postgresql/postgresql/templates/primary/networkpolicy.yaml b/charts/postgresql/postgresql/templates/primary/networkpolicy.yaml deleted file mode 100644 index 6e5e28f..0000000 --- a/charts/postgresql/postgresql/templates/primary/networkpolicy.yaml +++ /dev/null @@ -1,78 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.primary.networkPolicy.enabled }} -kind: NetworkPolicy -apiVersion: {{ include "common.capabilities.networkPolicy.apiVersion" . }} -metadata: - name: {{ include "postgresql.v1.primary.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.podLabels .Values.commonLabels ) "context" . ) }} - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: primary - policyTypes: - - Ingress - - Egress - {{- if .Values.primary.networkPolicy.allowExternalEgress }} - egress: - - {} - {{- else }} - egress: - # Allow dns resolution - - ports: - - port: 53 - protocol: UDP - - port: 53 - protocol: TCP - # Allow outbound connections to read-replicas - - ports: - - port: {{ .Values.containerPorts.postgresql }} - to: - - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 14 }} - app.kubernetes.io/component: read - {{- if .Values.primary.networkPolicy.extraEgress }} - {{- include "common.tplvalues.render" ( dict "value" .Values.primary.networkPolicy.extraEgress "context" $ ) | nindent 4 }} - {{- end }} - {{- end }} - ingress: - - ports: - - port: {{ .Values.containerPorts.postgresql }} - {{- if .Values.metrics.enabled }} - - port: {{ .Values.metrics.containerPorts.metrics }} - {{- end }} - {{- if not .Values.primary.networkPolicy.allowExternal }} - from: - - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 14 }} - - podSelector: - matchLabels: - {{ template "postgresql.v1.primary.fullname" . }}-client: "true" - {{- if .Values.primary.networkPolicy.ingressNSMatchLabels }} - - namespaceSelector: - matchLabels: - {{- range $key, $value := .Values.primary.networkPolicy.ingressNSMatchLabels }} - {{ $key | quote }}: {{ $value | quote }} - {{- end }} - {{- if .Values.primary.networkPolicy.ingressNSPodMatchLabels }} - podSelector: - matchLabels: - {{- range $key, $value := .Values.primary.networkPolicy.ingressNSPodMatchLabels }} - {{ $key | quote }}: {{ $value | quote }} - {{- end }} - {{- end }} - {{- end }} - {{- end }} - {{- if .Values.primary.networkPolicy.extraIngress }} - {{- include "common.tplvalues.render" ( dict "value" .Values.primary.networkPolicy.extraIngress "context" $ ) | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/primary/pdb.yaml b/charts/postgresql/postgresql/templates/primary/pdb.yaml deleted file mode 100644 index ca6caca..0000000 --- a/charts/postgresql/postgresql/templates/primary/pdb.yaml +++ /dev/null @@ -1,29 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.primary.pdb.create }} -apiVersion: {{ include "common.capabilities.policy.apiVersion" . }} -kind: PodDisruptionBudget -metadata: - name: {{ include "postgresql.v1.primary.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - {{- $labels := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $labels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- if .Values.primary.pdb.minAvailable }} - minAvailable: {{ .Values.primary.pdb.minAvailable }} - {{- end }} - {{- if or .Values.primary.pdb.maxUnavailable ( not .Values.primary.pdb.minAvailable ) }} - maxUnavailable: {{ .Values.primary.pdb.maxUnavailable | default 1 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.podLabels .Values.commonLabels ) "context" . ) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: primary -{{- end }} diff --git a/charts/postgresql/postgresql/templates/primary/preinitialization-configmap.yaml b/charts/postgresql/postgresql/templates/primary/preinitialization-configmap.yaml deleted file mode 100644 index 65606a4..0000000 --- a/charts/postgresql/postgresql/templates/primary/preinitialization-configmap.yaml +++ /dev/null @@ -1,17 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.primary.preInitDb.scripts (not .Values.primary.preInitDb.scriptsConfigMap) }} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ printf "%s-preinit-scripts" (include "postgresql.v1.primary.fullname" .) }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: {{- include "common.tplvalues.render" (dict "value" .Values.primary.preInitDb.scripts "context" .) | nindent 2 }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/primary/servicemonitor.yaml b/charts/postgresql/postgresql/templates/primary/servicemonitor.yaml deleted file mode 100644 index 22a0ce7..0000000 --- a/charts/postgresql/postgresql/templates/primary/servicemonitor.yaml +++ /dev/null @@ -1,46 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled .Values.metrics.serviceMonitor.enabled }} -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ include "postgresql.v1.primary.fullname" . }} - namespace: {{ default .Release.Namespace .Values.metrics.serviceMonitor.namespace | quote }} - {{- $labels := include "common.tplvalues.merge" ( dict "values" ( list .Values.metrics.serviceMonitor.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $labels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: metrics - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- if .Values.metrics.serviceMonitor.jobLabel }} - jobLabel: {{ .Values.metrics.serviceMonitor.jobLabel }} - {{- end }} - selector: - {{- $svcLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.metrics.serviceMonitor.selector .Values.commonLabels ) "context" . ) }} - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $svcLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: metrics - endpoints: - - port: http-metrics - {{- if .Values.metrics.serviceMonitor.interval }} - interval: {{ .Values.metrics.serviceMonitor.interval }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.scrapeTimeout }} - scrapeTimeout: {{ .Values.metrics.serviceMonitor.scrapeTimeout }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.relabelings }} - relabelings: {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.serviceMonitor.relabelings "context" $) | nindent 6 }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.metricRelabelings }} - metricRelabelings: {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.serviceMonitor.metricRelabelings "context" $) | nindent 6 }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.honorLabels }} - honorLabels: {{ .Values.metrics.serviceMonitor.honorLabels }} - {{- end }} - namespaceSelector: - matchNames: - - {{ .Release.Namespace | quote }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/primary/statefulset.yaml b/charts/postgresql/postgresql/templates/primary/statefulset.yaml deleted file mode 100644 index c110f91..0000000 --- a/charts/postgresql/postgresql/templates/primary/statefulset.yaml +++ /dev/null @@ -1,703 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- $customUser := include "postgresql.v1.username" . }} -apiVersion: {{ include "common.capabilities.statefulset.apiVersion" . }} -kind: StatefulSet -metadata: - name: {{ include "postgresql.v1.primary.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - {{- $labels := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $labels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary - {{- if or .Values.commonAnnotations .Values.primary.annotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - replicas: 1 - serviceName: {{ include "postgresql.v1.primary.svc.headless" . }} - {{- if .Values.primary.updateStrategy }} - updateStrategy: {{- toYaml .Values.primary.updateStrategy | nindent 4 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.podLabels .Values.commonLabels ) "context" . ) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: primary - template: - metadata: - name: {{ include "postgresql.v1.primary.fullname" . }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $podLabels "context" $ ) | nindent 8 }} - app.kubernetes.io/component: primary - {{- if or (include "postgresql.v1.primary.createConfigmap" .) (include "postgresql.v1.primary.createExtendedConfigmap" .) .Values.primary.podAnnotations }} - annotations: - {{- if (include "postgresql.v1.primary.createConfigmap" .) }} - checksum/configuration: {{ pick (include (print $.Template.BasePath "/primary/configmap.yaml") . | fromYaml) "data" | toYaml | sha256sum }} - {{- end }} - {{- if (include "postgresql.v1.primary.createExtendedConfigmap" .) }} - checksum/extended-configuration: {{ pick (include (print $.Template.BasePath "/primary/extended-configmap.yaml") . | fromYaml) "data" | toYaml | sha256sum }} - {{- end }} - {{- if .Values.primary.podAnnotations }} - {{- include "common.tplvalues.render" ( dict "value" .Values.primary.podAnnotations "context" $ ) | nindent 8 }} - {{- end }} - {{- end }} - spec: - {{- if .Values.primary.extraPodSpec }} - {{- include "common.tplvalues.render" (dict "value" .Values.primary.extraPodSpec "context" $) | nindent 6 }} - {{- end }} - serviceAccountName: {{ include "postgresql.v1.serviceAccountName" . }} - {{- include "postgresql.v1.imagePullSecrets" . | nindent 6 }} - automountServiceAccountToken: {{ .Values.primary.automountServiceAccountToken }} - {{- if .Values.primary.hostAliases }} - hostAliases: {{- include "common.tplvalues.render" (dict "value" .Values.primary.hostAliases "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.primary.affinity }} - affinity: {{- include "common.tplvalues.render" (dict "value" .Values.primary.affinity "context" $) | nindent 8 }} - {{- else }} - affinity: - podAffinity: {{- include "common.affinities.pods" (dict "type" .Values.primary.podAffinityPreset "component" "primary" "customLabels" $podLabels "context" $) | nindent 10 }} - podAntiAffinity: {{- include "common.affinities.pods" (dict "type" .Values.primary.podAntiAffinityPreset "component" "primary" "customLabels" $podLabels "context" $) | nindent 10 }} - nodeAffinity: {{- include "common.affinities.nodes" (dict "type" .Values.primary.nodeAffinityPreset.type "key" .Values.primary.nodeAffinityPreset.key "values" .Values.primary.nodeAffinityPreset.values) | nindent 10 }} - {{- end }} - {{- if .Values.primary.nodeSelector }} - nodeSelector: {{- include "common.tplvalues.render" (dict "value" .Values.primary.nodeSelector "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.primary.tolerations }} - tolerations: {{- include "common.tplvalues.render" (dict "value" .Values.primary.tolerations "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.primary.topologySpreadConstraints }} - topologySpreadConstraints: {{- include "common.tplvalues.render" (dict "value" .Values.primary.topologySpreadConstraints "context" .) | nindent 8 }} - {{- end }} - {{- if .Values.primary.priorityClassName }} - priorityClassName: {{ .Values.primary.priorityClassName }} - {{- end }} - {{- if .Values.primary.schedulerName }} - schedulerName: {{ .Values.primary.schedulerName | quote }} - {{- end }} - {{- if .Values.primary.terminationGracePeriodSeconds }} - terminationGracePeriodSeconds: {{ .Values.primary.terminationGracePeriodSeconds }} - {{- end }} - {{- if .Values.primary.podSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.primary.podSecurityContext "context" $) | nindent 8 }} - {{- end }} - hostNetwork: {{ .Values.primary.hostNetwork }} - hostIPC: {{ .Values.primary.hostIPC }} - {{- if or (and .Values.tls.enabled (not .Values.volumePermissions.enabled)) (and .Values.volumePermissions.enabled (or .Values.primary.persistence.enabled .Values.shmVolume.enabled)) .Values.primary.initContainers }} - initContainers: - {{- if and .Values.tls.enabled (not .Values.volumePermissions.enabled) }} - - name: copy-certs - image: {{ include "postgresql.v1.volumePermissions.image" . }} - imagePullPolicy: {{ .Values.volumePermissions.image.pullPolicy | quote }} - {{- if .Values.primary.resources }} - resources: {{- toYaml .Values.primary.resources | nindent 12 }} - {{- else if ne .Values.primary.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.primary.resourcesPreset) | nindent 12 }} - {{- end }} - # We don't require a privileged container in this case - {{- if .Values.primary.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.primary.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - command: - - /bin/sh - - -ec - - | - cp /tmp/certs/* /opt/bitnami/postgresql/certs/ - chmod 600 {{ include "postgresql.v1.tlsCertKey" . }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: raw-certificates - mountPath: /tmp/certs - - name: postgresql-certificates - mountPath: /opt/bitnami/postgresql/certs - {{- else if and .Values.volumePermissions.enabled (or .Values.primary.persistence.enabled .Values.shmVolume.enabled) }} - - name: init-chmod-data - image: {{ include "postgresql.v1.volumePermissions.image" . }} - imagePullPolicy: {{ .Values.volumePermissions.image.pullPolicy | quote }} - {{- if .Values.volumePermissions.resources }} - resources: {{- toYaml .Values.volumePermissions.resources | nindent 12 }} - {{- else if ne .Values.volumePermissions.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.volumePermissions.resourcesPreset) | nindent 12 }} - {{- end }} - command: - - /bin/sh - - -ec - - | - {{- if .Values.primary.persistence.enabled }} - {{- if eq ( toString ( .Values.volumePermissions.containerSecurityContext.runAsUser )) "auto" }} - chown `id -u`:`id -G | cut -d " " -f2` {{ .Values.primary.persistence.mountPath }} - {{- else }} - chown {{ .Values.primary.containerSecurityContext.runAsUser }}:{{ .Values.primary.podSecurityContext.fsGroup }} {{ .Values.primary.persistence.mountPath }} - {{- end }} - mkdir -p {{ .Values.primary.persistence.mountPath }}/data {{- if (include "postgresql.v1.mountConfigurationCM" .) }} {{ .Values.primary.persistence.mountPath }}/conf {{- end }} - chmod 700 {{ .Values.primary.persistence.mountPath }}/data {{- if (include "postgresql.v1.mountConfigurationCM" .) }} {{ .Values.primary.persistence.mountPath }}/conf {{- end }} - find {{ .Values.primary.persistence.mountPath }} -mindepth 1 -maxdepth 1 {{- if not (include "postgresql.v1.mountConfigurationCM" .) }} -not -name "conf" {{- end }} -not -name ".snapshot" -not -name "lost+found" | \ - {{- if eq ( toString ( .Values.volumePermissions.containerSecurityContext.runAsUser )) "auto" }} - xargs -r chown -R `id -u`:`id -G | cut -d " " -f2` - {{- else }} - xargs -r chown -R {{ .Values.primary.containerSecurityContext.runAsUser }}:{{ .Values.primary.podSecurityContext.fsGroup }} - {{- end }} - {{- end }} - {{- if .Values.shmVolume.enabled }} - chmod -R 777 /dev/shm - {{- end }} - {{- if .Values.tls.enabled }} - cp /tmp/certs/* /opt/bitnami/postgresql/certs/ - {{- if eq ( toString ( .Values.volumePermissions.containerSecurityContext.runAsUser )) "auto" }} - chown -R `id -u`:`id -G | cut -d " " -f2` /opt/bitnami/postgresql/certs/ - {{- else }} - chown -R {{ .Values.primary.containerSecurityContext.runAsUser }}:{{ .Values.primary.podSecurityContext.fsGroup }} /opt/bitnami/postgresql/certs/ - {{- end }} - chmod 600 {{ include "postgresql.v1.tlsCertKey" . }} - {{- end }} - {{- if eq ( toString ( .Values.volumePermissions.containerSecurityContext.runAsUser )) "auto" }} - securityContext: {{- omit .Values.volumePermissions.containerSecurityContext "runAsUser" | toYaml | nindent 12 }} - {{- else }} - securityContext: {{- .Values.volumePermissions.containerSecurityContext | toYaml | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: {{ .Values.primary.persistence.volumeName }} - mountPath: {{ .Values.primary.persistence.mountPath }} - {{- if .Values.primary.persistence.subPath }} - subPath: {{ .Values.primary.persistence.subPath }} - {{- end }} - {{- if .Values.shmVolume.enabled }} - - name: dshm - mountPath: /dev/shm - {{- end }} - {{- if .Values.tls.enabled }} - - name: raw-certificates - mountPath: /tmp/certs - - name: postgresql-certificates - mountPath: /opt/bitnami/postgresql/certs - {{- end }} - {{- end }} - {{- if .Values.primary.initContainers }} - {{- include "common.tplvalues.render" ( dict "value" .Values.primary.initContainers "context" $ ) | nindent 8 }} - {{- end }} - {{- end }} - containers: - - name: postgresql - image: {{ include "postgresql.v1.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy | quote }} - {{- if .Values.primary.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.primary.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - {{- else if .Values.primary.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.primary.command "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- else if .Values.primary.args }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.primary.args "context" $) | nindent 12 }} - {{- end }} - env: - - name: BITNAMI_DEBUG - value: {{ ternary "true" "false" (or .Values.image.debug .Values.diagnosticMode.enabled) | quote }} - - name: POSTGRESQL_PORT_NUMBER - value: {{ .Values.containerPorts.postgresql | quote }} - - name: POSTGRESQL_VOLUME_DIR - value: {{ .Values.primary.persistence.mountPath | quote }} - {{- if .Values.primary.persistence.mountPath }} - - name: PGDATA - value: {{ .Values.postgresqlDataDir | quote }} - {{- end }} - # Authentication - {{- if or (eq $customUser "postgres") (empty $customUser) }} - {{- if .Values.auth.enablePostgresUser }} - {{- if .Values.auth.usePasswordFiles }} - - name: POSTGRES_PASSWORD_FILE - value: {{ printf "/opt/bitnami/postgresql/secrets/%s" (include "postgresql.v1.adminPasswordKey" .) }} - {{- else }} - - name: POSTGRES_PASSWORD - valueFrom: - secretKeyRef: - name: {{ include "postgresql.v1.secretName" . }} - key: {{ include "postgresql.v1.adminPasswordKey" . }} - {{- end }} - {{- else }} - - name: ALLOW_EMPTY_PASSWORD - value: "true" - {{- end }} - {{- else }} - - name: POSTGRES_USER - value: {{ $customUser | quote }} - {{- if .Values.auth.usePasswordFiles }} - - name: POSTGRES_PASSWORD_FILE - value: {{ printf "/opt/bitnami/postgresql/secrets/%s" (include "postgresql.v1.userPasswordKey" .) }} - {{- else }} - - name: POSTGRES_PASSWORD - valueFrom: - secretKeyRef: - name: {{ include "postgresql.v1.secretName" . }} - key: {{ include "postgresql.v1.userPasswordKey" . }} - {{- end }} - {{- if .Values.auth.enablePostgresUser }} - {{- if .Values.auth.usePasswordFiles }} - - name: POSTGRES_POSTGRES_PASSWORD_FILE - value: {{ printf "/opt/bitnami/postgresql/secrets/%s" (include "postgresql.v1.adminPasswordKey" .) }} - {{- else }} - - name: POSTGRES_POSTGRES_PASSWORD - valueFrom: - secretKeyRef: - name: {{ include "postgresql.v1.secretName" . }} - key: {{ include "postgresql.v1.adminPasswordKey" . }} - {{- end }} - {{- end }} - {{- end }} - {{- if (include "postgresql.v1.database" .) }} - - name: POSTGRES_DATABASE - value: {{ (include "postgresql.v1.database" .) | quote }} - {{- end }} - {{- if or (eq .Values.architecture "replication") .Values.primary.standby.enabled }} - # Replication - - name: POSTGRES_REPLICATION_MODE - value: {{ ternary "slave" "master" .Values.primary.standby.enabled | quote }} - - name: POSTGRES_REPLICATION_USER - value: {{ .Values.auth.replicationUsername | quote }} - {{- if .Values.auth.usePasswordFiles }} - - name: POSTGRES_REPLICATION_PASSWORD_FILE - value: {{ printf "/opt/bitnami/postgresql/secrets/%s" (include "postgresql.v1.replicationPasswordKey" .) }} - {{- else }} - - name: POSTGRES_REPLICATION_PASSWORD - valueFrom: - secretKeyRef: - name: {{ include "postgresql.v1.secretName" . }} - key: {{ include "postgresql.v1.replicationPasswordKey" . }} - {{- end }} - {{- if ne .Values.replication.synchronousCommit "off" }} - - name: POSTGRES_SYNCHRONOUS_COMMIT_MODE - value: {{ .Values.replication.synchronousCommit | quote }} - - name: POSTGRES_NUM_SYNCHRONOUS_REPLICAS - value: {{ .Values.replication.numSynchronousReplicas | quote }} - {{- end }} - - name: POSTGRES_CLUSTER_APP_NAME - value: {{ .Values.replication.applicationName }} - {{- end }} - {{- if .Values.primary.initdb.args }} - # Initdb - - name: POSTGRES_INITDB_ARGS - value: {{ .Values.primary.initdb.args | quote }} - {{- end }} - {{- if .Values.primary.initdb.postgresqlWalDir }} - - name: POSTGRES_INITDB_WALDIR - value: {{ .Values.primary.initdb.postgresqlWalDir | quote }} - {{- end }} - {{- if .Values.primary.initdb.user }} - - name: POSTGRES_INITSCRIPTS_USERNAME - value: {{ .Values.primary.initdb.user }} - {{- end }} - {{- if .Values.primary.initdb.password }} - - name: POSTGRES_INITSCRIPTS_PASSWORD - value: {{ .Values.primary.initdb.password | quote }} - {{- end }} - {{- if .Values.primary.standby.enabled }} - # Standby - - name: POSTGRES_MASTER_HOST - value: {{ .Values.primary.standby.primaryHost }} - - name: POSTGRES_MASTER_PORT_NUMBER - value: {{ .Values.primary.standby.primaryPort | quote }} - {{- end }} - # LDAP - - name: POSTGRESQL_ENABLE_LDAP - value: {{ ternary "yes" "no" .Values.ldap.enabled | quote }} - {{- if .Values.ldap.enabled }} - {{- if or .Values.ldap.url .Values.ldap.uri }} - - name: POSTGRESQL_LDAP_URL - value: {{ coalesce .Values.ldap.url .Values.ldap.uri }} - {{- else }} - - name: POSTGRESQL_LDAP_SERVER - value: {{ .Values.ldap.server }} - - name: POSTGRESQL_LDAP_PORT - value: {{ .Values.ldap.port | quote }} - - name: POSTGRESQL_LDAP_SCHEME - value: {{ .Values.ldap.scheme }} - {{- if (include "postgresql.v1.ldap.tls.enabled" .) }} - - name: POSTGRESQL_LDAP_TLS - value: "1" - {{- end }} - - name: POSTGRESQL_LDAP_PREFIX - value: {{ .Values.ldap.prefix | quote }} - - name: POSTGRESQL_LDAP_SUFFIX - value: {{ .Values.ldap.suffix | quote }} - - name: POSTGRESQL_LDAP_BASE_DN - value: {{ coalesce .Values.ldap.baseDN .Values.ldap.basedn }} - - name: POSTGRESQL_LDAP_BIND_DN - value: {{ coalesce .Values.ldap.bindDN .Values.ldap.binddn}} - {{- if or (not (empty .Values.ldap.bind_password)) (not (empty .Values.ldap.bindpw)) }} - - name: POSTGRESQL_LDAP_BIND_PASSWORD - valueFrom: - secretKeyRef: - name: {{ include "postgresql.v1.secretName" . }} - key: ldap-password - {{- end }} - - name: POSTGRESQL_LDAP_SEARCH_ATTR - value: {{ coalesce .Values.ldap.search_attr .Values.ldap.searchAttribute }} - - name: POSTGRESQL_LDAP_SEARCH_FILTER - value: {{ coalesce .Values.ldap.search_filter .Values.ldap.searchFilter }} - {{- end }} - {{- end }} - # TLS - - name: POSTGRESQL_ENABLE_TLS - value: {{ ternary "yes" "no" .Values.tls.enabled | quote }} - {{- if .Values.tls.enabled }} - - name: POSTGRESQL_TLS_PREFER_SERVER_CIPHERS - value: {{ ternary "yes" "no" .Values.tls.preferServerCiphers | quote }} - - name: POSTGRESQL_TLS_CERT_FILE - value: {{ include "postgresql.v1.tlsCert" . }} - - name: POSTGRESQL_TLS_KEY_FILE - value: {{ include "postgresql.v1.tlsCertKey" . }} - {{- if .Values.tls.certCAFilename }} - - name: POSTGRESQL_TLS_CA_FILE - value: {{ include "postgresql.v1.tlsCACert" . }} - {{- end }} - {{- if .Values.tls.crlFilename }} - - name: POSTGRESQL_TLS_CRL_FILE - value: {{ include "postgresql.v1.tlsCRL" . }} - {{- end }} - {{- end }} - # Audit - - name: POSTGRESQL_LOG_HOSTNAME - value: {{ .Values.audit.logHostname | quote }} - - name: POSTGRESQL_LOG_CONNECTIONS - value: {{ .Values.audit.logConnections | quote }} - - name: POSTGRESQL_LOG_DISCONNECTIONS - value: {{ .Values.audit.logDisconnections | quote }} - {{- if .Values.audit.logLinePrefix }} - - name: POSTGRESQL_LOG_LINE_PREFIX - value: {{ .Values.audit.logLinePrefix | quote }} - {{- end }} - {{- if .Values.audit.logTimezone }} - - name: POSTGRESQL_LOG_TIMEZONE - value: {{ .Values.audit.logTimezone | quote }} - {{- end }} - {{- if .Values.audit.pgAuditLog }} - - name: POSTGRESQL_PGAUDIT_LOG - value: {{ .Values.audit.pgAuditLog | quote }} - {{- end }} - - name: POSTGRESQL_PGAUDIT_LOG_CATALOG - value: {{ .Values.audit.pgAuditLogCatalog | quote }} - # Others - - name: POSTGRESQL_CLIENT_MIN_MESSAGES - value: {{ .Values.audit.clientMinMessages | quote }} - - name: POSTGRESQL_SHARED_PRELOAD_LIBRARIES - value: {{ .Values.postgresqlSharedPreloadLibraries | quote }} - {{- if .Values.primary.extraEnvVars }} - {{- include "common.tplvalues.render" (dict "value" .Values.primary.extraEnvVars "context" $) | nindent 12 }} - {{- end }} - {{- if or .Values.primary.extraEnvVarsCM .Values.primary.extraEnvVarsSecret }} - envFrom: - {{- if .Values.primary.extraEnvVarsCM }} - - configMapRef: - name: {{ .Values.primary.extraEnvVarsCM }} - {{- end }} - {{- if .Values.primary.extraEnvVarsSecret }} - - secretRef: - name: {{ .Values.primary.extraEnvVarsSecret }} - {{- end }} - {{- end }} - ports: - - name: tcp-postgresql - containerPort: {{ .Values.containerPorts.postgresql }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.primary.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.primary.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.primary.startupProbe.enabled }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.primary.startupProbe "enabled") "context" $) | nindent 12 }} - exec: - command: - - /bin/sh - - -c - {{- if (include "postgresql.v1.database" .) }} - - exec pg_isready -U {{ default "postgres" $customUser | quote }} -d "dbname={{ include "postgresql.v1.database" . }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} sslcert={{ include "postgresql.v1.tlsCert" . }} sslkey={{ include "postgresql.v1.tlsCertKey" . }}{{- end }}" -h 127.0.0.1 -p {{ .Values.containerPorts.postgresql }} - {{- else }} - - exec pg_isready -U {{ default "postgres" $customUser | quote }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} -d "sslcert={{ include "postgresql.v1.tlsCert" . }} sslkey={{ include "postgresql.v1.tlsCertKey" . }}"{{- end }} -h 127.0.0.1 -p {{ .Values.containerPorts.postgresql }} - {{- end }} - {{- end }} - {{- if .Values.primary.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.primary.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.primary.livenessProbe.enabled }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.primary.livenessProbe "enabled") "context" $) | nindent 12 }} - exec: - command: - - /bin/sh - - -c - {{- if (include "postgresql.v1.database" .) }} - - exec pg_isready -U {{ default "postgres" $customUser | quote }} -d "dbname={{ include "postgresql.v1.database" . }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} sslcert={{ include "postgresql.v1.tlsCert" . }} sslkey={{ include "postgresql.v1.tlsCertKey" . }}{{- end }}" -h 127.0.0.1 -p {{ .Values.containerPorts.postgresql }} - {{- else }} - - exec pg_isready -U {{ default "postgres" $customUser | quote }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} -d "sslcert={{ include "postgresql.v1.tlsCert" . }} sslkey={{ include "postgresql.v1.tlsCertKey" . }}"{{- end }} -h 127.0.0.1 -p {{ .Values.containerPorts.postgresql }} - {{- end }} - {{- end }} - {{- if .Values.primary.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.primary.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.primary.readinessProbe.enabled }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.primary.readinessProbe "enabled") "context" $) | nindent 12 }} - exec: - command: - - /bin/sh - - -c - - -e - {{- include "postgresql.v1.readinessProbeCommand" . | nindent 16 }} - {{- end }} - {{- end }} - {{- if .Values.primary.resources }} - resources: {{- toYaml .Values.primary.resources | nindent 12 }} - {{- else if ne .Values.primary.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.primary.resourcesPreset) | nindent 12 }} - {{- end }} - {{- if .Values.primary.lifecycleHooks }} - lifecycle: {{- include "common.tplvalues.render" (dict "value" .Values.primary.lifecycleHooks "context" $) | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: empty-dir - mountPath: /opt/bitnami/postgresql/conf - subPath: app-conf-dir - - name: empty-dir - mountPath: /opt/bitnami/postgresql/tmp - subPath: app-tmp-dir - {{- if or .Values.primary.preInitDb.scriptsConfigMap .Values.primary.preInitDb.scripts }} - - name: custom-preinit-scripts - mountPath: /docker-entrypoint-preinitdb.d/ - {{- end }} - {{- if .Values.primary.preInitDb.scriptsSecret }} - - name: custom-preinit-scripts-secret - mountPath: /docker-entrypoint-preinitdb.d/secret - {{- end }} - {{- if or .Values.primary.initdb.scriptsConfigMap .Values.primary.initdb.scripts }} - - name: custom-init-scripts - mountPath: /docker-entrypoint-initdb.d/ - {{- end }} - {{- if .Values.primary.initdb.scriptsSecret }} - - name: custom-init-scripts-secret - mountPath: /docker-entrypoint-initdb.d/secret - {{- end }} - {{- if or .Values.primary.extendedConfiguration .Values.primary.existingExtendedConfigmap }} - - name: postgresql-extended-config - mountPath: {{ .Values.primary.persistence.mountPath }}/conf/conf.d/ - {{- end }} - {{- if .Values.auth.usePasswordFiles }} - - name: postgresql-password - mountPath: /opt/bitnami/postgresql/secrets/ - {{- end }} - {{- if .Values.tls.enabled }} - - name: postgresql-certificates - mountPath: /opt/bitnami/postgresql/certs - readOnly: true - {{- end }} - {{- if .Values.shmVolume.enabled }} - - name: dshm - mountPath: /dev/shm - {{- end }} - - name: {{ .Values.primary.persistence.volumeName }} - mountPath: {{ .Values.primary.persistence.mountPath }} - {{- if .Values.primary.persistence.subPath }} - subPath: {{ .Values.primary.persistence.subPath }} - {{- end }} - {{- if or .Values.primary.configuration .Values.primary.pgHbaConfiguration .Values.primary.existingConfigmap }} - - name: postgresql-config - mountPath: {{ .Values.primary.persistence.mountPath }}/conf - {{- end }} - {{- if .Values.primary.extraVolumeMounts }} - {{- include "common.tplvalues.render" (dict "value" .Values.primary.extraVolumeMounts "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.metrics.enabled }} - - name: metrics - image: {{ include "postgresql.v1.metrics.image" . }} - imagePullPolicy: {{ .Values.metrics.image.pullPolicy | quote }} - {{- if .Values.metrics.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.metrics.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- else if or .Values.metrics.customMetrics .Values.metrics.collectors }} - args: - {{- if .Values.metrics.customMetrics }} - - --extend.query-path - - /conf/custom-metrics.yaml - {{- end }} - {{- range $name, $enabled := .Values.metrics.collectors }} - - --{{ if not $enabled }}no-{{ end }}collector.{{ $name }} - {{- end }} - {{- end }} - env: - - name: DATA_SOURCE_URI - value: {{ printf "127.0.0.1:%d/postgres?sslmode=disable" (int (include "postgresql.v1.service.port" .)) }} - {{- $pwdKey := ternary (include "postgresql.v1.adminPasswordKey" .) (include "postgresql.v1.userPasswordKey" .) (or (eq $customUser "postgres") (empty $customUser)) }} - {{- if .Values.auth.usePasswordFiles }} - - name: DATA_SOURCE_PASS_FILE - value: {{ printf "/opt/bitnami/postgresql/secrets/%s" $pwdKey }} - {{- else }} - - name: DATA_SOURCE_PASS - valueFrom: - secretKeyRef: - name: {{ include "postgresql.v1.secretName" . }} - key: {{ $pwdKey }} - {{- end }} - - name: DATA_SOURCE_USER - value: {{ default "postgres" $customUser | quote }} - {{- if .Values.metrics.extraEnvVars }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.extraEnvVars "context" $) | nindent 12 }} - {{- end }} - ports: - - name: http-metrics - containerPort: {{ .Values.metrics.containerPorts.metrics }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.metrics.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.startupProbe.enabled }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.startupProbe "enabled") "context" $) | nindent 12 }} - tcpSocket: - port: http-metrics - {{- end }} - {{- if .Values.metrics.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.livenessProbe.enabled }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.livenessProbe "enabled") "context" $) | nindent 12 }} - httpGet: - path: / - port: http-metrics - {{- end }} - {{- if .Values.metrics.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.readinessProbe.enabled }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.readinessProbe "enabled") "context" $) | nindent 12 }} - httpGet: - path: / - port: http-metrics - {{- end }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - {{- if .Values.auth.usePasswordFiles }} - - name: postgresql-password - mountPath: /opt/bitnami/postgresql/secrets/ - {{- end }} - {{- if .Values.metrics.customMetrics }} - - name: custom-metrics - mountPath: /conf - readOnly: true - {{- end }} - {{- if .Values.metrics.resources }} - resources: {{- toYaml .Values.metrics.resources | nindent 12 }} - {{- else if ne .Values.metrics.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.metrics.resourcesPreset) | nindent 12 }} - {{- end }} - {{- end }} - {{- if .Values.primary.sidecars }} - {{- include "common.tplvalues.render" ( dict "value" .Values.primary.sidecars "context" $ ) | nindent 8 }} - {{- end }} - volumes: - - name: empty-dir - emptyDir: {} - {{- if or .Values.primary.configuration .Values.primary.pgHbaConfiguration .Values.primary.existingConfigmap }} - - name: postgresql-config - configMap: - name: {{ include "postgresql.v1.primary.configmapName" . }} - {{- end }} - {{- if or .Values.primary.extendedConfiguration .Values.primary.existingExtendedConfigmap }} - - name: postgresql-extended-config - configMap: - name: {{ include "postgresql.v1.primary.extendedConfigmapName" . }} - {{- end }} - {{- if .Values.auth.usePasswordFiles }} - - name: postgresql-password - secret: - secretName: {{ include "postgresql.v1.secretName" . }} - {{- end }} - {{- if or .Values.primary.preInitDb.scriptsConfigMap .Values.primary.preInitDb.scripts }} - - name: custom-preinit-scripts - configMap: - name: {{ include "postgresql.v1.preInitDb.scriptsCM" . }} - {{- end }} - {{- if .Values.primary.preInitDb.scriptsSecret }} - - name: custom-preinit-scripts-secret - secret: - secretName: {{ tpl .Values.primary.preInitDb.scriptsSecret $ }} - {{- end }} - {{- if or .Values.primary.initdb.scriptsConfigMap .Values.primary.initdb.scripts }} - - name: custom-init-scripts - configMap: - name: {{ include "postgresql.v1.initdb.scriptsCM" . }} - {{- end }} - {{- if .Values.primary.initdb.scriptsSecret }} - - name: custom-init-scripts-secret - secret: - secretName: {{ tpl .Values.primary.initdb.scriptsSecret $ }} - {{- end }} - {{- if .Values.tls.enabled }} - - name: raw-certificates - secret: - secretName: {{ include "postgresql.v1.tlsSecretName" . }} - - name: postgresql-certificates - emptyDir: {} - {{- end }} - {{- if .Values.primary.extraVolumes }} - {{- include "common.tplvalues.render" ( dict "value" .Values.primary.extraVolumes "context" $ ) | nindent 8 }} - {{- end }} - {{- if and .Values.metrics.enabled .Values.metrics.customMetrics }} - - name: custom-metrics - configMap: - name: {{ printf "%s-metrics" (include "postgresql.v1.primary.fullname" .) }} - {{- end }} - {{- if .Values.shmVolume.enabled }} - - name: dshm - emptyDir: - medium: Memory - {{- if .Values.shmVolume.sizeLimit }} - sizeLimit: {{ .Values.shmVolume.sizeLimit }} - {{- end }} - {{- end }} - {{- if and .Values.primary.persistence.enabled .Values.primary.persistence.existingClaim }} - - name: {{ .Values.primary.persistence.volumeName }} - persistentVolumeClaim: - claimName: {{ tpl .Values.primary.persistence.existingClaim $ }} - {{- else if not .Values.primary.persistence.enabled }} - - name: {{ .Values.primary.persistence.volumeName }} - emptyDir: {} - {{- else }} - {{- if .Values.primary.persistentVolumeClaimRetentionPolicy.enabled }} - persistentVolumeClaimRetentionPolicy: - whenDeleted: {{ .Values.primary.persistentVolumeClaimRetentionPolicy.whenDeleted }} - whenScaled: {{ .Values.primary.persistentVolumeClaimRetentionPolicy.whenScaled }} - {{- end }} - volumeClaimTemplates: - - apiVersion: v1 - kind: PersistentVolumeClaim - metadata: - name: {{ .Values.primary.persistence.volumeName }} - {{- if .Values.primary.persistence.annotations }} - annotations: {{- include "common.tplvalues.render" (dict "value" .Values.primary.persistence.annotations "context" $) | nindent 10 }} - {{- end }} - {{- if .Values.primary.persistence.labels }} - labels: {{- include "common.tplvalues.render" (dict "value" .Values.primary.persistence.labels "context" $) | nindent 10 }} - {{- end }} - spec: - accessModes: - {{- range .Values.primary.persistence.accessModes }} - - {{ . | quote }} - {{- end }} - {{- if .Values.primary.persistence.dataSource }} - dataSource: {{- include "common.tplvalues.render" (dict "value" .Values.primary.persistence.dataSource "context" $) | nindent 10 }} - {{- end }} - resources: - requests: - storage: {{ .Values.primary.persistence.size | quote }} - {{- if .Values.primary.persistence.selector }} - selector: {{- include "common.tplvalues.render" (dict "value" .Values.primary.persistence.selector "context" $) | nindent 10 }} - {{- end }} - {{- include "common.storage.class" (dict "persistence" .Values.primary.persistence "global" .Values.global) | nindent 8 }} - {{- end }} diff --git a/charts/postgresql/postgresql/templates/primary/svc-headless.yaml b/charts/postgresql/postgresql/templates/primary/svc-headless.yaml deleted file mode 100644 index ba918f4..0000000 --- a/charts/postgresql/postgresql/templates/primary/svc-headless.yaml +++ /dev/null @@ -1,31 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -apiVersion: v1 -kind: Service -metadata: - name: {{ include "postgresql.v1.primary.svc.headless" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary - annotations: - {{- if or .Values.primary.service.headless.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.service.headless.annotations .Values.commonAnnotations ) "context" . ) }} - {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: ClusterIP - clusterIP: None - # We want all pods in the StatefulSet to have their addresses published for - # the sake of the other Postgresql pods even before they're ready, since they - # have to be able to talk to each other in order to become ready. - publishNotReadyAddresses: true - ports: - - name: tcp-postgresql - port: {{ template "postgresql.v1.service.port" . }} - targetPort: tcp-postgresql - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary diff --git a/charts/postgresql/postgresql/templates/primary/svc.yaml b/charts/postgresql/postgresql/templates/primary/svc.yaml deleted file mode 100644 index 62fe7d4..0000000 --- a/charts/postgresql/postgresql/templates/primary/svc.yaml +++ /dev/null @@ -1,54 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -apiVersion: v1 -kind: Service -metadata: - name: {{ include "postgresql.v1.primary.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary - {{- if or .Values.commonAnnotations .Values.primary.service.annotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.service.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: {{ .Values.primary.service.type }} - {{- if or (eq .Values.primary.service.type "LoadBalancer") (eq .Values.primary.service.type "NodePort") }} - externalTrafficPolicy: {{ .Values.primary.service.externalTrafficPolicy | quote }} - {{- end }} - {{- if and (eq .Values.primary.service.type "LoadBalancer") (not (empty .Values.primary.service.loadBalancerSourceRanges)) }} - loadBalancerSourceRanges: {{ .Values.primary.service.loadBalancerSourceRanges | toJson}} - {{- end }} - {{- if and (eq .Values.primary.service.type "LoadBalancer") (not (empty .Values.primary.service.loadBalancerClass)) }} - loadBalancerClass: {{ .Values.primary.service.loadBalancerClass }} - {{- end }} - {{- if and (eq .Values.primary.service.type "LoadBalancer") (not (empty .Values.primary.service.loadBalancerIP)) }} - loadBalancerIP: {{ .Values.primary.service.loadBalancerIP }} - {{- end }} - {{- if and .Values.primary.service.clusterIP (eq .Values.primary.service.type "ClusterIP") }} - clusterIP: {{ .Values.primary.service.clusterIP }} - {{- end }} - {{- if .Values.primary.service.sessionAffinity }} - sessionAffinity: {{ .Values.primary.service.sessionAffinity }} - {{- end }} - {{- if .Values.primary.service.sessionAffinityConfig }} - sessionAffinityConfig: {{- include "common.tplvalues.render" (dict "value" .Values.primary.service.sessionAffinityConfig "context" $) | nindent 4 }} - {{- end }} - ports: - - name: tcp-postgresql - port: {{ template "postgresql.v1.service.port" . }} - targetPort: tcp-postgresql - {{- if and (or (eq .Values.primary.service.type "NodePort") (eq .Values.primary.service.type "LoadBalancer")) (not (empty .Values.primary.service.nodePorts.postgresql)) }} - nodePort: {{ .Values.primary.service.nodePorts.postgresql }} - {{- else if eq .Values.primary.service.type "ClusterIP" }} - nodePort: null - {{- end }} - {{- if .Values.primary.service.extraPorts }} - {{- include "common.tplvalues.render" (dict "value" .Values.primary.service.extraPorts "context" $) | nindent 4 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.primary.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: primary diff --git a/charts/postgresql/postgresql/templates/prometheusrule.yaml b/charts/postgresql/postgresql/templates/prometheusrule.yaml deleted file mode 100644 index b6d17ba..0000000 --- a/charts/postgresql/postgresql/templates/prometheusrule.yaml +++ /dev/null @@ -1,22 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled .Values.metrics.prometheusRule.enabled }} -apiVersion: monitoring.coreos.com/v1 -kind: PrometheusRule -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ default .Release.Namespace .Values.metrics.prometheusRule.namespace | quote }} - {{- $labels := include "common.tplvalues.merge" ( dict "values" ( list .Values.metrics.prometheusRule.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $labels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: metrics - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - groups: - - name: {{ include "common.names.fullname" . }} - rules: {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.prometheusRule.rules "context" $ ) | nindent 8 }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/psp.yaml b/charts/postgresql/postgresql/templates/psp.yaml deleted file mode 100644 index b522002..0000000 --- a/charts/postgresql/postgresql/templates/psp.yaml +++ /dev/null @@ -1,42 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and (include "common.capabilities.psp.supported" .) .Values.psp.create }} -apiVersion: policy/v1beta1 -kind: PodSecurityPolicy -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - privileged: false - volumes: - - 'configMap' - - 'secret' - - 'persistentVolumeClaim' - - 'emptyDir' - - 'projected' - hostNetwork: false - hostIPC: false - hostPID: false - runAsUser: - rule: 'RunAsAny' - seLinux: - rule: 'RunAsAny' - supplementalGroups: - rule: 'MustRunAs' - ranges: - - min: 1 - max: 65535 - fsGroup: - rule: 'MustRunAs' - ranges: - - min: 1 - max: 65535 - readOnlyRootFilesystem: false -{{- end }} diff --git a/charts/postgresql/postgresql/templates/read/extended-configmap.yaml b/charts/postgresql/postgresql/templates/read/extended-configmap.yaml deleted file mode 100644 index d4db2ae..0000000 --- a/charts/postgresql/postgresql/templates/read/extended-configmap.yaml +++ /dev/null @@ -1,20 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if (include "postgresql.v1.readReplicas.createExtendedConfigmap" .) }} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ printf "%s-extended-configuration" (include "postgresql.v1.readReplica.fullname" .) }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: read - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: - override.conf: |- - {{- include "common.tplvalues.render" ( dict "value" .Values.readReplicas.extendedConfiguration "context" $ ) | nindent 4 }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/read/metrics-configmap.yaml b/charts/postgresql/postgresql/templates/read/metrics-configmap.yaml deleted file mode 100644 index ca019c9..0000000 --- a/charts/postgresql/postgresql/templates/read/metrics-configmap.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled .Values.metrics.customMetrics (eq .Values.architecture "replication") }} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ printf "%s-metrics" (include "postgresql.v1.readReplica.fullname" .) }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: - custom-metrics.yaml: {{ toYaml .Values.metrics.customMetrics | quote }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/read/metrics-svc.yaml b/charts/postgresql/postgresql/templates/read/metrics-svc.yaml deleted file mode 100644 index cdb2647..0000000 --- a/charts/postgresql/postgresql/templates/read/metrics-svc.yaml +++ /dev/null @@ -1,31 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled (eq .Values.architecture "replication") }} -apiVersion: v1 -kind: Service -metadata: - name: {{ printf "%s-metrics" (include "postgresql.v1.readReplica.fullname" .) }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: metrics-read - {{- if or .Values.commonAnnotations .Values.metrics.service.annotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.metrics.service.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: ClusterIP - sessionAffinity: {{ .Values.metrics.service.sessionAffinity }} - {{- if .Values.metrics.service.clusterIP }} - clusterIP: {{ .Values.metrics.service.clusterIP }} - {{- end }} - ports: - - name: http-metrics - port: {{ .Values.metrics.service.ports.metrics }} - targetPort: http-metrics - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.readReplicas.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: read -{{- end }} diff --git a/charts/postgresql/postgresql/templates/read/networkpolicy.yaml b/charts/postgresql/postgresql/templates/read/networkpolicy.yaml deleted file mode 100644 index 9c730d8..0000000 --- a/charts/postgresql/postgresql/templates/read/networkpolicy.yaml +++ /dev/null @@ -1,80 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if eq .Values.architecture "replication" }} -{{- if .Values.readReplicas.networkPolicy.enabled }} -kind: NetworkPolicy -apiVersion: {{ include "common.capabilities.networkPolicy.apiVersion" . }} -metadata: - name: {{ include "postgresql.v1.readReplica.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: read - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.readReplicas.podLabels .Values.commonLabels ) "context" . ) }} - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: read - policyTypes: - - Ingress - - Egress - {{- if .Values.readReplicas.networkPolicy.allowExternalEgress }} - egress: - - {} - {{- else }} - egress: - # Allow dns resolution - - ports: - - port: 53 - protocol: UDP - - port: 53 - protocol: TCP - # Allow outbound connections to primary - - ports: - - port: {{ .Values.containerPorts.postgresql }} - to: - - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 14 }} - app.kubernetes.io/component: primary - {{- if .Values.readReplicas.networkPolicy.extraEgress }} - {{- include "common.tplvalues.render" ( dict "value" .Values.readReplicas.networkPolicy.extraEgress "context" $ ) | nindent 4 }} - {{- end }} - {{- end }} - ingress: - - ports: - - port: {{ .Values.containerPorts.postgresql }} - {{- if .Values.metrics.enabled }} - - port: {{ .Values.metrics.containerPorts.metrics }} - {{- end }} - {{- if not .Values.readReplicas.networkPolicy.allowExternal }} - from: - - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 14 }} - - podSelector: - matchLabels: - {{ template "postgresql.v1.readReplica.fullname" . }}-client: "true" - {{- if .Values.readReplicas.networkPolicy.ingressNSMatchLabels }} - - namespaceSelector: - matchLabels: - {{- range $key, $value := .Values.readReplicas.networkPolicy.ingressNSMatchLabels }} - {{ $key | quote }}: {{ $value | quote }} - {{- end }} - {{- if .Values.readReplicas.networkPolicy.ingressNSPodMatchLabels }} - podSelector: - matchLabels: - {{- range $key, $value := .Values.readReplicas.networkPolicy.ingressNSPodMatchLabels }} - {{ $key | quote }}: {{ $value | quote }} - {{- end }} - {{- end }} - {{- end }} - {{- end }} - {{- if .Values.readReplicas.networkPolicy.extraIngress }} - {{- include "common.tplvalues.render" ( dict "value" .Values.readReplicas.networkPolicy.extraIngress "context" $ ) | nindent 4 }} - {{- end }} -{{- end }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/read/pdb.yaml b/charts/postgresql/postgresql/templates/read/pdb.yaml deleted file mode 100644 index 1d25b68..0000000 --- a/charts/postgresql/postgresql/templates/read/pdb.yaml +++ /dev/null @@ -1,29 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and ( eq .Values.architecture "replication" ) .Values.readReplicas.pdb.create }} -apiVersion: {{ include "common.capabilities.policy.apiVersion" . }} -kind: PodDisruptionBudget -metadata: - name: {{ include "postgresql.v1.readReplica.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - {{- $labels := include "common.tplvalues.merge" ( dict "values" ( list .Values.readReplicas.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $labels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: read - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- if .Values.readReplicas.pdb.minAvailable }} - minAvailable: {{ .Values.readReplicas.pdb.minAvailable }} - {{- end }} - {{- if or .Values.readReplicas.pdb.maxUnavailable ( not .Values.readReplicas.pdb.minAvailable ) }} - maxUnavailable: {{ .Values.readReplicas.pdb.maxUnavailable | default 1 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.readReplicas.podLabels .Values.commonLabels ) "context" . ) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: read -{{- end }} diff --git a/charts/postgresql/postgresql/templates/read/servicemonitor.yaml b/charts/postgresql/postgresql/templates/read/servicemonitor.yaml deleted file mode 100644 index 2b63559..0000000 --- a/charts/postgresql/postgresql/templates/read/servicemonitor.yaml +++ /dev/null @@ -1,46 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled .Values.metrics.serviceMonitor.enabled (eq .Values.architecture "replication") }} -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ include "postgresql.v1.readReplica.fullname" . }} - namespace: {{ default .Release.Namespace .Values.metrics.serviceMonitor.namespace | quote }} - {{- $labels := include "common.tplvalues.merge" ( dict "values" ( list .Values.metrics.serviceMonitor.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $labels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: metrics-read - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- if .Values.metrics.serviceMonitor.jobLabel }} - jobLabel: {{ .Values.metrics.serviceMonitor.jobLabel }} - {{- end }} - selector: - {{- $svcLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.metrics.serviceMonitor.selector .Values.commonLabels ) "context" . ) }} - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $svcLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: metrics-read - endpoints: - - port: http-metrics - {{- if .Values.metrics.serviceMonitor.interval }} - interval: {{ .Values.metrics.serviceMonitor.interval }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.scrapeTimeout }} - scrapeTimeout: {{ .Values.metrics.serviceMonitor.scrapeTimeout }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.relabelings }} - relabelings: {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.serviceMonitor.relabelings "context" $) | nindent 6 }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.metricRelabelings }} - metricRelabelings: {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.serviceMonitor.metricRelabelings "context" $) | nindent 6 }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.honorLabels }} - honorLabels: {{ .Values.metrics.serviceMonitor.honorLabels }} - {{- end }} - namespaceSelector: - matchNames: - - {{ .Release.Namespace | quote }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/read/statefulset.yaml b/charts/postgresql/postgresql/templates/read/statefulset.yaml deleted file mode 100644 index 4f537e6..0000000 --- a/charts/postgresql/postgresql/templates/read/statefulset.yaml +++ /dev/null @@ -1,578 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if eq .Values.architecture "replication" }} -{{- $customUser := include "postgresql.v1.username" . }} -apiVersion: {{ include "common.capabilities.statefulset.apiVersion" . }} -kind: StatefulSet -metadata: - name: {{ include "postgresql.v1.readReplica.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - {{- $labels := include "common.tplvalues.merge" ( dict "values" ( list .Values.readReplicas.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $labels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: read - {{- if or .Values.commonAnnotations .Values.readReplicas.annotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.readReplicas.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - replicas: {{ .Values.readReplicas.replicaCount }} - serviceName: {{ include "postgresql.v1.readReplica.svc.headless" . }} - {{- if .Values.readReplicas.updateStrategy }} - updateStrategy: {{- toYaml .Values.readReplicas.updateStrategy | nindent 4 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.readReplicas.podLabels .Values.commonLabels ) "context" . ) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: read - template: - metadata: - name: {{ include "postgresql.v1.readReplica.fullname" . }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $podLabels "context" $ ) | nindent 8 }} - app.kubernetes.io/component: read - {{- if or (include "postgresql.v1.readReplicas.createExtendedConfigmap" .) .Values.readReplicas.podAnnotations }} - annotations: - {{- if (include "postgresql.v1.readReplicas.createExtendedConfigmap" .) }} - checksum/extended-configuration: {{ pick (include (print $.Template.BasePath "/primary/extended-configmap.yaml") . | fromYaml) "data" | toYaml | sha256sum }} - {{- end }} - {{- if .Values.readReplicas.podAnnotations }} - {{- include "common.tplvalues.render" ( dict "value" .Values.readReplicas.podAnnotations "context" $ ) | nindent 8 }} - {{- end }} - {{- end }} - spec: - {{- if .Values.readReplicas.extraPodSpec }} - {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.extraPodSpec "context" $) | nindent 6 }} - {{- end }} - serviceAccountName: {{ include "postgresql.v1.serviceAccountName" . }} - {{- include "postgresql.v1.imagePullSecrets" . | nindent 6 }} - automountServiceAccountToken: {{ .Values.readReplicas.automountServiceAccountToken }} - {{- if .Values.readReplicas.hostAliases }} - hostAliases: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.hostAliases "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.readReplicas.affinity }} - affinity: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.affinity "context" $) | nindent 8 }} - {{- else }} - affinity: - podAffinity: {{- include "common.affinities.pods" (dict "type" .Values.readReplicas.podAffinityPreset "component" "read" "customLabels" $podLabels "context" $) | nindent 10 }} - podAntiAffinity: {{- include "common.affinities.pods" (dict "type" .Values.readReplicas.podAntiAffinityPreset "component" "read" "customLabels" $podLabels "context" $) | nindent 10 }} - nodeAffinity: {{- include "common.affinities.nodes" (dict "type" .Values.readReplicas.nodeAffinityPreset.type "key" .Values.readReplicas.nodeAffinityPreset.key "values" .Values.readReplicas.nodeAffinityPreset.values) | nindent 10 }} - {{- end }} - {{- if .Values.readReplicas.nodeSelector }} - nodeSelector: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.nodeSelector "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.readReplicas.tolerations }} - tolerations: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.tolerations "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.readReplicas.topologySpreadConstraints }} - topologySpreadConstraints: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.topologySpreadConstraints "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.readReplicas.priorityClassName }} - priorityClassName: {{ .Values.readReplicas.priorityClassName }} - {{- end }} - {{- if .Values.readReplicas.schedulerName }} - schedulerName: {{ .Values.readReplicas.schedulerName | quote }} - {{- end }} - {{- if .Values.readReplicas.terminationGracePeriodSeconds }} - terminationGracePeriodSeconds: {{ .Values.readReplicas.terminationGracePeriodSeconds }} - {{- end }} - {{- if .Values.readReplicas.podSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.readReplicas.podSecurityContext "context" $) | nindent 8 }} - {{- end }} - hostNetwork: {{ .Values.readReplicas.hostNetwork }} - hostIPC: {{ .Values.readReplicas.hostIPC }} - {{- if or (and .Values.tls.enabled (not .Values.volumePermissions.enabled)) (and .Values.volumePermissions.enabled (or .Values.readReplicas.persistence.enabled .Values.shmVolume.enabled)) .Values.readReplicas.initContainers }} - initContainers: - {{- if and .Values.tls.enabled (not .Values.volumePermissions.enabled) }} - - name: copy-certs - image: {{ include "postgresql.v1.volumePermissions.image" . }} - imagePullPolicy: {{ .Values.volumePermissions.image.pullPolicy | quote }} - {{- if .Values.readReplicas.resources }} - resources: {{- toYaml .Values.readReplicas.resources | nindent 12 }} - {{- else if ne .Values.readReplicas.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.readReplicas.resourcesPreset) | nindent 12 }} - {{- end }} - # We don't require a privileged container in this case - {{- if .Values.readReplicas.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.readReplicas.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - command: - - /bin/sh - - -ec - - | - cp /tmp/certs/* /opt/bitnami/postgresql/certs/ - chmod 600 {{ include "postgresql.v1.tlsCertKey" . }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: raw-certificates - mountPath: /tmp/certs - - name: postgresql-certificates - mountPath: /opt/bitnami/postgresql/certs - {{- else if and .Values.volumePermissions.enabled (or .Values.readReplicas.persistence.enabled .Values.shmVolume.enabled) }} - - name: init-chmod-data - image: {{ include "postgresql.v1.volumePermissions.image" . }} - imagePullPolicy: {{ .Values.volumePermissions.image.pullPolicy | quote }} - {{- if .Values.readReplicas.resources }} - resources: {{- toYaml .Values.readReplicas.resources | nindent 12 }} - {{- else if ne .Values.readReplicas.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.readReplicas.resourcesPreset) | nindent 12 }} - {{- end }} - command: - - /bin/sh - - -ec - - | - {{- if .Values.readReplicas.persistence.enabled }} - {{- if eq ( toString ( .Values.volumePermissions.containerSecurityContext.runAsUser )) "auto" }} - chown `id -u`:`id -G | cut -d " " -f2` {{ .Values.readReplicas.persistence.mountPath }} - {{- else }} - chown {{ .Values.readReplicas.containerSecurityContext.runAsUser }}:{{ .Values.readReplicas.podSecurityContext.fsGroup }} {{ .Values.readReplicas.persistence.mountPath }} - {{- end }} - mkdir -p {{ .Values.readReplicas.persistence.mountPath }}/data {{- if (include "postgresql.v1.mountConfigurationCM" .) }} {{ .Values.readReplicas.persistence.mountPath }}/conf {{- end }} - chmod 700 {{ .Values.readReplicas.persistence.mountPath }}/data {{- if (include "postgresql.v1.mountConfigurationCM" .) }} {{ .Values.readReplicas.persistence.mountPath }}/conf {{- end }} - find {{ .Values.readReplicas.persistence.mountPath }} -mindepth 1 -maxdepth 1 {{- if not (include "postgresql.v1.mountConfigurationCM" .) }} -not -name "conf" {{- end }} -not -name ".snapshot" -not -name "lost+found" | \ - {{- if eq ( toString ( .Values.volumePermissions.containerSecurityContext.runAsUser )) "auto" }} - xargs -r chown -R `id -u`:`id -G | cut -d " " -f2` - {{- else }} - xargs -r chown -R {{ .Values.readReplicas.containerSecurityContext.runAsUser }}:{{ .Values.readReplicas.podSecurityContext.fsGroup }} - {{- end }} - {{- end }} - {{- if .Values.shmVolume.enabled }} - chmod -R 777 /dev/shm - {{- end }} - {{- if .Values.tls.enabled }} - cp /tmp/certs/* /opt/bitnami/postgresql/certs/ - {{- if eq ( toString ( .Values.volumePermissions.containerSecurityContext.runAsUser )) "auto" }} - chown -R `id -u`:`id -G | cut -d " " -f2` /opt/bitnami/postgresql/certs/ - {{- else }} - chown -R {{ .Values.readReplicas.containerSecurityContext.runAsUser }}:{{ .Values.readReplicas.podSecurityContext.fsGroup }} /opt/bitnami/postgresql/certs/ - {{- end }} - chmod 600 {{ include "postgresql.v1.tlsCertKey" . }} - {{- end }} - {{- if eq ( toString ( .Values.volumePermissions.containerSecurityContext.runAsUser )) "auto" }} - securityContext: {{- omit .Values.volumePermissions.containerSecurityContext "runAsUser" | toYaml | nindent 12 }} - {{- else }} - securityContext: {{- .Values.volumePermissions.containerSecurityContext | toYaml | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: data - mountPath: {{ .Values.readReplicas.persistence.mountPath }} - {{- if .Values.readReplicas.persistence.subPath }} - subPath: {{ .Values.readReplicas.persistence.subPath }} - {{- end }} - {{- if .Values.shmVolume.enabled }} - - name: dshm - mountPath: /dev/shm - {{- end }} - {{- if .Values.tls.enabled }} - - name: raw-certificates - mountPath: /tmp/certs - - name: postgresql-certificates - mountPath: /opt/bitnami/postgresql/certs - {{- end }} - {{- end }} - {{- if .Values.readReplicas.initContainers }} - {{- include "common.tplvalues.render" ( dict "value" .Values.readReplicas.initContainers "context" $ ) | nindent 8 }} - {{- end }} - {{- end }} - containers: - - name: postgresql - image: {{ include "postgresql.v1.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy | quote }} - {{- if .Values.readReplicas.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.readReplicas.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - {{- else if .Values.readReplicas.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.command "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- else if .Values.readReplicas.args }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.args "context" $) | nindent 12 }} - {{- end }} - env: - - name: BITNAMI_DEBUG - value: {{ ternary "true" "false" (or .Values.image.debug .Values.diagnosticMode.enabled) | quote }} - - name: POSTGRESQL_PORT_NUMBER - value: {{ .Values.containerPorts.postgresql | quote }} - - name: POSTGRESQL_VOLUME_DIR - value: {{ .Values.readReplicas.persistence.mountPath | quote }} - {{- if .Values.readReplicas.persistence.mountPath }} - - name: PGDATA - value: {{ .Values.postgresqlDataDir | quote }} - {{- end }} - # Authentication - {{- if or (eq $customUser "postgres") (empty $customUser) }} - {{- if .Values.auth.enablePostgresUser }} - {{- if .Values.auth.usePasswordFiles }} - - name: POSTGRES_PASSWORD_FILE - value: {{ printf "/opt/bitnami/postgresql/secrets/%s" (include "postgresql.v1.adminPasswordKey" .) }} - {{- else }} - - name: POSTGRES_PASSWORD - valueFrom: - secretKeyRef: - name: {{ include "postgresql.v1.secretName" . }} - key: {{ include "postgresql.v1.adminPasswordKey" . }} - {{- end }} - {{- else }} - - name: ALLOW_EMPTY_PASSWORD - value: "true" - {{- end }} - {{- else }} - - name: POSTGRES_USER - value: {{ $customUser | quote }} - {{- if .Values.auth.usePasswordFiles }} - - name: POSTGRES_PASSWORD_FILE - value: {{ printf "/opt/bitnami/postgresql/secrets/%s" (include "postgresql.v1.userPasswordKey" .) }} - {{- else }} - - name: POSTGRES_PASSWORD - valueFrom: - secretKeyRef: - name: {{ include "postgresql.v1.secretName" . }} - key: {{ include "postgresql.v1.userPasswordKey" . }} - {{- end }} - {{- if .Values.auth.enablePostgresUser }} - {{- if .Values.auth.usePasswordFiles }} - - name: POSTGRES_POSTGRES_PASSWORD_FILE - value: {{ printf "/opt/bitnami/postgresql/secrets/%s" (include "postgresql.v1.adminPasswordKey" .) }} - {{- else }} - - name: POSTGRES_POSTGRES_PASSWORD - valueFrom: - secretKeyRef: - name: {{ include "postgresql.v1.secretName" . }} - key: {{ include "postgresql.v1.adminPasswordKey" . }} - {{- end }} - {{- end }} - {{- end }} - # Replication - - name: POSTGRES_REPLICATION_MODE - value: "slave" - - name: POSTGRES_REPLICATION_USER - value: {{ .Values.auth.replicationUsername | quote }} - {{- if .Values.auth.usePasswordFiles }} - - name: POSTGRES_REPLICATION_PASSWORD_FILE - value: {{ printf "/opt/bitnami/postgresql/secrets/%s" (include "postgresql.v1.replicationPasswordKey" .) }} - {{- else }} - - name: POSTGRES_REPLICATION_PASSWORD - valueFrom: - secretKeyRef: - name: {{ include "postgresql.v1.secretName" . }} - key: {{ include "postgresql.v1.replicationPasswordKey" . }} - {{- end }} - - name: POSTGRES_CLUSTER_APP_NAME - value: {{ .Values.replication.applicationName }} - - name: POSTGRES_MASTER_HOST - value: {{ include "postgresql.v1.primary.fullname" . }} - - name: POSTGRES_MASTER_PORT_NUMBER - value: {{ include "postgresql.v1.service.port" . | quote }} - # TLS - - name: POSTGRESQL_ENABLE_TLS - value: {{ ternary "yes" "no" .Values.tls.enabled | quote }} - {{- if .Values.tls.enabled }} - - name: POSTGRESQL_TLS_PREFER_SERVER_CIPHERS - value: {{ ternary "yes" "no" .Values.tls.preferServerCiphers | quote }} - - name: POSTGRESQL_TLS_CERT_FILE - value: {{ include "postgresql.v1.tlsCert" . }} - - name: POSTGRESQL_TLS_KEY_FILE - value: {{ include "postgresql.v1.tlsCertKey" . }} - {{- if .Values.tls.certCAFilename }} - - name: POSTGRESQL_TLS_CA_FILE - value: {{ include "postgresql.v1.tlsCACert" . }} - {{- end }} - {{- if .Values.tls.crlFilename }} - - name: POSTGRESQL_TLS_CRL_FILE - value: {{ include "postgresql.v1.tlsCRL" . }} - {{- end }} - {{- end }} - # Audit - - name: POSTGRESQL_LOG_HOSTNAME - value: {{ .Values.audit.logHostname | quote }} - - name: POSTGRESQL_LOG_CONNECTIONS - value: {{ .Values.audit.logConnections | quote }} - - name: POSTGRESQL_LOG_DISCONNECTIONS - value: {{ .Values.audit.logDisconnections | quote }} - {{- if .Values.audit.logLinePrefix }} - - name: POSTGRESQL_LOG_LINE_PREFIX - value: {{ .Values.audit.logLinePrefix | quote }} - {{- end }} - {{- if .Values.audit.logTimezone }} - - name: POSTGRESQL_LOG_TIMEZONE - value: {{ .Values.audit.logTimezone | quote }} - {{- end }} - {{- if .Values.audit.pgAuditLog }} - - name: POSTGRESQL_PGAUDIT_LOG - value: {{ .Values.audit.pgAuditLog | quote }} - {{- end }} - - name: POSTGRESQL_PGAUDIT_LOG_CATALOG - value: {{ .Values.audit.pgAuditLogCatalog | quote }} - # Others - - name: POSTGRESQL_CLIENT_MIN_MESSAGES - value: {{ .Values.audit.clientMinMessages | quote }} - - name: POSTGRESQL_SHARED_PRELOAD_LIBRARIES - value: {{ .Values.postgresqlSharedPreloadLibraries | quote }} - {{- if .Values.readReplicas.extraEnvVars }} - {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.extraEnvVars "context" $) | nindent 12 }} - {{- end }} - {{- if or .Values.readReplicas.extraEnvVarsCM .Values.readReplicas.extraEnvVarsSecret }} - envFrom: - {{- if .Values.readReplicas.extraEnvVarsCM }} - - configMapRef: - name: {{ .Values.readReplicas.extraEnvVarsCM }} - {{- end }} - {{- if .Values.readReplicas.extraEnvVarsSecret }} - - secretRef: - name: {{ .Values.readReplicas.extraEnvVarsSecret }} - {{- end }} - {{- end }} - ports: - - name: tcp-postgresql - containerPort: {{ .Values.containerPorts.postgresql }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.readReplicas.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.readReplicas.startupProbe.enabled }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.readReplicas.startupProbe "enabled") "context" $) | nindent 12 }} - exec: - command: - - /bin/sh - - -c - {{- if (include "postgresql.v1.database" .) }} - - exec pg_isready -U {{ default "postgres" $customUser| quote }} -d "dbname={{ include "postgresql.v1.database" . }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} sslcert={{ include "postgresql.v1.tlsCert" . }} sslkey={{ include "postgresql.v1.tlsCertKey" . }}{{- end }}" -h 127.0.0.1 -p {{ .Values.containerPorts.postgresql }} - {{- else }} - - exec pg_isready -U {{ default "postgres" $customUser | quote }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} -d "sslcert={{ include "postgresql.v1.tlsCert" . }} sslkey={{ include "postgresql.v1.tlsCertKey" . }}"{{- end }} -h 127.0.0.1 -p {{ .Values.containerPorts.postgresql }} - {{- end }} - {{- end }} - {{- if .Values.readReplicas.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.readReplicas.livenessProbe.enabled }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.readReplicas.livenessProbe "enabled") "context" $) | nindent 12 }} - exec: - command: - - /bin/sh - - -c - {{- if (include "postgresql.v1.database" .) }} - - exec pg_isready -U {{ default "postgres" $customUser | quote }} -d "dbname={{ include "postgresql.v1.database" . }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} sslcert={{ include "postgresql.v1.tlsCert" . }} sslkey={{ include "postgresql.v1.tlsCertKey" . }}{{- end }}" -h 127.0.0.1 -p {{ .Values.containerPorts.postgresql }} - {{- else }} - - exec pg_isready -U {{default "postgres" $customUser | quote }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} -d "sslcert={{ include "postgresql.v1.tlsCert" . }} sslkey={{ include "postgresql.v1.tlsCertKey" . }}"{{- end }} -h 127.0.0.1 -p {{ .Values.containerPorts.postgresql }} - {{- end }} - {{- end }} - {{- if .Values.readReplicas.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.readReplicas.readinessProbe.enabled }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.readReplicas.readinessProbe "enabled") "context" $) | nindent 12 }} - exec: - command: - - /bin/sh - - -c - - -e - {{- include "postgresql.v1.readinessProbeCommand" . | nindent 16 }} - {{- end }} - {{- end }} - {{- if .Values.readReplicas.resources }} - resources: {{- toYaml .Values.readReplicas.resources | nindent 12 }} - {{- else if ne .Values.readReplicas.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.readReplicas.resourcesPreset) | nindent 12 }} - {{- end }} - {{- if .Values.readReplicas.lifecycleHooks }} - lifecycle: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.lifecycleHooks "context" $) | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: empty-dir - mountPath: /opt/bitnami/postgresql/conf - subPath: app-conf-dir - - name: empty-dir - mountPath: /opt/bitnami/postgresql/tmp - subPath: app-tmp-dir - {{- if .Values.auth.usePasswordFiles }} - - name: postgresql-password - mountPath: /opt/bitnami/postgresql/secrets/ - {{- end }} - {{- if .Values.readReplicas.extendedConfiguration }} - - name: postgresql-extended-config - mountPath: {{ .Values.readReplicas.persistence.mountPath }}/conf/conf.d/ - {{- end }} - {{- if .Values.tls.enabled }} - - name: postgresql-certificates - mountPath: /opt/bitnami/postgresql/certs - readOnly: true - {{- end }} - {{- if .Values.shmVolume.enabled }} - - name: dshm - mountPath: /dev/shm - {{- end }} - - name: data - mountPath: {{ .Values.readReplicas.persistence.mountPath }} - {{- if .Values.readReplicas.persistence.subPath }} - subPath: {{ .Values.readReplicas.persistence.subPath }} - {{- end }} - {{- if .Values.readReplicas.extraVolumeMounts }} - {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.extraVolumeMounts "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.metrics.enabled }} - - name: metrics - image: {{ include "postgresql.v1.metrics.image" . }} - imagePullPolicy: {{ .Values.metrics.image.pullPolicy | quote }} - {{- if .Values.metrics.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.metrics.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- else if .Values.metrics.customMetrics }} - args: [ "--extend.query-path", "/conf/custom-metrics.yaml" ] - {{- end }} - env: - - name: DATA_SOURCE_URI - value: {{ printf "127.0.0.1:%d/postgres?sslmode=disable" (int (include "postgresql.v1.service.port" .)) }} - {{- if .Values.auth.usePasswordFiles }} - - name: DATA_SOURCE_PASS_FILE - value: {{ printf "/opt/bitnami/postgresql/secrets/%s" (include "postgresql.v1.userPasswordKey" .) }} - {{- else }} - - name: DATA_SOURCE_PASS - valueFrom: - secretKeyRef: - name: {{ include "postgresql.v1.secretName" . }} - key: {{ include "postgresql.v1.userPasswordKey" . }} - {{- end }} - - name: DATA_SOURCE_USER - value: {{ default "postgres" $customUser | quote }} - {{- if .Values.metrics.extraEnvVars }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.extraEnvVars "context" $) | nindent 12 }} - {{- end }} - ports: - - name: http-metrics - containerPort: {{ .Values.metrics.containerPorts.metrics }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.metrics.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.startupProbe.enabled }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.startupProbe "enabled") "context" $) | nindent 12 }} - tcpSocket: - port: http-metrics - {{- end }} - {{- if .Values.metrics.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.livenessProbe.enabled }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.livenessProbe "enabled") "context" $) | nindent 12 }} - httpGet: - path: / - port: http-metrics - {{- end }} - {{- if .Values.metrics.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.readinessProbe.enabled }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.readinessProbe "enabled") "context" $) | nindent 12 }} - httpGet: - path: / - port: http-metrics - {{- end }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - {{- if .Values.auth.usePasswordFiles }} - - name: postgresql-password - mountPath: /opt/bitnami/postgresql/secrets/ - {{- end }} - {{- if .Values.metrics.customMetrics }} - - name: custom-metrics - mountPath: /conf - readOnly: true - {{- end }} - {{- if .Values.metrics.resources }} - resources: {{- toYaml .Values.metrics.resources | nindent 12 }} - {{- else if ne .Values.metrics.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.metrics.resourcesPreset) | nindent 12 }} - {{- end }} - {{- end }} - {{- if .Values.readReplicas.sidecars }} - {{- include "common.tplvalues.render" ( dict "value" .Values.readReplicas.sidecars "context" $ ) | nindent 8 }} - {{- end }} - volumes: - {{- if .Values.readReplicas.extendedConfiguration }} - - name: postgresql-extended-config - configMap: - name: {{ include "postgresql.v1.readReplicas.extendedConfigmapName" . }} - {{- end }} - {{- if .Values.auth.usePasswordFiles }} - - name: postgresql-password - secret: - secretName: {{ include "postgresql.v1.secretName" . }} - {{- end }} - {{- if .Values.tls.enabled }} - - name: raw-certificates - secret: - secretName: {{ include "postgresql.v1.tlsSecretName" . }} - - name: postgresql-certificates - emptyDir: {} - {{- end }} - {{- if and .Values.metrics.enabled .Values.metrics.customMetrics }} - - name: custom-metrics - configMap: - name: {{ printf "%s-metrics" (include "postgresql.v1.readReplica.fullname" .) }} - {{- end }} - {{- if .Values.shmVolume.enabled }} - - name: dshm - emptyDir: - medium: Memory - {{- if .Values.shmVolume.sizeLimit }} - sizeLimit: {{ .Values.shmVolume.sizeLimit }} - {{- end }} - {{- end }} - - name: empty-dir - emptyDir: {} - {{- if .Values.readReplicas.extraVolumes }} - {{- include "common.tplvalues.render" ( dict "value" .Values.readReplicas.extraVolumes "context" $ ) | nindent 8 }} - {{- end }} - {{- if and .Values.readReplicas.persistence.enabled .Values.readReplicas.persistence.existingClaim }} - - name: data - persistentVolumeClaim: - claimName: {{ tpl .Values.readReplicas.persistence.existingClaim $ }} - {{- else if not .Values.readReplicas.persistence.enabled }} - - name: data - emptyDir: {} - {{- else }} - {{- if .Values.readReplicas.persistentVolumeClaimRetentionPolicy.enabled }} - persistentVolumeClaimRetentionPolicy: - whenDeleted: {{ .Values.readReplicas.persistentVolumeClaimRetentionPolicy.whenDeleted }} - whenScaled: {{ .Values.readReplicas.persistentVolumeClaimRetentionPolicy.whenScaled }} - {{- end }} - volumeClaimTemplates: - - apiVersion: v1 - kind: PersistentVolumeClaim - metadata: - name: data - {{- if .Values.readReplicas.persistence.annotations }} - annotations: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.persistence.annotations "context" $) | nindent 10 }} - {{- end }} - {{- if .Values.readReplicas.persistence.labels }} - labels: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.persistence.labels "context" $) | nindent 10 }} - {{- end }} - spec: - accessModes: - {{- range .Values.readReplicas.persistence.accessModes }} - - {{ . | quote }} - {{- end }} - {{- if .Values.readReplicas.persistence.dataSource }} - dataSource: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.persistence.dataSource "context" $) | nindent 10 }} - {{- end }} - resources: - requests: - storage: {{ .Values.readReplicas.persistence.size | quote }} - {{- if .Values.readReplicas.persistence.selector }} - selector: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.persistence.selector "context" $) | nindent 10 }} - {{- end -}} - {{- include "common.storage.class" (dict "persistence" .Values.readReplicas.persistence "global" .Values.global) | nindent 8 }} - {{- end }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/read/svc-headless.yaml b/charts/postgresql/postgresql/templates/read/svc-headless.yaml deleted file mode 100644 index ef44601..0000000 --- a/charts/postgresql/postgresql/templates/read/svc-headless.yaml +++ /dev/null @@ -1,33 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if eq .Values.architecture "replication" }} -apiVersion: v1 -kind: Service -metadata: - name: {{ include "postgresql.v1.readReplica.svc.headless" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: read - annotations: - {{- if or .Values.readReplicas.service.headless.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.readReplicas.service.headless.annotations .Values.commonAnnotations ) "context" . ) }} - {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: ClusterIP - clusterIP: None - # We want all pods in the StatefulSet to have their addresses published for - # the sake of the other Postgresql pods even before they're ready, since they - # have to be able to talk to each other in order to become ready. - publishNotReadyAddresses: true - ports: - - name: tcp-postgresql - port: {{ include "postgresql.v1.readReplica.service.port" . }} - targetPort: tcp-postgresql - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.readReplicas.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: read -{{- end }} diff --git a/charts/postgresql/postgresql/templates/read/svc.yaml b/charts/postgresql/postgresql/templates/read/svc.yaml deleted file mode 100644 index d230acb..0000000 --- a/charts/postgresql/postgresql/templates/read/svc.yaml +++ /dev/null @@ -1,56 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if eq .Values.architecture "replication" }} -apiVersion: v1 -kind: Service -metadata: - name: {{ include "postgresql.v1.readReplica.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: read - {{- if or .Values.commonAnnotations .Values.readReplicas.service.annotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.readReplicas.service.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: {{ .Values.readReplicas.service.type }} - {{- if or (eq .Values.readReplicas.service.type "LoadBalancer") (eq .Values.readReplicas.service.type "NodePort") }} - externalTrafficPolicy: {{ .Values.readReplicas.service.externalTrafficPolicy | quote }} - {{- end }} - {{- if and (eq .Values.readReplicas.service.type "LoadBalancer") (not (empty .Values.readReplicas.service.loadBalancerSourceRanges)) }} - loadBalancerSourceRanges: {{ .Values.readReplicas.service.loadBalancerSourceRanges }} - {{- end }} - {{- if and (eq .Values.readReplicas.service.type "LoadBalancer") (not (empty .Values.readReplicas.service.loadBalancerClass)) }} - loadBalancerClass: {{ .Values.readReplicas.service.loadBalancerClass }} - {{- end }} - {{- if and (eq .Values.readReplicas.service.type "LoadBalancer") (not (empty .Values.readReplicas.service.loadBalancerIP)) }} - loadBalancerIP: {{ .Values.readReplicas.service.loadBalancerIP }} - {{- end }} - {{- if and .Values.readReplicas.service.clusterIP (eq .Values.readReplicas.service.type "ClusterIP") }} - clusterIP: {{ .Values.readReplicas.service.clusterIP }} - {{- end }} - {{- if .Values.readReplicas.service.sessionAffinity }} - sessionAffinity: {{ .Values.readReplicas.service.sessionAffinity }} - {{- end }} - {{- if .Values.readReplicas.service.sessionAffinityConfig }} - sessionAffinityConfig: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.service.sessionAffinityConfig "context" $) | nindent 4 }} - {{- end }} - ports: - - name: tcp-postgresql - port: {{ include "postgresql.v1.readReplica.service.port" . }} - targetPort: tcp-postgresql - {{- if and (or (eq .Values.readReplicas.service.type "NodePort") (eq .Values.readReplicas.service.type "LoadBalancer")) (not (empty .Values.readReplicas.service.nodePorts.postgresql)) }} - nodePort: {{ .Values.readReplicas.service.nodePorts.postgresql }} - {{- else if eq .Values.readReplicas.service.type "ClusterIP" }} - nodePort: null - {{- end }} - {{- if .Values.readReplicas.service.extraPorts }} - {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.service.extraPorts "context" $) | nindent 4 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.readReplicas.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: read -{{- end }} diff --git a/charts/postgresql/postgresql/templates/role.yaml b/charts/postgresql/postgresql/templates/role.yaml deleted file mode 100644 index 12cd212..0000000 --- a/charts/postgresql/postgresql/templates/role.yaml +++ /dev/null @@ -1,32 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.rbac.create }} -kind: Role -apiVersion: {{ include "common.capabilities.rbac.apiVersion" . }} -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -# yamllint disable rule:indentation -rules: - {{- if and (include "common.capabilities.psp.supported" .) .Values.psp.create }} - - apiGroups: - - 'policy' - resources: - - 'podsecuritypolicies' - verbs: - - 'use' - resourceNames: - - {{ include "common.names.fullname" . }} - {{- end }} - {{- if .Values.rbac.rules }} - {{- include "common.tplvalues.render" ( dict "value" .Values.rbac.rules "context" $ ) | nindent 2 }} - {{- end }} -# yamllint enable rule:indentation -{{- end }} diff --git a/charts/postgresql/postgresql/templates/rolebinding.yaml b/charts/postgresql/postgresql/templates/rolebinding.yaml deleted file mode 100644 index 2ee30a4..0000000 --- a/charts/postgresql/postgresql/templates/rolebinding.yaml +++ /dev/null @@ -1,24 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.rbac.create }} -kind: RoleBinding -apiVersion: {{ include "common.capabilities.rbac.apiVersion" . }} -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -roleRef: - kind: Role - name: {{ include "common.names.fullname" . }} - apiGroup: rbac.authorization.k8s.io -subjects: - - kind: ServiceAccount - name: {{ include "postgresql.v1.serviceAccountName" . }} - namespace: {{ .Release.Namespace | quote }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/secrets.yaml b/charts/postgresql/postgresql/templates/secrets.yaml deleted file mode 100644 index 2fcf64c..0000000 --- a/charts/postgresql/postgresql/templates/secrets.yaml +++ /dev/null @@ -1,99 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- $host := include "postgresql.v1.primary.fullname" . }} -{{- $port := include "postgresql.v1.service.port" . }} -{{- $customUser := include "postgresql.v1.username" . }} -{{- $postgresPassword := include "common.secrets.lookup" (dict "secret" (include "postgresql.v1.secretName" .) "key" (coalesce .Values.global.postgresql.auth.secretKeys.adminPasswordKey .Values.auth.secretKeys.adminPasswordKey) "defaultValue" (ternary (coalesce .Values.global.postgresql.auth.password .Values.auth.password .Values.global.postgresql.auth.postgresPassword .Values.auth.postgresPassword) (coalesce .Values.global.postgresql.auth.postgresPassword .Values.auth.postgresPassword) (or (empty $customUser) (eq $customUser "postgres"))) "context" $) | trimAll "\"" | b64dec }} -{{- if and (not $postgresPassword) .Values.auth.enablePostgresUser }} -{{- $postgresPassword = randAlphaNum 10 }} -{{- end }} -{{- $replicationPassword := "" }} -{{- if eq .Values.architecture "replication" }} -{{- $replicationPassword = include "common.secrets.passwords.manage" (dict "secret" (include "postgresql.v1.secretName" .) "key" (coalesce .Values.global.postgresql.auth.secretKeys.replicationPasswordKey .Values.auth.secretKeys.replicationPasswordKey) "providedValues" (list "auth.replicationPassword") "context" $) | trimAll "\"" | b64dec }} -{{- end }} -{{- $ldapPassword := "" }} -{{- if and .Values.ldap.enabled (or .Values.ldap.bind_password .Values.ldap.bindpw) }} -{{- $ldapPassword = coalesce .Values.ldap.bind_password .Values.ldap.bindpw }} -{{- end }} -{{- $password := "" }} -{{- if and (not (empty $customUser)) (ne $customUser "postgres") }} -{{- $password = include "common.secrets.passwords.manage" (dict "secret" (include "postgresql.v1.secretName" .) "key" (coalesce .Values.global.postgresql.auth.secretKeys.userPasswordKey .Values.auth.secretKeys.userPasswordKey) "providedValues" (list "global.postgresql.auth.password" "auth.password") "context" $) | trimAll "\"" | b64dec }} -{{- end }} -{{- $database := include "postgresql.v1.database" . }} -{{- if (include "postgresql.v1.createSecret" .) }} -apiVersion: v1 -kind: Secret -metadata: - name: {{ include "common.names.fullname" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: Opaque -data: - {{- if $postgresPassword }} - postgres-password: {{ $postgresPassword | b64enc | quote }} - {{- end }} - {{- if $password }} - password: {{ $password | b64enc | quote }} - {{- end }} - {{- if $replicationPassword }} - replication-password: {{ $replicationPassword | b64enc | quote }} - {{- end }} - # We don't auto-generate LDAP password when it's not provided as we do for other passwords - {{- if and .Values.ldap.enabled (or .Values.ldap.bind_password .Values.ldap.bindpw) }} - ldap-password: {{ $ldapPassword | b64enc | quote }} - {{- end }} -{{- end }} -{{- if .Values.serviceBindings.enabled }} -{{- if $postgresPassword }} ---- -apiVersion: v1 -kind: Secret -metadata: - name: {{ include "common.names.fullname" . }}-svcbind-postgres - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: servicebinding.io/postgresql -data: - provider: {{ print "bitnami" | b64enc | quote }} - type: {{ print "postgresql" | b64enc | quote }} - host: {{ $host | b64enc | quote }} - port: {{ $port | b64enc | quote }} - username: {{ print "postgres" | b64enc | quote }} - database: {{ print "postgres" | b64enc | quote }} - password: {{ $postgresPassword | b64enc | quote }} - uri: {{ printf "postgresql://postgres:%s@%s:%s/postgres" $postgresPassword $host $port | b64enc | quote }} -{{- end }} -{{- if $password }} ---- -apiVersion: v1 -kind: Secret -metadata: - name: {{ include "common.names.fullname" . }}-svcbind-custom-user - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: servicebinding.io/postgresql -data: - provider: {{ print "bitnami" | b64enc | quote }} - type: {{ print "postgresql" | b64enc | quote }} - host: {{ $host | b64enc | quote }} - port: {{ $port | b64enc | quote }} - username: {{ $customUser | b64enc | quote }} - password: {{ $password | b64enc | quote }} - {{- if $database }} - database: {{ $database | b64enc | quote }} - {{- end }} - uri: {{ printf "postgresql://%s:%s@%s:%s/%s" $customUser $password $host $port $database | b64enc | quote }} -{{- end }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/serviceaccount.yaml b/charts/postgresql/postgresql/templates/serviceaccount.yaml deleted file mode 100644 index 93efa64..0000000 --- a/charts/postgresql/postgresql/templates/serviceaccount.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.serviceAccount.create }} -apiVersion: v1 -kind: ServiceAccount -metadata: - name: {{ include "postgresql.v1.serviceAccountName" . }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if or .Values.serviceAccount.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.serviceAccount.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -automountServiceAccountToken: {{ .Values.serviceAccount.automountServiceAccountToken }} -{{- end }} diff --git a/charts/postgresql/postgresql/templates/tls-secrets.yaml b/charts/postgresql/postgresql/templates/tls-secrets.yaml deleted file mode 100644 index 2e992d6..0000000 --- a/charts/postgresql/postgresql/templates/tls-secrets.yaml +++ /dev/null @@ -1,30 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if (include "postgresql.v1.createTlsSecret" . ) }} -{{- $secretName := printf "%s-crt" (include "common.names.fullname" .) }} -{{- $ca := genCA "postgresql-ca" 365 }} -{{- $fullname := include "common.names.fullname" . }} -{{- $releaseNamespace := .Release.Namespace }} -{{- $clusterDomain := .Values.clusterDomain }} -{{- $primaryHeadlessServiceName := include "postgresql.v1.primary.svc.headless" . }} -{{- $readHeadlessServiceName := include "postgresql.v1.readReplica.svc.headless" . }} -{{- $altNames := list (printf "*.%s.%s.svc.%s" $fullname $releaseNamespace $clusterDomain) (printf "%s.%s.svc.%s" $fullname $releaseNamespace $clusterDomain) (printf "*.%s.%s.svc.%s" $primaryHeadlessServiceName $releaseNamespace $clusterDomain) (printf "%s.%s.svc.%s" $primaryHeadlessServiceName $releaseNamespace $clusterDomain) (printf "*.%s.%s.svc.%s" $readHeadlessServiceName $releaseNamespace $clusterDomain) (printf "%s.%s.svc.%s" $readHeadlessServiceName $releaseNamespace $clusterDomain) $fullname }} -{{- $cert := genSignedCert $fullname nil $altNames 365 $ca }} -apiVersion: v1 -kind: Secret -metadata: - name: {{ $secretName }} - namespace: {{ .Release.Namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: kubernetes.io/tls -data: - tls.crt: {{ include "common.secrets.lookup" (dict "secret" $secretName "key" "tls.crt" "defaultValue" $cert.Cert "context" $) }} - tls.key: {{ include "common.secrets.lookup" (dict "secret" $secretName "key" "tls.key" "defaultValue" $cert.Key "context" $) }} - ca.crt: {{ include "common.secrets.lookup" (dict "secret" $secretName "key" "ca.crt" "defaultValue" $ca.Cert "context" $) }} -{{- end }} diff --git a/charts/postgresql/postgresql/values.schema.json b/charts/postgresql/postgresql/values.schema.json deleted file mode 100644 index fc41483..0000000 --- a/charts/postgresql/postgresql/values.schema.json +++ /dev/null @@ -1,156 +0,0 @@ -{ - "$schema": "http://json-schema.org/schema#", - "type": "object", - "properties": { - "architecture": { - "type": "string", - "title": "PostgreSQL architecture", - "form": true, - "description": "Allowed values: `standalone` or `replication`" - }, - "auth": { - "type": "object", - "title": "Authentication configuration", - "form": true, - "properties": { - "enablePostgresUser": { - "type": "boolean", - "title": "Enable \"postgres\" admin user", - "description": "Assign a password to the \"postgres\" admin user. Otherwise, remote access will be blocked for this user", - "form": true - }, - "postgresPassword": { - "type": "string", - "title": "Password for the \"postgres\" admin user", - "description": "Defaults to a random 10-character alphanumeric string if not set", - "form": true - }, - "database": { - "type": "string", - "title": "PostgreSQL custom database", - "description": "Name of the custom database to be created during the 1st initialization of PostgreSQL", - "form": true - }, - "username": { - "type": "string", - "title": "PostgreSQL custom user", - "description": "Name of the custom user to be created during the 1st initialization of PostgreSQL. This user only has permissions on the PostgreSQL custom database", - "form": true - }, - "password": { - "type": "string", - "title": "Password for the custom user to create", - "description": "Defaults to a random 10-character alphanumeric string if not set", - "form": true - }, - "replicationUsername": { - "type": "string", - "title": "PostgreSQL replication user", - "description": "Name of user used to manage replication.", - "form": true, - "hidden": { - "value": "standalone", - "path": "architecture" - } - }, - "replicationPassword": { - "type": "string", - "title": "Password for PostgreSQL replication user", - "description": "Defaults to a random 10-character alphanumeric string if not set", - "form": true, - "hidden": { - "value": "standalone", - "path": "architecture" - } - } - } - }, - "persistence": { - "type": "object", - "properties": { - "size": { - "type": "string", - "title": "Persistent Volume Size", - "form": true, - "render": "slider", - "sliderMin": 1, - "sliderMax": 100, - "sliderUnit": "Gi" - } - } - }, - "resources": { - "type": "object", - "title": "Required Resources", - "description": "Configure resource requests", - "form": true, - "properties": { - "requests": { - "type": "object", - "properties": { - "memory": { - "type": "string", - "form": true, - "render": "slider", - "title": "Memory Request", - "sliderMin": 10, - "sliderMax": 2048, - "sliderUnit": "Mi" - }, - "cpu": { - "type": "string", - "form": true, - "render": "slider", - "title": "CPU Request", - "sliderMin": 10, - "sliderMax": 2000, - "sliderUnit": "m" - } - } - } - } - }, - "replication": { - "type": "object", - "form": true, - "title": "Replication Details", - "properties": { - "enabled": { - "type": "boolean", - "title": "Enable Replication", - "form": true - }, - "readReplicas": { - "type": "integer", - "title": "read Replicas", - "form": true, - "hidden": { - "value": "standalone", - "path": "architecture" - } - } - } - }, - "volumePermissions": { - "type": "object", - "properties": { - "enabled": { - "type": "boolean", - "form": true, - "title": "Enable Init Containers", - "description": "Change the owner of the persist volume mountpoint to RunAsUser:fsGroup" - } - } - }, - "metrics": { - "type": "object", - "properties": { - "enabled": { - "type": "boolean", - "title": "Configure metrics exporter", - "form": true - } - } - } - } -} diff --git a/charts/postgresql/postgresql/values.yaml b/charts/postgresql/postgresql/values.yaml deleted file mode 100644 index fba84bc..0000000 --- a/charts/postgresql/postgresql/values.yaml +++ /dev/null @@ -1,1780 +0,0 @@ -# Copyright Broadcom, Inc. All Rights Reserved. -# SPDX-License-Identifier: APACHE-2.0 - -## @section Global parameters -## Please, note that this will override the parameters, including dependencies, configured to use the global value -## -global: - ## @param global.imageRegistry Global Docker image registry - ## - imageRegistry: "" - ## @param global.imagePullSecrets Global Docker registry secret names as an array - ## e.g. - ## imagePullSecrets: - ## - myRegistryKeySecretName - ## - imagePullSecrets: [] - ## @param global.defaultStorageClass Global default StorageClass for Persistent Volume(s) -## @param global.storageClass DEPRECATED: use global.defaultStorageClass instead - ## - defaultStorageClass: "" - storageClass: "" - postgresql: - ## @param global.postgresql.auth.postgresPassword Password for the "postgres" admin user (overrides `auth.postgresPassword`) - ## @param global.postgresql.auth.username Name for a custom user to create (overrides `auth.username`) - ## @param global.postgresql.auth.password Password for the custom user to create (overrides `auth.password`) - ## @param global.postgresql.auth.database Name for a custom database to create (overrides `auth.database`) - ## @param global.postgresql.auth.existingSecret Name of existing secret to use for PostgreSQL credentials (overrides `auth.existingSecret`). - ## @param global.postgresql.auth.secretKeys.adminPasswordKey Name of key in existing secret to use for PostgreSQL credentials (overrides `auth.secretKeys.adminPasswordKey`). Only used when `global.postgresql.auth.existingSecret` is set. - ## @param global.postgresql.auth.secretKeys.userPasswordKey Name of key in existing secret to use for PostgreSQL credentials (overrides `auth.secretKeys.userPasswordKey`). Only used when `global.postgresql.auth.existingSecret` is set. - ## @param global.postgresql.auth.secretKeys.replicationPasswordKey Name of key in existing secret to use for PostgreSQL credentials (overrides `auth.secretKeys.replicationPasswordKey`). Only used when `global.postgresql.auth.existingSecret` is set. - ## - auth: - postgresPassword: "" - username: "" - password: "" - database: "" - existingSecret: "" - secretKeys: - adminPasswordKey: "" - userPasswordKey: "" - replicationPasswordKey: "" - ## @param global.postgresql.service.ports.postgresql PostgreSQL service port (overrides `service.ports.postgresql`) - ## - service: - ports: - postgresql: "" - ## Compatibility adaptations for Kubernetes platforms - ## - compatibility: - ## Compatibility adaptations for Openshift - ## - openshift: - ## @param global.compatibility.openshift.adaptSecurityContext Adapt the securityContext sections of the deployment to make them compatible with Openshift restricted-v2 SCC: remove runAsUser, runAsGroup and fsGroup and let the platform use their allowed default IDs. Possible values: auto (apply if the detected running cluster is Openshift), force (perform the adaptation always), disabled (do not perform adaptation) - ## - adaptSecurityContext: auto -## @section Common parameters -## - -## @param kubeVersion Override Kubernetes version -## -kubeVersion: "" -## @param nameOverride String to partially override common.names.fullname template (will maintain the release name) -## -nameOverride: "" -## @param fullnameOverride String to fully override common.names.fullname template -## -fullnameOverride: "" -## @param clusterDomain Kubernetes Cluster Domain -## -clusterDomain: cluster.local -## @param extraDeploy Array of extra objects to deploy with the release (evaluated as a template) -## -extraDeploy: [] -## @param commonLabels Add labels to all the deployed resources -## -commonLabels: {} -## @param commonAnnotations Add annotations to all the deployed resources -## -commonAnnotations: {} -## Enable diagnostic mode in the statefulset -## -diagnosticMode: - ## @param diagnosticMode.enabled Enable diagnostic mode (all probes will be disabled and the command will be overridden) - ## - enabled: false - ## @param diagnosticMode.command Command to override all containers in the statefulset - ## - command: - - sleep - ## @param diagnosticMode.args Args to override all containers in the statefulset - ## - args: - - infinity -## @section PostgreSQL common parameters -## - -## Bitnami PostgreSQL image version -## ref: https://hub.docker.com/r/bitnami/postgresql/tags/ -## @param image.registry [default: REGISTRY_NAME] PostgreSQL image registry -## @param image.repository [default: REPOSITORY_NAME/postgresql] PostgreSQL image repository -## @skip image.tag PostgreSQL image tag (immutable tags are recommended) -## @param image.digest PostgreSQL image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag -## @param image.pullPolicy PostgreSQL image pull policy -## @param image.pullSecrets Specify image pull secrets -## @param image.debug Specify if debug values should be set -## -image: - registry: docker.io - repository: bitnami/postgresql - tag: 16.3.0-debian-12-r19 - digest: "" - ## Specify a imagePullPolicy - ## Defaults to 'Always' if image tag is 'latest', else set to 'IfNotPresent' - ## ref: https://kubernetes.io/docs/concepts/containers/images/#pre-pulled-images - ## - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets. - ## Secrets must be manually created in the namespace. - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## Example: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## Set to true if you would like to see extra information on logs - ## - debug: false -## Authentication parameters -## ref: https://github.com/bitnami/containers/tree/main/bitnami/postgresql#setting-the-root-password-on-first-run -## ref: https://github.com/bitnami/containers/tree/main/bitnami/postgresql#creating-a-database-on-first-run -## ref: https://github.com/bitnami/containers/tree/main/bitnami/postgresql#creating-a-database-user-on-first-run -## -auth: - ## @param auth.enablePostgresUser Assign a password to the "postgres" admin user. Otherwise, remote access will be blocked for this user - ## - enablePostgresUser: true - ## @param auth.postgresPassword Password for the "postgres" admin user. Ignored if `auth.existingSecret` is provided - ## - postgresPassword: "" - ## @param auth.username Name for a custom user to create - ## - username: "" - ## @param auth.password Password for the custom user to create. Ignored if `auth.existingSecret` is provided - ## - password: "" - ## @param auth.database Name for a custom database to create - ## - database: "" - ## @param auth.replicationUsername Name of the replication user - ## - replicationUsername: repl_user - ## @param auth.replicationPassword Password for the replication user. Ignored if `auth.existingSecret` is provided - ## - replicationPassword: "" - ## @param auth.existingSecret Name of existing secret to use for PostgreSQL credentials. `auth.postgresPassword`, `auth.password`, and `auth.replicationPassword` will be ignored and picked up from this secret. The secret might also contains the key `ldap-password` if LDAP is enabled. `ldap.bind_password` will be ignored and picked from this secret in this case. - ## - existingSecret: "" - ## @param auth.secretKeys.adminPasswordKey Name of key in existing secret to use for PostgreSQL credentials. Only used when `auth.existingSecret` is set. - ## @param auth.secretKeys.userPasswordKey Name of key in existing secret to use for PostgreSQL credentials. Only used when `auth.existingSecret` is set. - ## @param auth.secretKeys.replicationPasswordKey Name of key in existing secret to use for PostgreSQL credentials. Only used when `auth.existingSecret` is set. - ## - secretKeys: - adminPasswordKey: postgres-password - userPasswordKey: password - replicationPasswordKey: replication-password - ## @param auth.usePasswordFiles Mount credentials as a files instead of using an environment variable - ## - usePasswordFiles: false -## @param architecture PostgreSQL architecture (`standalone` or `replication`) -## -architecture: standalone -## Replication configuration -## Ignored if `architecture` is `standalone` -## -replication: - ## @param replication.synchronousCommit Set synchronous commit mode. Allowed values: `on`, `remote_apply`, `remote_write`, `local` and `off` - ## @param replication.numSynchronousReplicas Number of replicas that will have synchronous replication. Note: Cannot be greater than `readReplicas.replicaCount`. - ## ref: https://www.postgresql.org/docs/current/runtime-config-wal.html#GUC-SYNCHRONOUS-COMMIT - ## - synchronousCommit: "off" - numSynchronousReplicas: 0 - ## @param replication.applicationName Cluster application name. Useful for advanced replication settings - ## - applicationName: my_application -## @param containerPorts.postgresql PostgreSQL container port -## -containerPorts: - postgresql: 5432 -## Audit settings -## https://github.com/bitnami/containers/tree/main/bitnami/postgresql#auditing -## @param audit.logHostname Log client hostnames -## @param audit.logConnections Add client log-in operations to the log file -## @param audit.logDisconnections Add client log-outs operations to the log file -## @param audit.pgAuditLog Add operations to log using the pgAudit extension -## @param audit.pgAuditLogCatalog Log catalog using pgAudit -## @param audit.clientMinMessages Message log level to share with the user -## @param audit.logLinePrefix Template for log line prefix (default if not set) -## @param audit.logTimezone Timezone for the log timestamps -## -audit: - logHostname: false - logConnections: false - logDisconnections: false - pgAuditLog: "" - pgAuditLogCatalog: "off" - clientMinMessages: error - logLinePrefix: "" - logTimezone: "" -## LDAP configuration -## @param ldap.enabled Enable LDAP support -## DEPRECATED ldap.url It will removed in a future, please use 'ldap.uri' instead -## @param ldap.server IP address or name of the LDAP server. -## @param ldap.port Port number on the LDAP server to connect to -## @param ldap.prefix String to prepend to the user name when forming the DN to bind -## @param ldap.suffix String to append to the user name when forming the DN to bind -## DEPRECATED ldap.baseDN It will removed in a future, please use 'ldap.basedn' instead -## DEPRECATED ldap.bindDN It will removed in a future, please use 'ldap.binddn' instead -## DEPRECATED ldap.bind_password It will removed in a future, please use 'ldap.bindpw' instead -## @param ldap.basedn Root DN to begin the search for the user in -## @param ldap.binddn DN of user to bind to LDAP -## @param ldap.bindpw Password for the user to bind to LDAP -## DEPRECATED ldap.search_attr It will removed in a future, please use 'ldap.searchAttribute' instead -## DEPRECATED ldap.search_filter It will removed in a future, please use 'ldap.searchFilter' instead -## @param ldap.searchAttribute Attribute to match against the user name in the search -## @param ldap.searchFilter The search filter to use when doing search+bind authentication -## @param ldap.scheme Set to `ldaps` to use LDAPS -## DEPRECATED ldap.tls as string is deprecated, please use 'ldap.tls.enabled' instead -## @param ldap.tls.enabled Se to true to enable TLS encryption -## -ldap: - enabled: false - server: "" - port: "" - prefix: "" - suffix: "" - basedn: "" - binddn: "" - bindpw: "" - searchAttribute: "" - searchFilter: "" - scheme: "" - tls: - enabled: false - ## @param ldap.uri LDAP URL beginning in the form `ldap[s]://host[:port]/basedn`. If provided, all the other LDAP parameters will be ignored. - ## Ref: https://www.postgresql.org/docs/current/auth-ldap.html - ## - uri: "" -## @param postgresqlDataDir PostgreSQL data dir folder -## -postgresqlDataDir: /bitnami/postgresql/data -## @param postgresqlSharedPreloadLibraries Shared preload libraries (comma-separated list) -## -postgresqlSharedPreloadLibraries: "pgaudit" -## Start PostgreSQL pod(s) without limitations on shm memory. -## By default docker and containerd (and possibly other container runtimes) limit `/dev/shm` to `64M` -## ref: https://github.com/docker-library/postgres/issues/416 -## ref: https://github.com/containerd/containerd/issues/3654 -## -shmVolume: - ## @param shmVolume.enabled Enable emptyDir volume for /dev/shm for PostgreSQL pod(s) - ## - enabled: true - ## @param shmVolume.sizeLimit Set this to enable a size limit on the shm tmpfs - ## Note: the size of the tmpfs counts against container's memory limit - ## e.g: - ## sizeLimit: 1Gi - ## - sizeLimit: "" -## TLS configuration -## -tls: - ## @param tls.enabled Enable TLS traffic support - ## - enabled: false - ## @param tls.autoGenerated Generate automatically self-signed TLS certificates - ## - autoGenerated: false - ## @param tls.preferServerCiphers Whether to use the server's TLS cipher preferences rather than the client's - ## - preferServerCiphers: true - ## @param tls.certificatesSecret Name of an existing secret that contains the certificates - ## - certificatesSecret: "" - ## @param tls.certFilename Certificate filename - ## - certFilename: "" - ## @param tls.certKeyFilename Certificate key filename - ## - certKeyFilename: "" - ## @param tls.certCAFilename CA Certificate filename - ## If provided, PostgreSQL will authenticate TLS/SSL clients by requesting them a certificate - ## ref: https://www.postgresql.org/docs/9.6/auth-methods.html - ## - certCAFilename: "" - ## @param tls.crlFilename File containing a Certificate Revocation List - ## - crlFilename: "" -## @section PostgreSQL Primary parameters -## -primary: - ## @param primary.name Name of the primary database (eg primary, master, leader, ...) - ## - name: primary - ## @param primary.configuration PostgreSQL Primary main configuration to be injected as ConfigMap - ## ref: https://www.postgresql.org/docs/current/static/runtime-config.html - ## - configuration: "" - ## @param primary.pgHbaConfiguration PostgreSQL Primary client authentication configuration - ## ref: https://www.postgresql.org/docs/current/static/auth-pg-hba-conf.html - ## e.g:# - ## pgHbaConfiguration: |- - ## local all all trust - ## host all all localhost trust - ## host mydatabase mysuser 192.168.0.0/24 md5 - ## - pgHbaConfiguration: "" - ## @param primary.existingConfigmap Name of an existing ConfigMap with PostgreSQL Primary configuration - ## NOTE: `primary.configuration` and `primary.pgHbaConfiguration` will be ignored - ## - existingConfigmap: "" - ## @param primary.extendedConfiguration Extended PostgreSQL Primary configuration (appended to main or default configuration) - ## ref: https://github.com/bitnami/containers/tree/main/bitnami/postgresql#allow-settings-to-be-loaded-from-files-other-than-the-default-postgresqlconf - ## - extendedConfiguration: "" - ## @param primary.existingExtendedConfigmap Name of an existing ConfigMap with PostgreSQL Primary extended configuration - ## NOTE: `primary.extendedConfiguration` will be ignored - ## - existingExtendedConfigmap: "" - ## Initdb configuration - ## ref: https://github.com/bitnami/containers/tree/main/bitnami/postgresql#specifying-initdb-arguments - ## - initdb: - ## @param primary.initdb.args PostgreSQL initdb extra arguments - ## - args: "" - ## @param primary.initdb.postgresqlWalDir Specify a custom location for the PostgreSQL transaction log - ## - postgresqlWalDir: "" - ## @param primary.initdb.scripts Dictionary of initdb scripts - ## Specify dictionary of scripts to be run at first boot - ## e.g: - ## scripts: - ## my_init_script.sh: | - ## #!/bin/sh - ## echo "Do something." - ## - scripts: {} - ## @param primary.initdb.scriptsConfigMap ConfigMap with scripts to be run at first boot - ## NOTE: This will override `primary.initdb.scripts` - ## - scriptsConfigMap: "" - ## @param primary.initdb.scriptsSecret Secret with scripts to be run at first boot (in case it contains sensitive information) - ## NOTE: This can work along `primary.initdb.scripts` or `primary.initdb.scriptsConfigMap` - ## - scriptsSecret: "" - ## @param primary.initdb.user Specify the PostgreSQL username to execute the initdb scripts - ## - user: "" - ## @param primary.initdb.password Specify the PostgreSQL password to execute the initdb scripts - ## - password: "" - ## Pre-init configuration - ## ref: https://github.com/bitnami/containers/tree/main/bitnami/postgresql/#on-container-start - preInitDb: - ## @param primary.preInitDb.scripts Dictionary of pre-init scripts - ## Specify dictionary of shell scripts to be run before db boot - ## e.g: - ## scripts: - ## my_pre_init_script.sh: | - ## #!/bin/sh - ## echo "Do something." - scripts: {} - ## @param primary.preInitDb.scriptsConfigMap ConfigMap with pre-init scripts to be run - ## NOTE: This will override `primary.preInitDb.scripts` - scriptsConfigMap: "" - ## @param primary.preInitDb.scriptsSecret Secret with pre-init scripts to be run - ## NOTE: This can work along `primary.preInitDb.scripts` or `primary.preInitDb.scriptsConfigMap` - scriptsSecret: "" - ## Configure current cluster's primary server to be the standby server in other cluster. - ## This will allow cross cluster replication and provide cross cluster high availability. - ## You will need to configure pgHbaConfiguration if you want to enable this feature with local cluster replication enabled. - ## @param primary.standby.enabled Whether to enable current cluster's primary as standby server of another cluster or not - ## @param primary.standby.primaryHost The Host of replication primary in the other cluster - ## @param primary.standby.primaryPort The Port of replication primary in the other cluster - ## - standby: - enabled: false - primaryHost: "" - primaryPort: "" - ## @param primary.extraEnvVars Array with extra environment variables to add to PostgreSQL Primary nodes - ## e.g: - ## extraEnvVars: - ## - name: FOO - ## value: "bar" - ## - extraEnvVars: [] - ## @param primary.extraEnvVarsCM Name of existing ConfigMap containing extra env vars for PostgreSQL Primary nodes - ## - extraEnvVarsCM: "" - ## @param primary.extraEnvVarsSecret Name of existing Secret containing extra env vars for PostgreSQL Primary nodes - ## - extraEnvVarsSecret: "" - ## @param primary.command Override default container command (useful when using custom images) - ## - command: [] - ## @param primary.args Override default container args (useful when using custom images) - ## - args: [] - ## Configure extra options for PostgreSQL Primary containers' liveness, readiness and startup probes - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/#configure-probes - ## @param primary.livenessProbe.enabled Enable livenessProbe on PostgreSQL Primary containers - ## @param primary.livenessProbe.initialDelaySeconds Initial delay seconds for livenessProbe - ## @param primary.livenessProbe.periodSeconds Period seconds for livenessProbe - ## @param primary.livenessProbe.timeoutSeconds Timeout seconds for livenessProbe - ## @param primary.livenessProbe.failureThreshold Failure threshold for livenessProbe - ## @param primary.livenessProbe.successThreshold Success threshold for livenessProbe - ## - livenessProbe: - enabled: true - initialDelaySeconds: 30 - periodSeconds: 10 - timeoutSeconds: 5 - failureThreshold: 6 - successThreshold: 1 - ## @param primary.readinessProbe.enabled Enable readinessProbe on PostgreSQL Primary containers - ## @param primary.readinessProbe.initialDelaySeconds Initial delay seconds for readinessProbe - ## @param primary.readinessProbe.periodSeconds Period seconds for readinessProbe - ## @param primary.readinessProbe.timeoutSeconds Timeout seconds for readinessProbe - ## @param primary.readinessProbe.failureThreshold Failure threshold for readinessProbe - ## @param primary.readinessProbe.successThreshold Success threshold for readinessProbe - ## - readinessProbe: - enabled: true - initialDelaySeconds: 5 - periodSeconds: 10 - timeoutSeconds: 5 - failureThreshold: 6 - successThreshold: 1 - ## @param primary.startupProbe.enabled Enable startupProbe on PostgreSQL Primary containers - ## @param primary.startupProbe.initialDelaySeconds Initial delay seconds for startupProbe - ## @param primary.startupProbe.periodSeconds Period seconds for startupProbe - ## @param primary.startupProbe.timeoutSeconds Timeout seconds for startupProbe - ## @param primary.startupProbe.failureThreshold Failure threshold for startupProbe - ## @param primary.startupProbe.successThreshold Success threshold for startupProbe - ## - startupProbe: - enabled: false - initialDelaySeconds: 30 - periodSeconds: 10 - timeoutSeconds: 1 - failureThreshold: 15 - successThreshold: 1 - ## @param primary.customLivenessProbe Custom livenessProbe that overrides the default one - ## - customLivenessProbe: {} - ## @param primary.customReadinessProbe Custom readinessProbe that overrides the default one - ## - customReadinessProbe: {} - ## @param primary.customStartupProbe Custom startupProbe that overrides the default one - ## - customStartupProbe: {} - ## @param primary.lifecycleHooks for the PostgreSQL Primary container to automate configuration before or after startup - ## - lifecycleHooks: {} - ## PostgreSQL Primary resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## @param primary.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if primary.resources is set (primary.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "nano" - ## @param primary.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} - ## Pod Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/ - ## @param primary.podSecurityContext.enabled Enable security context - ## @param primary.podSecurityContext.fsGroupChangePolicy Set filesystem group change policy - ## @param primary.podSecurityContext.sysctls Set kernel settings using the sysctl interface - ## @param primary.podSecurityContext.supplementalGroups Set filesystem extra groups - ## @param primary.podSecurityContext.fsGroup Group ID for the pod - ## - podSecurityContext: - enabled: true - fsGroupChangePolicy: Always - sysctls: [] - supplementalGroups: [] - fsGroup: 1001 - ## Container Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/ - ## @param primary.containerSecurityContext.enabled Enabled containers' Security Context - ## @param primary.containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container - ## @param primary.containerSecurityContext.runAsUser Set containers' Security Context runAsUser - ## @param primary.containerSecurityContext.runAsGroup Set containers' Security Context runAsGroup - ## @param primary.containerSecurityContext.runAsNonRoot Set container's Security Context runAsNonRoot - ## @param primary.containerSecurityContext.privileged Set container's Security Context privileged - ## @param primary.containerSecurityContext.readOnlyRootFilesystem Set container's Security Context readOnlyRootFilesystem - ## @param primary.containerSecurityContext.allowPrivilegeEscalation Set container's Security Context allowPrivilegeEscalation - ## @param primary.containerSecurityContext.capabilities.drop List of capabilities to be dropped - ## @param primary.containerSecurityContext.seccompProfile.type Set container's Security Context seccomp profile - ## - containerSecurityContext: - enabled: true - seLinuxOptions: {} - runAsUser: 1001 - runAsGroup: 1001 - runAsNonRoot: true - privileged: false - readOnlyRootFilesystem: true - allowPrivilegeEscalation: false - capabilities: - drop: ["ALL"] - seccompProfile: - type: "RuntimeDefault" - ## @param primary.automountServiceAccountToken Mount Service Account token in pod - ## - automountServiceAccountToken: false - ## @param primary.hostAliases PostgreSQL primary pods host aliases - ## https://kubernetes.io/docs/concepts/services-networking/add-entries-to-pod-etc-hosts-with-host-aliases/ - ## - hostAliases: [] - ## @param primary.hostNetwork Specify if host network should be enabled for PostgreSQL pod (postgresql primary) - ## - hostNetwork: false - ## @param primary.hostIPC Specify if host IPC should be enabled for PostgreSQL pod (postgresql primary) - ## - hostIPC: false - ## @param primary.labels Map of labels to add to the statefulset (postgresql primary) - ## - labels: {} - ## @param primary.annotations Annotations for PostgreSQL primary pods - ## - annotations: {} - ## @param primary.podLabels Map of labels to add to the pods (postgresql primary) - ## - podLabels: {} - ## @param primary.podAnnotations Map of annotations to add to the pods (postgresql primary) - ## - podAnnotations: {} - ## @param primary.podAffinityPreset PostgreSQL primary pod affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity - ## - podAffinityPreset: "" - ## @param primary.podAntiAffinityPreset PostgreSQL primary pod anti-affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity - ## - podAntiAffinityPreset: soft - ## PostgreSQL Primary node affinity preset - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity - ## - nodeAffinityPreset: - ## @param primary.nodeAffinityPreset.type PostgreSQL primary node affinity preset type. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` - ## - type: "" - ## @param primary.nodeAffinityPreset.key PostgreSQL primary node label key to match Ignored if `primary.affinity` is set. - ## E.g. - ## key: "kubernetes.io/e2e-az-name" - ## - key: "" - ## @param primary.nodeAffinityPreset.values PostgreSQL primary node label values to match. Ignored if `primary.affinity` is set. - ## E.g. - ## values: - ## - e2e-az1 - ## - e2e-az2 - ## - values: [] - ## @param primary.affinity Affinity for PostgreSQL primary pods assignment - ## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity - ## Note: primary.podAffinityPreset, primary.podAntiAffinityPreset, and primary.nodeAffinityPreset will be ignored when it's set - ## - affinity: {} - ## @param primary.nodeSelector Node labels for PostgreSQL primary pods assignment - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/ - ## - nodeSelector: {} - ## @param primary.tolerations Tolerations for PostgreSQL primary pods assignment - ## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ - ## - tolerations: [] - ## @param primary.topologySpreadConstraints Topology Spread Constraints for pod assignment spread across your cluster among failure-domains. Evaluated as a template - ## Ref: https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/#spread-constraints-for-pods - ## - topologySpreadConstraints: [] - ## @param primary.priorityClassName Priority Class to use for each pod (postgresql primary) - ## - priorityClassName: "" - ## @param primary.schedulerName Use an alternate scheduler, e.g. "stork". - ## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ - ## - schedulerName: "" - ## @param primary.terminationGracePeriodSeconds Seconds PostgreSQL primary pod needs to terminate gracefully - ## ref: https://kubernetes.io/docs/concepts/workloads/pods/pod/#termination-of-pods - ## - terminationGracePeriodSeconds: "" - ## @param primary.updateStrategy.type PostgreSQL Primary statefulset strategy type - ## @param primary.updateStrategy.rollingUpdate PostgreSQL Primary statefulset rolling update configuration parameters - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#update-strategies - ## - updateStrategy: - type: RollingUpdate - rollingUpdate: {} - ## @param primary.extraVolumeMounts Optionally specify extra list of additional volumeMounts for the PostgreSQL Primary container(s) - ## - extraVolumeMounts: [] - ## @param primary.extraVolumes Optionally specify extra list of additional volumes for the PostgreSQL Primary pod(s) - ## - extraVolumes: [] - ## @param primary.sidecars Add additional sidecar containers to the PostgreSQL Primary pod(s) - ## For example: - ## sidecars: - ## - name: your-image-name - ## image: your-image - ## imagePullPolicy: Always - ## ports: - ## - name: portname - ## containerPort: 1234 - ## - sidecars: [] - ## @param primary.initContainers Add additional init containers to the PostgreSQL Primary pod(s) - ## Example - ## - ## initContainers: - ## - name: do-something - ## image: busybox - ## command: ['do', 'something'] - ## - initContainers: [] - ## Pod Disruption Budget configuration - ## ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb - ## @param primary.pdb.create Enable/disable a Pod Disruption Budget creation - ## @param primary.pdb.minAvailable Minimum number/percentage of pods that should remain scheduled - ## @param primary.pdb.maxUnavailable Maximum number/percentage of pods that may be made unavailable. Defaults to `1` if both `primary.pdb.minAvailable` and `primary.pdb.maxUnavailable` are empty. - ## - pdb: - create: true - minAvailable: "" - maxUnavailable: "" - ## @param primary.extraPodSpec Optionally specify extra PodSpec for the PostgreSQL Primary pod(s) - ## - extraPodSpec: {} - ## Network Policies - ## Ref: https://kubernetes.io/docs/concepts/services-networking/network-policies/ - ## - networkPolicy: - ## @param primary.networkPolicy.enabled Specifies whether a NetworkPolicy should be created - ## - enabled: true - ## @param primary.networkPolicy.allowExternal Don't require server label for connections - ## The Policy model to apply. When set to false, only pods with the correct - ## server label will have network access to the ports server is listening - ## on. When true, server will accept connections from any source - ## (with the correct destination port). - ## - allowExternal: true - ## @param primary.networkPolicy.allowExternalEgress Allow the pod to access any range of port and all destinations. - ## - allowExternalEgress: true - ## @param primary.networkPolicy.extraIngress [array] Add extra ingress rules to the NetworkPolicy - ## e.g: - ## extraIngress: - ## - ports: - ## - port: 1234 - ## from: - ## - podSelector: - ## - matchLabels: - ## - role: frontend - ## - podSelector: - ## - matchExpressions: - ## - key: role - ## operator: In - ## values: - ## - frontend - extraIngress: [] - ## @param primary.networkPolicy.extraEgress [array] Add extra ingress rules to the NetworkPolicy - ## e.g: - ## extraEgress: - ## - ports: - ## - port: 1234 - ## to: - ## - podSelector: - ## - matchLabels: - ## - role: frontend - ## - podSelector: - ## - matchExpressions: - ## - key: role - ## operator: In - ## values: - ## - frontend - ## - extraEgress: [] - ## @param primary.networkPolicy.ingressNSMatchLabels [object] Labels to match to allow traffic from other namespaces - ## @param primary.networkPolicy.ingressNSPodMatchLabels [object] Pod labels to match to allow traffic from other namespaces - ## - ingressNSMatchLabels: {} - ingressNSPodMatchLabels: {} - ## PostgreSQL Primary service configuration - ## - service: - ## @param primary.service.type Kubernetes Service type - ## - type: ClusterIP - ## @param primary.service.ports.postgresql PostgreSQL service port - ## - ports: - postgresql: 5432 - ## Node ports to expose - ## NOTE: choose port between <30000-32767> - ## @param primary.service.nodePorts.postgresql Node port for PostgreSQL - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport - ## - nodePorts: - postgresql: "" - ## @param primary.service.clusterIP Static clusterIP or None for headless services - ## e.g: - ## clusterIP: None - ## - clusterIP: "" - ## @param primary.service.annotations Annotations for PostgreSQL primary service - ## - annotations: {} - ## @param primary.service.loadBalancerClass Load balancer class if service type is `LoadBalancer` - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#load-balancer-class - ## - loadBalancerClass: "" - ## @param primary.service.loadBalancerIP Load balancer IP if service type is `LoadBalancer` - ## Set the LoadBalancer service type to internal only - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#internal-load-balancer - ## - loadBalancerIP: "" - ## @param primary.service.externalTrafficPolicy Enable client source IP preservation - ## ref https://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer/#preserving-the-client-source-ip - ## - externalTrafficPolicy: Cluster - ## @param primary.service.loadBalancerSourceRanges Addresses that are allowed when service is LoadBalancer - ## https://kubernetes.io/docs/tasks/access-application-cluster/configure-cloud-provider-firewall/#restrict-access-for-loadbalancer-service - ## - ## loadBalancerSourceRanges: - ## - 10.10.10.0/24 - ## - loadBalancerSourceRanges: [] - ## @param primary.service.extraPorts Extra ports to expose in the PostgreSQL primary service - ## - extraPorts: [] - ## @param primary.service.sessionAffinity Session Affinity for Kubernetes service, can be "None" or "ClientIP" - ## If "ClientIP", consecutive client requests will be directed to the same Pod - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies - ## - sessionAffinity: None - ## @param primary.service.sessionAffinityConfig Additional settings for the sessionAffinity - ## sessionAffinityConfig: - ## clientIP: - ## timeoutSeconds: 300 - ## - sessionAffinityConfig: {} - ## Headless service properties - ## - headless: - ## @param primary.service.headless.annotations Additional custom annotations for headless PostgreSQL primary service - ## - annotations: {} - ## PostgreSQL Primary persistence configuration - ## - persistence: - ## @param primary.persistence.enabled Enable PostgreSQL Primary data persistence using PVC - ## - enabled: true - ## @param primary.persistence.volumeName Name to assign the volume - ## - volumeName: "data" - ## @param primary.persistence.existingClaim Name of an existing PVC to use - ## - existingClaim: "" - ## @param primary.persistence.mountPath The path the volume will be mounted at - ## Note: useful when using custom PostgreSQL images - ## - mountPath: /bitnami/postgresql - ## @param primary.persistence.subPath The subdirectory of the volume to mount to - ## Useful in dev environments and one PV for multiple services - ## - subPath: "" - ## @param primary.persistence.storageClass PVC Storage Class for PostgreSQL Primary data volume - ## If defined, storageClassName: - ## If set to "-", storageClassName: "", which disables dynamic provisioning - ## If undefined (the default) or set to null, no storageClassName spec is - ## set, choosing the default provisioner. (gp2 on AWS, standard on - ## GKE, AWS & OpenStack) - ## - storageClass: "" - ## @param primary.persistence.accessModes PVC Access Mode for PostgreSQL volume - ## - accessModes: - - ReadWriteOnce - ## @param primary.persistence.size PVC Storage Request for PostgreSQL volume - ## - size: 8Gi - ## @param primary.persistence.annotations Annotations for the PVC - ## - annotations: {} - ## @param primary.persistence.labels Labels for the PVC - ## - labels: {} - ## @param primary.persistence.selector Selector to match an existing Persistent Volume (this value is evaluated as a template) - ## selector: - ## matchLabels: - ## app: my-app - ## - selector: {} - ## @param primary.persistence.dataSource Custom PVC data source - ## - dataSource: {} - ## PostgreSQL Primary Persistent Volume Claim Retention Policy - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#persistentvolumeclaim-retention - ## - persistentVolumeClaimRetentionPolicy: - ## @param primary.persistentVolumeClaimRetentionPolicy.enabled Enable Persistent volume retention policy for Primary Statefulset - ## - enabled: false - ## @param primary.persistentVolumeClaimRetentionPolicy.whenScaled Volume retention behavior when the replica count of the StatefulSet is reduced - ## - whenScaled: Retain - ## @param primary.persistentVolumeClaimRetentionPolicy.whenDeleted Volume retention behavior that applies when the StatefulSet is deleted - ## - whenDeleted: Retain -## @section PostgreSQL read only replica parameters (only used when `architecture` is set to `replication`) -## -readReplicas: - ## @param readReplicas.name Name of the read replicas database (eg secondary, slave, ...) - ## - name: read - ## @param readReplicas.replicaCount Number of PostgreSQL read only replicas - ## - replicaCount: 1 - ## @param readReplicas.extendedConfiguration Extended PostgreSQL read only replicas configuration (appended to main or default configuration) - ## ref: https://github.com/bitnami/containers/tree/main/bitnami/postgresql#allow-settings-to-be-loaded-from-files-other-than-the-default-postgresqlconf - ## - extendedConfiguration: "" - ## @param readReplicas.extraEnvVars Array with extra environment variables to add to PostgreSQL read only nodes - ## e.g: - ## extraEnvVars: - ## - name: FOO - ## value: "bar" - ## - extraEnvVars: [] - ## @param readReplicas.extraEnvVarsCM Name of existing ConfigMap containing extra env vars for PostgreSQL read only nodes - ## - extraEnvVarsCM: "" - ## @param readReplicas.extraEnvVarsSecret Name of existing Secret containing extra env vars for PostgreSQL read only nodes - ## - extraEnvVarsSecret: "" - ## @param readReplicas.command Override default container command (useful when using custom images) - ## - command: [] - ## @param readReplicas.args Override default container args (useful when using custom images) - ## - args: [] - ## Configure extra options for PostgreSQL read only containers' liveness, readiness and startup probes - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/#configure-probes - ## @param readReplicas.livenessProbe.enabled Enable livenessProbe on PostgreSQL read only containers - ## @param readReplicas.livenessProbe.initialDelaySeconds Initial delay seconds for livenessProbe - ## @param readReplicas.livenessProbe.periodSeconds Period seconds for livenessProbe - ## @param readReplicas.livenessProbe.timeoutSeconds Timeout seconds for livenessProbe - ## @param readReplicas.livenessProbe.failureThreshold Failure threshold for livenessProbe - ## @param readReplicas.livenessProbe.successThreshold Success threshold for livenessProbe - ## - livenessProbe: - enabled: true - initialDelaySeconds: 30 - periodSeconds: 10 - timeoutSeconds: 5 - failureThreshold: 6 - successThreshold: 1 - ## @param readReplicas.readinessProbe.enabled Enable readinessProbe on PostgreSQL read only containers - ## @param readReplicas.readinessProbe.initialDelaySeconds Initial delay seconds for readinessProbe - ## @param readReplicas.readinessProbe.periodSeconds Period seconds for readinessProbe - ## @param readReplicas.readinessProbe.timeoutSeconds Timeout seconds for readinessProbe - ## @param readReplicas.readinessProbe.failureThreshold Failure threshold for readinessProbe - ## @param readReplicas.readinessProbe.successThreshold Success threshold for readinessProbe - ## - readinessProbe: - enabled: true - initialDelaySeconds: 5 - periodSeconds: 10 - timeoutSeconds: 5 - failureThreshold: 6 - successThreshold: 1 - ## @param readReplicas.startupProbe.enabled Enable startupProbe on PostgreSQL read only containers - ## @param readReplicas.startupProbe.initialDelaySeconds Initial delay seconds for startupProbe - ## @param readReplicas.startupProbe.periodSeconds Period seconds for startupProbe - ## @param readReplicas.startupProbe.timeoutSeconds Timeout seconds for startupProbe - ## @param readReplicas.startupProbe.failureThreshold Failure threshold for startupProbe - ## @param readReplicas.startupProbe.successThreshold Success threshold for startupProbe - ## - startupProbe: - enabled: false - initialDelaySeconds: 30 - periodSeconds: 10 - timeoutSeconds: 1 - failureThreshold: 15 - successThreshold: 1 - ## @param readReplicas.customLivenessProbe Custom livenessProbe that overrides the default one - ## - customLivenessProbe: {} - ## @param readReplicas.customReadinessProbe Custom readinessProbe that overrides the default one - ## - customReadinessProbe: {} - ## @param readReplicas.customStartupProbe Custom startupProbe that overrides the default one - ## - customStartupProbe: {} - ## @param readReplicas.lifecycleHooks for the PostgreSQL read only container to automate configuration before or after startup - ## - lifecycleHooks: {} - ## PostgreSQL read only resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## @param readReplicas.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if readReplicas.resources is set (readReplicas.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "nano" - ## @param readReplicas.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} - ## Pod Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/ - ## @param readReplicas.podSecurityContext.enabled Enable security context - ## @param readReplicas.podSecurityContext.fsGroupChangePolicy Set filesystem group change policy - ## @param readReplicas.podSecurityContext.sysctls Set kernel settings using the sysctl interface - ## @param readReplicas.podSecurityContext.supplementalGroups Set filesystem extra groups - ## @param readReplicas.podSecurityContext.fsGroup Group ID for the pod - ## - podSecurityContext: - enabled: true - fsGroupChangePolicy: Always - sysctls: [] - supplementalGroups: [] - fsGroup: 1001 - ## Container Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/ - ## @param readReplicas.containerSecurityContext.enabled Enabled containers' Security Context - ## @param readReplicas.containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container - ## @param readReplicas.containerSecurityContext.runAsUser Set containers' Security Context runAsUser - ## @param readReplicas.containerSecurityContext.runAsGroup Set containers' Security Context runAsGroup - ## @param readReplicas.containerSecurityContext.runAsNonRoot Set container's Security Context runAsNonRoot - ## @param readReplicas.containerSecurityContext.privileged Set container's Security Context privileged - ## @param readReplicas.containerSecurityContext.readOnlyRootFilesystem Set container's Security Context readOnlyRootFilesystem - ## @param readReplicas.containerSecurityContext.allowPrivilegeEscalation Set container's Security Context allowPrivilegeEscalation - ## @param readReplicas.containerSecurityContext.capabilities.drop List of capabilities to be dropped - ## @param readReplicas.containerSecurityContext.seccompProfile.type Set container's Security Context seccomp profile - ## - containerSecurityContext: - enabled: true - seLinuxOptions: {} - runAsUser: 1001 - runAsGroup: 1001 - runAsNonRoot: true - privileged: false - readOnlyRootFilesystem: true - allowPrivilegeEscalation: false - capabilities: - drop: ["ALL"] - seccompProfile: - type: "RuntimeDefault" - ## @param readReplicas.automountServiceAccountToken Mount Service Account token in pod - ## - automountServiceAccountToken: false - ## @param readReplicas.hostAliases PostgreSQL read only pods host aliases - ## https://kubernetes.io/docs/concepts/services-networking/add-entries-to-pod-etc-hosts-with-host-aliases/ - ## - hostAliases: [] - ## @param readReplicas.hostNetwork Specify if host network should be enabled for PostgreSQL pod (PostgreSQL read only) - ## - hostNetwork: false - ## @param readReplicas.hostIPC Specify if host IPC should be enabled for PostgreSQL pod (postgresql primary) - ## - hostIPC: false - ## @param readReplicas.labels Map of labels to add to the statefulset (PostgreSQL read only) - ## - labels: {} - ## @param readReplicas.annotations Annotations for PostgreSQL read only pods - ## - annotations: {} - ## @param readReplicas.podLabels Map of labels to add to the pods (PostgreSQL read only) - ## - podLabels: {} - ## @param readReplicas.podAnnotations Map of annotations to add to the pods (PostgreSQL read only) - ## - podAnnotations: {} - ## @param readReplicas.podAffinityPreset PostgreSQL read only pod affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity - ## - podAffinityPreset: "" - ## @param readReplicas.podAntiAffinityPreset PostgreSQL read only pod anti-affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity - ## - podAntiAffinityPreset: soft - ## PostgreSQL read only node affinity preset - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity - ## - nodeAffinityPreset: - ## @param readReplicas.nodeAffinityPreset.type PostgreSQL read only node affinity preset type. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` - ## - type: "" - ## @param readReplicas.nodeAffinityPreset.key PostgreSQL read only node label key to match Ignored if `primary.affinity` is set. - ## E.g. - ## key: "kubernetes.io/e2e-az-name" - ## - key: "" - ## @param readReplicas.nodeAffinityPreset.values PostgreSQL read only node label values to match. Ignored if `primary.affinity` is set. - ## E.g. - ## values: - ## - e2e-az1 - ## - e2e-az2 - ## - values: [] - ## @param readReplicas.affinity Affinity for PostgreSQL read only pods assignment - ## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity - ## Note: primary.podAffinityPreset, primary.podAntiAffinityPreset, and primary.nodeAffinityPreset will be ignored when it's set - ## - affinity: {} - ## @param readReplicas.nodeSelector Node labels for PostgreSQL read only pods assignment - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/ - ## - nodeSelector: {} - ## @param readReplicas.tolerations Tolerations for PostgreSQL read only pods assignment - ## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ - ## - tolerations: [] - ## @param readReplicas.topologySpreadConstraints Topology Spread Constraints for pod assignment spread across your cluster among failure-domains. Evaluated as a template - ## Ref: https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/#spread-constraints-for-pods - ## - topologySpreadConstraints: [] - ## @param readReplicas.priorityClassName Priority Class to use for each pod (PostgreSQL read only) - ## - priorityClassName: "" - ## @param readReplicas.schedulerName Use an alternate scheduler, e.g. "stork". - ## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ - ## - schedulerName: "" - ## @param readReplicas.terminationGracePeriodSeconds Seconds PostgreSQL read only pod needs to terminate gracefully - ## ref: https://kubernetes.io/docs/concepts/workloads/pods/pod/#termination-of-pods - ## - terminationGracePeriodSeconds: "" - ## @param readReplicas.updateStrategy.type PostgreSQL read only statefulset strategy type - ## @param readReplicas.updateStrategy.rollingUpdate PostgreSQL read only statefulset rolling update configuration parameters - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#update-strategies - ## - updateStrategy: - type: RollingUpdate - rollingUpdate: {} - ## @param readReplicas.extraVolumeMounts Optionally specify extra list of additional volumeMounts for the PostgreSQL read only container(s) - ## - extraVolumeMounts: [] - ## @param readReplicas.extraVolumes Optionally specify extra list of additional volumes for the PostgreSQL read only pod(s) - ## - extraVolumes: [] - ## @param readReplicas.sidecars Add additional sidecar containers to the PostgreSQL read only pod(s) - ## For example: - ## sidecars: - ## - name: your-image-name - ## image: your-image - ## imagePullPolicy: Always - ## ports: - ## - name: portname - ## containerPort: 1234 - ## - sidecars: [] - ## @param readReplicas.initContainers Add additional init containers to the PostgreSQL read only pod(s) - ## Example - ## - ## initContainers: - ## - name: do-something - ## image: busybox - ## command: ['do', 'something'] - ## - initContainers: [] - ## Pod Disruption Budget configuration - ## ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb - ## @param readReplicas.pdb.create Enable/disable a Pod Disruption Budget creation - ## @param readReplicas.pdb.minAvailable Minimum number/percentage of pods that should remain scheduled - ## @param readReplicas.pdb.maxUnavailable Maximum number/percentage of pods that may be made unavailable. Defaults to `1` if both `readReplicas.pdb.minAvailable` and `readReplicas.pdb.maxUnavailable` are empty. - ## - pdb: - create: true - minAvailable: "" - maxUnavailable: "" - ## @param readReplicas.extraPodSpec Optionally specify extra PodSpec for the PostgreSQL read only pod(s) - ## - extraPodSpec: {} - ## Network Policies - ## Ref: https://kubernetes.io/docs/concepts/services-networking/network-policies/ - ## - networkPolicy: - ## @param readReplicas.networkPolicy.enabled Specifies whether a NetworkPolicy should be created - ## - enabled: true - ## @param readReplicas.networkPolicy.allowExternal Don't require server label for connections - ## The Policy model to apply. When set to false, only pods with the correct - ## server label will have network access to the ports server is listening - ## on. When true, server will accept connections from any source - ## (with the correct destination port). - ## - allowExternal: true - ## @param readReplicas.networkPolicy.allowExternalEgress Allow the pod to access any range of port and all destinations. - ## - allowExternalEgress: true - ## @param readReplicas.networkPolicy.extraIngress [array] Add extra ingress rules to the NetworkPolicy - ## e.g: - ## extraIngress: - ## - ports: - ## - port: 1234 - ## from: - ## - podSelector: - ## - matchLabels: - ## - role: frontend - ## - podSelector: - ## - matchExpressions: - ## - key: role - ## operator: In - ## values: - ## - frontend - extraIngress: [] - ## @param readReplicas.networkPolicy.extraEgress [array] Add extra ingress rules to the NetworkPolicy - ## e.g: - ## extraEgress: - ## - ports: - ## - port: 1234 - ## to: - ## - podSelector: - ## - matchLabels: - ## - role: frontend - ## - podSelector: - ## - matchExpressions: - ## - key: role - ## operator: In - ## values: - ## - frontend - ## - extraEgress: [] - ## @param readReplicas.networkPolicy.ingressNSMatchLabels [object] Labels to match to allow traffic from other namespaces - ## @param readReplicas.networkPolicy.ingressNSPodMatchLabels [object] Pod labels to match to allow traffic from other namespaces - ## - ingressNSMatchLabels: {} - ingressNSPodMatchLabels: {} - ## PostgreSQL read only service configuration - ## - service: - ## @param readReplicas.service.type Kubernetes Service type - ## - type: ClusterIP - ## @param readReplicas.service.ports.postgresql PostgreSQL service port - ## - ports: - postgresql: 5432 - ## Node ports to expose - ## NOTE: choose port between <30000-32767> - ## @param readReplicas.service.nodePorts.postgresql Node port for PostgreSQL - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport - ## - nodePorts: - postgresql: "" - ## @param readReplicas.service.clusterIP Static clusterIP or None for headless services - ## e.g: - ## clusterIP: None - ## - clusterIP: "" - ## @param readReplicas.service.annotations Annotations for PostgreSQL read only service - ## - annotations: {} - ## @param readReplicas.service.loadBalancerClass Load balancer class if service type is `LoadBalancer` - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#load-balancer-class - ## - loadBalancerClass: "" - ## @param readReplicas.service.loadBalancerIP Load balancer IP if service type is `LoadBalancer` - ## Set the LoadBalancer service type to internal only - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#internal-load-balancer - ## - loadBalancerIP: "" - ## @param readReplicas.service.externalTrafficPolicy Enable client source IP preservation - ## ref https://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer/#preserving-the-client-source-ip - ## - externalTrafficPolicy: Cluster - ## @param readReplicas.service.loadBalancerSourceRanges Addresses that are allowed when service is LoadBalancer - ## https://kubernetes.io/docs/tasks/access-application-cluster/configure-cloud-provider-firewall/#restrict-access-for-loadbalancer-service - ## - ## loadBalancerSourceRanges: - ## - 10.10.10.0/24 - ## - loadBalancerSourceRanges: [] - ## @param readReplicas.service.extraPorts Extra ports to expose in the PostgreSQL read only service - ## - extraPorts: [] - ## @param readReplicas.service.sessionAffinity Session Affinity for Kubernetes service, can be "None" or "ClientIP" - ## If "ClientIP", consecutive client requests will be directed to the same Pod - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies - ## - sessionAffinity: None - ## @param readReplicas.service.sessionAffinityConfig Additional settings for the sessionAffinity - ## sessionAffinityConfig: - ## clientIP: - ## timeoutSeconds: 300 - ## - sessionAffinityConfig: {} - ## Headless service properties - ## - headless: - ## @param readReplicas.service.headless.annotations Additional custom annotations for headless PostgreSQL read only service - ## - annotations: {} - ## PostgreSQL read only persistence configuration - ## - persistence: - ## @param readReplicas.persistence.enabled Enable PostgreSQL read only data persistence using PVC - ## - enabled: true - ## @param readReplicas.persistence.existingClaim Name of an existing PVC to use - ## - existingClaim: "" - ## @param readReplicas.persistence.mountPath The path the volume will be mounted at - ## Note: useful when using custom PostgreSQL images - ## - mountPath: /bitnami/postgresql - ## @param readReplicas.persistence.subPath The subdirectory of the volume to mount to - ## Useful in dev environments and one PV for multiple services - ## - subPath: "" - ## @param readReplicas.persistence.storageClass PVC Storage Class for PostgreSQL read only data volume - ## If defined, storageClassName: - ## If set to "-", storageClassName: "", which disables dynamic provisioning - ## If undefined (the default) or set to null, no storageClassName spec is - ## set, choosing the default provisioner. (gp2 on AWS, standard on - ## GKE, AWS & OpenStack) - ## - storageClass: "" - ## @param readReplicas.persistence.accessModes PVC Access Mode for PostgreSQL volume - ## - accessModes: - - ReadWriteOnce - ## @param readReplicas.persistence.size PVC Storage Request for PostgreSQL volume - ## - size: 8Gi - ## @param readReplicas.persistence.annotations Annotations for the PVC - ## - annotations: {} - ## @param readReplicas.persistence.labels Labels for the PVC - ## - labels: {} - ## @param readReplicas.persistence.selector Selector to match an existing Persistent Volume (this value is evaluated as a template) - ## selector: - ## matchLabels: - ## app: my-app - ## - selector: {} - ## @param readReplicas.persistence.dataSource Custom PVC data source - ## - dataSource: {} - ## PostgreSQL Read only Persistent Volume Claim Retention Policy - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#persistentvolumeclaim-retention - ## - persistentVolumeClaimRetentionPolicy: - ## @param readReplicas.persistentVolumeClaimRetentionPolicy.enabled Enable Persistent volume retention policy for read only Statefulset - ## - enabled: false - ## @param readReplicas.persistentVolumeClaimRetentionPolicy.whenScaled Volume retention behavior when the replica count of the StatefulSet is reduced - ## - whenScaled: Retain - ## @param readReplicas.persistentVolumeClaimRetentionPolicy.whenDeleted Volume retention behavior that applies when the StatefulSet is deleted - ## - whenDeleted: Retain -## @section Backup parameters -## This section implements a trivial logical dump cronjob of the database. -## This only comes with the consistency guarantees of the dump program. -## This is not a snapshot based roll forward/backward recovery backup. -## ref: https://kubernetes.io/docs/concepts/workloads/controllers/cron-jobs/ -backup: - ## @param backup.enabled Enable the logical dump of the database "regularly" - enabled: false - cronjob: - ## @param backup.cronjob.schedule Set the cronjob parameter schedule - schedule: "@daily" - ## @param backup.cronjob.timeZone Set the cronjob parameter timeZone - timeZone: "" - ## @param backup.cronjob.concurrencyPolicy Set the cronjob parameter concurrencyPolicy - concurrencyPolicy: Allow - ## @param backup.cronjob.failedJobsHistoryLimit Set the cronjob parameter failedJobsHistoryLimit - failedJobsHistoryLimit: 1 - ## @param backup.cronjob.successfulJobsHistoryLimit Set the cronjob parameter successfulJobsHistoryLimit - successfulJobsHistoryLimit: 3 - ## @param backup.cronjob.startingDeadlineSeconds Set the cronjob parameter startingDeadlineSeconds - startingDeadlineSeconds: "" - ## @param backup.cronjob.ttlSecondsAfterFinished Set the cronjob parameter ttlSecondsAfterFinished - ttlSecondsAfterFinished: "" - ## @param backup.cronjob.restartPolicy Set the cronjob parameter restartPolicy - restartPolicy: OnFailure - ## @param backup.cronjob.podSecurityContext.enabled Enable PodSecurityContext for CronJob/Backup - ## @param backup.cronjob.podSecurityContext.fsGroupChangePolicy Set filesystem group change policy - ## @param backup.cronjob.podSecurityContext.sysctls Set kernel settings using the sysctl interface - ## @param backup.cronjob.podSecurityContext.supplementalGroups Set filesystem extra groups - ## @param backup.cronjob.podSecurityContext.fsGroup Group ID for the CronJob - podSecurityContext: - enabled: true - fsGroupChangePolicy: Always - sysctls: [] - supplementalGroups: [] - fsGroup: 1001 - ## backup container's Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container - ## @param backup.cronjob.containerSecurityContext.enabled Enabled containers' Security Context - ## @param backup.cronjob.containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container - ## @param backup.cronjob.containerSecurityContext.runAsUser Set containers' Security Context runAsUser - ## @param backup.cronjob.containerSecurityContext.runAsGroup Set containers' Security Context runAsGroup - ## @param backup.cronjob.containerSecurityContext.runAsNonRoot Set container's Security Context runAsNonRoot - ## @param backup.cronjob.containerSecurityContext.privileged Set container's Security Context privileged - ## @param backup.cronjob.containerSecurityContext.readOnlyRootFilesystem Set container's Security Context readOnlyRootFilesystem - ## @param backup.cronjob.containerSecurityContext.allowPrivilegeEscalation Set container's Security Context allowPrivilegeEscalation - ## @param backup.cronjob.containerSecurityContext.capabilities.drop List of capabilities to be dropped - ## @param backup.cronjob.containerSecurityContext.seccompProfile.type Set container's Security Context seccomp profile - containerSecurityContext: - enabled: true - seLinuxOptions: {} - runAsUser: 1001 - runAsGroup: 1001 - runAsNonRoot: true - privileged: false - readOnlyRootFilesystem: true - allowPrivilegeEscalation: false - capabilities: - drop: ["ALL"] - seccompProfile: - type: "RuntimeDefault" - ## @param backup.cronjob.command Set backup container's command to run - command: - - /bin/sh - - -c - - "pg_dumpall --clean --if-exists --load-via-partition-root --quote-all-identifiers --no-password --file=${PGDUMP_DIR}/pg_dumpall-$(date '+%Y-%m-%d-%H-%M').pgdump" - ## @param backup.cronjob.labels Set the cronjob labels - labels: {} - ## @param backup.cronjob.annotations Set the cronjob annotations - annotations: {} - ## @param backup.cronjob.nodeSelector Node labels for PostgreSQL backup CronJob pod assignment - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/assign-pods-nodes/ - ## - nodeSelector: {} - ## @param backup.cronjob.tolerations Tolerations for PostgreSQL backup CronJob pods assignment - ## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ - ## - tolerations: [] - ## backup cronjob container resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## @param backup.cronjob.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if backup.cronjob.resources is set (backup.cronjob.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "nano" - ## @param backup.cronjob.resources Set container requests and limits for different resources like CPU or memory - ## Example: - resources: {} - ## resources: - ## requests: - ## cpu: 1 - ## memory: 512Mi - ## limits: - ## cpu: 2 - ## memory: 1024Mi - networkPolicy: - ## @param backup.cronjob.networkPolicy.enabled Specifies whether a NetworkPolicy should be created - ## - enabled: true - storage: - ## @param backup.cronjob.storage.enabled Enable using a `PersistentVolumeClaim` as backup data volume - ## - enabled: true - ## @param backup.cronjob.storage.existingClaim Provide an existing `PersistentVolumeClaim` (only when `architecture=standalone`) - ## If defined, PVC must be created manually before volume will be bound - ## - existingClaim: "" - ## @param backup.cronjob.storage.resourcePolicy Setting it to "keep" to avoid removing PVCs during a helm delete operation. Leaving it empty will delete PVCs after the chart deleted - ## - resourcePolicy: "" - ## @param backup.cronjob.storage.storageClass PVC Storage Class for the backup data volume - ## If defined, storageClassName: - ## If set to "-", storageClassName: "", which disables dynamic provisioning - ## If undefined (the default) or set to null, no storageClassName spec is - ## set, choosing the default provisioner. - ## - storageClass: "" - ## @param backup.cronjob.storage.accessModes PV Access Mode - ## - accessModes: - - ReadWriteOnce - ## @param backup.cronjob.storage.size PVC Storage Request for the backup data volume - ## - size: 8Gi - ## @param backup.cronjob.storage.annotations PVC annotations - ## - annotations: {} - ## @param backup.cronjob.storage.mountPath Path to mount the volume at - ## - mountPath: /backup/pgdump - ## @param backup.cronjob.storage.subPath Subdirectory of the volume to mount at - ## and one PV for multiple services. - ## - subPath: "" - ## Fine tuning for volumeClaimTemplates - ## - volumeClaimTemplates: - ## @param backup.cronjob.storage.volumeClaimTemplates.selector A label query over volumes to consider for binding (e.g. when using local volumes) - ## A label query over volumes to consider for binding (e.g. when using local volumes) - ## See https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.20/#labelselector-v1-meta for more details - ## - selector: {} - ## @param backup.cronjob.extraVolumeMounts Optionally specify extra list of additional volumeMounts for the backup container - ## - extraVolumeMounts: [] - ## @param backup.cronjob.extraVolumes Optionally specify extra list of additional volumes for the backup container - ## - extraVolumes: [] -## @section Volume Permissions parameters -## - -## Init containers parameters: -## volumePermissions: Change the owner and group of the persistent volume(s) mountpoint(s) to 'runAsUser:fsGroup' on each node -## -volumePermissions: - ## @param volumePermissions.enabled Enable init container that changes the owner and group of the persistent volume - ## - enabled: false - ## @param volumePermissions.image.registry [default: REGISTRY_NAME] Init container volume-permissions image registry - ## @param volumePermissions.image.repository [default: REPOSITORY_NAME/os-shell] Init container volume-permissions image repository - ## @skip volumePermissions.image.tag Init container volume-permissions image tag (immutable tags are recommended) - ## @param volumePermissions.image.digest Init container volume-permissions image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag - ## @param volumePermissions.image.pullPolicy Init container volume-permissions image pull policy - ## @param volumePermissions.image.pullSecrets Init container volume-permissions image pull secrets - ## - image: - registry: docker.io - repository: bitnami/os-shell - tag: 12-debian-12-r24 - digest: "" - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets. - ## Secrets must be manually created in the namespace. - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## Example: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## Init container resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## @param volumePermissions.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if volumePermissions.resources is set (volumePermissions.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "nano" - ## @param volumePermissions.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} - ## Init container' Security Context - ## Note: the chown of the data folder is done to containerSecurityContext.runAsUser - ## and not the below volumePermissions.containerSecurityContext.runAsUser - ## @param volumePermissions.containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container - ## @param volumePermissions.containerSecurityContext.runAsUser User ID for the init container - ## @param volumePermissions.containerSecurityContext.runAsGroup Group ID for the init container - ## @param volumePermissions.containerSecurityContext.runAsNonRoot runAsNonRoot for the init container - ## @param volumePermissions.containerSecurityContext.seccompProfile.type seccompProfile.type for the init container - ## - containerSecurityContext: - seLinuxOptions: {} - runAsUser: 0 - runAsGroup: 0 - runAsNonRoot: false - seccompProfile: - type: RuntimeDefault -## @section Other Parameters -## - -## @param serviceBindings.enabled Create secret for service binding (Experimental) -## Ref: https://servicebinding.io/service-provider/ -## -serviceBindings: - enabled: false -## Service account for PostgreSQL to use. -## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/ -## -serviceAccount: - ## @param serviceAccount.create Enable creation of ServiceAccount for PostgreSQL pod - ## - create: true - ## @param serviceAccount.name The name of the ServiceAccount to use. - ## If not set and create is true, a name is generated using the common.names.fullname template - ## - name: "" - ## @param serviceAccount.automountServiceAccountToken Allows auto mount of ServiceAccountToken on the serviceAccount created - ## Can be set to false if pods using this serviceAccount do not need to use K8s API - ## - automountServiceAccountToken: false - ## @param serviceAccount.annotations Additional custom annotations for the ServiceAccount - ## - annotations: {} -## Creates role for ServiceAccount -## @param rbac.create Create Role and RoleBinding (required for PSP to work) -## -rbac: - create: false - ## @param rbac.rules Custom RBAC rules to set - ## e.g: - ## rules: - ## - apiGroups: - ## - "" - ## resources: - ## - pods - ## verbs: - ## - get - ## - list - ## - rules: [] -## Pod Security Policy -## ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/ -## @param psp.create Whether to create a PodSecurityPolicy. WARNING: PodSecurityPolicy is deprecated in Kubernetes v1.21 or later, unavailable in v1.25 or later -## -psp: - create: false -## @section Metrics Parameters -## -metrics: - ## @param metrics.enabled Start a prometheus exporter - ## - enabled: false - ## @param metrics.image.registry [default: REGISTRY_NAME] PostgreSQL Prometheus Exporter image registry - ## @param metrics.image.repository [default: REPOSITORY_NAME/postgres-exporter] PostgreSQL Prometheus Exporter image repository - ## @skip metrics.image.tag PostgreSQL Prometheus Exporter image tag (immutable tags are recommended) - ## @param metrics.image.digest PostgreSQL image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag - ## @param metrics.image.pullPolicy PostgreSQL Prometheus Exporter image pull policy - ## @param metrics.image.pullSecrets Specify image pull secrets - ## - image: - registry: docker.io - repository: bitnami/postgres-exporter - tag: 0.15.0-debian-12-r36 - digest: "" - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets. - ## Secrets must be manually created in the namespace. - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## Example: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## @param metrics.collectors Control enabled collectors - ## ref: https://github.com/prometheus-community/postgres_exporter#flags - ## Example: - ## collectors: - ## wal: false - collectors: {} - ## @param metrics.customMetrics Define additional custom metrics - ## ref: https://github.com/prometheus-community/postgres_exporter#adding-new-metrics-via-a-config-file-deprecated - ## customMetrics: - ## pg_database: - ## query: "SELECT d.datname AS name, CASE WHEN pg_catalog.has_database_privilege(d.datname, 'CONNECT') THEN pg_catalog.pg_database_size(d.datname) ELSE 0 END AS size_bytes FROM pg_catalog.pg_database d where datname not in ('template0', 'template1', 'postgres')" - ## metrics: - ## - name: - ## usage: "LABEL" - ## description: "Name of the database" - ## - size_bytes: - ## usage: "GAUGE" - ## description: "Size of the database in bytes" - ## - customMetrics: {} - ## @param metrics.extraEnvVars Extra environment variables to add to PostgreSQL Prometheus exporter - ## see: https://github.com/prometheus-community/postgres_exporter#environment-variables - ## For example: - ## extraEnvVars: - ## - name: PG_EXPORTER_DISABLE_DEFAULT_METRICS - ## value: "true" - ## - extraEnvVars: [] - ## PostgreSQL Prometheus exporter containers' Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container - ## @param metrics.containerSecurityContext.enabled Enabled containers' Security Context - ## @param metrics.containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container - ## @param metrics.containerSecurityContext.runAsUser Set containers' Security Context runAsUser - ## @param metrics.containerSecurityContext.runAsGroup Set containers' Security Context runAsGroup - ## @param metrics.containerSecurityContext.runAsNonRoot Set container's Security Context runAsNonRoot - ## @param metrics.containerSecurityContext.privileged Set container's Security Context privileged - ## @param metrics.containerSecurityContext.readOnlyRootFilesystem Set container's Security Context readOnlyRootFilesystem - ## @param metrics.containerSecurityContext.allowPrivilegeEscalation Set container's Security Context allowPrivilegeEscalation - ## @param metrics.containerSecurityContext.capabilities.drop List of capabilities to be dropped - ## @param metrics.containerSecurityContext.seccompProfile.type Set container's Security Context seccomp profile - ## - containerSecurityContext: - enabled: true - seLinuxOptions: {} - runAsUser: 1001 - runAsGroup: 1001 - runAsNonRoot: true - privileged: false - readOnlyRootFilesystem: true - allowPrivilegeEscalation: false - capabilities: - drop: ["ALL"] - seccompProfile: - type: "RuntimeDefault" - ## Configure extra options for PostgreSQL Prometheus exporter containers' liveness, readiness and startup probes - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/#configure-probes - ## @param metrics.livenessProbe.enabled Enable livenessProbe on PostgreSQL Prometheus exporter containers - ## @param metrics.livenessProbe.initialDelaySeconds Initial delay seconds for livenessProbe - ## @param metrics.livenessProbe.periodSeconds Period seconds for livenessProbe - ## @param metrics.livenessProbe.timeoutSeconds Timeout seconds for livenessProbe - ## @param metrics.livenessProbe.failureThreshold Failure threshold for livenessProbe - ## @param metrics.livenessProbe.successThreshold Success threshold for livenessProbe - ## - livenessProbe: - enabled: true - initialDelaySeconds: 5 - periodSeconds: 10 - timeoutSeconds: 5 - failureThreshold: 6 - successThreshold: 1 - ## @param metrics.readinessProbe.enabled Enable readinessProbe on PostgreSQL Prometheus exporter containers - ## @param metrics.readinessProbe.initialDelaySeconds Initial delay seconds for readinessProbe - ## @param metrics.readinessProbe.periodSeconds Period seconds for readinessProbe - ## @param metrics.readinessProbe.timeoutSeconds Timeout seconds for readinessProbe - ## @param metrics.readinessProbe.failureThreshold Failure threshold for readinessProbe - ## @param metrics.readinessProbe.successThreshold Success threshold for readinessProbe - ## - readinessProbe: - enabled: true - initialDelaySeconds: 5 - periodSeconds: 10 - timeoutSeconds: 5 - failureThreshold: 6 - successThreshold: 1 - ## @param metrics.startupProbe.enabled Enable startupProbe on PostgreSQL Prometheus exporter containers - ## @param metrics.startupProbe.initialDelaySeconds Initial delay seconds for startupProbe - ## @param metrics.startupProbe.periodSeconds Period seconds for startupProbe - ## @param metrics.startupProbe.timeoutSeconds Timeout seconds for startupProbe - ## @param metrics.startupProbe.failureThreshold Failure threshold for startupProbe - ## @param metrics.startupProbe.successThreshold Success threshold for startupProbe - ## - startupProbe: - enabled: false - initialDelaySeconds: 10 - periodSeconds: 10 - timeoutSeconds: 1 - failureThreshold: 15 - successThreshold: 1 - ## @param metrics.customLivenessProbe Custom livenessProbe that overrides the default one - ## - customLivenessProbe: {} - ## @param metrics.customReadinessProbe Custom readinessProbe that overrides the default one - ## - customReadinessProbe: {} - ## @param metrics.customStartupProbe Custom startupProbe that overrides the default one - ## - customStartupProbe: {} - ## @param metrics.containerPorts.metrics PostgreSQL Prometheus exporter metrics container port - ## - containerPorts: - metrics: 9187 - ## PostgreSQL Prometheus exporter resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## @param metrics.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if metrics.resources is set (metrics.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "nano" - ## @param metrics.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} - ## Service configuration - ## - service: - ## @param metrics.service.ports.metrics PostgreSQL Prometheus Exporter service port - ## - ports: - metrics: 9187 - ## @param metrics.service.clusterIP Static clusterIP or None for headless services - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#choosing-your-own-ip-address - ## - clusterIP: "" - ## @param metrics.service.sessionAffinity Control where client requests go, to the same pod or round-robin - ## Values: ClientIP or None - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/ - ## - sessionAffinity: None - ## @param metrics.service.annotations [object] Annotations for Prometheus to auto-discover the metrics endpoint - ## - annotations: - prometheus.io/scrape: "true" - prometheus.io/port: "{{ .Values.metrics.service.ports.metrics }}" - ## Prometheus Operator ServiceMonitor configuration - ## - serviceMonitor: - ## @param metrics.serviceMonitor.enabled Create ServiceMonitor Resource for scraping metrics using Prometheus Operator - ## - enabled: false - ## @param metrics.serviceMonitor.namespace Namespace for the ServiceMonitor Resource (defaults to the Release Namespace) - ## - namespace: "" - ## @param metrics.serviceMonitor.interval Interval at which metrics should be scraped. - ## ref: https://github.com/coreos/prometheus-operator/blob/master/Documentation/api.md#endpoint - ## - interval: "" - ## @param metrics.serviceMonitor.scrapeTimeout Timeout after which the scrape is ended - ## ref: https://github.com/coreos/prometheus-operator/blob/master/Documentation/api.md#endpoint - ## - scrapeTimeout: "" - ## @param metrics.serviceMonitor.labels Additional labels that can be used so ServiceMonitor will be discovered by Prometheus - ## - labels: {} - ## @param metrics.serviceMonitor.selector Prometheus instance selector labels - ## ref: https://github.com/bitnami/charts/tree/main/bitnami/prometheus-operator#prometheus-configuration - ## - selector: {} - ## @param metrics.serviceMonitor.relabelings RelabelConfigs to apply to samples before scraping - ## - relabelings: [] - ## @param metrics.serviceMonitor.metricRelabelings MetricRelabelConfigs to apply to samples before ingestion - ## - metricRelabelings: [] - ## @param metrics.serviceMonitor.honorLabels Specify honorLabels parameter to add the scrape endpoint - ## - honorLabels: false - ## @param metrics.serviceMonitor.jobLabel The name of the label on the target service to use as the job name in prometheus. - ## - jobLabel: "" - ## Custom PrometheusRule to be defined - ## The value is evaluated as a template, so, for example, the value can depend on .Release or .Chart - ## ref: https://github.com/coreos/prometheus-operator#customresourcedefinitions - ## - prometheusRule: - ## @param metrics.prometheusRule.enabled Create a PrometheusRule for Prometheus Operator - ## - enabled: false - ## @param metrics.prometheusRule.namespace Namespace for the PrometheusRule Resource (defaults to the Release Namespace) - ## - namespace: "" - ## @param metrics.prometheusRule.labels Additional labels that can be used so PrometheusRule will be discovered by Prometheus - ## - labels: {} - ## @param metrics.prometheusRule.rules PrometheusRule definitions - ## Make sure to constraint the rules to the current postgresql service. - ## rules: - ## - alert: HugeReplicationLag - ## expr: pg_replication_lag{service="{{ printf "%s-metrics" (include "common.names.fullname" .) }}"} / 3600 > 1 - ## for: 1m - ## labels: - ## severity: critical - ## annotations: - ## description: replication for {{ include "common.names.fullname" . }} PostgreSQL is lagging by {{ "{{ $value }}" }} hour(s). - ## summary: PostgreSQL replication is lagging by {{ "{{ $value }}" }} hour(s). - ## - rules: [] diff --git a/charts/postgresql/values-overrides.yaml b/charts/postgresql/values-overrides.yaml deleted file mode 100644 index 925c771..0000000 --- a/charts/postgresql/values-overrides.yaml +++ /dev/null @@ -1,14 +0,0 @@ -auth: - enablePostgresUser: true - existingSecret: postgresql - usePasswordFiles: false -architecture: replication -readReplicas: - replicaCount: 1 -replication: - syncronousCommit: on - numSyncronousReplicas: 1 - applicationName: futureporn -image: - debug: true -fullnameOverride: postgresql \ No newline at end of file diff --git a/charts/redis/redis/.helmignore b/charts/redis/redis/.helmignore deleted file mode 100644 index 207983f..0000000 --- a/charts/redis/redis/.helmignore +++ /dev/null @@ -1,25 +0,0 @@ -# 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 -# img folder -img/ -# Changelog -CHANGELOG.md diff --git a/charts/redis/redis/Chart.lock b/charts/redis/redis/Chart.lock deleted file mode 100644 index 6a37b39..0000000 --- a/charts/redis/redis/Chart.lock +++ /dev/null @@ -1,6 +0,0 @@ -dependencies: -- name: common - repository: oci://registry-1.docker.io/bitnamicharts - version: 2.22.0 -digest: sha256:a8fb2fc887ead658a89598a48acde5324196fbc0509503a3eaed50a710fbfe74 -generated: "2024-08-09T13:57:12.563276536Z" diff --git a/charts/redis/redis/Chart.yaml b/charts/redis/redis/Chart.yaml deleted file mode 100644 index ce7a625..0000000 --- a/charts/redis/redis/Chart.yaml +++ /dev/null @@ -1,38 +0,0 @@ -annotations: - category: Database - images: | - - name: kubectl - image: docker.io/bitnami/kubectl:1.30.3-debian-12-r5 - - name: os-shell - image: docker.io/bitnami/os-shell:12-debian-12-r27 - - name: redis - image: docker.io/bitnami/redis:7.4.0-debian-12-r0 - - name: redis-exporter - image: docker.io/bitnami/redis-exporter:1.62.0-debian-12-r3 - - name: redis-sentinel - image: docker.io/bitnami/redis-sentinel:7.4.0-debian-12-r0 - licenses: Apache-2.0 -apiVersion: v2 -appVersion: 7.4.0 -dependencies: -- name: common - repository: oci://registry-1.docker.io/bitnamicharts - tags: - - bitnami-common - version: 2.x.x -description: Redis(R) is an open source, advanced key-value store. It is often referred - to as a data structure server since keys can contain strings, hashes, lists, sets - and sorted sets. -home: https://bitnami.com -icon: https://bitnami.com/assets/stacks/redis/img/redis-stack-220x234.png -keywords: -- redis -- keyvalue -- database -maintainers: -- name: Broadcom, Inc. All Rights Reserved. - url: https://github.com/bitnami/charts -name: redis -sources: -- https://github.com/bitnami/charts/tree/main/bitnami/redis -version: 20.0.1 diff --git a/charts/redis/redis/README.md b/charts/redis/redis/README.md deleted file mode 100644 index 250e24a..0000000 --- a/charts/redis/redis/README.md +++ /dev/null @@ -1,1280 +0,0 @@ - - -# Bitnami package for Redis(R) - -Redis(R) is an open source, advanced key-value store. It is often referred to as a data structure server since keys can contain strings, hashes, lists, sets and sorted sets. - -[Overview of Redis®](http://redis.io) - -Disclaimer: Redis is a registered trademark of Redis Ltd. Any rights therein are reserved to Redis Ltd. Any use by Bitnami is for referential purposes only and does not indicate any sponsorship, endorsement, or affiliation between Redis Ltd. - -## TL;DR - -```console -helm install my-release oci://registry-1.docker.io/bitnamicharts/redis -``` - -Looking to use Redis® in production? Try [VMware Tanzu Application Catalog](https://bitnami.com/enterprise), the commercial edition of the Bitnami catalog. - -## Introduction - -This chart bootstraps a [Redis®](https://github.com/bitnami/containers/tree/main/bitnami/redis) deployment on a [Kubernetes](https://kubernetes.io) cluster using the [Helm](https://helm.sh) package manager. - -Bitnami charts can be used with [Kubeapps](https://kubeapps.dev/) for deployment and management of Helm Charts in clusters. - -### Choose between Redis® Helm Chart and Redis® Cluster Helm Chart - -You can choose any of the two Redis® Helm charts for deploying a Redis® cluster. - -1. [Redis® Helm Chart](https://github.com/bitnami/charts/tree/main/bitnami/redis) will deploy a master-replica cluster, with the [option](https://github.com/bitnami/charts/tree/main/bitnami/redis#redis-sentinel-configuration-parameters) of enabling using Redis® Sentinel. -2. [Redis® Cluster Helm Chart](https://github.com/bitnami/charts/tree/main/bitnami/redis-cluster) will deploy a Redis® Cluster topology with sharding. - -The main features of each chart are the following: - -| Redis® | Redis® Cluster | -|--------------------------------------------------------|------------------------------------------------------------------------| -| Supports multiple databases | Supports only one database. Better if you have a big dataset | -| Single write point (single master) | Multiple write points (multiple masters) | -| ![Redis® Topology](img/redis-topology.png) | ![Redis® Cluster Topology](img/redis-cluster-topology.png) | - -## Prerequisites - -- Kubernetes 1.23+ -- Helm 3.8.0+ -- PV provisioner support in the underlying infrastructure - -## Installing the Chart - -To install the chart with the release name `my-release`: - -```console -helm install my-release oci://REGISTRY_NAME/REPOSITORY_NAME/redis -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -The command deploys Redis® on the Kubernetes cluster in the default configuration. The [Parameters](#parameters) section lists the parameters that can be configured during installation. - -> **Tip**: List all releases using `helm list` - -## Configuration and installation details - -### Resource requests and limits - -Bitnami charts allow setting resource requests and limits for all containers inside the chart deployment. These are inside the `resources` value (check parameter table). Setting requests is essential for production workloads and these should be adapted to your specific use case. - -To make this process easier, the chart contains the `resourcesPreset` values, which automatically sets the `resources` section according to different presets. Check these presets in [the bitnami/common chart](https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15). However, in production workloads using `resourcePreset` is discouraged as it may not fully adapt to your specific needs. Find more information on container resource management in the [official Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/). - -### [Rolling VS Immutable tags](https://docs.vmware.com/en/VMware-Tanzu-Application-Catalog/services/tutorials/GUID-understand-rolling-tags-containers-index.html) - -It is strongly recommended to use immutable tags in a production environment. This ensures your deployment does not change automatically if the same tag is updated with a different image. - -Bitnami will release a new chart updating its containers if a new version of the main container, significant changes, or critical vulnerabilities exist. - -### Use a different Redis® version - -To modify the application version used in this chart, specify a different version of the image using the `image.tag` parameter and/or a different repository using the `image.repository` parameter. - -### Bootstrapping with an External Cluster - -This chart is equipped with the ability to bring online a set of Pods that connect to an existing Redis deployment that lies outside of Kubernetes. This effectively creates a hybrid Redis Deployment where both Pods in Kubernetes and Instances such as Virtual Machines can partake in a single Redis Deployment. This is helpful in situations where one may be migrating Redis from Virtual Machines into Kubernetes, for example. To take advantage of this, use the following as an example configuration: - -```yaml -replica: - externalMaster: - enabled: true - host: external-redis-0.internal -sentinel: - externalMaster: - enabled: true - host: external-redis-0.internal -``` - -:warning: This is currently limited to clusters in which Sentinel and Redis run on the same node! :warning: - -Please also note that the external sentinel must be listening on port `26379`, and this is currently not configurable. - -Once the Kubernetes Redis Deployment is online and confirmed to be working with the existing cluster, the configuration can then be removed and the cluster will remain connected. - -### External DNS - -This chart is equipped to allow leveraging the ExternalDNS project. Doing so will enable ExternalDNS to publish the FQDN for each instance, in the format of `..`. -Example, when using the following configuration: - -```yaml -useExternalDNS: - enabled: true - suffix: prod.example.org - additionalAnnotations: - ttl: 10 -``` - -On a cluster where the name of the Helm release is `a`, the hostname of a Pod is generated as: `a-redis-node-0.a-redis.prod.example.org`. The IP of that FQDN will match that of the associated Pod. This modifies the following parameters of the Redis/Sentinel configuration using this new FQDN: - -- `replica-announce-ip` -- `known-sentinel` -- `known-replica` -- `announce-ip` - -:warning: This requires a working installation of `external-dns` to be fully functional. :warning: - -See the [official ExternalDNS documentation](https://github.com/kubernetes-sigs/external-dns) for additional configuration options. - -### Cluster topologies - -#### Default: Master-Replicas - -When installing the chart with `architecture=replication`, it will deploy a Redis® master StatefulSet and a Redis® replicas StatefulSet. The replicas will be read-replicas of the master. Two services will be exposed: - -- Redis® Master service: Points to the master, where read-write operations can be performed -- Redis® Replicas service: Points to the replicas, where only read operations are allowed by default. - -In case the master crashes, the replicas will wait until the master node is respawned again by the Kubernetes Controller Manager. - -#### Standalone - -When installing the chart with `architecture=standalone`, it will deploy a standalone Redis® StatefulSet. A single service will be exposed: - -- Redis® Master service: Points to the master, where read-write operations can be performed - -#### Master-Replicas with Sentinel - -When installing the chart with `architecture=replication` and `sentinel.enabled=true`, it will deploy a Redis® master StatefulSet (only one master allowed) and a Redis® replicas StatefulSet. In this case, the pods will contain an extra container with Redis® Sentinel. This container will form a cluster of Redis® Sentinel nodes, which will promote a new master in case the actual one fails. - -On graceful termination of the Redis® master pod, a failover of the master is initiated to promote a new master. The Redis® Sentinel container in this pod will wait for the failover to occur before terminating. If `sentinel.redisShutdownWaitFailover=true` is set (the default), the Redis® container will wait for the failover as well before terminating. This increases availability for reads during failover, but may cause stale reads until all clients have switched to the new master. - -In addition to this, only one service is exposed: - -- Redis® service: Exposes port 6379 for Redis® read-only operations and port 26379 for accessing Redis® Sentinel. - -For read-only operations, access the service using port 6379. For write operations, it's necessary to access the Redis® Sentinel cluster and query the current master using the command below (using redis-cli or similar): - -```console -SENTINEL get-master-addr-by-name -``` - -This command will return the address of the current master, which can be accessed from inside the cluster. - -In case the current master crashes, the Sentinel containers will elect a new master node. - -`master.count` greater than `1` is not designed for use when `sentinel.enabled=true`. - -### Multiple masters (experimental) - -When `master.count` is greater than `1`, special care must be taken to create a consistent setup. - -An example of use case is the creation of a redundant set of standalone masters or master-replicas per Kubernetes node where you must ensure: - -- No more than `1` master can be deployed per Kubernetes node -- Replicas and writers can only see the single master of their own Kubernetes node - -One way of achieving this is by setting `master.service.internalTrafficPolicy=Local` in combination with a `master.affinity.podAntiAffinity` spec to never schedule more than one master per Kubernetes node. - -It's recommended to only change `master.count` if you know what you are doing. -`master.count` greater than `1` is not designed for use when `sentinel.enabled=true`. - -### Using a password file - -To use a password file for Redis® you need to create a secret containing the password and then deploy the chart using that secret. Follow these instructions: - -- Create the secret with the password. It is important that the file with the password must be called `redis-password`. - -```console -kubectl create secret generic redis-password-secret --from-file=redis-password.yaml -``` - -- Deploy the Helm Chart using the secret name as parameter: - -```text -usePassword=true -usePasswordFile=true -existingSecret=redis-password-secret -sentinels.enabled=true -metrics.enabled=true -``` - -### Securing traffic using TLS - -TLS support can be enabled in the chart by specifying the `tls.` parameters while creating a release. The following parameters should be configured to properly enable the TLS support in the cluster: - -- `tls.enabled`: Enable TLS support. Defaults to `false` -- `tls.existingSecret`: Name of the secret that contains the certificates. No defaults. -- `tls.certFilename`: Certificate filename. No defaults. -- `tls.certKeyFilename`: Certificate key filename. No defaults. -- `tls.certCAFilename`: CA Certificate filename. No defaults. - -For example: - -First, create the secret with the certificates files: - -```console -kubectl create secret generic certificates-tls-secret --from-file=./cert.pem --from-file=./cert.key --from-file=./ca.pem -``` - -Then, use the following parameters: - -```console -tls.enabled="true" -tls.existingSecret="certificates-tls-secret" -tls.certFilename="cert.pem" -tls.certKeyFilename="cert.key" -tls.certCAFilename="ca.pem" -``` - -### Metrics - -The chart optionally can start a metrics exporter for [prometheus](https://prometheus.io). The metrics endpoint (port 9121) is exposed in the service. Metrics can be scraped from within the cluster using something similar as the described in the [example Prometheus scrape configuration](https://github.com/prometheus/prometheus/blob/master/documentation/examples/prometheus-kubernetes.yml). If metrics are to be scraped from outside the cluster, the Kubernetes API proxy can be utilized to access the endpoint. - -If you have enabled TLS by specifying `tls.enabled=true` you also need to specify TLS option to the metrics exporter. You can do that via `metrics.extraArgs`. You can find the metrics exporter CLI flags for TLS [here](https://github.com/oliver006/redis_exporter#command-line-flags). For example: - -You can either specify `metrics.extraArgs.skip-tls-verification=true` to skip TLS verification or providing the following values under `metrics.extraArgs` for TLS client authentication: - -```console -tls-client-key-file -tls-client-cert-file -tls-ca-cert-file -``` - -### Deploy a custom metrics script in the sidecar - -A custom Lua script can be added to the `redis-exporter` sidecar by way of the `metrics.extraArgs.script` parameter. The pathname of the script must exist on the container, or the `redis_exporter` process (and therefore the whole pod) will refuse to start. The script can be provided to the sidecar containers via the `metrics.extraVolumes` and `metrics.extraVolumeMounts` parameters: - -```yaml -metrics: - extraVolumeMounts: - - name: '{{ printf "%s-metrics-script-file" (include "common.names.fullname" .) }}' - mountPath: '{{ printf "/mnt/%s/" (include "common.names.name" .) }}' - readOnly: true - extraVolumes: - - name: '{{ printf "%s-metrics-script-file" (include "common.names.fullname" .) }}' - configMap: - name: '{{ printf "%s-metrics-script" (include "common.names.fullname" .) }}' - extraArgs: - script: '{{ printf "/mnt/%s/my_custom_metrics.lua" (include "common.names.name" .) }}' -``` - -Then deploy the script into the correct location via `extraDeploy`: - -```yaml -extraDeploy: - - apiVersion: v1 - kind: ConfigMap - metadata: - name: '{{ printf "%s-metrics-script" (include "common.names.fullname" .) }}' - data: - my_custom_metrics.lua: | - -- LUA SCRIPT CODE HERE, e.g., - return {'bitnami_makes_the_best_charts', '1'} -``` - -### Host Kernel Settings - -Redis® may require some changes in the kernel of the host machine to work as expected, in particular increasing the `somaxconn` value and disabling transparent huge pages. To do so, you can set up a privileged `initContainer` with the `sysctlImage` config values, for example: - -```yaml -sysctlImage: - enabled: true - mountHostSys: true - command: - - /bin/sh - - -c - - |- - install_packages procps - sysctl -w net.core.somaxconn=10000 - echo never > /host-sys/kernel/mm/transparent_hugepage/enabled -``` - -Alternatively, for Kubernetes 1.12+ you can set `securityContext.sysctls` which will configure `sysctls` for master and slave pods. Example: - -```yaml -securityContext: - sysctls: - - name: net.core.somaxconn - value: "10000" -``` - -Note that this will not disable transparent huge tables. - -### Backup and restore - -To backup and restore Redis deployments on Kubernetes, you will need to create a snapshot of the data in the source cluster, and later restore it in a new cluster with the new parameters. Follow the instructions below: - -#### Step 1: Backup the deployment - -- Connect to one of the nodes and start the Redis CLI tool. Then, run the commands below: - - ```text - $ kubectl exec -it my-release-master-0 bash - $ redis-cli - 127.0.0.1:6379> auth your_current_redis_password - OK - 127.0.0.1:6379> save - OK - ``` - -- Copy the dump file from the Redis node: - - ```console - kubectl cp my-release-master-0:/data/dump.rdb dump.rdb -c redis - ``` - -#### Step 2: Restore the data on the destination cluster - -To restore the data in a new cluster, you will need to create a PVC and then upload the *dump.rdb* file to the new volume. - -Follow the following steps: - -- In the [*values.yaml*](https://github.com/bitnami/charts/blob/main/bitnami/redis/values.yaml) file set the *appendonly* parameter to *no*. You can skip this step if it is already configured as *no* - - ```yaml - commonConfiguration: |- - # Enable AOF https://redis.io/topics/persistence#append-only-file - appendonly no - # Disable RDB persistence, AOF persistence already enabled. - save "" - ``` - - > *Note that the `Enable AOF` comment belongs to the original config file and what you're actually doing is disabling it. This change will only be neccessary for the temporal cluster you're creating to upload the dump.* - -- Start the new cluster to create the PVCs. Use the command below as an example: - - ```console - helm install new-redis -f values.yaml . --set cluster.enabled=true --set cluster.slaveCount=3 - ``` - -- Now that the PVC were created, stop it and copy the *dump.rdp* file on the persisted data by using a helping pod. - - ```text - $ helm delete new-redis - - $ kubectl run --generator=run-pod/v1 -i --rm --tty volpod --overrides=' - { - "apiVersion": "v1", - "kind": "Pod", - "metadata": { - "name": "redisvolpod" - }, - "spec": { - "containers": [{ - "command": [ - "tail", - "-f", - "/dev/null" - ], - "image": "bitnami/minideb", - "name": "mycontainer", - "volumeMounts": [{ - "mountPath": "/mnt", - "name": "redisdata" - }] - }], - "restartPolicy": "Never", - "volumes": [{ - "name": "redisdata", - "persistentVolumeClaim": { - "claimName": "redis-data-new-redis-master-0" - } - }] - } - }' --image="bitnami/minideb" - - $ kubectl cp dump.rdb redisvolpod:/mnt/dump.rdb - $ kubectl delete pod volpod - ``` - -- Restart the cluster: - - > **INFO:** The *appendonly* parameter can be safely restored to your desired value. - - ```console - helm install new-redis -f values.yaml . --set cluster.enabled=true --set cluster.slaveCount=3 - ``` - -### NetworkPolicy - -To enable network policy for Redis®, install [a networking plugin that implements the Kubernetes NetworkPolicy spec](https://kubernetes.io/docs/tasks/administer-cluster/declare-network-policy#before-you-begin), and set `networkPolicy.enabled` to `true`. - -With NetworkPolicy enabled, only pods with the generated client label will be able to connect to Redis. This label will be displayed in the output after a successful install. - -With `networkPolicy.ingressNSMatchLabels` pods from other namespaces can connect to Redis. Set `networkPolicy.ingressNSPodMatchLabels` to match pod labels in matched namespace. For example, for a namespace labeled `redis=external` and pods in that namespace labeled `redis-client=true` the fields should be set: - -```yaml -networkPolicy: - enabled: true - ingressNSMatchLabels: - redis: external - ingressNSPodMatchLabels: - redis-client: true -``` - -#### Setting Pod's affinity - -This chart allows you to set your custom affinity using the `XXX.affinity` parameter(s). Find more information about Pod's affinity in the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity). - -As an alternative, you can use of the preset configurations for pod affinity, pod anti-affinity, and node affinity available at the [bitnami/common](https://github.com/bitnami/charts/tree/main/bitnami/common#affinities) chart. To do so, set the `XXX.podAffinityPreset`, `XXX.podAntiAffinityPreset`, or `XXX.nodeAffinityPreset` parameters. - -## Persistence - -By default, the chart mounts a [Persistent Volume](https://kubernetes.io/docs/concepts/storage/persistent-volumes/) at the `/data` path. The volume is created using dynamic volume provisioning. If a Persistent Volume Claim already exists, specify it during installation. - -### Existing PersistentVolumeClaim - -1. Create the PersistentVolume -2. Create the PersistentVolumeClaim -3. Install the chart - -```console -helm install my-release --set master.persistence.existingClaim=PVC_NAME oci://REGISTRY_NAME/REPOSITORY_NAME/redis -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -## Parameters - -### Global parameters - -| Name | Description | Value | -| ----------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------ | -| `global.imageRegistry` | Global Docker image registry | `""` | -| `global.imagePullSecrets` | Global Docker registry secret names as an array | `[]` | -| `global.defaultStorageClass` | Global default StorageClass for Persistent Volume(s) | `""` | -| `global.storageClass` | DEPRECATED: use global.defaultStorageClass instead | `""` | -| `global.redis.password` | Global Redis® password (overrides `auth.password`) | `""` | -| `global.compatibility.openshift.adaptSecurityContext` | Adapt the securityContext sections of the deployment to make them compatible with Openshift restricted-v2 SCC: remove runAsUser, runAsGroup and fsGroup and let the platform use their allowed default IDs. Possible values: auto (apply if the detected running cluster is Openshift), force (perform the adaptation always), disabled (do not perform adaptation) | `auto` | - -### Common parameters - -| Name | Description | Value | -| ------------------------- | -------------------------------------------------------------------------------------------------------------- | --------------- | -| `kubeVersion` | Override Kubernetes version | `""` | -| `nameOverride` | String to partially override common.names.fullname | `""` | -| `fullnameOverride` | String to fully override common.names.fullname | `""` | -| `namespaceOverride` | String to fully override common.names.namespace | `""` | -| `commonLabels` | Labels to add to all deployed objects | `{}` | -| `commonAnnotations` | Annotations to add to all deployed objects | `{}` | -| `secretAnnotations` | Annotations to add to secret | `{}` | -| `clusterDomain` | Kubernetes cluster domain name | `cluster.local` | -| `extraDeploy` | Array of extra objects to deploy with the release | `[]` | -| `useHostnames` | Use hostnames internally when announcing replication. If false, the hostname will be resolved to an IP address | `true` | -| `nameResolutionThreshold` | Failure threshold for internal hostnames resolution | `5` | -| `nameResolutionTimeout` | Timeout seconds between probes for internal hostnames resolution | `5` | -| `diagnosticMode.enabled` | Enable diagnostic mode (all probes will be disabled and the command will be overridden) | `false` | -| `diagnosticMode.command` | Command to override all containers in the deployment | `["sleep"]` | -| `diagnosticMode.args` | Args to override all containers in the deployment | `["infinity"]` | - -### Redis® Image parameters - -| Name | Description | Value | -| ------------------- | ---------------------------------------------------------------------------------------------------------- | ----------------------- | -| `image.registry` | Redis® image registry | `REGISTRY_NAME` | -| `image.repository` | Redis® image repository | `REPOSITORY_NAME/redis` | -| `image.digest` | Redis® image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `image.pullPolicy` | Redis® image pull policy | `IfNotPresent` | -| `image.pullSecrets` | Redis® image pull secrets | `[]` | -| `image.debug` | Enable image debug mode | `false` | - -### Redis® common configuration parameters - -| Name | Description | Value | -| -------------------------------- | ------------------------------------------------------------------------------------- | ------------- | -| `architecture` | Redis® architecture. Allowed values: `standalone` or `replication` | `replication` | -| `auth.enabled` | Enable password authentication | `true` | -| `auth.sentinel` | Enable password authentication on sentinels too | `true` | -| `auth.password` | Redis® password | `""` | -| `auth.existingSecret` | The name of an existing secret with Redis® credentials | `""` | -| `auth.existingSecretPasswordKey` | Password key to be retrieved from existing secret | `""` | -| `auth.usePasswordFiles` | Mount credentials as files instead of using an environment variable | `false` | -| `auth.usePasswordFileFromSecret` | Mount password file from secret | `true` | -| `commonConfiguration` | Common configuration to be added into the ConfigMap | `""` | -| `existingConfigmap` | The name of an existing ConfigMap with your custom configuration for Redis® nodes | `""` | - -### Redis® master configuration parameters - -| Name | Description | Value | -| ---------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------------ | -| `master.count` | Number of Redis® master instances to deploy (experimental, requires additional configuration) | `1` | -| `master.revisionHistoryLimit` | The number of old history to retain to allow rollback | `10` | -| `master.configuration` | Configuration for Redis® master nodes | `""` | -| `master.disableCommands` | Array with Redis® commands to disable on master nodes | `["FLUSHDB","FLUSHALL"]` | -| `master.command` | Override default container command (useful when using custom images) | `[]` | -| `master.args` | Override default container args (useful when using custom images) | `[]` | -| `master.enableServiceLinks` | Whether information about services should be injected into pod's environment variable | `true` | -| `master.preExecCmds` | Additional commands to run prior to starting Redis® master | `[]` | -| `master.extraFlags` | Array with additional command line flags for Redis® master | `[]` | -| `master.extraEnvVars` | Array with extra environment variables to add to Redis® master nodes | `[]` | -| `master.extraEnvVarsCM` | Name of existing ConfigMap containing extra env vars for Redis® master nodes | `""` | -| `master.extraEnvVarsSecret` | Name of existing Secret containing extra env vars for Redis® master nodes | `""` | -| `master.containerPorts.redis` | Container port to open on Redis® master nodes | `6379` | -| `master.startupProbe.enabled` | Enable startupProbe on Redis® master nodes | `false` | -| `master.startupProbe.initialDelaySeconds` | Initial delay seconds for startupProbe | `20` | -| `master.startupProbe.periodSeconds` | Period seconds for startupProbe | `5` | -| `master.startupProbe.timeoutSeconds` | Timeout seconds for startupProbe | `5` | -| `master.startupProbe.failureThreshold` | Failure threshold for startupProbe | `5` | -| `master.startupProbe.successThreshold` | Success threshold for startupProbe | `1` | -| `master.livenessProbe.enabled` | Enable livenessProbe on Redis® master nodes | `true` | -| `master.livenessProbe.initialDelaySeconds` | Initial delay seconds for livenessProbe | `20` | -| `master.livenessProbe.periodSeconds` | Period seconds for livenessProbe | `5` | -| `master.livenessProbe.timeoutSeconds` | Timeout seconds for livenessProbe | `5` | -| `master.livenessProbe.failureThreshold` | Failure threshold for livenessProbe | `5` | -| `master.livenessProbe.successThreshold` | Success threshold for livenessProbe | `1` | -| `master.readinessProbe.enabled` | Enable readinessProbe on Redis® master nodes | `true` | -| `master.readinessProbe.initialDelaySeconds` | Initial delay seconds for readinessProbe | `20` | -| `master.readinessProbe.periodSeconds` | Period seconds for readinessProbe | `5` | -| `master.readinessProbe.timeoutSeconds` | Timeout seconds for readinessProbe | `1` | -| `master.readinessProbe.failureThreshold` | Failure threshold for readinessProbe | `5` | -| `master.readinessProbe.successThreshold` | Success threshold for readinessProbe | `1` | -| `master.customStartupProbe` | Custom startupProbe that overrides the default one | `{}` | -| `master.customLivenessProbe` | Custom livenessProbe that overrides the default one | `{}` | -| `master.customReadinessProbe` | Custom readinessProbe that overrides the default one | `{}` | -| `master.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if master.resources is set (master.resources is recommended for production). | `nano` | -| `master.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `master.podSecurityContext.enabled` | Enabled Redis® master pods' Security Context | `true` | -| `master.podSecurityContext.fsGroupChangePolicy` | Set filesystem group change policy | `Always` | -| `master.podSecurityContext.sysctls` | Set kernel settings using the sysctl interface | `[]` | -| `master.podSecurityContext.supplementalGroups` | Set filesystem extra groups | `[]` | -| `master.podSecurityContext.fsGroup` | Set Redis® master pod's Security Context fsGroup | `1001` | -| `master.containerSecurityContext.enabled` | Enabled Redis® master containers' Security Context | `true` | -| `master.containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `master.containerSecurityContext.runAsUser` | Set Redis® master containers' Security Context runAsUser | `1001` | -| `master.containerSecurityContext.runAsGroup` | Set Redis® master containers' Security Context runAsGroup | `1001` | -| `master.containerSecurityContext.runAsNonRoot` | Set Redis® master containers' Security Context runAsNonRoot | `true` | -| `master.containerSecurityContext.allowPrivilegeEscalation` | Is it possible to escalate Redis® pod(s) privileges | `false` | -| `master.containerSecurityContext.readOnlyRootFilesystem` | Set container's Security Context read-only root filesystem | `true` | -| `master.containerSecurityContext.seccompProfile.type` | Set Redis® master containers' Security Context seccompProfile | `RuntimeDefault` | -| `master.containerSecurityContext.capabilities.drop` | Set Redis® master containers' Security Context capabilities to drop | `["ALL"]` | -| `master.kind` | Use either Deployment, StatefulSet (default) or DaemonSet | `StatefulSet` | -| `master.schedulerName` | Alternate scheduler for Redis® master pods | `""` | -| `master.updateStrategy.type` | Redis® master statefulset strategy type | `RollingUpdate` | -| `master.minReadySeconds` | How many seconds a pod needs to be ready before killing the next, during update | `0` | -| `master.priorityClassName` | Redis® master pods' priorityClassName | `""` | -| `master.automountServiceAccountToken` | Mount Service Account token in pod | `false` | -| `master.hostAliases` | Redis® master pods host aliases | `[]` | -| `master.podLabels` | Extra labels for Redis® master pods | `{}` | -| `master.podAnnotations` | Annotations for Redis® master pods | `{}` | -| `master.shareProcessNamespace` | Share a single process namespace between all of the containers in Redis® master pods | `false` | -| `master.podAffinityPreset` | Pod affinity preset. Ignored if `master.affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `master.podAntiAffinityPreset` | Pod anti-affinity preset. Ignored if `master.affinity` is set. Allowed values: `soft` or `hard` | `soft` | -| `master.nodeAffinityPreset.type` | Node affinity preset type. Ignored if `master.affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `master.nodeAffinityPreset.key` | Node label key to match. Ignored if `master.affinity` is set | `""` | -| `master.nodeAffinityPreset.values` | Node label values to match. Ignored if `master.affinity` is set | `[]` | -| `master.affinity` | Affinity for Redis® master pods assignment | `{}` | -| `master.nodeSelector` | Node labels for Redis® master pods assignment | `{}` | -| `master.tolerations` | Tolerations for Redis® master pods assignment | `[]` | -| `master.topologySpreadConstraints` | Spread Constraints for Redis® master pod assignment | `[]` | -| `master.dnsPolicy` | DNS Policy for Redis® master pod | `""` | -| `master.dnsConfig` | DNS Configuration for Redis® master pod | `{}` | -| `master.lifecycleHooks` | for the Redis® master container(s) to automate configuration before or after startup | `{}` | -| `master.extraVolumes` | Optionally specify extra list of additional volumes for the Redis® master pod(s) | `[]` | -| `master.extraVolumeMounts` | Optionally specify extra list of additional volumeMounts for the Redis® master container(s) | `[]` | -| `master.sidecars` | Add additional sidecar containers to the Redis® master pod(s) | `[]` | -| `master.initContainers` | Add additional init containers to the Redis® master pod(s) | `[]` | -| `master.persistence.enabled` | Enable persistence on Redis® master nodes using Persistent Volume Claims | `true` | -| `master.persistence.medium` | Provide a medium for `emptyDir` volumes. | `""` | -| `master.persistence.sizeLimit` | Set this to enable a size limit for `emptyDir` volumes. | `""` | -| `master.persistence.path` | The path the volume will be mounted at on Redis® master containers | `/data` | -| `master.persistence.subPath` | The subdirectory of the volume to mount on Redis® master containers | `""` | -| `master.persistence.subPathExpr` | Used to construct the subPath subdirectory of the volume to mount on Redis® master containers | `""` | -| `master.persistence.storageClass` | Persistent Volume storage class | `""` | -| `master.persistence.accessModes` | Persistent Volume access modes | `["ReadWriteOnce"]` | -| `master.persistence.size` | Persistent Volume size | `8Gi` | -| `master.persistence.annotations` | Additional custom annotations for the PVC | `{}` | -| `master.persistence.labels` | Additional custom labels for the PVC | `{}` | -| `master.persistence.selector` | Additional labels to match for the PVC | `{}` | -| `master.persistence.dataSource` | Custom PVC data source | `{}` | -| `master.persistence.existingClaim` | Use a existing PVC which must be created manually before bound | `""` | -| `master.persistentVolumeClaimRetentionPolicy.enabled` | Controls if and how PVCs are deleted during the lifecycle of a StatefulSet | `false` | -| `master.persistentVolumeClaimRetentionPolicy.whenScaled` | Volume retention behavior when the replica count of the StatefulSet is reduced | `Retain` | -| `master.persistentVolumeClaimRetentionPolicy.whenDeleted` | Volume retention behavior that applies when the StatefulSet is deleted | `Retain` | -| `master.service.type` | Redis® master service type | `ClusterIP` | -| `master.service.portNames.redis` | Redis® master service port name | `tcp-redis` | -| `master.service.ports.redis` | Redis® master service port | `6379` | -| `master.service.nodePorts.redis` | Node port for Redis® master | `""` | -| `master.service.externalTrafficPolicy` | Redis® master service external traffic policy | `Cluster` | -| `master.service.extraPorts` | Extra ports to expose (normally used with the `sidecar` value) | `[]` | -| `master.service.internalTrafficPolicy` | Redis® master service internal traffic policy (requires Kubernetes v1.22 or greater to be usable) | `Cluster` | -| `master.service.clusterIP` | Redis® master service Cluster IP | `""` | -| `master.service.loadBalancerIP` | Redis® master service Load Balancer IP | `""` | -| `master.service.loadBalancerClass` | master service Load Balancer class if service type is `LoadBalancer` (optional, cloud specific) | `""` | -| `master.service.loadBalancerSourceRanges` | Redis® master service Load Balancer sources | `[]` | -| `master.service.externalIPs` | Redis® master service External IPs | `[]` | -| `master.service.annotations` | Additional custom annotations for Redis® master service | `{}` | -| `master.service.sessionAffinity` | Session Affinity for Kubernetes service, can be "None" or "ClientIP" | `None` | -| `master.service.sessionAffinityConfig` | Additional settings for the sessionAffinity | `{}` | -| `master.terminationGracePeriodSeconds` | Integer setting the termination grace period for the redis-master pods | `30` | -| `master.serviceAccount.create` | Specifies whether a ServiceAccount should be created | `true` | -| `master.serviceAccount.name` | The name of the ServiceAccount to use. | `""` | -| `master.serviceAccount.automountServiceAccountToken` | Whether to auto mount the service account token | `false` | -| `master.serviceAccount.annotations` | Additional custom annotations for the ServiceAccount | `{}` | -| `master.pdb.create` | Enable/disable a Pod Disruption Budget creation | `true` | -| `master.pdb.minAvailable` | Minimum number/percentage of pods that should remain scheduled | `{}` | -| `master.pdb.maxUnavailable` | Maximum number/percentage of pods that may be made unavailable. Defaults to `1` if both `master.pdb.minAvailable` and `master.pdb.maxUnavailable` are empty. | `{}` | - -### Redis® replicas configuration parameters - -| Name | Description | Value | -| ----------------------------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------------ | -| `replica.kind` | Use either DaemonSet or StatefulSet (default) | `StatefulSet` | -| `replica.replicaCount` | Number of Redis® replicas to deploy | `3` | -| `replica.revisionHistoryLimit` | The number of old history to retain to allow rollback | `10` | -| `replica.configuration` | Configuration for Redis® replicas nodes | `""` | -| `replica.disableCommands` | Array with Redis® commands to disable on replicas nodes | `["FLUSHDB","FLUSHALL"]` | -| `replica.command` | Override default container command (useful when using custom images) | `[]` | -| `replica.args` | Override default container args (useful when using custom images) | `[]` | -| `replica.enableServiceLinks` | Whether information about services should be injected into pod's environment variable | `true` | -| `replica.preExecCmds` | Additional commands to run prior to starting Redis® replicas | `[]` | -| `replica.extraFlags` | Array with additional command line flags for Redis® replicas | `[]` | -| `replica.extraEnvVars` | Array with extra environment variables to add to Redis® replicas nodes | `[]` | -| `replica.extraEnvVarsCM` | Name of existing ConfigMap containing extra env vars for Redis® replicas nodes | `""` | -| `replica.extraEnvVarsSecret` | Name of existing Secret containing extra env vars for Redis® replicas nodes | `""` | -| `replica.externalMaster.enabled` | Use external master for bootstrapping | `false` | -| `replica.externalMaster.host` | External master host to bootstrap from | `""` | -| `replica.externalMaster.port` | Port for Redis service external master host | `6379` | -| `replica.containerPorts.redis` | Container port to open on Redis® replicas nodes | `6379` | -| `replica.startupProbe.enabled` | Enable startupProbe on Redis® replicas nodes | `true` | -| `replica.startupProbe.initialDelaySeconds` | Initial delay seconds for startupProbe | `10` | -| `replica.startupProbe.periodSeconds` | Period seconds for startupProbe | `10` | -| `replica.startupProbe.timeoutSeconds` | Timeout seconds for startupProbe | `5` | -| `replica.startupProbe.failureThreshold` | Failure threshold for startupProbe | `22` | -| `replica.startupProbe.successThreshold` | Success threshold for startupProbe | `1` | -| `replica.livenessProbe.enabled` | Enable livenessProbe on Redis® replicas nodes | `true` | -| `replica.livenessProbe.initialDelaySeconds` | Initial delay seconds for livenessProbe | `20` | -| `replica.livenessProbe.periodSeconds` | Period seconds for livenessProbe | `5` | -| `replica.livenessProbe.timeoutSeconds` | Timeout seconds for livenessProbe | `5` | -| `replica.livenessProbe.failureThreshold` | Failure threshold for livenessProbe | `5` | -| `replica.livenessProbe.successThreshold` | Success threshold for livenessProbe | `1` | -| `replica.readinessProbe.enabled` | Enable readinessProbe on Redis® replicas nodes | `true` | -| `replica.readinessProbe.initialDelaySeconds` | Initial delay seconds for readinessProbe | `20` | -| `replica.readinessProbe.periodSeconds` | Period seconds for readinessProbe | `5` | -| `replica.readinessProbe.timeoutSeconds` | Timeout seconds for readinessProbe | `1` | -| `replica.readinessProbe.failureThreshold` | Failure threshold for readinessProbe | `5` | -| `replica.readinessProbe.successThreshold` | Success threshold for readinessProbe | `1` | -| `replica.customStartupProbe` | Custom startupProbe that overrides the default one | `{}` | -| `replica.customLivenessProbe` | Custom livenessProbe that overrides the default one | `{}` | -| `replica.customReadinessProbe` | Custom readinessProbe that overrides the default one | `{}` | -| `replica.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if replica.resources is set (replica.resources is recommended for production). | `nano` | -| `replica.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `replica.podSecurityContext.enabled` | Enabled Redis® replicas pods' Security Context | `true` | -| `replica.podSecurityContext.fsGroupChangePolicy` | Set filesystem group change policy | `Always` | -| `replica.podSecurityContext.sysctls` | Set kernel settings using the sysctl interface | `[]` | -| `replica.podSecurityContext.supplementalGroups` | Set filesystem extra groups | `[]` | -| `replica.podSecurityContext.fsGroup` | Set Redis® replicas pod's Security Context fsGroup | `1001` | -| `replica.containerSecurityContext.enabled` | Enabled Redis® replicas containers' Security Context | `true` | -| `replica.containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `replica.containerSecurityContext.runAsUser` | Set Redis® replicas containers' Security Context runAsUser | `1001` | -| `replica.containerSecurityContext.runAsGroup` | Set Redis® replicas containers' Security Context runAsGroup | `1001` | -| `replica.containerSecurityContext.runAsNonRoot` | Set Redis® replicas containers' Security Context runAsNonRoot | `true` | -| `replica.containerSecurityContext.allowPrivilegeEscalation` | Set Redis® replicas pod's Security Context allowPrivilegeEscalation | `false` | -| `replica.containerSecurityContext.readOnlyRootFilesystem` | Set container's Security Context read-only root filesystem | `true` | -| `replica.containerSecurityContext.seccompProfile.type` | Set Redis® replicas containers' Security Context seccompProfile | `RuntimeDefault` | -| `replica.containerSecurityContext.capabilities.drop` | Set Redis® replicas containers' Security Context capabilities to drop | `["ALL"]` | -| `replica.schedulerName` | Alternate scheduler for Redis® replicas pods | `""` | -| `replica.updateStrategy.type` | Redis® replicas statefulset strategy type | `RollingUpdate` | -| `replica.minReadySeconds` | How many seconds a pod needs to be ready before killing the next, during update | `0` | -| `replica.priorityClassName` | Redis® replicas pods' priorityClassName | `""` | -| `replica.podManagementPolicy` | podManagementPolicy to manage scaling operation of %%MAIN_CONTAINER_NAME%% pods | `""` | -| `replica.automountServiceAccountToken` | Mount Service Account token in pod | `false` | -| `replica.hostAliases` | Redis® replicas pods host aliases | `[]` | -| `replica.podLabels` | Extra labels for Redis® replicas pods | `{}` | -| `replica.podAnnotations` | Annotations for Redis® replicas pods | `{}` | -| `replica.shareProcessNamespace` | Share a single process namespace between all of the containers in Redis® replicas pods | `false` | -| `replica.podAffinityPreset` | Pod affinity preset. Ignored if `replica.affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `replica.podAntiAffinityPreset` | Pod anti-affinity preset. Ignored if `replica.affinity` is set. Allowed values: `soft` or `hard` | `soft` | -| `replica.nodeAffinityPreset.type` | Node affinity preset type. Ignored if `replica.affinity` is set. Allowed values: `soft` or `hard` | `""` | -| `replica.nodeAffinityPreset.key` | Node label key to match. Ignored if `replica.affinity` is set | `""` | -| `replica.nodeAffinityPreset.values` | Node label values to match. Ignored if `replica.affinity` is set | `[]` | -| `replica.affinity` | Affinity for Redis® replicas pods assignment | `{}` | -| `replica.nodeSelector` | Node labels for Redis® replicas pods assignment | `{}` | -| `replica.tolerations` | Tolerations for Redis® replicas pods assignment | `[]` | -| `replica.topologySpreadConstraints` | Spread Constraints for Redis® replicas pod assignment | `[]` | -| `replica.dnsPolicy` | DNS Policy for Redis® replica pods | `""` | -| `replica.dnsConfig` | DNS Configuration for Redis® replica pods | `{}` | -| `replica.lifecycleHooks` | for the Redis® replica container(s) to automate configuration before or after startup | `{}` | -| `replica.extraVolumes` | Optionally specify extra list of additional volumes for the Redis® replicas pod(s) | `[]` | -| `replica.extraVolumeMounts` | Optionally specify extra list of additional volumeMounts for the Redis® replicas container(s) | `[]` | -| `replica.sidecars` | Add additional sidecar containers to the Redis® replicas pod(s) | `[]` | -| `replica.initContainers` | Add additional init containers to the Redis® replicas pod(s) | `[]` | -| `replica.persistence.enabled` | Enable persistence on Redis® replicas nodes using Persistent Volume Claims | `true` | -| `replica.persistence.medium` | Provide a medium for `emptyDir` volumes. | `""` | -| `replica.persistence.sizeLimit` | Set this to enable a size limit for `emptyDir` volumes. | `""` | -| `replica.persistence.path` | The path the volume will be mounted at on Redis® replicas containers | `/data` | -| `replica.persistence.subPath` | The subdirectory of the volume to mount on Redis® replicas containers | `""` | -| `replica.persistence.subPathExpr` | Used to construct the subPath subdirectory of the volume to mount on Redis® replicas containers | `""` | -| `replica.persistence.storageClass` | Persistent Volume storage class | `""` | -| `replica.persistence.accessModes` | Persistent Volume access modes | `["ReadWriteOnce"]` | -| `replica.persistence.size` | Persistent Volume size | `8Gi` | -| `replica.persistence.annotations` | Additional custom annotations for the PVC | `{}` | -| `replica.persistence.labels` | Additional custom labels for the PVC | `{}` | -| `replica.persistence.selector` | Additional labels to match for the PVC | `{}` | -| `replica.persistence.dataSource` | Custom PVC data source | `{}` | -| `replica.persistence.existingClaim` | Use a existing PVC which must be created manually before bound | `""` | -| `replica.persistentVolumeClaimRetentionPolicy.enabled` | Controls if and how PVCs are deleted during the lifecycle of a StatefulSet | `false` | -| `replica.persistentVolumeClaimRetentionPolicy.whenScaled` | Volume retention behavior when the replica count of the StatefulSet is reduced | `Retain` | -| `replica.persistentVolumeClaimRetentionPolicy.whenDeleted` | Volume retention behavior that applies when the StatefulSet is deleted | `Retain` | -| `replica.service.type` | Redis® replicas service type | `ClusterIP` | -| `replica.service.ports.redis` | Redis® replicas service port | `6379` | -| `replica.service.nodePorts.redis` | Node port for Redis® replicas | `""` | -| `replica.service.externalTrafficPolicy` | Redis® replicas service external traffic policy | `Cluster` | -| `replica.service.internalTrafficPolicy` | Redis® replicas service internal traffic policy (requires Kubernetes v1.22 or greater to be usable) | `Cluster` | -| `replica.service.extraPorts` | Extra ports to expose (normally used with the `sidecar` value) | `[]` | -| `replica.service.clusterIP` | Redis® replicas service Cluster IP | `""` | -| `replica.service.loadBalancerIP` | Redis® replicas service Load Balancer IP | `""` | -| `replica.service.loadBalancerClass` | replicas service Load Balancer class if service type is `LoadBalancer` (optional, cloud specific) | `""` | -| `replica.service.loadBalancerSourceRanges` | Redis® replicas service Load Balancer sources | `[]` | -| `replica.service.annotations` | Additional custom annotations for Redis® replicas service | `{}` | -| `replica.service.sessionAffinity` | Session Affinity for Kubernetes service, can be "None" or "ClientIP" | `None` | -| `replica.service.sessionAffinityConfig` | Additional settings for the sessionAffinity | `{}` | -| `replica.terminationGracePeriodSeconds` | Integer setting the termination grace period for the redis-replicas pods | `30` | -| `replica.autoscaling.enabled` | Enable replica autoscaling settings | `false` | -| `replica.autoscaling.minReplicas` | Minimum replicas for the pod autoscaling | `1` | -| `replica.autoscaling.maxReplicas` | Maximum replicas for the pod autoscaling | `11` | -| `replica.autoscaling.targetCPU` | Percentage of CPU to consider when autoscaling | `""` | -| `replica.autoscaling.targetMemory` | Percentage of Memory to consider when autoscaling | `""` | -| `replica.serviceAccount.create` | Specifies whether a ServiceAccount should be created | `true` | -| `replica.serviceAccount.name` | The name of the ServiceAccount to use. | `""` | -| `replica.serviceAccount.automountServiceAccountToken` | Whether to auto mount the service account token | `false` | -| `replica.serviceAccount.annotations` | Additional custom annotations for the ServiceAccount | `{}` | -| `replica.pdb.create` | Enable/disable a Pod Disruption Budget creation | `true` | -| `replica.pdb.minAvailable` | Minimum number/percentage of pods that should remain scheduled | `{}` | -| `replica.pdb.maxUnavailable` | Maximum number/percentage of pods that may be made unavailable. Defaults to `1` if both `replica.pdb.minAvailable` and `replica.pdb.maxUnavailable` are empty. | `{}` | - -### Redis® Sentinel configuration parameters - -| Name | Description | Value | -| ------------------------------------------------------------ | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | -------------------------------- | -| `sentinel.enabled` | Use Redis® Sentinel on Redis® pods. | `false` | -| `sentinel.image.registry` | Redis® Sentinel image registry | `REGISTRY_NAME` | -| `sentinel.image.repository` | Redis® Sentinel image repository | `REPOSITORY_NAME/redis-sentinel` | -| `sentinel.image.digest` | Redis® Sentinel image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `sentinel.image.pullPolicy` | Redis® Sentinel image pull policy | `IfNotPresent` | -| `sentinel.image.pullSecrets` | Redis® Sentinel image pull secrets | `[]` | -| `sentinel.image.debug` | Enable image debug mode | `false` | -| `sentinel.annotations` | Additional custom annotations for Redis® Sentinel resource | `{}` | -| `sentinel.masterSet` | Master set name | `mymaster` | -| `sentinel.quorum` | Sentinel Quorum | `2` | -| `sentinel.getMasterTimeout` | Amount of time to allow before get_sentinel_master_info() times out. | `90` | -| `sentinel.automateClusterRecovery` | Automate cluster recovery in cases where the last replica is not considered a good replica and Sentinel won't automatically failover to it. | `false` | -| `sentinel.redisShutdownWaitFailover` | Whether the Redis® master container waits for the failover at shutdown (in addition to the Redis® Sentinel container). | `true` | -| `sentinel.downAfterMilliseconds` | Timeout for detecting a Redis® node is down | `60000` | -| `sentinel.failoverTimeout` | Timeout for performing a election failover | `180000` | -| `sentinel.parallelSyncs` | Number of replicas that can be reconfigured in parallel to use the new master after a failover | `1` | -| `sentinel.configuration` | Configuration for Redis® Sentinel nodes | `""` | -| `sentinel.command` | Override default container command (useful when using custom images) | `[]` | -| `sentinel.args` | Override default container args (useful when using custom images) | `[]` | -| `sentinel.enableServiceLinks` | Whether information about services should be injected into pod's environment variable | `true` | -| `sentinel.preExecCmds` | Additional commands to run prior to starting Redis® Sentinel | `[]` | -| `sentinel.extraEnvVars` | Array with extra environment variables to add to Redis® Sentinel nodes | `[]` | -| `sentinel.extraEnvVarsCM` | Name of existing ConfigMap containing extra env vars for Redis® Sentinel nodes | `""` | -| `sentinel.extraEnvVarsSecret` | Name of existing Secret containing extra env vars for Redis® Sentinel nodes | `""` | -| `sentinel.externalMaster.enabled` | Use external master for bootstrapping | `false` | -| `sentinel.externalMaster.host` | External master host to bootstrap from | `""` | -| `sentinel.externalMaster.port` | Port for Redis service external master host | `6379` | -| `sentinel.containerPorts.sentinel` | Container port to open on Redis® Sentinel nodes | `26379` | -| `sentinel.startupProbe.enabled` | Enable startupProbe on Redis® Sentinel nodes | `true` | -| `sentinel.startupProbe.initialDelaySeconds` | Initial delay seconds for startupProbe | `10` | -| `sentinel.startupProbe.periodSeconds` | Period seconds for startupProbe | `10` | -| `sentinel.startupProbe.timeoutSeconds` | Timeout seconds for startupProbe | `5` | -| `sentinel.startupProbe.failureThreshold` | Failure threshold for startupProbe | `22` | -| `sentinel.startupProbe.successThreshold` | Success threshold for startupProbe | `1` | -| `sentinel.livenessProbe.enabled` | Enable livenessProbe on Redis® Sentinel nodes | `true` | -| `sentinel.livenessProbe.initialDelaySeconds` | Initial delay seconds for livenessProbe | `20` | -| `sentinel.livenessProbe.periodSeconds` | Period seconds for livenessProbe | `10` | -| `sentinel.livenessProbe.timeoutSeconds` | Timeout seconds for livenessProbe | `5` | -| `sentinel.livenessProbe.failureThreshold` | Failure threshold for livenessProbe | `6` | -| `sentinel.livenessProbe.successThreshold` | Success threshold for livenessProbe | `1` | -| `sentinel.readinessProbe.enabled` | Enable readinessProbe on Redis® Sentinel nodes | `true` | -| `sentinel.readinessProbe.initialDelaySeconds` | Initial delay seconds for readinessProbe | `20` | -| `sentinel.readinessProbe.periodSeconds` | Period seconds for readinessProbe | `5` | -| `sentinel.readinessProbe.timeoutSeconds` | Timeout seconds for readinessProbe | `1` | -| `sentinel.readinessProbe.failureThreshold` | Failure threshold for readinessProbe | `6` | -| `sentinel.readinessProbe.successThreshold` | Success threshold for readinessProbe | `1` | -| `sentinel.customStartupProbe` | Custom startupProbe that overrides the default one | `{}` | -| `sentinel.customLivenessProbe` | Custom livenessProbe that overrides the default one | `{}` | -| `sentinel.customReadinessProbe` | Custom readinessProbe that overrides the default one | `{}` | -| `sentinel.persistence.enabled` | Enable persistence on Redis® sentinel nodes using Persistent Volume Claims (Experimental) | `false` | -| `sentinel.persistence.storageClass` | Persistent Volume storage class | `""` | -| `sentinel.persistence.accessModes` | Persistent Volume access modes | `["ReadWriteOnce"]` | -| `sentinel.persistence.size` | Persistent Volume size | `100Mi` | -| `sentinel.persistence.annotations` | Additional custom annotations for the PVC | `{}` | -| `sentinel.persistence.labels` | Additional custom labels for the PVC | `{}` | -| `sentinel.persistence.selector` | Additional labels to match for the PVC | `{}` | -| `sentinel.persistence.dataSource` | Custom PVC data source | `{}` | -| `sentinel.persistence.medium` | Provide a medium for `emptyDir` volumes. | `""` | -| `sentinel.persistence.sizeLimit` | Set this to enable a size limit for `emptyDir` volumes. | `""` | -| `sentinel.persistentVolumeClaimRetentionPolicy.enabled` | Controls if and how PVCs are deleted during the lifecycle of a StatefulSet | `false` | -| `sentinel.persistentVolumeClaimRetentionPolicy.whenScaled` | Volume retention behavior when the replica count of the StatefulSet is reduced | `Retain` | -| `sentinel.persistentVolumeClaimRetentionPolicy.whenDeleted` | Volume retention behavior that applies when the StatefulSet is deleted | `Retain` | -| `sentinel.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if sentinel.resources is set (sentinel.resources is recommended for production). | `nano` | -| `sentinel.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `sentinel.containerSecurityContext.enabled` | Enabled Redis® Sentinel containers' Security Context | `true` | -| `sentinel.containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `sentinel.containerSecurityContext.runAsUser` | Set Redis® Sentinel containers' Security Context runAsUser | `1001` | -| `sentinel.containerSecurityContext.runAsGroup` | Set Redis® Sentinel containers' Security Context runAsGroup | `1001` | -| `sentinel.containerSecurityContext.runAsNonRoot` | Set Redis® Sentinel containers' Security Context runAsNonRoot | `true` | -| `sentinel.containerSecurityContext.readOnlyRootFilesystem` | Set container's Security Context read-only root filesystem | `true` | -| `sentinel.containerSecurityContext.allowPrivilegeEscalation` | Set Redis® Sentinel containers' Security Context allowPrivilegeEscalation | `false` | -| `sentinel.containerSecurityContext.seccompProfile.type` | Set Redis® Sentinel containers' Security Context seccompProfile | `RuntimeDefault` | -| `sentinel.containerSecurityContext.capabilities.drop` | Set Redis® Sentinel containers' Security Context capabilities to drop | `["ALL"]` | -| `sentinel.lifecycleHooks` | for the Redis® sentinel container(s) to automate configuration before or after startup | `{}` | -| `sentinel.extraVolumes` | Optionally specify extra list of additional volumes for the Redis® Sentinel | `[]` | -| `sentinel.extraVolumeMounts` | Optionally specify extra list of additional volumeMounts for the Redis® Sentinel container(s) | `[]` | -| `sentinel.service.type` | Redis® Sentinel service type | `ClusterIP` | -| `sentinel.service.ports.redis` | Redis® service port for Redis® | `6379` | -| `sentinel.service.ports.sentinel` | Redis® service port for Redis® Sentinel | `26379` | -| `sentinel.service.nodePorts.redis` | Node port for Redis® | `""` | -| `sentinel.service.nodePorts.sentinel` | Node port for Sentinel | `""` | -| `sentinel.service.externalTrafficPolicy` | Redis® Sentinel service external traffic policy | `Cluster` | -| `sentinel.service.extraPorts` | Extra ports to expose (normally used with the `sidecar` value) | `[]` | -| `sentinel.service.clusterIP` | Redis® Sentinel service Cluster IP | `""` | -| `sentinel.service.createMaster` | Enable master service pointing to the current master (experimental) | `false` | -| `sentinel.service.loadBalancerIP` | Redis® Sentinel service Load Balancer IP | `""` | -| `sentinel.service.loadBalancerClass` | sentinel service Load Balancer class if service type is `LoadBalancer` (optional, cloud specific) | `""` | -| `sentinel.service.loadBalancerSourceRanges` | Redis® Sentinel service Load Balancer sources | `[]` | -| `sentinel.service.annotations` | Additional custom annotations for Redis® Sentinel service | `{}` | -| `sentinel.service.sessionAffinity` | Session Affinity for Kubernetes service, can be "None" or "ClientIP" | `None` | -| `sentinel.service.sessionAffinityConfig` | Additional settings for the sessionAffinity | `{}` | -| `sentinel.service.headless.annotations` | Annotations for the headless service. | `{}` | -| `sentinel.masterService.enabled` | Enable master service pointing to the current master (experimental) | `false` | -| `sentinel.masterService.type` | Redis® Sentinel master service type | `ClusterIP` | -| `sentinel.masterService.ports.redis` | Redis® service port for Redis® | `6379` | -| `sentinel.masterService.nodePorts.redis` | Node port for Redis® | `""` | -| `sentinel.masterService.externalTrafficPolicy` | Redis® master service external traffic policy | `""` | -| `sentinel.masterService.extraPorts` | Extra ports to expose (normally used with the `sidecar` value) | `[]` | -| `sentinel.masterService.clusterIP` | Redis® master service Cluster IP | `""` | -| `sentinel.masterService.loadBalancerIP` | Redis® master service Load Balancer IP | `""` | -| `sentinel.masterService.loadBalancerClass` | master service Load Balancer class if service type is `LoadBalancer` (optional, cloud specific) | `""` | -| `sentinel.masterService.loadBalancerSourceRanges` | Redis® master service Load Balancer sources | `[]` | -| `sentinel.masterService.annotations` | Additional custom annotations for Redis® master service | `{}` | -| `sentinel.masterService.sessionAffinity` | Session Affinity for Kubernetes service, can be "None" or "ClientIP" | `None` | -| `sentinel.masterService.sessionAffinityConfig` | Additional settings for the sessionAffinity | `{}` | -| `sentinel.terminationGracePeriodSeconds` | Integer setting the termination grace period for the redis-node pods | `30` | - -### Other Parameters - -| Name | Description | Value | -| ----------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------- | ------- | -| `serviceBindings.enabled` | Create secret for service binding (Experimental) | `false` | -| `networkPolicy.enabled` | Enable creation of NetworkPolicy resources | `true` | -| `networkPolicy.allowExternal` | Don't require client label for connections | `true` | -| `networkPolicy.allowExternalEgress` | Allow the pod to access any range of port and all destinations. | `true` | -| `networkPolicy.extraIngress` | Add extra ingress rules to the NetworkPolicy | `[]` | -| `networkPolicy.extraEgress` | Add extra egress rules to the NetworkPolicy | `[]` | -| `networkPolicy.ingressNSMatchLabels` | Labels to match to allow traffic from other namespaces | `{}` | -| `networkPolicy.ingressNSPodMatchLabels` | Pod labels to match to allow traffic from other namespaces | `{}` | -| `networkPolicy.metrics.allowExternal` | Don't require client label for connections for metrics endpoint | `true` | -| `networkPolicy.metrics.ingressNSMatchLabels` | Labels to match to allow traffic from other namespaces to metrics endpoint | `{}` | -| `networkPolicy.metrics.ingressNSPodMatchLabels` | Pod labels to match to allow traffic from other namespaces to metrics endpoint | `{}` | -| `podSecurityPolicy.create` | Whether to create a PodSecurityPolicy. WARNING: PodSecurityPolicy is deprecated in Kubernetes v1.21 or later, unavailable in v1.25 or later | `false` | -| `podSecurityPolicy.enabled` | Enable PodSecurityPolicy's RBAC rules | `false` | -| `rbac.create` | Specifies whether RBAC resources should be created | `false` | -| `rbac.rules` | Custom RBAC rules to set | `[]` | -| `serviceAccount.create` | Specifies whether a ServiceAccount should be created | `true` | -| `serviceAccount.name` | The name of the ServiceAccount to use. | `""` | -| `serviceAccount.automountServiceAccountToken` | Whether to auto mount the service account token | `false` | -| `serviceAccount.annotations` | Additional custom annotations for the ServiceAccount | `{}` | -| `pdb` | DEPRECATED Please use `master.pdb` and `replica.pdb` values instead | `{}` | -| `tls.enabled` | Enable TLS traffic | `false` | -| `tls.authClients` | Require clients to authenticate | `true` | -| `tls.autoGenerated` | Enable autogenerated certificates | `false` | -| `tls.existingSecret` | The name of the existing secret that contains the TLS certificates | `""` | -| `tls.certificatesSecret` | DEPRECATED. Use existingSecret instead. | `""` | -| `tls.certFilename` | Certificate filename | `""` | -| `tls.certKeyFilename` | Certificate Key filename | `""` | -| `tls.certCAFilename` | CA Certificate filename | `""` | -| `tls.dhParamsFilename` | File containing DH params (in order to support DH based ciphers) | `""` | - -### Metrics Parameters - -| Name | Description | Value | -| ----------------------------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | -------------------------------- | -| `metrics.enabled` | Start a sidecar prometheus exporter to expose Redis® metrics | `false` | -| `metrics.image.registry` | Redis® Exporter image registry | `REGISTRY_NAME` | -| `metrics.image.repository` | Redis® Exporter image repository | `REPOSITORY_NAME/redis-exporter` | -| `metrics.image.digest` | Redis® Exporter image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `metrics.image.pullPolicy` | Redis® Exporter image pull policy | `IfNotPresent` | -| `metrics.image.pullSecrets` | Redis® Exporter image pull secrets | `[]` | -| `metrics.containerPorts.http` | Metrics HTTP container port | `9121` | -| `metrics.startupProbe.enabled` | Enable startupProbe on Redis® replicas nodes | `false` | -| `metrics.startupProbe.initialDelaySeconds` | Initial delay seconds for startupProbe | `10` | -| `metrics.startupProbe.periodSeconds` | Period seconds for startupProbe | `10` | -| `metrics.startupProbe.timeoutSeconds` | Timeout seconds for startupProbe | `5` | -| `metrics.startupProbe.failureThreshold` | Failure threshold for startupProbe | `5` | -| `metrics.startupProbe.successThreshold` | Success threshold for startupProbe | `1` | -| `metrics.livenessProbe.enabled` | Enable livenessProbe on Redis® replicas nodes | `true` | -| `metrics.livenessProbe.initialDelaySeconds` | Initial delay seconds for livenessProbe | `10` | -| `metrics.livenessProbe.periodSeconds` | Period seconds for livenessProbe | `10` | -| `metrics.livenessProbe.timeoutSeconds` | Timeout seconds for livenessProbe | `5` | -| `metrics.livenessProbe.failureThreshold` | Failure threshold for livenessProbe | `5` | -| `metrics.livenessProbe.successThreshold` | Success threshold for livenessProbe | `1` | -| `metrics.readinessProbe.enabled` | Enable readinessProbe on Redis® replicas nodes | `true` | -| `metrics.readinessProbe.initialDelaySeconds` | Initial delay seconds for readinessProbe | `5` | -| `metrics.readinessProbe.periodSeconds` | Period seconds for readinessProbe | `10` | -| `metrics.readinessProbe.timeoutSeconds` | Timeout seconds for readinessProbe | `1` | -| `metrics.readinessProbe.failureThreshold` | Failure threshold for readinessProbe | `3` | -| `metrics.readinessProbe.successThreshold` | Success threshold for readinessProbe | `1` | -| `metrics.customStartupProbe` | Custom startupProbe that overrides the default one | `{}` | -| `metrics.customLivenessProbe` | Custom livenessProbe that overrides the default one | `{}` | -| `metrics.customReadinessProbe` | Custom readinessProbe that overrides the default one | `{}` | -| `metrics.command` | Override default metrics container init command (useful when using custom images) | `[]` | -| `metrics.redisTargetHost` | A way to specify an alternative Redis® hostname | `localhost` | -| `metrics.extraArgs` | Extra arguments for Redis® exporter, for example: | `{}` | -| `metrics.extraEnvVars` | Array with extra environment variables to add to Redis® exporter | `[]` | -| `metrics.containerSecurityContext.enabled` | Enabled Redis® exporter containers' Security Context | `true` | -| `metrics.containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `metrics.containerSecurityContext.runAsUser` | Set Redis® exporter containers' Security Context runAsUser | `1001` | -| `metrics.containerSecurityContext.runAsGroup` | Set Redis® exporter containers' Security Context runAsGroup | `1001` | -| `metrics.containerSecurityContext.runAsNonRoot` | Set Redis® exporter containers' Security Context runAsNonRoot | `true` | -| `metrics.containerSecurityContext.allowPrivilegeEscalation` | Set Redis® exporter containers' Security Context allowPrivilegeEscalation | `false` | -| `metrics.containerSecurityContext.readOnlyRootFilesystem` | Set container's Security Context read-only root filesystem | `true` | -| `metrics.containerSecurityContext.seccompProfile.type` | Set Redis® exporter containers' Security Context seccompProfile | `RuntimeDefault` | -| `metrics.containerSecurityContext.capabilities.drop` | Set Redis® exporter containers' Security Context capabilities to drop | `["ALL"]` | -| `metrics.extraVolumes` | Optionally specify extra list of additional volumes for the Redis® metrics sidecar | `[]` | -| `metrics.extraVolumeMounts` | Optionally specify extra list of additional volumeMounts for the Redis® metrics sidecar | `[]` | -| `metrics.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if metrics.resources is set (metrics.resources is recommended for production). | `nano` | -| `metrics.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `metrics.podLabels` | Extra labels for Redis® exporter pods | `{}` | -| `metrics.podAnnotations` | Annotations for Redis® exporter pods | `{}` | -| `metrics.service.enabled` | Create Service resource(s) for scraping metrics using PrometheusOperator ServiceMonitor, can be disabled when using a PodMonitor | `true` | -| `metrics.service.type` | Redis® exporter service type | `ClusterIP` | -| `metrics.service.ports.http` | Redis® exporter service port | `9121` | -| `metrics.service.externalTrafficPolicy` | Redis® exporter service external traffic policy | `Cluster` | -| `metrics.service.extraPorts` | Extra ports to expose (normally used with the `sidecar` value) | `[]` | -| `metrics.service.loadBalancerIP` | Redis® exporter service Load Balancer IP | `""` | -| `metrics.service.loadBalancerClass` | exporter service Load Balancer class if service type is `LoadBalancer` (optional, cloud specific) | `""` | -| `metrics.service.loadBalancerSourceRanges` | Redis® exporter service Load Balancer sources | `[]` | -| `metrics.service.annotations` | Additional custom annotations for Redis® exporter service | `{}` | -| `metrics.service.clusterIP` | Redis® exporter service Cluster IP | `""` | -| `metrics.serviceMonitor.port` | the service port to scrape metrics from | `http-metrics` | -| `metrics.serviceMonitor.enabled` | Create ServiceMonitor resource(s) for scraping metrics using PrometheusOperator | `false` | -| `metrics.serviceMonitor.namespace` | The namespace in which the ServiceMonitor will be created | `""` | -| `metrics.serviceMonitor.interval` | The interval at which metrics should be scraped | `30s` | -| `metrics.serviceMonitor.scrapeTimeout` | The timeout after which the scrape is ended | `""` | -| `metrics.serviceMonitor.relabelings` | Metrics RelabelConfigs to apply to samples before scraping. | `[]` | -| `metrics.serviceMonitor.metricRelabelings` | Metrics RelabelConfigs to apply to samples before ingestion. | `[]` | -| `metrics.serviceMonitor.honorLabels` | Specify honorLabels parameter to add the scrape endpoint | `false` | -| `metrics.serviceMonitor.additionalLabels` | Additional labels that can be used so ServiceMonitor resource(s) can be discovered by Prometheus | `{}` | -| `metrics.serviceMonitor.podTargetLabels` | Labels from the Kubernetes pod to be transferred to the created metrics | `[]` | -| `metrics.serviceMonitor.sampleLimit` | Limit of how many samples should be scraped from every Pod | `false` | -| `metrics.serviceMonitor.targetLimit` | Limit of how many targets should be scraped | `false` | -| `metrics.serviceMonitor.additionalEndpoints` | Additional endpoints to scrape (e.g sentinel) | `[]` | -| `metrics.podMonitor.port` | the pod port to scrape metrics from | `metrics` | -| `metrics.podMonitor.enabled` | Create PodMonitor resource(s) for scraping metrics using PrometheusOperator | `false` | -| `metrics.podMonitor.namespace` | The namespace in which the PodMonitor will be created | `""` | -| `metrics.podMonitor.interval` | The interval at which metrics should be scraped | `30s` | -| `metrics.podMonitor.scrapeTimeout` | The timeout after which the scrape is ended | `""` | -| `metrics.podMonitor.relabelings` | Metrics RelabelConfigs to apply to samples before scraping. | `[]` | -| `metrics.podMonitor.metricRelabelings` | Metrics RelabelConfigs to apply to samples before ingestion. | `[]` | -| `metrics.podMonitor.honorLabels` | Specify honorLabels parameter to add the scrape endpoint | `false` | -| `metrics.podMonitor.additionalLabels` | Additional labels that can be used so PodMonitor resource(s) can be discovered by Prometheus | `{}` | -| `metrics.podMonitor.podTargetLabels` | Labels from the Kubernetes pod to be transferred to the created metrics | `[]` | -| `metrics.podMonitor.sampleLimit` | Limit of how many samples should be scraped from every Pod | `false` | -| `metrics.podMonitor.targetLimit` | Limit of how many targets should be scraped | `false` | -| `metrics.podMonitor.additionalEndpoints` | Additional endpoints to scrape (e.g sentinel) | `[]` | -| `metrics.prometheusRule.enabled` | Create a custom prometheusRule Resource for scraping metrics using PrometheusOperator | `false` | -| `metrics.prometheusRule.namespace` | The namespace in which the prometheusRule will be created | `""` | -| `metrics.prometheusRule.additionalLabels` | Additional labels for the prometheusRule | `{}` | -| `metrics.prometheusRule.rules` | Custom Prometheus rules | `[]` | - -### Init Container Parameters - -| Name | Description | Value | -| ----------------------------------------------------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ----------------------------------------------------------------- | -| `volumePermissions.enabled` | Enable init container that changes the owner/group of the PV mount point to `runAsUser:fsGroup` | `false` | -| `volumePermissions.image.registry` | OS Shell + Utility image registry | `REGISTRY_NAME` | -| `volumePermissions.image.repository` | OS Shell + Utility image repository | `REPOSITORY_NAME/os-shell` | -| `volumePermissions.image.digest` | OS Shell + Utility image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `volumePermissions.image.pullPolicy` | OS Shell + Utility image pull policy | `IfNotPresent` | -| `volumePermissions.image.pullSecrets` | OS Shell + Utility image pull secrets | `[]` | -| `volumePermissions.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if volumePermissions.resources is set (volumePermissions.resources is recommended for production). | `nano` | -| `volumePermissions.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | -| `volumePermissions.containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `volumePermissions.containerSecurityContext.runAsUser` | Set init container's Security Context runAsUser | `0` | -| `kubectl.image.registry` | Kubectl image registry | `REGISTRY_NAME` | -| `kubectl.image.repository` | Kubectl image repository | `REPOSITORY_NAME/kubectl` | -| `kubectl.image.digest` | Kubectl image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `kubectl.image.pullPolicy` | Kubectl image pull policy | `IfNotPresent` | -| `kubectl.image.pullSecrets` | Kubectl pull secrets | `[]` | -| `kubectl.command` | kubectl command to execute | `["/opt/bitnami/scripts/kubectl-scripts/update-master-label.sh"]` | -| `kubectl.containerSecurityContext.enabled` | Enabled kubectl containers' Security Context | `true` | -| `kubectl.containerSecurityContext.seLinuxOptions` | Set SELinux options in container | `{}` | -| `kubectl.containerSecurityContext.runAsUser` | Set kubectl containers' Security Context runAsUser | `1001` | -| `kubectl.containerSecurityContext.runAsGroup` | Set kubectl containers' Security Context runAsGroup | `1001` | -| `kubectl.containerSecurityContext.runAsNonRoot` | Set kubectl containers' Security Context runAsNonRoot | `true` | -| `kubectl.containerSecurityContext.allowPrivilegeEscalation` | Set kubectl containers' Security Context allowPrivilegeEscalation | `false` | -| `kubectl.containerSecurityContext.readOnlyRootFilesystem` | Set container's Security Context read-only root filesystem | `true` | -| `kubectl.containerSecurityContext.seccompProfile.type` | Set kubectl containers' Security Context seccompProfile | `RuntimeDefault` | -| `kubectl.containerSecurityContext.capabilities.drop` | Set kubectl containers' Security Context capabilities to drop | `["ALL"]` | -| `kubectl.resources.limits` | The resources limits for the kubectl containers | `{}` | -| `kubectl.resources.requests` | The requested resources for the kubectl containers | `{}` | -| `sysctl.enabled` | Enable init container to modify Kernel settings | `false` | -| `sysctl.image.registry` | OS Shell + Utility image registry | `REGISTRY_NAME` | -| `sysctl.image.repository` | OS Shell + Utility image repository | `REPOSITORY_NAME/os-shell` | -| `sysctl.image.digest` | OS Shell + Utility image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag | `""` | -| `sysctl.image.pullPolicy` | OS Shell + Utility image pull policy | `IfNotPresent` | -| `sysctl.image.pullSecrets` | OS Shell + Utility image pull secrets | `[]` | -| `sysctl.command` | Override default init-sysctl container command (useful when using custom images) | `[]` | -| `sysctl.mountHostSys` | Mount the host `/sys` folder to `/host-sys` | `false` | -| `sysctl.resourcesPreset` | Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if sysctl.resources is set (sysctl.resources is recommended for production). | `nano` | -| `sysctl.resources` | Set container requests and limits for different resources like CPU or memory (essential for production workloads) | `{}` | - -### useExternalDNS Parameters - -| Name | Description | Value | -| -------------------------------------- | ---------------------------------------------------------------------------------------------------------------------------------------- | ----------------------------------- | -| `useExternalDNS.enabled` | Enable various syntax that would enable external-dns to work. Note this requires a working installation of `external-dns` to be usable. | `false` | -| `useExternalDNS.additionalAnnotations` | Extra annotations to be utilized when `external-dns` is enabled. | `{}` | -| `useExternalDNS.annotationKey` | The annotation key utilized when `external-dns` is enabled. Setting this to `false` will disable annotations. | `external-dns.alpha.kubernetes.io/` | -| `useExternalDNS.suffix` | The DNS suffix utilized when `external-dns` is enabled. Note that we prepend the suffix with the full name of the release. | `""` | - -Specify each parameter using the `--set key=value[,key=value]` argument to `helm install`. For example, - -```console -helm install my-release \ - --set auth.password=secretpassword \ - oci://REGISTRY_NAME/REPOSITORY_NAME/redis -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -The above command sets the Redis® server password to `secretpassword`. - -> NOTE: Once this chart is deployed, it is not possible to change the application's access credentials, such as usernames or passwords, using Helm. To change these application credentials after deployment, delete any persistent volumes (PVs) used by the chart and re-deploy it, or use the application's built-in administrative tools if available. - -Alternatively, a YAML file that specifies the values for the parameters can be provided while installing the chart. For example, - -```console -helm install my-release -f values.yaml oci://REGISTRY_NAME/REPOSITORY_NAME/redis -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. -> **Tip**: You can use the default [values.yaml](https://github.com/bitnami/charts/tree/main/bitnami/redis/values.yaml) - -## Troubleshooting - -Find more information about how to deal with common errors related to Bitnami's Helm charts in [this troubleshooting guide](https://docs.bitnami.com/general/how-to/troubleshoot-helm-chart-issues). - -## Upgrading - -A major chart version change (like v1.2.3 -> v2.0.0) indicates that there is an incompatible breaking change needing manual actions. - -### RDB compatibility - -It's common to have RDB format changes across Redis® releases where we see backward compatibility but no forward compatibility. For example, v7.0 can load an RDB created by v6.2 , but the opposite is not true. -When that's the case, the rolling update can cause replicas to temporarily stop synchronizing while they are running a lower version than master. -For example, on a rolling update `master-0` and `replica-2` are updated first from version v6.2 to v7.0; `replica-0` and `replica-1` won't be able to start a full sync with `master-0` because they are still running v6.2 and can't support the RDB format from version 7.0 that master is now using. -This issue can be mitigated by splitting the upgrade into two stages: one for all replicas and another for any master. - -- Stage 1 (replicas only, as there's no master with an ordinal higher than 99): -`helm upgrade oci://REGISTRY_NAME/REPOSITORY_NAME/redis --set master.updateStrategy.rollingUpdate.partition=99` -- Stage 2 (anything else that is not up to date, in this case only master): -`helm upgrade oci://REGISTRY_NAME/REPOSITORY_NAME/redis` - -### To 19.0.0 - -This major bump changes the following security defaults: - -- `runAsGroup` is changed from `0` to `1001` -- `readOnlyRootFilesystem` is set to `true` -- `resourcesPreset` is changed from `none` to the minimum size working in our test suites (NOTE: `resourcesPreset` is not meant for production usage, but `resources` adapted to your use case). -- `global.compatibility.openshift.adaptSecurityContext` is changed from `disabled` to `auto`. - -This could potentially break any customization or init scripts used in your deployment. If this is the case, change the default values to the previous ones. - -### To 18.0.0 - -This major version updates the Redis® docker image version used from `7.0` to `7.2`, the new stable version. There are no major changes in the chart, but we recommend checking the [Redis® 7.2 release notes](https://raw.githubusercontent.com/redis/redis/7.2/00-RELEASENOTES) before upgrading. - -NOTE: Due to an error in our release process, versions higher or equal than 17.15.4 already use 7.2 by default. - -### To 17.0.0 - -This major version updates the Redis® docker image version used from `6.2` to `7.0`, the new stable version. There are no major changes in the chart, but we recommend checking the [Redis® 7.0 release notes](https://raw.githubusercontent.com/redis/redis/7.0/00-RELEASENOTES) before upgrading. - -### To 16.0.0 - -This major release renames several values in this chart and adds missing features, in order to be inline with the rest of assets in the Bitnami charts repository. - -Affected values: - -- `master.service.port` renamed as `master.service.ports.redis`. -- `master.service.nodePort` renamed as `master.service.nodePorts.redis`. -- `replica.service.port` renamed as `replica.service.ports.redis`. -- `replica.service.nodePort` renamed as `replica.service.nodePorts.redis`. -- `sentinel.service.port` renamed as `sentinel.service.ports.redis`. -- `sentinel.service.sentinelPort` renamed as `sentinel.service.ports.sentinel`. -- `master.containerPort` renamed as `master.containerPorts.redis`. -- `replica.containerPort` renamed as `replica.containerPorts.redis`. -- `sentinel.containerPort` renamed as `sentinel.containerPorts.sentinel`. -- `master.spreadConstraints` renamed as `master.topologySpreadConstraints` -- `replica.spreadConstraints` renamed as `replica.topologySpreadConstraints` - -### To 15.0.0 - -The parameter to enable the usage of StaticIDs was removed. The behavior is to [always use StaticIDs](https://github.com/bitnami/charts/pull/7278). - -### To 14.8.0 - -The Redis® sentinel exporter was removed in this version because the upstream project was deprecated. The regular Redis® exporter is included in the sentinel scenario as usual. - -### To 14.0.0 - -- Several parameters were renamed or disappeared in favor of new ones on this major version: - - The term *slave* has been replaced by the term *replica*. Therefore, parameters prefixed with `slave` are now prefixed with `replicas`. - - Credentials parameter are reorganized under the `auth` parameter. - - `cluster.enabled` parameter is deprecated in favor of `architecture` parameter that accepts two values: `standalone` and `replication`. - - `securityContext.*` is deprecated in favor of `XXX.podSecurityContext` and `XXX.containerSecurityContext`. - - `sentinel.metrics.*` parameters are deprecated in favor of `metrics.sentinel.*` ones. -- New parameters to add custom command, environment variables, sidecars, init containers, etc. were added. -- Chart labels were adapted to follow the [Helm charts standard labels](https://helm.sh/docs/chart_best_practices/labels/#standard-labels). -- values.yaml metadata was adapted to follow the format supported by [Readme Generator for Helm](https://github.com/bitnami/readme-generator-for-helm). - -Consequences: - -Backwards compatibility is not guaranteed. To upgrade to `14.0.0`, install a new release of the Redis® chart, and migrate the data from your previous release. You have 2 alternatives to do so: - -- Create a backup of the database, and restore it on the new release as explained in the [Backup and restore](#backup-and-restore) section. -- Reuse the PVC used to hold the master data on your previous release. To do so, use the `master.persistence.existingClaim` parameter. The following example assumes that the release name is `redis`: - -```console -helm install redis oci://REGISTRY_NAME/REPOSITORY_NAME/redis --set auth.password=[PASSWORD] --set master.persistence.existingClaim=[EXISTING_PVC] -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -| Note: you need to substitute the placeholder *[EXISTING_PVC]* with the name of the PVC used on your previous release, and *[PASSWORD]* with the password used in your previous release. - -### To 13.0.0 - -This major version updates the Redis® docker image version used from `6.0` to `6.2`, the new stable version. There are no major changes in the chart and there shouldn't be any breaking changes in it as `6.2` is basically a stricter superset of `6.0`. For more information, please refer to [Redis® 6.2 release notes](https://raw.githubusercontent.com/redis/redis/6.2/00-RELEASENOTES). - -### To 12.3.0 - -This version also introduces `bitnami/common`, a [library chart](https://helm.sh/docs/topics/library_charts/#helm) as a dependency. More documentation about this new utility could be found [here](https://github.com/bitnami/charts/tree/main/bitnami/common#bitnami-common-library-chart). Please, make sure that you have updated the chart dependencies before executing any upgrade. - -### To 12.0.0 - -[On November 13, 2020, Helm v2 support was formally finished](https://github.com/helm/charts#status-of-the-project), this major version is the result of the required changes applied to the Helm Chart to be able to incorporate the different features added in Helm v3 and to be consistent with the Helm project itself regarding the Helm v2 EOL. - -#### What changes were introduced in this major version? - -- Previous versions of this Helm Chart use `apiVersion: v1` (installable by both Helm 2 and 3), this Helm Chart was updated to `apiVersion: v2` (installable by Helm 3 only). [Here](https://helm.sh/docs/topics/charts/#the-apiversion-field) you can find more information about the `apiVersion` field. -- The different fields present in the *Chart.yaml* file has been ordered alphabetically in a homogeneous way for all the Bitnami Helm Charts - -#### Considerations when upgrading to this version - -- If you want to upgrade to this version from a previous one installed with Helm v3, you shouldn't face any issues -- If you want to upgrade to this version using Helm v2, this scenario is not supported as this version doesn't support Helm v2 anymore -- If you installed the previous version with Helm v2 and wants to upgrade to this version with Helm v3, please refer to the [official Helm documentation](https://helm.sh/docs/topics/v2_v3_migration/#migration-use-cases) about migrating from Helm v2 to v3 - -#### Useful links - -- -- -- - -### To 11.0.0 - -When using sentinel, a new statefulset called `-node` was introduced. This will break upgrading from a previous version where the statefulsets are called master and slave. Hence the PVC will not match the new naming and won't be reused. If you want to keep your data, you will need to perform a backup and then a restore the data in this new version. - -When deployed with sentinel enabled, only a group of nodes is deployed and the master/slave role is handled in the group. To avoid breaking the compatibility, the settings for this nodes are given through the `slave.xxxx` parameters in `values.yaml` - -### To 10.0.0 - -For releases with `usePassword: true`, the value `sentinel.usePassword` controls whether the password authentication also applies to the sentinel port. This defaults to `true` for a secure configuration, however it is possible to disable to account for the following cases: - -- Using a version of redis-sentinel prior to `5.0.1` where the authentication feature was introduced. -- Where redis clients need to be updated to support sentinel authentication. - -If using a master/slave topology, or with `usePassword: false`, no action is required. - -### To 9.0.0 - -The metrics exporter has been changed from a separate deployment to a sidecar container, due to the latest changes in the Redis® exporter code. Check the [official page](https://github.com/oliver006/redis_exporter/) for more information. The metrics container image was changed from oliver006/redis_exporter to bitnami/redis-exporter (Bitnami's maintained package of oliver006/redis_exporter). - -### To 8.0.18 - -For releases with `metrics.enabled: true` the default tag for the exporter image is now `v1.x.x`. This introduces many changes including metrics names. You'll want to use [this dashboard](https://github.com/oliver006/redis_exporter/blob/master/contrib/grafana_prometheus_redis_dashboard.json) now. Please see the [redis_exporter github page](https://github.com/oliver006/redis_exporter#upgrading-from-0x-to-1x) for more details. - -### To 7.0.0 - -This version causes a change in the Redis® Master StatefulSet definition, so the command helm upgrade would not work out of the box. As an alternative, one of the following could be done: - -- Recommended: Create a clone of the Redis® Master PVC (for example, using projects like [this one](https://github.com/edseymour/pvc-transfer)). Then launch a fresh release reusing this cloned PVC. - -```console -helm install my-release oci://REGISTRY_NAME/REPOSITORY_NAME/redis --set persistence.existingClaim= -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -- Alternative (not recommended, do at your own risk): `helm delete --purge` does not remove the PVC assigned to the Redis® Master StatefulSet. As a consequence, the following commands can be done to upgrade the release - -```console -helm delete --purge -helm install oci://REGISTRY_NAME/REPOSITORY_NAME/redis -``` - -> Note: You need to substitute the placeholders `REGISTRY_NAME` and `REPOSITORY_NAME` with a reference to your Helm chart registry and repository. For example, in the case of Bitnami, you need to use `REGISTRY_NAME=registry-1.docker.io` and `REPOSITORY_NAME=bitnamicharts`. - -Previous versions of the chart were not using persistence in the slaves, so this upgrade would add it to them. Another important change is that no values are inherited from master to slaves. For example, in 6.0.0 `slaves.readinessProbe.periodSeconds`, if empty, would be set to `master.readinessProbe.periodSeconds`. This approach lacked transparency and was difficult to maintain. From now on, all the slave parameters must be configured just as it is done with the masters. - -Some values have changed as well: - -- `master.port` and `slave.port` have been changed to `redisPort` (same value for both master and slaves) -- `master.securityContext` and `slave.securityContext` have been changed to `securityContext`(same values for both master and slaves) - -By default, the upgrade will not change the cluster topology. In case you want to use Redis® Sentinel, you must explicitly set `sentinel.enabled` to `true`. - -### To 6.0.0 - -Previous versions of the chart were using an init-container to change the permissions of the volumes. This was done in case the `securityContext` directive in the template was not enough for that (for example, with cephFS). In this new version of the chart, this container is disabled by default (which should not affect most of the deployments). If your installation still requires that init container, execute `helm upgrade` with the `--set volumePermissions.enabled=true`. - -### To 5.0.0 - -The default image in this release may be switched out for any image containing the `redis-server` -and `redis-cli` binaries. If `redis-server` is not the default image ENTRYPOINT, `master.command` -must be specified. - -#### Breaking changes - -- `master.args` and `slave.args` are removed. Use `master.command` or `slave.command` instead in order to override the image entrypoint, or `master.extraFlags` to pass additional flags to `redis-server`. -- `disableCommands` is now interpreted as an array of strings instead of a string of comma separated values. -- `master.persistence.path` now defaults to `/data`. - -### To 4.0.0 - -This version removes the `chart` label from the `spec.selector.matchLabels` -which is immutable since `StatefulSet apps/v1beta2`. It has been inadvertently -added, causing any subsequent upgrade to fail. See . - -It also fixes where a deployment `extensions/v1beta1` can not be upgraded if `spec.selector` is not explicitly set. - -Finally, it fixes by removing mutable labels in `spec.VolumeClaimTemplate.metadata.labels` so that it is upgradable. - -In order to upgrade, delete the Redis® StatefulSet before upgrading: - -```console -kubectl delete statefulsets.apps --cascade=false my-release-redis-master -``` - -And edit the Redis® slave (and metrics if enabled) deployment: - -```console -kubectl patch deployments my-release-redis-slave --type=json -p='[{"op": "remove", "path": "/spec/selector/matchLabels/chart"}]' -kubectl patch deployments my-release-redis-metrics --type=json -p='[{"op": "remove", "path": "/spec/selector/matchLabels/chart"}]' -``` - -## License - -Copyright © 2024 Broadcom. The term "Broadcom" refers to Broadcom Inc. and/or its subsidiaries. - -Licensed under the Apache License, Version 2.0 (the "License"); -you may not use this file except in compliance with the License. -You may obtain a copy of the License at - - - -Unless required by applicable law or agreed to in writing, software -distributed under the License is distributed on an "AS IS" BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -See the License for the specific language governing permissions and -limitations under the License. \ No newline at end of file diff --git a/charts/redis/redis/templates/NOTES.txt b/charts/redis/redis/templates/NOTES.txt deleted file mode 100644 index 69bdec3..0000000 --- a/charts/redis/redis/templates/NOTES.txt +++ /dev/null @@ -1,213 +0,0 @@ -CHART NAME: {{ .Chart.Name }} -CHART VERSION: {{ .Chart.Version }} -APP VERSION: {{ .Chart.AppVersion }} - -** Please be patient while the chart is being deployed ** - -{{- if .Values.diagnosticMode.enabled }} -The chart has been deployed in diagnostic mode. All probes have been disabled and the command has been overwritten with: - - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 4 }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 4 }} - -Get the list of pods by executing: - - kubectl get pods --namespace {{ include "common.names.namespace" . }} -l app.kubernetes.io/instance={{ .Release.Name }} - -Access the pod you want to debug by executing - - kubectl exec --namespace {{ include "common.names.namespace" . }} -ti -- bash - -In order to replicate the container startup scripts execute this command: - -For Redis: - - /opt/bitnami/scripts/redis/entrypoint.sh /opt/bitnami/scripts/redis/run.sh - -{{- if .Values.sentinel.enabled }} - -For Redis Sentinel: - - /opt/bitnami/scripts/redis-sentinel/entrypoint.sh /opt/bitnami/scripts/redis-sentinel/run.sh - -{{- end }} -{{- else }} - -{{- if contains .Values.master.service.type "LoadBalancer" }} -{{- if not .Values.auth.enabled }} -{{ if and (not .Values.networkPolicy.enabled) (.Values.networkPolicy.allowExternal) }} - -------------------------------------------------------------------------------- - WARNING - - By specifying "master.service.type=LoadBalancer" and "auth.enabled=false" you have - most likely exposed the Redis® service externally without any authentication - mechanism. - - For security reasons, we strongly suggest that you switch to "ClusterIP" or - "NodePort". As alternative, you can also switch to "auth.enabled=true" - providing a valid password on "password" parameter. - -------------------------------------------------------------------------------- -{{- end }} -{{- end }} -{{- end }} - -{{- if and .Values.auth.usePasswordFiles (not .Values.auth.usePasswordFileFromSecret) (or (empty .Values.master.initContainers) (empty .Values.replica.initContainers)) }} - -------------------------------------------------------------------------------- - WARNING - - By specifying ".Values.auth.usePasswordFiles=true" and ".Values.auth.usePasswordFileFromSecret=false" - Redis is expecting that the password is mounted as a file in each pod - (by default in /opt/bitnami/redis/secrets/redis-password) - - Ensure that you specify the respective initContainers in - both .Values.master.initContainers and .Values.replica.initContainers - in order to populate the contents of this file. - -------------------------------------------------------------------------------- -{{- end }} - -{{- if eq .Values.architecture "replication" }} -{{- if .Values.sentinel.enabled }} - -Redis® can be accessed via port {{ .Values.sentinel.service.ports.redis }} on the following DNS name from within your cluster: - - {{ template "common.names.fullname" . }}.{{ include "common.names.namespace" . }}.svc.{{ .Values.clusterDomain }} for read only operations - -For read/write operations, first access the Redis® Sentinel cluster, which is available in port {{ .Values.sentinel.service.ports.sentinel }} using the same domain name above. - -{{- else }} - -Redis® can be accessed on the following DNS names from within your cluster: - - {{ printf "%s-master.%s.svc.%s" (include "common.names.fullname" .) (include "common.names.namespace" . ) .Values.clusterDomain }} for read/write operations (port {{ .Values.master.service.ports.redis }}) - {{ printf "%s-replicas.%s.svc.%s" (include "common.names.fullname" .) (include "common.names.namespace" . ) .Values.clusterDomain }} for read-only operations (port {{ .Values.replica.service.ports.redis }}) - -{{- end }} -{{- else }} - -Redis® can be accessed via port {{ .Values.master.service.ports.redis }} on the following DNS name from within your cluster: - - {{ template "common.names.fullname" . }}-master.{{ include "common.names.namespace" . }}.svc.{{ .Values.clusterDomain }} - -{{- end }} - -{{ if .Values.auth.enabled }} - -To get your password run: - - export REDIS_PASSWORD=$(kubectl get secret --namespace {{ include "common.names.namespace" . }} {{ template "redis.secretName" . }} -o jsonpath="{.data.redis-password}" | base64 -d) - -{{- end }} - -To connect to your Redis® server: - -1. Run a Redis® pod that you can use as a client: - - kubectl run --namespace {{ include "common.names.namespace" . }} redis-client --restart='Never' {{ if .Values.auth.enabled }} --env REDIS_PASSWORD=$REDIS_PASSWORD {{ end }} --image {{ template "redis.image" . }} --command -- sleep infinity - -{{- if .Values.tls.enabled }} - - Copy your TLS certificates to the pod: - - kubectl cp --namespace {{ include "common.names.namespace" . }} /path/to/client.cert redis-client:/tmp/client.cert - kubectl cp --namespace {{ include "common.names.namespace" . }} /path/to/client.key redis-client:/tmp/client.key - kubectl cp --namespace {{ include "common.names.namespace" . }} /path/to/CA.cert redis-client:/tmp/CA.cert - -{{- end }} - - Use the following command to attach to the pod: - - kubectl exec --tty -i redis-client \ - {{- if and (.Values.networkPolicy.enabled) (not .Values.networkPolicy.allowExternal) }}--labels="{{ template "common.names.fullname" . }}-client=true" \{{- end }} - --namespace {{ include "common.names.namespace" . }} -- bash - -2. Connect using the Redis® CLI: - -{{- if eq .Values.architecture "replication" }} - {{- if .Values.sentinel.enabled }} - {{ if .Values.auth.enabled }}REDISCLI_AUTH="$REDIS_PASSWORD" {{ end }}redis-cli -h {{ template "common.names.fullname" . }} -p {{ .Values.sentinel.service.ports.redis }}{{ if .Values.tls.enabled }} --tls --cert /tmp/client.cert --key /tmp/client.key --cacert /tmp/CA.cert{{ end }} # Read only operations - {{ if .Values.auth.enabled }}REDISCLI_AUTH="$REDIS_PASSWORD" {{ end }}redis-cli -h {{ template "common.names.fullname" . }} -p {{ .Values.sentinel.service.ports.sentinel }}{{ if .Values.tls.enabled }} --tls --cert /tmp/client.cert --key /tmp/client.key --cacert /tmp/CA.cert{{ end }} # Sentinel access - {{- else }} - {{ if .Values.auth.enabled }}REDISCLI_AUTH="$REDIS_PASSWORD" {{ end }}redis-cli -h {{ printf "%s-master" (include "common.names.fullname" .) }}{{ if .Values.tls.enabled }} --tls --cert /tmp/client.cert --key /tmp/client.key --cacert /tmp/CA.cert{{ end }} - {{ if .Values.auth.enabled }}REDISCLI_AUTH="$REDIS_PASSWORD" {{ end }}redis-cli -h {{ printf "%s-replicas" (include "common.names.fullname" .) }}{{ if .Values.tls.enabled }} --tls --cert /tmp/client.cert --key /tmp/client.key --cacert /tmp/CA.cert{{ end }} - {{- end }} -{{- else }} - {{ if .Values.auth.enabled }}REDISCLI_AUTH="$REDIS_PASSWORD" {{ end }}redis-cli -h {{ template "common.names.fullname" . }}-master{{ if .Values.tls.enabled }} --tls --cert /tmp/client.cert --key /tmp/client.key --cacert /tmp/CA.cert{{ end }} -{{- end }} - -{{- if and (.Values.networkPolicy.enabled) (not .Values.networkPolicy.allowExternal) }} - -Note: Since NetworkPolicy is enabled, only pods with label {{ template "common.names.fullname" . }}-client=true" will be able to connect to redis. - -{{- else }} - -To connect to your database from outside the cluster execute the following commands: - -{{- if and (eq .Values.architecture "replication") .Values.sentinel.enabled }} -{{- if contains "NodePort" .Values.sentinel.service.type }} - - export NODE_IP=$(kubectl get nodes --namespace {{ include "common.names.namespace" . }} -o jsonpath="{.items[0].status.addresses[0].address}") - export NODE_PORT=$(kubectl get --namespace {{ include "common.names.namespace" . }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "common.names.fullname" . }}) - {{ if .Values.auth.enabled }}REDISCLI_AUTH="$REDIS_PASSWORD" {{ end }}redis-cli -h $NODE_IP -p $NODE_PORT {{- if .Values.tls.enabled }} --tls --cert /tmp/client.cert --key /tmp/client.key --cacert /tmp/CA.cert{{ end }} - -{{- else if contains "LoadBalancer" .Values.sentinel.service.type }} - - NOTE: It may take a few minutes for the LoadBalancer IP to be available. - Watch the status with: 'kubectl get svc --namespace {{ include "common.names.namespace" . }} -w {{ template "common.names.fullname" . }}' - - export SERVICE_IP=$(kubectl get svc --namespace {{ include "common.names.namespace" . }} {{ template "common.names.fullname" . }} --template "{{ "{{ range (index .status.loadBalancer.ingress 0) }}{{ . }}{{ end }}" }}") - {{ if .Values.auth.enabled }}REDISCLI_AUTH="$REDIS_PASSWORD" {{ end }}redis-cli -h $SERVICE_IP -p {{ .Values.sentinel.service.ports.redis }} {{- if .Values.tls.enabled }} --tls --cert /tmp/client.cert --key /tmp/client.key --cacert /tmp/CA.cert{{ end }} - -{{- else if contains "ClusterIP" .Values.sentinel.service.type }} - - kubectl port-forward --namespace {{ include "common.names.namespace" . }} svc/{{ template "common.names.fullname" . }} {{ .Values.sentinel.service.ports.redis }}:{{ .Values.sentinel.service.ports.redis }} & - {{ if .Values.auth.enabled }}REDISCLI_AUTH="$REDIS_PASSWORD" {{ end }}redis-cli -h 127.0.0.1 -p {{ .Values.sentinel.service.ports.redis }} {{- if .Values.tls.enabled }} --tls --cert /tmp/client.cert --key /tmp/client.key --cacert /tmp/CA.cert{{ end }} - -{{- end }} -{{- else }} -{{- if contains "NodePort" .Values.master.service.type }} - - export NODE_IP=$(kubectl get nodes --namespace {{ include "common.names.namespace" . }} -o jsonpath="{.items[0].status.addresses[0].address}") - export NODE_PORT=$(kubectl get --namespace {{ include "common.names.namespace" . }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ printf "%s-master" (include "common.names.fullname" .) }}) - {{ if .Values.auth.enabled }}REDISCLI_AUTH="$REDIS_PASSWORD" {{ end }}redis-cli -h $NODE_IP -p $NODE_PORT {{- if .Values.tls.enabled }} --tls --cert /tmp/client.cert --key /tmp/client.key --cacert /tmp/CA.cert{{ end }} - -{{- else if contains "LoadBalancer" .Values.master.service.type }} - - NOTE: It may take a few minutes for the LoadBalancer IP to be available. - Watch the status with: 'kubectl get svc --namespace {{ include "common.names.namespace" . }} -w {{ template "common.names.fullname" . }}' - - export SERVICE_IP=$(kubectl get svc --namespace {{ include "common.names.namespace" . }} {{ printf "%s-master" (include "common.names.fullname" .) }} --template "{{ "{{ range (index .status.loadBalancer.ingress 0) }}{{ . }}{{ end }}" }}") - {{ if .Values.auth.enabled }}REDISCLI_AUTH="$REDIS_PASSWORD" {{ end }}redis-cli -h $SERVICE_IP -p {{ .Values.master.service.ports.redis }} {{- if .Values.tls.enabled }} --tls --cert /tmp/client.cert --key /tmp/client.key --cacert /tmp/CA.cert{{ end }} - -{{- else if contains "ClusterIP" .Values.master.service.type }} - - kubectl port-forward --namespace {{ include "common.names.namespace" . }} svc/{{ printf "%s-master" (include "common.names.fullname" .) }} {{ .Values.master.service.ports.redis }}:{{ .Values.master.service.ports.redis }} & - {{ if .Values.auth.enabled }}REDISCLI_AUTH="$REDIS_PASSWORD" {{ end }}redis-cli -h 127.0.0.1 -p {{ .Values.master.service.ports.redis }} {{- if .Values.tls.enabled }} --tls --cert /tmp/client.cert --key /tmp/client.key --cacert /tmp/CA.cert{{ end }} - -{{- end }} -{{- end }} - -{{- end }} -{{- end }} -{{- include "redis.checkRollingTags" . }} -{{- include "common.warnings.rollingTag" .Values.volumePermissions.image }} -{{- include "common.warnings.rollingTag" .Values.sysctl.image }} -{{- include "redis.validateValues" . }} - -{{- if and (eq .Values.architecture "replication") .Values.sentinel.enabled (eq .Values.sentinel.service.type "NodePort") (not .Release.IsUpgrade ) }} -{{- if $.Values.sentinel.service.nodePorts.sentinel }} -No need to upgrade, ports and nodeports have been set from values -{{- else }} -#!#!#!#!#!#!#!# IMPORTANT #!#!#!#!#!#!#!# -YOU NEED TO PERFORM AN UPGRADE FOR THE SERVICES AND WORKLOAD TO BE CREATED -{{- end }} -{{- end }} -{{- $resourceSections := list "metrics" "replica" "sentinel" "sysctl" "volumePermissions" }} -{{- if not (and (eq .Values.architecture "replication") .Values.sentinel.enabled) }} - {{- $resourceSections = append $resourceSections "master" -}} -{{- end }} -{{- include "common.warnings.resources" (dict "sections" $resourceSections "context" $) }} -{{- include "common.warnings.modifiedImages" (dict "images" (list .Values.image .Values.sentinel.image .Values.metrics.image .Values.volumePermissions.image .Values.kubectl.image .Values.sysctl.image) "context" $) }} \ No newline at end of file diff --git a/charts/redis/redis/templates/_helpers.tpl b/charts/redis/redis/templates/_helpers.tpl deleted file mode 100644 index 0912b98..0000000 --- a/charts/redis/redis/templates/_helpers.tpl +++ /dev/null @@ -1,346 +0,0 @@ -{{/* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{/* vim: set filetype=mustache: */}} - -{{/* -Return the proper Redis image name -*/}} -{{- define "redis.image" -}} -{{ include "common.images.image" (dict "imageRoot" .Values.image "global" .Values.global) }} -{{- end -}} - -{{/* -Return the proper Redis Sentinel image name -*/}} -{{- define "redis.sentinel.image" -}} -{{ include "common.images.image" (dict "imageRoot" .Values.sentinel.image "global" .Values.global) }} -{{- end -}} - -{{/* -Return the proper image name (for the metrics image) -*/}} -{{- define "redis.metrics.image" -}} -{{ include "common.images.image" (dict "imageRoot" .Values.metrics.image "global" .Values.global) }} -{{- end -}} - -{{/* -Return the proper image name (for the init container volume-permissions image) -*/}} -{{- define "redis.volumePermissions.image" -}} -{{ include "common.images.image" (dict "imageRoot" .Values.volumePermissions.image "global" .Values.global) }} -{{- end -}} - -{{/* -Return kubectl image -*/}} -{{- define "redis.kubectl.image" -}} -{{ include "common.images.image" (dict "imageRoot" .Values.kubectl.image "global" .Values.global) }} -{{- end -}} - -{{/* -Return sysctl image -*/}} -{{- define "redis.sysctl.image" -}} -{{ include "common.images.image" (dict "imageRoot" .Values.sysctl.image "global" .Values.global) }} -{{- end -}} - -{{/* -Return the proper Docker Image Registry Secret Names -*/}} -{{- define "redis.imagePullSecrets" -}} -{{- include "common.images.renderPullSecrets" (dict "images" (list .Values.image .Values.sentinel.image .Values.metrics.image .Values.volumePermissions.image .Values.sysctl.image) "context" $) -}} -{{- end -}} - -{{/* -Return the appropriate apiVersion for networkpolicy. -*/}} -{{- define "networkPolicy.apiVersion" -}} -{{- if semverCompare ">=1.4-0, <1.7-0" .Capabilities.KubeVersion.GitVersion -}} -{{- print "extensions/v1beta1" -}} -{{- else -}} -{{- print "networking.k8s.io/v1" -}} -{{- end -}} -{{- end -}} - -{{/* -Return the appropriate apiGroup for PodSecurityPolicy. -*/}} -{{- define "podSecurityPolicy.apiGroup" -}} -{{- if semverCompare ">=1.14-0" .Capabilities.KubeVersion.GitVersion -}} -{{- print "policy" -}} -{{- else -}} -{{- print "extensions" -}} -{{- end -}} -{{- end -}} - -{{/* -Return true if a TLS secret object should be created -*/}} -{{- define "redis.createTlsSecret" -}} -{{- if and .Values.tls.enabled .Values.tls.autoGenerated (and (not .Values.tls.existingSecret) (not .Values.tls.certificatesSecret)) }} - {{- true -}} -{{- end -}} -{{- end -}} - -{{/* -Return the secret containing Redis TLS certificates -*/}} -{{- define "redis.tlsSecretName" -}} -{{- $secretName := coalesce .Values.tls.existingSecret .Values.tls.certificatesSecret -}} -{{- if $secretName -}} - {{- printf "%s" (tpl $secretName $) -}} -{{- else -}} - {{- printf "%s-crt" (include "common.names.fullname" .) -}} -{{- end -}} -{{- end -}} - -{{/* -Return the path to the cert file. -*/}} -{{- define "redis.tlsCert" -}} -{{- if (include "redis.createTlsSecret" . ) -}} - {{- printf "/opt/bitnami/redis/certs/%s" "tls.crt" -}} -{{- else -}} - {{- required "Certificate filename is required when TLS in enabled" .Values.tls.certFilename | printf "/opt/bitnami/redis/certs/%s" -}} -{{- end -}} -{{- end -}} - -{{/* -Return the path to the cert key file. -*/}} -{{- define "redis.tlsCertKey" -}} -{{- if (include "redis.createTlsSecret" . ) -}} - {{- printf "/opt/bitnami/redis/certs/%s" "tls.key" -}} -{{- else -}} - {{- required "Certificate Key filename is required when TLS in enabled" .Values.tls.certKeyFilename | printf "/opt/bitnami/redis/certs/%s" -}} -{{- end -}} -{{- end -}} - -{{/* -Return the path to the CA cert file. -*/}} -{{- define "redis.tlsCACert" -}} -{{- if (include "redis.createTlsSecret" . ) -}} - {{- printf "/opt/bitnami/redis/certs/%s" "ca.crt" -}} -{{- else -}} - {{- required "Certificate CA filename is required when TLS in enabled" .Values.tls.certCAFilename | printf "/opt/bitnami/redis/certs/%s" -}} -{{- end -}} -{{- end -}} - -{{/* -Return the path to the DH params file. -*/}} -{{- define "redis.tlsDHParams" -}} -{{- if .Values.tls.dhParamsFilename -}} -{{- printf "/opt/bitnami/redis/certs/%s" .Values.tls.dhParamsFilename -}} -{{- end -}} -{{- end -}} - -{{/* -Create the name of the shared service account to use -*/}} -{{- define "redis.serviceAccountName" -}} -{{- if .Values.serviceAccount.create -}} - {{ default (include "common.names.fullname" .) .Values.serviceAccount.name }} -{{- else -}} - {{ default "default" .Values.serviceAccount.name }} -{{- end -}} -{{- end -}} - -{{/* -Create the name of the master service account to use -*/}} -{{- define "redis.masterServiceAccountName" -}} -{{- if .Values.master.serviceAccount.create -}} - {{ default (printf "%s-master" (include "common.names.fullname" .)) .Values.master.serviceAccount.name }} -{{- else -}} - {{- if .Values.serviceAccount.create -}} - {{ template "redis.serviceAccountName" . }} - {{- else -}} - {{ default "default" .Values.master.serviceAccount.name }} - {{- end -}} -{{- end -}} -{{- end -}} - -{{/* -Create the name of the replicas service account to use -*/}} -{{- define "redis.replicaServiceAccountName" -}} -{{- if .Values.replica.serviceAccount.create -}} - {{ default (printf "%s-replica" (include "common.names.fullname" .)) .Values.replica.serviceAccount.name }} -{{- else -}} - {{- if .Values.serviceAccount.create -}} - {{ template "redis.serviceAccountName" . }} - {{- else -}} - {{ default "default" .Values.replica.serviceAccount.name }} - {{- end -}} -{{- end -}} -{{- end -}} - -{{/* -Return the configuration configmap name -*/}} -{{- define "redis.configmapName" -}} -{{- if .Values.existingConfigmap -}} - {{- printf "%s" (tpl .Values.existingConfigmap $) -}} -{{- else -}} - {{- printf "%s-configuration" (include "common.names.fullname" .) -}} -{{- end -}} -{{- end -}} - -{{/* -Return true if a configmap object should be created -*/}} -{{- define "redis.createConfigmap" -}} -{{- if empty .Values.existingConfigmap }} - {{- true -}} -{{- end -}} -{{- end -}} - -{{/* -Get the password secret. -*/}} -{{- define "redis.secretName" -}} -{{- if .Values.auth.existingSecret -}} -{{- printf "%s" (tpl .Values.auth.existingSecret $) -}} -{{- else -}} -{{- printf "%s" (include "common.names.fullname" .) -}} -{{- end -}} -{{- end -}} - -{{/* -Get the password key to be retrieved from Redis® secret. -*/}} -{{- define "redis.secretPasswordKey" -}} -{{- if and .Values.auth.existingSecret .Values.auth.existingSecretPasswordKey -}} -{{- printf "%s" (tpl .Values.auth.existingSecretPasswordKey $) -}} -{{- else -}} -{{- printf "redis-password" -}} -{{- end -}} -{{- end -}} - - -{{/* -Returns the available value for certain key in an existing secret (if it exists), -otherwise it generates a random value. -*/}} -{{- define "getValueFromSecret" }} - {{- $len := (default 16 .Length) | int -}} - {{- $obj := (lookup "v1" "Secret" .Namespace .Name).data -}} - {{- if $obj }} - {{- index $obj .Key | b64dec -}} - {{- else -}} - {{- randAlphaNum $len -}} - {{- end -}} -{{- end }} - -{{/* -Return Redis® password -*/}} -{{- define "redis.password" -}} -{{- if or .Values.auth.enabled .Values.global.redis.password }} - {{- if not (empty .Values.global.redis.password) }} - {{- .Values.global.redis.password -}} - {{- else if not (empty .Values.auth.password) -}} - {{- .Values.auth.password -}} - {{- else -}} - {{- include "getValueFromSecret" (dict "Namespace" (include "common.names.namespace" .) "Name" (include "redis.secretName" .) "Length" 10 "Key" (include "redis.secretPasswordKey" .)) -}} - {{- end -}} -{{- end -}} -{{- end }} - -{{/* Check if there are rolling tags in the images */}} -{{- define "redis.checkRollingTags" -}} -{{- include "common.warnings.rollingTag" .Values.image }} -{{- include "common.warnings.rollingTag" .Values.sentinel.image }} -{{- include "common.warnings.rollingTag" .Values.metrics.image }} -{{- end -}} - -{{/* -Compile all warnings into a single message, and call fail. -*/}} -{{- define "redis.validateValues" -}} -{{- $messages := list -}} -{{- $messages := append $messages (include "redis.validateValues.topologySpreadConstraints" .) -}} -{{- $messages := append $messages (include "redis.validateValues.architecture" .) -}} -{{- $messages := append $messages (include "redis.validateValues.podSecurityPolicy.create" .) -}} -{{- $messages := append $messages (include "redis.validateValues.tls" .) -}} -{{- $messages := append $messages (include "redis.validateValues.createMaster" .) -}} -{{- $messages := without $messages "" -}} -{{- $message := join "\n" $messages -}} - -{{- if $message -}} -{{- printf "\nVALUES VALIDATION:\n%s" $message | fail -}} -{{- end -}} -{{- end -}} - -{{/* Validate values of Redis® - spreadConstrainsts K8s version */}} -{{- define "redis.validateValues.topologySpreadConstraints" -}} -{{- if and (semverCompare "<1.16-0" .Capabilities.KubeVersion.GitVersion) .Values.replica.topologySpreadConstraints -}} -redis: topologySpreadConstraints - Pod Topology Spread Constraints are only available on K8s >= 1.16 - Find more information at https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/ -{{- end -}} -{{- end -}} - -{{/* Validate values of Redis® - must provide a valid architecture */}} -{{- define "redis.validateValues.architecture" -}} -{{- if and (ne .Values.architecture "standalone") (ne .Values.architecture "replication") -}} -redis: architecture - Invalid architecture selected. Valid values are "standalone" and - "replication". Please set a valid architecture (--set architecture="xxxx") -{{- end -}} -{{- if and .Values.sentinel.enabled (not (eq .Values.architecture "replication")) }} -redis: architecture - Using redis sentinel on standalone mode is not supported. - To deploy redis sentinel, please select the "replication" mode - (--set "architecture=replication,sentinel.enabled=true") -{{- end -}} -{{- end -}} - -{{/* Validate values of Redis® - PodSecurityPolicy create */}} -{{- define "redis.validateValues.podSecurityPolicy.create" -}} -{{- if and .Values.podSecurityPolicy.create (not .Values.podSecurityPolicy.enabled) }} -redis: podSecurityPolicy.create - In order to create PodSecurityPolicy, you also need to enable - podSecurityPolicy.enabled field -{{- end -}} -{{- end -}} - -{{/* Validate values of Redis® - TLS enabled */}} -{{- define "redis.validateValues.tls" -}} -{{- if and .Values.tls.enabled (not .Values.tls.autoGenerated) (not .Values.tls.existingSecret) (not .Values.tls.certificatesSecret) }} -redis: tls.enabled - In order to enable TLS, you also need to provide - an existing secret containing the TLS certificates or - enable auto-generated certificates. -{{- end -}} -{{- end -}} - -{{/* Validate values of Redis® - master service enabled */}} -{{- define "redis.validateValues.createMaster" -}} -{{- if and (or .Values.sentinel.masterService.enabled .Values.sentinel.service.createMaster) (or (not .Values.rbac.create) (not .Values.replica.automountServiceAccountToken) (not .Values.serviceAccount.create)) }} -redis: sentinel.masterService.enabled - In order to redirect requests only to the master pod via the service, you also need to - create rbac and serviceAccount. In addition, you need to enable - replica.automountServiceAccountToken. -{{- end -}} -{{- end -}} - -{{/* Define the suffix utilized for external-dns */}} -{{- define "redis.externalDNS.suffix" -}} -{{ printf "%s.%s" (include "common.names.fullname" .) .Values.useExternalDNS.suffix }} -{{- end -}} - -{{/* Compile all annotations utilized for external-dns */}} -{{- define "redis.externalDNS.annotations" -}} -{{- if and .Values.useExternalDNS.enabled .Values.useExternalDNS.annotationKey }} -{{ .Values.useExternalDNS.annotationKey }}hostname: {{ include "redis.externalDNS.suffix" . }} -{{- range $key, $val := .Values.useExternalDNS.additionalAnnotations }} -{{ $.Values.useExternalDNS.annotationKey }}{{ $key }}: {{ $val | quote }} -{{- end }} -{{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/configmap.yaml b/charts/redis/redis/templates/configmap.yaml deleted file mode 100644 index 22df358..0000000 --- a/charts/redis/redis/templates/configmap.yaml +++ /dev/null @@ -1,64 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if (include "redis.createConfigmap" .) }} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ printf "%s-configuration" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: - redis.conf: |- - # User-supplied common configuration: - {{- if .Values.commonConfiguration }} - {{- include "common.tplvalues.render" ( dict "value" .Values.commonConfiguration "context" $ ) | nindent 4 }} - {{- end }} - # End of common configuration - master.conf: |- - dir {{ .Values.master.persistence.path }} - # User-supplied master configuration: - {{- if .Values.master.configuration }} - {{- include "common.tplvalues.render" ( dict "value" .Values.master.configuration "context" $ ) | nindent 4 }} - {{- end }} - {{- if .Values.master.disableCommands }} - {{- range .Values.master.disableCommands }} - rename-command {{ . }} "" - {{- end }} - {{- end }} - # End of master configuration - replica.conf: |- - dir {{ .Values.replica.persistence.path }} - # User-supplied replica configuration: - {{- if .Values.replica.configuration }} - {{- include "common.tplvalues.render" ( dict "value" .Values.replica.configuration "context" $ ) | nindent 4 }} - {{- end }} - {{- if .Values.replica.disableCommands }} - {{- range .Values.replica.disableCommands }} - rename-command {{ . }} "" - {{- end }} - {{- end }} - # End of replica configuration - {{- if .Values.sentinel.enabled }} - sentinel.conf: |- - dir "/tmp" - port {{ .Values.sentinel.containerPorts.sentinel }} - sentinel monitor {{ .Values.sentinel.masterSet }} {{ template "common.names.fullname" . }}-node-0.{{ template "common.names.fullname" . }}-headless.{{ include "common.names.namespace" . }}.svc.{{ .Values.clusterDomain }} {{ .Values.sentinel.service.ports.redis }} {{ .Values.sentinel.quorum }} - sentinel down-after-milliseconds {{ .Values.sentinel.masterSet }} {{ .Values.sentinel.downAfterMilliseconds }} - sentinel failover-timeout {{ .Values.sentinel.masterSet }} {{ .Values.sentinel.failoverTimeout }} - sentinel parallel-syncs {{ .Values.sentinel.masterSet }} {{ .Values.sentinel.parallelSyncs }} - {{- if or .Values.sentinel.masterService.enabled .Values.sentinel.service.createMaster }} - sentinel client-reconfig-script {{ .Values.sentinel.masterSet }} /opt/bitnami/scripts/start-scripts/push-master-label.sh - {{- end }} - # User-supplied sentinel configuration: - {{- if .Values.sentinel.configuration }} - {{- include "common.tplvalues.render" ( dict "value" .Values.sentinel.configuration "context" $ ) | nindent 4 }} - {{- end }} - # End of sentinel configuration - {{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/extra-list.yaml b/charts/redis/redis/templates/extra-list.yaml deleted file mode 100644 index 329f5c6..0000000 --- a/charts/redis/redis/templates/extra-list.yaml +++ /dev/null @@ -1,9 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- range .Values.extraDeploy }} ---- -{{ include "common.tplvalues.render" (dict "value" . "context" $) }} -{{- end }} diff --git a/charts/redis/redis/templates/headless-svc.yaml b/charts/redis/redis/templates/headless-svc.yaml deleted file mode 100644 index 280d9de..0000000 --- a/charts/redis/redis/templates/headless-svc.yaml +++ /dev/null @@ -1,35 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -apiVersion: v1 -kind: Service -metadata: - name: {{ printf "%s-headless" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if or .Values.sentinel.service.headless.annotations .Values.commonAnnotations (include "redis.externalDNS.annotations" .) }} - annotations: - {{- if or .Values.sentinel.service.headless.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.sentinel.service.headless.annotations .Values.commonAnnotations ) "context" . ) }} - {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} - {{- include "redis.externalDNS.annotations" . | nindent 4 }} - {{- end }} -spec: - type: ClusterIP - clusterIP: None - {{- if .Values.sentinel.enabled }} - publishNotReadyAddresses: true - {{- end }} - ports: - - name: tcp-redis - port: {{ if .Values.sentinel.enabled }}{{ .Values.sentinel.service.ports.redis }}{{ else }}{{ .Values.master.service.ports.redis }}{{ end }} - targetPort: redis - {{- if .Values.sentinel.enabled }} - - name: tcp-sentinel - port: {{ .Values.sentinel.service.ports.sentinel }} - targetPort: redis-sentinel - {{- end }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} diff --git a/charts/redis/redis/templates/health-configmap.yaml b/charts/redis/redis/templates/health-configmap.yaml deleted file mode 100644 index bdd72a0..0000000 --- a/charts/redis/redis/templates/health-configmap.yaml +++ /dev/null @@ -1,194 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ printf "%s-health" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: - ping_readiness_local.sh: |- - #!/bin/bash - - [[ -f $REDIS_PASSWORD_FILE ]] && export REDIS_PASSWORD="$(< "${REDIS_PASSWORD_FILE}")" - [[ -n "$REDIS_PASSWORD" ]] && export REDISCLI_AUTH="$REDIS_PASSWORD" - response=$( - timeout -s 15 $1 \ - redis-cli \ - -h localhost \ -{{- if .Values.tls.enabled }} - -p $REDIS_TLS_PORT \ - --tls \ - --cacert {{ template "redis.tlsCACert" . }} \ - {{- if .Values.tls.authClients }} - --cert {{ template "redis.tlsCert" . }} \ - --key {{ template "redis.tlsCertKey" . }} \ - {{- end }} -{{- else }} - -p $REDIS_PORT \ -{{- end }} - ping - ) - if [ "$?" -eq "124" ]; then - echo "Timed out" - exit 1 - fi - if [ "$response" != "PONG" ]; then - echo "$response" - exit 1 - fi - ping_liveness_local.sh: |- - #!/bin/bash - - [[ -f $REDIS_PASSWORD_FILE ]] && export REDIS_PASSWORD="$(< "${REDIS_PASSWORD_FILE}")" - [[ -n "$REDIS_PASSWORD" ]] && export REDISCLI_AUTH="$REDIS_PASSWORD" - response=$( - timeout -s 15 $1 \ - redis-cli \ - -h localhost \ -{{- if .Values.tls.enabled }} - -p $REDIS_TLS_PORT \ - --tls \ - --cacert {{ template "redis.tlsCACert" . }} \ - {{- if .Values.tls.authClients }} - --cert {{ template "redis.tlsCert" . }} \ - --key {{ template "redis.tlsCertKey" . }} \ - {{- end }} -{{- else }} - -p $REDIS_PORT \ -{{- end }} - ping - ) - if [ "$?" -eq "124" ]; then - echo "Timed out" - exit 1 - fi - responseFirstWord=$(echo $response | head -n1 | awk '{print $1;}') - if [ "$response" != "PONG" ] && [ "$responseFirstWord" != "LOADING" ] && [ "$responseFirstWord" != "MASTERDOWN" ]; then - echo "$response" - exit 1 - fi -{{- if .Values.sentinel.enabled }} - ping_sentinel.sh: |- - #!/bin/bash - -{{- if .Values.auth.sentinel }} - [[ -f $REDIS_PASSWORD_FILE ]] && export REDIS_PASSWORD="$(< "${REDIS_PASSWORD_FILE}")" - [[ -n "$REDIS_PASSWORD" ]] && export REDISCLI_AUTH="$REDIS_PASSWORD" -{{- end }} - response=$( - timeout -s 15 $1 \ - redis-cli \ - -h localhost \ -{{- if .Values.tls.enabled }} - -p $REDIS_SENTINEL_TLS_PORT_NUMBER \ - --tls \ - --cacert "$REDIS_SENTINEL_TLS_CA_FILE" \ - {{- if .Values.tls.authClients }} - --cert "$REDIS_SENTINEL_TLS_CERT_FILE" \ - --key "$REDIS_SENTINEL_TLS_KEY_FILE" \ - {{- end }} -{{- else }} - -p $REDIS_SENTINEL_PORT \ -{{- end }} - ping - ) - if [ "$?" -eq "124" ]; then - echo "Timed out" - exit 1 - fi - if [ "$response" != "PONG" ]; then - echo "$response" - exit 1 - fi - parse_sentinels.awk: |- - /ip/ {FOUND_IP=1} - /port/ {FOUND_PORT=1} - /runid/ {FOUND_RUNID=1} - !/ip|port|runid/ { - if (FOUND_IP==1) { - IP=$1; FOUND_IP=0; - } - else if (FOUND_PORT==1) { - PORT=$1; - FOUND_PORT=0; - } else if (FOUND_RUNID==1) { - printf "\nsentinel known-sentinel {{ .Values.sentinel.masterSet }} %s %s %s", IP, PORT, $0; FOUND_RUNID=0; - } - } -{{- end }} - ping_readiness_master.sh: |- - #!/bin/bash - - [[ -f $REDIS_MASTER_PASSWORD_FILE ]] && export REDIS_MASTER_PASSWORD="$(< "${REDIS_MASTER_PASSWORD_FILE}")" - [[ -n "$REDIS_MASTER_PASSWORD" ]] && export REDISCLI_AUTH="$REDIS_MASTER_PASSWORD" - response=$( - timeout -s 15 $1 \ - redis-cli \ - -h $REDIS_MASTER_HOST \ - -p $REDIS_MASTER_PORT_NUMBER \ -{{- if .Values.tls.enabled }} - --tls \ - --cacert {{ template "redis.tlsCACert" . }} \ - {{- if .Values.tls.authClients }} - --cert {{ template "redis.tlsCert" . }} \ - --key {{ template "redis.tlsCertKey" . }} \ - {{- end }} -{{- end }} - ping - ) - if [ "$?" -eq "124" ]; then - echo "Timed out" - exit 1 - fi - if [ "$response" != "PONG" ]; then - echo "$response" - exit 1 - fi - ping_liveness_master.sh: |- - #!/bin/bash - - [[ -f $REDIS_MASTER_PASSWORD_FILE ]] && export REDIS_MASTER_PASSWORD="$(< "${REDIS_MASTER_PASSWORD_FILE}")" - [[ -n "$REDIS_MASTER_PASSWORD" ]] && export REDISCLI_AUTH="$REDIS_MASTER_PASSWORD" - response=$( - timeout -s 15 $1 \ - redis-cli \ - -h $REDIS_MASTER_HOST \ - -p $REDIS_MASTER_PORT_NUMBER \ -{{- if .Values.tls.enabled }} - --tls \ - --cacert {{ template "redis.tlsCACert" . }} \ - {{- if .Values.tls.authClients }} - --cert {{ template "redis.tlsCert" . }} \ - --key {{ template "redis.tlsCertKey" . }} \ - {{- end }} -{{- end }} - ping - ) - if [ "$?" -eq "124" ]; then - echo "Timed out" - exit 1 - fi - responseFirstWord=$(echo $response | head -n1 | awk '{print $1;}') - if [ "$response" != "PONG" ] && [ "$responseFirstWord" != "LOADING" ]; then - echo "$response" - exit 1 - fi - ping_readiness_local_and_master.sh: |- - script_dir="$(dirname "$0")" - exit_status=0 - "$script_dir/ping_readiness_local.sh" $1 || exit_status=$? - "$script_dir/ping_readiness_master.sh" $1 || exit_status=$? - exit $exit_status - ping_liveness_local_and_master.sh: |- - script_dir="$(dirname "$0")" - exit_status=0 - "$script_dir/ping_liveness_local.sh" $1 || exit_status=$? - "$script_dir/ping_liveness_master.sh" $1 || exit_status=$? - exit $exit_status diff --git a/charts/redis/redis/templates/master/application.yaml b/charts/redis/redis/templates/master/application.yaml deleted file mode 100644 index b17a240..0000000 --- a/charts/redis/redis/templates/master/application.yaml +++ /dev/null @@ -1,546 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if gt (int64 .Values.master.count) 0 -}} -{{- if or (not (eq .Values.architecture "replication")) (not .Values.sentinel.enabled) }} -apiVersion: {{ include "common.capabilities.statefulset.apiVersion" . }} -kind: {{ .Values.master.kind }} -metadata: - name: {{ printf "%s-master" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: master - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- if not (eq .Values.master.kind "DaemonSet") }} - replicas: {{ .Values.master.count }} - {{- end }} - revisionHistoryLimit: {{ .Values.master.revisionHistoryLimit }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.master.podLabels .Values.commonLabels ) "context" . ) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: master - {{- if (eq .Values.master.kind "StatefulSet") }} - serviceName: {{ printf "%s-headless" (include "common.names.fullname" .) }} - {{- end }} - {{- if .Values.master.updateStrategy }} - {{- if (eq .Values.master.kind "Deployment") }} - strategy: {{- toYaml .Values.master.updateStrategy | nindent 4 }} - {{- else }} - updateStrategy: {{- toYaml .Values.master.updateStrategy | nindent 4 }} - {{- end }} - {{- if and .Values.master.minReadySeconds (semverCompare ">= 1.23-0" (include "common.capabilities.kubeVersion" .)) }} - minReadySeconds: {{ .Values.master.minReadySeconds }} - {{- end }} - {{- end }} - template: - metadata: - labels: {{- include "common.labels.standard" ( dict "customLabels" $podLabels "context" $ ) | nindent 8 }} - app.kubernetes.io/component: master - {{- if and .Values.metrics.enabled .Values.metrics.podLabels }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.podLabels "context" $ ) | nindent 8 }} - {{- end }} - annotations: - {{- if (include "redis.createConfigmap" .) }} - checksum/configmap: {{ pick ( include (print $.Template.BasePath "/configmap.yaml") . | fromYaml ) "data" | toYaml | sha256sum }} - {{- end }} - checksum/health: {{ pick ( include (print $.Template.BasePath "/health-configmap.yaml") . | fromYaml ) "data" | toYaml | sha256sum }} - checksum/scripts: {{ pick ( include (print $.Template.BasePath "/scripts-configmap.yaml") . | fromYaml ) "data" | toYaml | sha256sum }} - checksum/secret: {{ pick ( include (print $.Template.BasePath "/secret.yaml") . | fromYaml ) "data" | toYaml | sha256sum }} - {{- if .Values.master.podAnnotations }} - {{- include "common.tplvalues.render" ( dict "value" .Values.master.podAnnotations "context" $ ) | nindent 8 }} - {{- end }} - {{- if and .Values.metrics.enabled .Values.metrics.podAnnotations }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.podAnnotations "context" $ ) | nindent 8 }} - {{- end }} - spec: - {{- include "redis.imagePullSecrets" . | nindent 6 }} - {{- if .Values.master.hostAliases }} - hostAliases: {{- include "common.tplvalues.render" (dict "value" .Values.master.hostAliases "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.master.podSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.master.podSecurityContext "context" $) | nindent 8 }} - {{- end }} - serviceAccountName: {{ template "redis.masterServiceAccountName" . }} - automountServiceAccountToken: {{ .Values.master.automountServiceAccountToken }} - {{- if .Values.master.priorityClassName }} - priorityClassName: {{ .Values.master.priorityClassName | quote }} - {{- end }} - {{- if .Values.master.affinity }} - affinity: {{- include "common.tplvalues.render" (dict "value" .Values.master.affinity "context" $) | nindent 8 }} - {{- else }} - affinity: - podAffinity: {{- include "common.affinities.pods" (dict "type" .Values.master.podAffinityPreset "component" "master" "customLabels" $podLabels "context" $) | nindent 10 }} - podAntiAffinity: {{- include "common.affinities.pods" (dict "type" .Values.master.podAntiAffinityPreset "component" "master" "customLabels" $podLabels "context" $) | nindent 10 }} - nodeAffinity: {{- include "common.affinities.nodes" (dict "type" .Values.master.nodeAffinityPreset.type "key" .Values.master.nodeAffinityPreset.key "values" .Values.master.nodeAffinityPreset.values) | nindent 10 }} - {{- end }} - {{- if .Values.master.nodeSelector }} - nodeSelector: {{- include "common.tplvalues.render" (dict "value" .Values.master.nodeSelector "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.master.tolerations }} - tolerations: {{- include "common.tplvalues.render" (dict "value" .Values.master.tolerations "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.master.topologySpreadConstraints }} - topologySpreadConstraints: {{- include "common.tplvalues.render" (dict "value" .Values.master.topologySpreadConstraints "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.master.shareProcessNamespace }} - shareProcessNamespace: {{ .Values.master.shareProcessNamespace }} - {{- end }} - {{- if .Values.master.schedulerName }} - schedulerName: {{ .Values.master.schedulerName | quote }} - {{- end }} - {{- if .Values.master.dnsPolicy }} - dnsPolicy: {{ .Values.master.dnsPolicy }} - {{- end }} - {{- if .Values.master.dnsConfig }} - dnsConfig: {{- include "common.tplvalues.render" (dict "value" .Values.master.dnsConfig "context" $) | nindent 8 }} - {{- end }} - enableServiceLinks: {{ .Values.master.enableServiceLinks }} - terminationGracePeriodSeconds: {{ .Values.master.terminationGracePeriodSeconds }} - containers: - - name: redis - image: {{ template "redis.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy | quote }} - {{- if .Values.master.lifecycleHooks }} - lifecycle: {{- include "common.tplvalues.render" (dict "value" .Values.master.lifecycleHooks "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.master.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.master.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - {{- else if .Values.master.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.master.command "context" $) | nindent 12 }} - {{- else }} - command: - - /bin/bash - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- else if .Values.master.args }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.master.args "context" $) | nindent 12 }} - {{- else }} - args: - - -c - - /opt/bitnami/scripts/start-scripts/start-master.sh - {{- end }} - env: - - name: BITNAMI_DEBUG - value: {{ ternary "true" "false" (or .Values.image.debug .Values.diagnosticMode.enabled) | quote }} - - name: REDIS_REPLICATION_MODE - value: master - - name: ALLOW_EMPTY_PASSWORD - value: {{ ternary "no" "yes" .Values.auth.enabled | quote }} - {{- if .Values.auth.enabled }} - {{- if .Values.auth.usePasswordFiles }} - - name: REDIS_PASSWORD_FILE - value: "/opt/bitnami/redis/secrets/redis-password" - {{- else }} - - name: REDIS_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "redis.secretName" . }} - key: {{ template "redis.secretPasswordKey" . }} - {{- end }} - {{- end }} - - name: REDIS_TLS_ENABLED - value: {{ ternary "yes" "no" .Values.tls.enabled | quote }} - {{- if .Values.tls.enabled }} - - name: REDIS_TLS_PORT - value: {{ .Values.master.containerPorts.redis | quote }} - - name: REDIS_TLS_AUTH_CLIENTS - value: {{ ternary "yes" "no" .Values.tls.authClients | quote }} - - name: REDIS_TLS_CERT_FILE - value: {{ template "redis.tlsCert" . }} - - name: REDIS_TLS_KEY_FILE - value: {{ template "redis.tlsCertKey" . }} - - name: REDIS_TLS_CA_FILE - value: {{ template "redis.tlsCACert" . }} - {{- if .Values.tls.dhParamsFilename }} - - name: REDIS_TLS_DH_PARAMS_FILE - value: {{ template "redis.tlsDHParams" . }} - {{- end }} - {{- else }} - - name: REDIS_PORT - value: {{ .Values.master.containerPorts.redis | quote }} - {{- end }} - {{- if .Values.master.extraEnvVars }} - {{- include "common.tplvalues.render" (dict "value" .Values.master.extraEnvVars "context" $) | nindent 12 }} - {{- end }} - {{- if or .Values.master.extraEnvVarsCM .Values.master.extraEnvVarsSecret }} - envFrom: - {{- if .Values.master.extraEnvVarsCM }} - - configMapRef: - name: {{ .Values.master.extraEnvVarsCM }} - {{- end }} - {{- if .Values.master.extraEnvVarsSecret }} - - secretRef: - name: {{ .Values.master.extraEnvVarsSecret }} - {{- end }} - {{- end }} - ports: - - name: redis - containerPort: {{ .Values.master.containerPorts.redis }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.master.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.master.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.master.startupProbe.enabled }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.master.startupProbe "enabled") "context" $) | nindent 12 }} - tcpSocket: - port: redis - {{- end }} - {{- if .Values.master.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.master.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.master.livenessProbe.enabled }} - livenessProbe: - initialDelaySeconds: {{ .Values.master.livenessProbe.initialDelaySeconds }} - periodSeconds: {{ .Values.master.livenessProbe.periodSeconds }} - # One second longer than command timeout should prevent generation of zombie processes. - timeoutSeconds: {{ add1 .Values.master.livenessProbe.timeoutSeconds }} - successThreshold: {{ .Values.master.livenessProbe.successThreshold }} - failureThreshold: {{ .Values.master.livenessProbe.failureThreshold }} - exec: - command: - - sh - - -c - - /health/ping_liveness_local.sh {{ .Values.master.livenessProbe.timeoutSeconds }} - {{- end }} - {{- if .Values.master.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.master.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.master.readinessProbe.enabled }} - readinessProbe: - initialDelaySeconds: {{ .Values.master.readinessProbe.initialDelaySeconds }} - periodSeconds: {{ .Values.master.readinessProbe.periodSeconds }} - timeoutSeconds: {{ add1 .Values.master.readinessProbe.timeoutSeconds }} - successThreshold: {{ .Values.master.readinessProbe.successThreshold }} - failureThreshold: {{ .Values.master.readinessProbe.failureThreshold }} - exec: - command: - - sh - - -c - - /health/ping_readiness_local.sh {{ .Values.master.readinessProbe.timeoutSeconds }} - {{- end }} - {{- end }} - {{- if .Values.master.resources }} - resources: {{- toYaml .Values.master.resources | nindent 12 }} - {{- else if ne .Values.master.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.master.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: start-scripts - mountPath: /opt/bitnami/scripts/start-scripts - - name: health - mountPath: /health - {{- if .Values.auth.usePasswordFiles }} - - name: redis-password - mountPath: /opt/bitnami/redis/secrets/ - {{- end }} - - name: redis-data - mountPath: {{ .Values.master.persistence.path }} - {{- if .Values.master.persistence.subPath }} - subPath: {{ .Values.master.persistence.subPath }} - {{- else if .Values.master.persistence.subPathExpr }} - subPathExpr: {{ .Values.master.persistence.subPathExpr }} - {{- end }} - - name: config - mountPath: /opt/bitnami/redis/mounted-etc - - name: empty-dir - mountPath: /opt/bitnami/redis/etc/ - subPath: app-conf-dir - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - {{- if .Values.tls.enabled }} - - name: redis-certificates - mountPath: /opt/bitnami/redis/certs - readOnly: true - {{- end }} - {{- if .Values.master.extraVolumeMounts }} - {{- include "common.tplvalues.render" ( dict "value" .Values.master.extraVolumeMounts "context" $ ) | nindent 12 }} - {{- end }} - {{- if .Values.metrics.enabled }} - - name: metrics - image: {{ include "redis.metrics.image" . }} - imagePullPolicy: {{ .Values.metrics.image.pullPolicy | quote }} - {{- if .Values.metrics.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.metrics.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - {{- else if .Values.metrics.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.command "context" $) | nindent 12 }} - {{- else }} - command: - - /bin/bash - - -c - - | - if [[ -f '/secrets/redis-password' ]]; then - export REDIS_PASSWORD=$(cat /secrets/redis-password) - fi - redis_exporter{{- range $key, $value := .Values.metrics.extraArgs }} --{{ $key }}={{ $value }}{{- end }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- end }} - env: - - name: REDIS_ALIAS - value: {{ template "common.names.fullname" . }} - - name: REDIS_EXPORTER_WEB_LISTEN_ADDRESS - value: {{ printf ":%v" .Values.metrics.containerPorts.http }} - {{- if .Values.auth.enabled }} - - name: REDIS_USER - value: default - {{- if (not .Values.auth.usePasswordFiles) }} - - name: REDIS_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "redis.secretName" . }} - key: {{ template "redis.secretPasswordKey" . }} - {{- end }} - {{- end }} - {{- if .Values.tls.enabled }} - - name: REDIS_ADDR - value: rediss://{{ .Values.metrics.redisTargetHost }}:{{ .Values.master.containerPorts.redis }} - {{- if .Values.tls.authClients }} - - name: REDIS_EXPORTER_TLS_CLIENT_KEY_FILE - value: {{ template "redis.tlsCertKey" . }} - - name: REDIS_EXPORTER_TLS_CLIENT_CERT_FILE - value: {{ template "redis.tlsCert" . }} - {{- end }} - - name: REDIS_EXPORTER_TLS_CA_CERT_FILE - value: {{ template "redis.tlsCACert" . }} - {{- end }} - {{- if .Values.metrics.extraEnvVars }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.extraEnvVars "context" $) | nindent 12 }} - {{- end }} - ports: - - name: metrics - containerPort: {{ .Values.metrics.containerPorts.http }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.metrics.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.startupProbe.enabled }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.startupProbe "enabled") "context" $) | nindent 12 }} - tcpSocket: - port: metrics - {{- end }} - {{- if .Values.metrics.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.livenessProbe.enabled }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.livenessProbe "enabled") "context" $) | nindent 12 }} - tcpSocket: - port: metrics - {{- end }} - {{- if .Values.metrics.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.readinessProbe.enabled }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.readinessProbe "enabled") "context" $) | nindent 12 }} - httpGet: - path: / - port: metrics - {{- end }} - {{- end }} - {{- if .Values.metrics.resources }} - resources: {{- toYaml .Values.metrics.resources | nindent 12 }} - {{- else if ne .Values.metrics.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.metrics.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: app-tmp-dir - {{- if .Values.auth.usePasswordFiles }} - - name: redis-password - mountPath: /secrets/ - {{- end }} - {{- if .Values.tls.enabled }} - - name: redis-certificates - mountPath: /opt/bitnami/redis/certs - readOnly: true - {{- end }} - {{- if .Values.metrics.extraVolumeMounts }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.extraVolumeMounts "context" $ ) | nindent 12 }} - {{- end }} - {{- end }} - {{- if .Values.master.sidecars }} - {{- include "common.tplvalues.render" (dict "value" .Values.master.sidecars "context" $) | nindent 8 }} - {{- end }} - {{- $needsVolumePermissions := and .Values.volumePermissions.enabled .Values.master.persistence.enabled .Values.master.podSecurityContext.enabled .Values.master.containerSecurityContext.enabled }} - {{- if or .Values.master.initContainers $needsVolumePermissions .Values.sysctl.enabled }} - initContainers: - {{- if .Values.master.initContainers }} - {{- include "common.tplvalues.render" (dict "value" .Values.master.initContainers "context" $) | nindent 8 }} - {{- end }} - {{- if $needsVolumePermissions }} - - name: volume-permissions - image: {{ include "redis.volumePermissions.image" . }} - imagePullPolicy: {{ .Values.volumePermissions.image.pullPolicy | quote }} - command: - - /bin/bash - - -ec - - | - {{- if eq ( toString ( .Values.volumePermissions.containerSecurityContext.runAsUser )) "auto" }} - chown -R `id -u`:`id -G | cut -d " " -f2` {{ .Values.master.persistence.path }} - {{- else }} - chown -R {{ .Values.master.containerSecurityContext.runAsUser }}:{{ .Values.master.podSecurityContext.fsGroup }} {{ .Values.master.persistence.path }} - {{- end }} - {{- if eq ( toString ( .Values.volumePermissions.containerSecurityContext.runAsUser )) "auto" }} - securityContext: {{- omit .Values.volumePermissions.containerSecurityContext "runAsUser" | toYaml | nindent 12 }} - {{- else }} - securityContext: {{- .Values.volumePermissions.containerSecurityContext | toYaml | nindent 12 }} - {{- end }} - {{- if .Values.volumePermissions.resources }} - resources: {{- toYaml .Values.volumePermissions.resources | nindent 12 }} - {{- else if ne .Values.volumePermissions.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.volumePermissions.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: redis-data - mountPath: {{ .Values.master.persistence.path }} - {{- if .Values.master.persistence.subPath }} - subPath: {{ .Values.master.persistence.subPath }} - {{- else if .Values.master.persistence.subPathExpr }} - subPathExpr: {{ .Values.master.persistence.subPathExpr }} - {{- end }} - {{- end }} - {{- if .Values.sysctl.enabled }} - - name: init-sysctl - image: {{ include "redis.sysctl.image" . }} - imagePullPolicy: {{ default "" .Values.sysctl.image.pullPolicy | quote }} - securityContext: - privileged: true - runAsUser: 0 - {{- if .Values.sysctl.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.sysctl.command "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.sysctl.resources }} - resources: {{- toYaml .Values.sysctl.resources | nindent 12 }} - {{- else if ne .Values.sysctl.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.sysctl.resourcesPreset) | nindent 12 }} - {{- end }} - {{- if .Values.sysctl.mountHostSys }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: host-sys - mountPath: /host-sys - {{- end }} - {{- end }} - {{- end }} - volumes: - - name: start-scripts - configMap: - name: {{ printf "%s-scripts" (include "common.names.fullname" .) }} - defaultMode: 0755 - - name: health - configMap: - name: {{ printf "%s-health" (include "common.names.fullname" .) }} - defaultMode: 0755 - {{- if .Values.auth.usePasswordFiles }} - - name: redis-password - {{ if .Values.auth.usePasswordFileFromSecret }} - secret: - secretName: {{ template "redis.secretName" . }} - items: - - key: {{ template "redis.secretPasswordKey" . }} - path: redis-password - {{- else }} - emptyDir: {} - {{- end }} - {{- end }} - - name: config - configMap: - name: {{ include "redis.configmapName" . }} - {{- if .Values.sysctl.mountHostSys }} - - name: host-sys - hostPath: - path: /sys - {{- end }} - - name: empty-dir - {{- if or .Values.master.persistence.medium .Values.master.persistence.sizeLimit }} - emptyDir: - {{- if .Values.master.persistence.medium }} - medium: {{ .Values.master.persistence.medium | quote }} - {{- end }} - {{- if .Values.master.persistence.sizeLimit }} - sizeLimit: {{ .Values.master.persistence.sizeLimit | quote }} - {{- end }} - {{- else }} - emptyDir: {} - {{- end }} - {{- if .Values.tls.enabled }} - - name: redis-certificates - secret: - secretName: {{ include "redis.tlsSecretName" . }} - defaultMode: 256 - {{- end }} - {{- if .Values.master.extraVolumes }} - {{- include "common.tplvalues.render" ( dict "value" .Values.master.extraVolumes "context" $ ) | nindent 8 }} - {{- end }} - {{- if .Values.metrics.extraVolumes }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.extraVolumes "context" $ ) | nindent 8 }} - {{- end }} - {{- if or (not .Values.master.persistence.enabled) (eq .Values.master.kind "DaemonSet") }} - - name: redis-data - {{- if or .Values.master.persistence.medium .Values.master.persistence.sizeLimit }} - emptyDir: - {{- if .Values.master.persistence.medium }} - medium: {{ .Values.master.persistence.medium | quote }} - {{- end }} - {{- if .Values.master.persistence.sizeLimit }} - sizeLimit: {{ .Values.master.persistence.sizeLimit | quote }} - {{- end }} - {{- else }} - emptyDir: {} - {{- end }} - {{- else if .Values.master.persistence.existingClaim }} - - name: redis-data - persistentVolumeClaim: - claimName: {{ printf "%s" (tpl .Values.master.persistence.existingClaim .) }} - {{- else if (eq .Values.master.kind "Deployment") }} - - name: redis-data - persistentVolumeClaim: - claimName: {{ printf "redis-data-%s-master" (include "common.names.fullname" .) }} - {{- else }} - {{- if .Values.master.persistentVolumeClaimRetentionPolicy.enabled }} - persistentVolumeClaimRetentionPolicy: - whenDeleted: {{ .Values.master.persistentVolumeClaimRetentionPolicy.whenDeleted }} - whenScaled: {{ .Values.master.persistentVolumeClaimRetentionPolicy.whenScaled }} - {{- end }} - volumeClaimTemplates: - - apiVersion: v1 - kind: PersistentVolumeClaim - metadata: - name: redis-data - {{- $claimLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.master.persistence.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.matchLabels" ( dict "customLabels" $claimLabels "context" $ ) | nindent 10 }} - app.kubernetes.io/component: master - {{- if .Values.master.persistence.annotations }} - annotations: {{- toYaml .Values.master.persistence.annotations | nindent 10 }} - {{- end }} - spec: - accessModes: - {{- range .Values.master.persistence.accessModes }} - - {{ . | quote }} - {{- end }} - resources: - requests: - storage: {{ .Values.master.persistence.size | quote }} - {{- if .Values.master.persistence.selector }} - selector: {{- include "common.tplvalues.render" (dict "value" .Values.master.persistence.selector "context" $) | nindent 10 }} - {{- end }} - {{- if .Values.master.persistence.dataSource }} - dataSource: {{- include "common.tplvalues.render" (dict "value" .Values.master.persistence.dataSource "context" $) | nindent 10 }} - {{- end }} - {{- include "common.storage.class" (dict "persistence" .Values.master.persistence "global" .Values.global) | nindent 8 }} - {{- end }} -{{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/master/pdb.yaml b/charts/redis/redis/templates/master/pdb.yaml deleted file mode 100644 index dab636d..0000000 --- a/charts/redis/redis/templates/master/pdb.yaml +++ /dev/null @@ -1,27 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} -{{- $pdb := coalesce .Values.pdb .Values.master.pdb }} -{{- if and $pdb.create (gt (int64 .Values.master.count) 0) (or (not (eq .Values.architecture "replication")) (not .Values.sentinel.enabled)) }} -apiVersion: {{ include "common.capabilities.policy.apiVersion" . }} -kind: PodDisruptionBudget -metadata: - name: {{ printf "%s-master" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: master - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- if $pdb.minAvailable }} - minAvailable: {{ $pdb.minAvailable }} - {{- end }} - {{- if or $pdb.maxUnavailable (not $pdb.minAvailable)}} - maxUnavailable: {{ $pdb.maxUnavailable | default 1 }} - {{- end }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: master -{{- end }} diff --git a/charts/redis/redis/templates/master/psp.yaml b/charts/redis/redis/templates/master/psp.yaml deleted file mode 100644 index 2a685f8..0000000 --- a/charts/redis/redis/templates/master/psp.yaml +++ /dev/null @@ -1,47 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and (include "common.capabilities.psp.supported" .) .Values.podSecurityPolicy.create }} -apiVersion: policy/v1beta1 -kind: PodSecurityPolicy -metadata: - name: {{ printf "%s-master" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - allowPrivilegeEscalation: false - fsGroup: - rule: 'MustRunAs' - ranges: - - min: {{ .Values.master.podSecurityContext.fsGroup }} - max: {{ .Values.master.podSecurityContext.fsGroup }} - hostIPC: false - hostNetwork: false - hostPID: false - privileged: false - readOnlyRootFilesystem: false - requiredDropCapabilities: - - ALL - runAsUser: - rule: 'MustRunAs' - ranges: - - min: {{ .Values.master.containerSecurityContext.runAsUser }} - max: {{ .Values.master.containerSecurityContext.runAsUser }} - seLinux: - rule: 'RunAsAny' - supplementalGroups: - rule: 'MustRunAs' - ranges: - - min: {{ .Values.master.containerSecurityContext.runAsUser }} - max: {{ .Values.master.containerSecurityContext.runAsUser }} - volumes: - - 'configMap' - - 'secret' - - 'emptyDir' - - 'persistentVolumeClaim' -{{- end }} diff --git a/charts/redis/redis/templates/master/pvc.yaml b/charts/redis/redis/templates/master/pvc.yaml deleted file mode 100644 index 13aee50..0000000 --- a/charts/redis/redis/templates/master/pvc.yaml +++ /dev/null @@ -1,33 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and (eq .Values.architecture "standalone") (eq .Values.master.kind "Deployment") (.Values.master.persistence.enabled) (not .Values.master.persistence.existingClaim) }} -kind: PersistentVolumeClaim -apiVersion: v1 -metadata: - name: {{ printf "redis-data-%s-master" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - {{- $labels := include "common.tplvalues.merge" ( dict "values" ( list .Values.master.persistence.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $labels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: master - {{- if .Values.master.persistence.annotations }} - annotations: {{- toYaml .Values.master.persistence.annotations | nindent 4 }} - {{- end }} -spec: - accessModes: - {{- range .Values.master.persistence.accessModes }} - - {{ . | quote }} - {{- end }} - resources: - requests: - storage: {{ .Values.master.persistence.size | quote }} - {{- if .Values.master.persistence.selector }} - selector: {{- include "common.tplvalues.render" (dict "value" .Values.master.persistence.selector "context" $) | nindent 4 }} - {{- end }} - {{- if .Values.master.persistence.dataSource }} - dataSource: {{- include "common.tplvalues.render" (dict "value" .Values.master.persistence.dataSource "context" $) | nindent 4 }} - {{- end }} - {{- include "common.storage.class" (dict "persistence" .Values.master.persistence "global" .Values.global) | nindent 2 }} -{{- end }} diff --git a/charts/redis/redis/templates/master/service.yaml b/charts/redis/redis/templates/master/service.yaml deleted file mode 100644 index b9bf47d..0000000 --- a/charts/redis/redis/templates/master/service.yaml +++ /dev/null @@ -1,62 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and (not .Values.sentinel.enabled) (gt (int64 .Values.master.count) 0) }} -apiVersion: v1 -kind: Service -metadata: - name: {{ printf "%s-master" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: master - {{- if or .Values.master.service.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.master.service.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: {{ .Values.master.service.type }} - {{- if or (eq .Values.master.service.type "LoadBalancer") (eq .Values.master.service.type "NodePort") }} - externalTrafficPolicy: {{ .Values.master.service.externalTrafficPolicy | quote }} - {{- end }} - {{- if (semverCompare ">=1.22-0" (include "common.capabilities.kubeVersion" .)) }} - internalTrafficPolicy: {{ .Values.master.service.internalTrafficPolicy }} - {{- end }} - {{- if and (eq .Values.master.service.type "LoadBalancer") (not (empty .Values.master.service.loadBalancerIP)) }} - loadBalancerIP: {{ .Values.master.service.loadBalancerIP }} - {{- end }} - {{- if and (eq .Values.master.service.type "LoadBalancer") .Values.master.service.loadBalancerClass }} - loadBalancerClass: {{ .Values.master.service.loadBalancerClass }} - {{- end }} - {{- if and (eq .Values.master.service.type "LoadBalancer") (not (empty .Values.master.service.loadBalancerSourceRanges)) }} - loadBalancerSourceRanges: {{ toYaml .Values.master.service.loadBalancerSourceRanges | nindent 4 }} - {{- end }} - {{- if and .Values.master.service.clusterIP (eq .Values.master.service.type "ClusterIP") }} - clusterIP: {{ .Values.master.service.clusterIP }} - {{- end }} - {{- if .Values.master.service.sessionAffinity }} - sessionAffinity: {{ .Values.master.service.sessionAffinity }} - {{- end }} - {{- if .Values.master.service.sessionAffinityConfig }} - sessionAffinityConfig: {{- include "common.tplvalues.render" (dict "value" .Values.master.service.sessionAffinityConfig "context" $) | nindent 4 }} - {{- end }} - {{- if .Values.master.service.externalIPs }} - externalIPs: {{- include "common.tplvalues.render" (dict "value" .Values.master.service.externalIPs "context" $) | nindent 4 }} - {{- end }} - ports: - - name: {{ .Values.master.service.portNames.redis }} - port: {{ .Values.master.service.ports.redis }} - targetPort: redis - {{- if and (or (eq .Values.master.service.type "NodePort") (eq .Values.master.service.type "LoadBalancer")) .Values.master.service.nodePorts.redis}} - nodePort: {{ .Values.master.service.nodePorts.redis}} - {{- else if eq .Values.master.service.type "ClusterIP" }} - nodePort: null - {{- end }} - {{- if .Values.master.service.extraPorts }} - {{- include "common.tplvalues.render" (dict "value" .Values.master.service.extraPorts "context" $) | nindent 4 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.master.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: master -{{- end }} diff --git a/charts/redis/redis/templates/master/serviceaccount.yaml b/charts/redis/redis/templates/master/serviceaccount.yaml deleted file mode 100644 index bf58cc5..0000000 --- a/charts/redis/redis/templates/master/serviceaccount.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.master.serviceAccount.create (or (not (eq .Values.architecture "replication")) (not .Values.sentinel.enabled)) }} -apiVersion: v1 -kind: ServiceAccount -automountServiceAccountToken: {{ .Values.master.serviceAccount.automountServiceAccountToken }} -metadata: - name: {{ template "redis.masterServiceAccountName" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if or .Values.master.serviceAccount.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.master.serviceAccount.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/metrics-svc.yaml b/charts/redis/redis/templates/metrics-svc.yaml deleted file mode 100644 index 529122e..0000000 --- a/charts/redis/redis/templates/metrics-svc.yaml +++ /dev/null @@ -1,44 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled .Values.metrics.service.enabled }} -apiVersion: v1 -kind: Service -metadata: - name: {{ printf "%s-metrics" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: metrics - {{- if or .Values.metrics.service.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.metrics.service.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: {{ .Values.metrics.service.type }} - {{- if and .Values.metrics.service.clusterIP (eq .Values.metrics.service.type "ClusterIP") }} - clusterIP: {{ .Values.metrics.service.clusterIP }} - {{- end }} - {{- if eq .Values.metrics.service.type "LoadBalancer" }} - externalTrafficPolicy: {{ .Values.metrics.service.externalTrafficPolicy }} - {{- end }} - {{- if and (eq .Values.metrics.service.type "LoadBalancer") .Values.metrics.service.loadBalancerIP }} - loadBalancerIP: {{ .Values.metrics.service.loadBalancerIP }} - {{- end }} - {{- if and (eq .Values.metrics.service.type "LoadBalancer") .Values.metrics.service.loadBalancerClass }} - loadBalancerClass: {{ .Values.metrics.service.loadBalancerClass }} - {{- end }} - {{- if and (eq .Values.metrics.service.type "LoadBalancer") .Values.metrics.service.loadBalancerSourceRanges }} - loadBalancerSourceRanges: {{- toYaml .Values.metrics.service.loadBalancerSourceRanges | nindent 4 }} - {{- end }} - ports: - - name: http-metrics - port: {{ coalesce .Values.metrics.service.ports.http .Values.metrics.service.port }} - protocol: TCP - targetPort: metrics - {{- if .Values.metrics.service.extraPorts }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.service.extraPorts "context" $) | nindent 4 }} - {{- end }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} -{{- end }} diff --git a/charts/redis/redis/templates/networkpolicy.yaml b/charts/redis/redis/templates/networkpolicy.yaml deleted file mode 100644 index 3d652c6..0000000 --- a/charts/redis/redis/templates/networkpolicy.yaml +++ /dev/null @@ -1,108 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.networkPolicy.enabled }} -kind: NetworkPolicy -apiVersion: {{ template "networkPolicy.apiVersion" . }} -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 6 }} - policyTypes: - - Ingress - - Egress - {{- if .Values.networkPolicy.allowExternalEgress }} - egress: - - {} - {{- else }} - egress: - {{- if eq .Values.architecture "replication" }} - # Allow dns resolution - - ports: - - port: 53 - protocol: UDP - # Allow outbound connections to other cluster pods - - ports: - - port: {{ .Values.master.containerPorts.redis }} - {{- if .Values.sentinel.enabled }} - - port: {{ .Values.sentinel.containerPorts.sentinel }} - {{- end }} - to: - - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 14 }} - {{- end }} - {{- if .Values.networkPolicy.extraEgress }} - {{- include "common.tplvalues.render" ( dict "value" .Values.networkPolicy.extraEgress "context" $ ) | nindent 4 }} - {{- end }} - {{- end }} - ingress: - # Allow inbound connections - - ports: - - port: {{ .Values.master.containerPorts.redis }} - {{- if .Values.sentinel.enabled }} - - port: {{ .Values.sentinel.containerPorts.sentinel }} - {{- end }} - {{- if not .Values.networkPolicy.allowExternal }} - from: - - podSelector: - matchLabels: - {{ template "common.names.fullname" . }}-client: "true" - - podSelector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 14 }} - {{- if or .Values.networkPolicy.ingressNSMatchLabels .Values.networkPolicy.ingressNSPodMatchLabels }} - - namespaceSelector: - matchLabels: - {{- if .Values.networkPolicy.ingressNSMatchLabels }} - {{- range $key, $value := .Values.networkPolicy.ingressNSMatchLabels }} - {{ $key | quote }}: {{ $value | quote }} - {{- end }} - {{ else }} - {} - {{- end }} - {{- if .Values.networkPolicy.ingressNSPodMatchLabels }} - podSelector: - matchLabels: - {{- range $key, $value := .Values.networkPolicy.ingressNSPodMatchLabels }} - {{ $key | quote }}: {{ $value | quote }} - {{- end }} - {{- end }} - {{- end }} - {{- end }} - {{- if .Values.metrics.enabled }} - # Allow prometheus scrapes for metrics - - ports: - - port: {{ .Values.metrics.containerPorts.http }} - {{- if not .Values.networkPolicy.metrics.allowExternal }} - from: - {{- if or .Values.networkPolicy.metrics.ingressNSMatchLabels .Values.networkPolicy.metrics.ingressNSPodMatchLabels }} - - namespaceSelector: - matchLabels: - {{- if .Values.networkPolicy.metrics.ingressNSMatchLabels }} - {{- range $key, $value := .Values.networkPolicy.metrics.ingressNSMatchLabels }} - {{ $key | quote }}: {{ $value | quote }} - {{- end }} - {{ else }} - {} - {{- end }} - {{- if .Values.networkPolicy.metrics.ingressNSPodMatchLabels }} - podSelector: - matchLabels: - {{- range $key, $value := .Values.networkPolicy.metrics.ingressNSPodMatchLabels }} - {{ $key | quote }}: {{ $value | quote }} - {{- end }} - {{- end }} - {{- end }} - {{- end }} - {{- end }} - {{- if .Values.networkPolicy.extraIngress }} - {{- include "common.tplvalues.render" ( dict "value" .Values.networkPolicy.extraIngress "context" $ ) | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/podmonitor.yaml b/charts/redis/redis/templates/podmonitor.yaml deleted file mode 100644 index 3297610..0000000 --- a/charts/redis/redis/templates/podmonitor.yaml +++ /dev/null @@ -1,81 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled .Values.metrics.podMonitor.enabled }} -apiVersion: monitoring.coreos.com/v1 -kind: PodMonitor -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ default (include "common.names.namespace" .) .Values.metrics.podMonitor.namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.metrics.podMonitor.additionalLabels }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.podMonitor.additionalLabels "context" $) | nindent 4 }} - {{- end }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - podMetricsEndpoints: - - port: {{ .Values.metrics.podMonitor.port }} - {{- if .Values.metrics.podMonitor.interval }} - interval: {{ .Values.metrics.podMonitor.interval }} - {{- end }} - {{- if .Values.metrics.podMonitor.scrapeTimeout }} - scrapeTimeout: {{ .Values.metrics.podMonitor.scrapeTimeout }} - {{- end }} - {{- if .Values.metrics.podMonitor.honorLabels }} - honorLabels: {{ .Values.metrics.podMonitor.honorLabels }} - {{- end }} - {{- with concat .Values.metrics.podMonitor.relabelings .Values.metrics.podMonitor.relabellings }} - relabelings: {{- toYaml . | nindent 6 }} - {{- end }} - {{- if .Values.metrics.podMonitor.metricRelabelings }} - metricRelabelings: {{- toYaml .Values.metrics.podMonitor.metricRelabelings | nindent 6 }} - {{- end }} - {{- range .Values.metrics.podMonitor.additionalEndpoints }} - - port: {{ .port }} - {{- if .interval }} - interval: {{ .interval }} - {{- end }} - {{- if .path }} - path: {{ .path }} - {{- end }} - {{- if .honorLabels }} - honorLabels: {{ .honorLabels }} - {{- end }} - {{- with concat .relabelings .relabellings }} - relabelings: {{- toYaml . | nindent 6 }} - {{- end }} - {{- if .metricRelabelings }} - metricRelabelings: {{- toYaml .metricRelabelings | nindent 6 }} - {{- end }} - {{- if .scrapeTimeout }} - scrapeTimeout: {{ .scrapeTimeout }} - {{- end }} - {{- if .params }} - params: - {{- range $key, $value := .params }} - {{ $key }}: - {{- range $value }} - - {{ . | quote }} - {{- end }} - {{- end }} - {{- end }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.podTargetLabels }} - podTargetLabels: {{- toYaml .Values.metrics.podMonitor.podTargetLabels | nindent 4 }} - {{- end }} - {{- with .Values.metrics.podMonitor.sampleLimit -}} - sampleLimit: {{ . }} - {{- end }} - {{- with .Values.metrics.podMonitor.targetLimit -}} - targetLimit: {{ . }} - {{- end }} - namespaceSelector: - matchNames: - - {{ include "common.names.namespace" . | quote }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 6 }} -{{- end }} diff --git a/charts/redis/redis/templates/prometheusrule.yaml b/charts/redis/redis/templates/prometheusrule.yaml deleted file mode 100644 index 56c013b..0000000 --- a/charts/redis/redis/templates/prometheusrule.yaml +++ /dev/null @@ -1,23 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled .Values.metrics.prometheusRule.enabled }} -apiVersion: monitoring.coreos.com/v1 -kind: PrometheusRule -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ default (include "common.names.namespace" .) .Values.metrics.prometheusRule.namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.metrics.prometheusRule.additionalLabels }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.prometheusRule.additionalLabels "context" $) | nindent 4 }} - {{- end }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - groups: - - name: {{ include "common.names.fullname" . }} - rules: {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.prometheusRule.rules "context" $ ) | nindent 8 }} -{{- end }} diff --git a/charts/redis/redis/templates/replicas/application.yaml b/charts/redis/redis/templates/replicas/application.yaml deleted file mode 100644 index fe4f334..0000000 --- a/charts/redis/redis/templates/replicas/application.yaml +++ /dev/null @@ -1,561 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and (eq .Values.architecture "replication") (not .Values.sentinel.enabled) }} -apiVersion: {{ include "common.capabilities.statefulset.apiVersion" . }} -kind: {{ .Values.replica.kind }} -metadata: - name: {{ printf "%s-replicas" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: replica - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- if and (not (eq .Values.replica.kind "DaemonSet")) (not .Values.replica.autoscaling.enabled) }} - replicas: {{ .Values.replica.replicaCount }} - {{- end }} - revisionHistoryLimit: {{ .Values.replica.revisionHistoryLimit }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.replica.podLabels .Values.commonLabels ) "context" . ) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: replica - {{- if (eq .Values.replica.kind "StatefulSet") }} - serviceName: {{ printf "%s-headless" (include "common.names.fullname" .) }} - {{- end }} - {{- if .Values.replica.updateStrategy }} - updateStrategy: {{- toYaml .Values.replica.updateStrategy | nindent 4 }} - {{- end }} - {{- if and .Values.replica.minReadySeconds (semverCompare ">= 1.23-0" (include "common.capabilities.kubeVersion" .)) }} - minReadySeconds: {{ .Values.replica.minReadySeconds }} - {{- end }} - {{- if .Values.replica.podManagementPolicy }} - podManagementPolicy: {{ .Values.replica.podManagementPolicy | quote }} - {{- end }} - template: - metadata: - labels: {{- include "common.labels.standard" ( dict "customLabels" $podLabels "context" $ ) | nindent 8 }} - app.kubernetes.io/component: replica - {{- if and .Values.metrics.enabled .Values.metrics.podLabels }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.podLabels "context" $ ) | nindent 8 }} - {{- end }} - annotations: - {{- if (include "redis.createConfigmap" .) }} - checksum/configmap: {{ pick ( include (print $.Template.BasePath "/configmap.yaml") . | fromYaml ) "data" | toYaml | sha256sum }} - {{- end }} - checksum/health: {{ pick ( include (print $.Template.BasePath "/health-configmap.yaml") . | fromYaml ) "data" | toYaml | sha256sum }} - checksum/scripts: {{ pick ( include (print $.Template.BasePath "/scripts-configmap.yaml") . | fromYaml ) "data" | toYaml | sha256sum }} - checksum/secret: {{ pick ( include (print $.Template.BasePath "/secret.yaml") . | fromYaml ) "data" | toYaml | sha256sum }} - {{- if .Values.replica.podAnnotations }} - {{- include "common.tplvalues.render" ( dict "value" .Values.replica.podAnnotations "context" $ ) | nindent 8 }} - {{- end }} - {{- if and .Values.metrics.enabled .Values.metrics.podAnnotations }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.podAnnotations "context" $ ) | nindent 8 }} - {{- end }} - spec: - {{- include "redis.imagePullSecrets" . | nindent 6 }} - {{- if .Values.replica.hostAliases }} - hostAliases: {{- include "common.tplvalues.render" (dict "value" .Values.replica.hostAliases "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.replica.podSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.replica.podSecurityContext "context" $) | nindent 8 }} - {{- end }} - serviceAccountName: {{ template "redis.replicaServiceAccountName" . }} - automountServiceAccountToken: {{ .Values.replica.automountServiceAccountToken }} - {{- if .Values.replica.priorityClassName }} - priorityClassName: {{ .Values.replica.priorityClassName | quote }} - {{- end }} - {{- if .Values.replica.affinity }} - affinity: {{- include "common.tplvalues.render" (dict "value" .Values.replica.affinity "context" $) | nindent 8 }} - {{- else }} - affinity: - podAffinity: {{- include "common.affinities.pods" (dict "type" .Values.replica.podAffinityPreset "component" "replica" "customLabels" $podLabels "context" $) | nindent 10 }} - podAntiAffinity: {{- include "common.affinities.pods" (dict "type" .Values.replica.podAntiAffinityPreset "component" "replica" "customLabels" $podLabels "context" $) | nindent 10 }} - nodeAffinity: {{- include "common.affinities.nodes" (dict "type" .Values.replica.nodeAffinityPreset.type "key" .Values.replica.nodeAffinityPreset.key "values" .Values.replica.nodeAffinityPreset.values) | nindent 10 }} - {{- end }} - {{- if .Values.replica.nodeSelector }} - nodeSelector: {{- include "common.tplvalues.render" (dict "value" .Values.replica.nodeSelector "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.replica.tolerations }} - tolerations: {{- include "common.tplvalues.render" (dict "value" .Values.replica.tolerations "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.replica.topologySpreadConstraints }} - topologySpreadConstraints: {{- include "common.tplvalues.render" (dict "value" .Values.replica.topologySpreadConstraints "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.replica.shareProcessNamespace }} - shareProcessNamespace: {{ .Values.replica.shareProcessNamespace }} - {{- end }} - {{- if .Values.replica.schedulerName }} - schedulerName: {{ .Values.replica.schedulerName | quote }} - {{- end }} - {{- if .Values.replica.dnsPolicy }} - dnsPolicy: {{ .Values.replica.dnsPolicy }} - {{- end }} - {{- if .Values.replica.dnsConfig }} - dnsConfig: {{- include "common.tplvalues.render" (dict "value" .Values.replica.dnsConfig "context" $) | nindent 8 }} - {{- end }} - enableServiceLinks: {{ .Values.replica.enableServiceLinks }} - terminationGracePeriodSeconds: {{ .Values.replica.terminationGracePeriodSeconds }} - containers: - - name: redis - image: {{ template "redis.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy | quote }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.replica.lifecycleHooks }} - lifecycle: {{- include "common.tplvalues.render" (dict "value" .Values.replica.lifecycleHooks "context" $) | nindent 12 }} - {{- end }} - {{- end }} - {{- if .Values.replica.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.replica.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - {{- else if .Values.replica.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.replica.command "context" $) | nindent 12 }} - {{- else }} - command: - - /bin/bash - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- else if .Values.replica.args }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.replica.args "context" $) | nindent 12 }} - {{- else }} - args: - - -c - - /opt/bitnami/scripts/start-scripts/start-replica.sh - {{- end }} - env: - - name: BITNAMI_DEBUG - value: {{ ternary "true" "false" (or .Values.image.debug .Values.diagnosticMode.enabled) | quote }} - - name: REDIS_REPLICATION_MODE - value: replica - - name: REDIS_MASTER_HOST - {{- if .Values.replica.externalMaster.enabled }} - value: {{ .Values.replica.externalMaster.host | quote }} - {{- else if and (eq (int64 .Values.master.count) 1) (eq .Values.master.kind "StatefulSet") }} - value: {{ template "common.names.fullname" . }}-master-0.{{ template "common.names.fullname" . }}-headless.{{ include "common.names.namespace" . }}.svc.{{ .Values.clusterDomain }} - {{- else }} - value: {{ template "common.names.fullname" . }}-master.{{ include "common.names.namespace" . }}.svc.{{ .Values.clusterDomain }} - {{- end }} - - name: REDIS_MASTER_PORT_NUMBER - {{- if .Values.replica.externalMaster.enabled }} - value: {{ .Values.replica.externalMaster.port | quote }} - {{- else }} - value: {{ .Values.master.containerPorts.redis | quote }} - {{- end }} - - name: ALLOW_EMPTY_PASSWORD - value: {{ ternary "no" "yes" .Values.auth.enabled | quote }} - {{- if .Values.auth.enabled }} - {{- if .Values.auth.usePasswordFiles }} - - name: REDIS_PASSWORD_FILE - value: "/opt/bitnami/redis/secrets/redis-password" - - name: REDIS_MASTER_PASSWORD_FILE - value: "/opt/bitnami/redis/secrets/redis-password" - {{- else }} - - name: REDIS_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "redis.secretName" . }} - key: {{ template "redis.secretPasswordKey" . }} - - name: REDIS_MASTER_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "redis.secretName" . }} - key: {{ template "redis.secretPasswordKey" . }} - {{- end }} - {{- end }} - - name: REDIS_TLS_ENABLED - value: {{ ternary "yes" "no" .Values.tls.enabled | quote }} - {{- if .Values.tls.enabled }} - - name: REDIS_TLS_PORT - value: {{ .Values.replica.containerPorts.redis | quote }} - - name: REDIS_TLS_AUTH_CLIENTS - value: {{ ternary "yes" "no" .Values.tls.authClients | quote }} - - name: REDIS_TLS_CERT_FILE - value: {{ template "redis.tlsCert" . }} - - name: REDIS_TLS_KEY_FILE - value: {{ template "redis.tlsCertKey" . }} - - name: REDIS_TLS_CA_FILE - value: {{ template "redis.tlsCACert" . }} - {{- if .Values.tls.dhParamsFilename }} - - name: REDIS_TLS_DH_PARAMS_FILE - value: {{ template "redis.tlsDHParams" . }} - {{- end }} - {{- else }} - - name: REDIS_PORT - value: {{ .Values.replica.containerPorts.redis | quote }} - {{- end }} - {{- if .Values.replica.extraEnvVars }} - {{- include "common.tplvalues.render" (dict "value" .Values.replica.extraEnvVars "context" $) | nindent 12 }} - {{- end }} - {{- if or .Values.replica.extraEnvVarsCM .Values.replica.extraEnvVarsSecret }} - envFrom: - {{- if .Values.replica.extraEnvVarsCM }} - - configMapRef: - name: {{ .Values.replica.extraEnvVarsCM }} - {{- end }} - {{- if .Values.replica.extraEnvVarsSecret }} - - secretRef: - name: {{ .Values.replica.extraEnvVarsSecret }} - {{- end }} - {{- end }} - ports: - - name: redis - containerPort: {{ .Values.replica.containerPorts.redis }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.replica.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.replica.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.replica.startupProbe.enabled }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.replica.startupProbe "enabled") "context" $) | nindent 12 }} - tcpSocket: - port: redis - {{- end }} - {{- if .Values.replica.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.replica.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.replica.livenessProbe.enabled }} - livenessProbe: - initialDelaySeconds: {{ .Values.replica.livenessProbe.initialDelaySeconds }} - periodSeconds: {{ .Values.replica.livenessProbe.periodSeconds }} - timeoutSeconds: {{ add1 .Values.replica.livenessProbe.timeoutSeconds }} - successThreshold: {{ .Values.replica.livenessProbe.successThreshold }} - failureThreshold: {{ .Values.replica.livenessProbe.failureThreshold }} - exec: - command: - - sh - - -c - - /health/ping_liveness_local_and_master.sh {{ .Values.replica.livenessProbe.timeoutSeconds }} - {{- end }} - {{- if .Values.replica.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.replica.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.replica.readinessProbe.enabled }} - readinessProbe: - initialDelaySeconds: {{ .Values.replica.readinessProbe.initialDelaySeconds }} - periodSeconds: {{ .Values.replica.readinessProbe.periodSeconds }} - timeoutSeconds: {{ add1 .Values.replica.readinessProbe.timeoutSeconds }} - successThreshold: {{ .Values.replica.readinessProbe.successThreshold }} - failureThreshold: {{ .Values.replica.readinessProbe.failureThreshold }} - exec: - command: - - sh - - -c - - /health/ping_readiness_local_and_master.sh {{ .Values.replica.readinessProbe.timeoutSeconds }} - {{- end }} - {{- end }} - {{- if .Values.replica.resources }} - resources: {{- toYaml .Values.replica.resources | nindent 12 }} - {{- else if ne .Values.replica.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.replica.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: start-scripts - mountPath: /opt/bitnami/scripts/start-scripts - - name: health - mountPath: /health - {{- if .Values.auth.usePasswordFiles }} - - name: redis-password - mountPath: /opt/bitnami/redis/secrets/ - {{- end }} - - name: redis-data - mountPath: /data - {{- if .Values.replica.persistence.subPath }} - subPath: {{ .Values.replica.persistence.subPath }} - {{- else if .Values.replica.persistence.subPathExpr }} - subPathExpr: {{ .Values.replica.persistence.subPathExpr }} - {{- end }} - - name: config - mountPath: /opt/bitnami/redis/mounted-etc - - name: empty-dir - mountPath: /opt/bitnami/redis/etc - subPath: app-conf-dir - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - {{- if .Values.tls.enabled }} - - name: redis-certificates - mountPath: /opt/bitnami/redis/certs - readOnly: true - {{- end }} - {{- if .Values.replica.extraVolumeMounts }} - {{- include "common.tplvalues.render" ( dict "value" .Values.replica.extraVolumeMounts "context" $ ) | nindent 12 }} - {{- end }} - {{- if .Values.metrics.enabled }} - - name: metrics - image: {{ include "redis.metrics.image" . }} - imagePullPolicy: {{ .Values.metrics.image.pullPolicy | quote }} - {{- if .Values.metrics.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.metrics.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - {{- else if .Values.metrics.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.command "context" $) | nindent 12 }} - {{- else }} - command: - - /bin/bash - - -c - - | - if [[ -f '/secrets/redis-password' ]]; then - export REDIS_PASSWORD=$(cat /secrets/redis-password) - fi - redis_exporter{{- range $key, $value := .Values.metrics.extraArgs }} --{{ $key }}={{ $value }}{{- end }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- end }} - env: - - name: REDIS_ALIAS - value: {{ template "common.names.fullname" . }} - - name: REDIS_EXPORTER_WEB_LISTEN_ADDRESS - value: {{ printf ":%v" .Values.metrics.containerPorts.http }} - {{- if .Values.auth.enabled }} - - name: REDIS_USER - value: default - {{- if (not .Values.auth.usePasswordFiles) }} - - name: REDIS_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "redis.secretName" . }} - key: {{ template "redis.secretPasswordKey" . }} - {{- end }} - {{- end }} - {{- if .Values.tls.enabled }} - - name: REDIS_ADDR - value: rediss://{{ .Values.metrics.redisTargetHost }}:{{ .Values.replica.containerPorts.redis }} - {{- if .Values.tls.authClients }} - - name: REDIS_EXPORTER_TLS_CLIENT_KEY_FILE - value: {{ template "redis.tlsCertKey" . }} - - name: REDIS_EXPORTER_TLS_CLIENT_CERT_FILE - value: {{ template "redis.tlsCert" . }} - {{- end }} - - name: REDIS_EXPORTER_TLS_CA_CERT_FILE - value: {{ template "redis.tlsCACert" . }} - {{- end }} - {{- if .Values.metrics.extraEnvVars }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.extraEnvVars "context" $) | nindent 12 }} - {{- end }} - ports: - - name: metrics - containerPort: {{ .Values.metrics.containerPorts.http }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.metrics.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.startupProbe.enabled }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.startupProbe "enabled") "context" $) | nindent 12 }} - tcpSocket: - port: metrics - {{- end }} - {{- if .Values.metrics.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.livenessProbe.enabled }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.livenessProbe "enabled") "context" $) | nindent 12 }} - tcpSocket: - port: metrics - {{- end }} - {{- if .Values.metrics.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.readinessProbe.enabled }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.readinessProbe "enabled") "context" $) | nindent 12 }} - httpGet: - path: / - port: metrics - {{- end }} - {{- end }} - {{- if .Values.metrics.resources }} - resources: {{- toYaml .Values.metrics.resources | nindent 12 }} - {{- else if ne .Values.metrics.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.metrics.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - {{- if .Values.auth.usePasswordFiles }} - - name: redis-password - mountPath: /secrets/ - {{- end }} - {{- if .Values.tls.enabled }} - - name: redis-certificates - mountPath: /opt/bitnami/redis/certs - readOnly: true - {{- end }} - {{- if .Values.metrics.extraVolumeMounts }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.extraVolumeMounts "context" $ ) | nindent 12 }} - {{- end }} - {{- end }} - {{- if .Values.replica.sidecars }} - {{- include "common.tplvalues.render" (dict "value" .Values.replica.sidecars "context" $) | nindent 8 }} - {{- end }} - {{- $needsVolumePermissions := and .Values.volumePermissions.enabled .Values.replica.persistence.enabled .Values.replica.podSecurityContext.enabled .Values.replica.containerSecurityContext.enabled }} - {{- if or .Values.replica.initContainers $needsVolumePermissions .Values.sysctl.enabled }} - initContainers: - {{- if .Values.replica.initContainers }} - {{- include "common.tplvalues.render" (dict "value" .Values.replica.initContainers "context" $) | nindent 8 }} - {{- end }} - {{- if $needsVolumePermissions }} - - name: volume-permissions - image: {{ include "redis.volumePermissions.image" . }} - imagePullPolicy: {{ .Values.volumePermissions.image.pullPolicy | quote }} - command: - - /bin/bash - - -ec - - | - {{- if eq ( toString ( .Values.volumePermissions.containerSecurityContext.runAsUser )) "auto" }} - chown -R `id -u`:`id -G | cut -d " " -f2` {{ .Values.replica.persistence.path }} - {{- else }} - chown -R {{ .Values.replica.containerSecurityContext.runAsUser }}:{{ .Values.replica.podSecurityContext.fsGroup }} {{ .Values.replica.persistence.path }} - {{- end }} - {{- if eq ( toString ( .Values.volumePermissions.containerSecurityContext.runAsUser )) "auto" }} - securityContext: {{- omit .Values.volumePermissions.containerSecurityContext "runAsUser" | toYaml | nindent 12 }} - {{- else }} - securityContext: {{- .Values.volumePermissions.containerSecurityContext | toYaml | nindent 12 }} - {{- end }} - {{- if .Values.volumePermissions.resources }} - resources: {{- toYaml .Values.volumePermissions.resources | nindent 12 }} - {{- else if ne .Values.volumePermissions.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.volumePermissions.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: redis-data - mountPath: {{ .Values.replica.persistence.path }} - {{- if .Values.replica.persistence.subPath }} - subPath: {{ .Values.replica.persistence.subPath }} - {{- else if .Values.replica.persistence.subPathExpr }} - subPathExpr: {{ .Values.replica.persistence.subPathExpr }} - {{- end }} - {{- end }} - {{- if .Values.sysctl.enabled }} - - name: init-sysctl - image: {{ include "redis.sysctl.image" . }} - imagePullPolicy: {{ default "" .Values.sysctl.image.pullPolicy | quote }} - securityContext: - privileged: true - runAsUser: 0 - {{- if .Values.sysctl.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.sysctl.command "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.sysctl.resources }} - resources: {{- toYaml .Values.sysctl.resources | nindent 12 }} - {{- else if ne .Values.sysctl.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.sysctl.resourcesPreset) | nindent 12 }} - {{- end }} - {{- if .Values.sysctl.mountHostSys }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: host-sys - mountPath: /host-sys - {{- end }} - {{- end }} - {{- end }} - volumes: - - name: start-scripts - configMap: - name: {{ printf "%s-scripts" (include "common.names.fullname" .) }} - defaultMode: 0755 - - name: health - configMap: - name: {{ printf "%s-health" (include "common.names.fullname" .) }} - defaultMode: 0755 - {{- if .Values.auth.usePasswordFiles }} - - name: redis-password - {{ if .Values.auth.usePasswordFileFromSecret }} - secret: - secretName: {{ template "redis.secretName" . }} - items: - - key: {{ template "redis.secretPasswordKey" . }} - path: redis-password - {{- else }} - emptyDir: {} - {{- end }} - {{- end }} - - name: config - configMap: - name: {{ include "redis.configmapName" . }} - {{- if .Values.sysctl.mountHostSys }} - - name: host-sys - hostPath: - path: /sys - {{- end }} - - name: empty-dir - {{- if or .Values.replica.persistence.medium .Values.replica.persistence.sizeLimit }} - emptyDir: - {{- if .Values.replica.persistence.medium }} - medium: {{ .Values.replica.persistence.medium | quote }} - {{- end }} - {{- if .Values.replica.persistence.sizeLimit }} - sizeLimit: {{ .Values.replica.persistence.sizeLimit | quote }} - {{- end }} - {{- else }} - emptyDir: {} - {{- end }} - {{- if .Values.tls.enabled }} - - name: redis-certificates - secret: - secretName: {{ include "redis.tlsSecretName" . }} - defaultMode: 256 - {{- end }} - {{- if .Values.replica.extraVolumes }} - {{- include "common.tplvalues.render" ( dict "value" .Values.replica.extraVolumes "context" $ ) | nindent 8 }} - {{- end }} - {{- if .Values.metrics.extraVolumes }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.extraVolumes "context" $ ) | nindent 8 }} - {{- end }} - {{- if or (not .Values.replica.persistence.enabled) (not (eq .Values.replica.kind "StatefulSet")) }} - - name: redis-data - {{- if or .Values.replica.persistence.medium .Values.replica.persistence.sizeLimit }} - emptyDir: - {{- if .Values.replica.persistence.medium }} - medium: {{ .Values.replica.persistence.medium | quote }} - {{- end }} - {{- if .Values.replica.persistence.sizeLimit }} - sizeLimit: {{ .Values.replica.persistence.sizeLimit | quote }} - {{- end }} - {{- else }} - emptyDir: {} - {{- end }} - {{- else if .Values.replica.persistence.existingClaim }} - - name: redis-data - persistentVolumeClaim: - claimName: {{ printf "%s" (tpl .Values.replica.persistence.existingClaim .) }} - {{- else }} - {{- if .Values.replica.persistentVolumeClaimRetentionPolicy.enabled }} - persistentVolumeClaimRetentionPolicy: - whenDeleted: {{ .Values.replica.persistentVolumeClaimRetentionPolicy.whenDeleted }} - whenScaled: {{ .Values.replica.persistentVolumeClaimRetentionPolicy.whenScaled }} - {{- end }} - volumeClaimTemplates: - - apiVersion: v1 - kind: PersistentVolumeClaim - metadata: - name: redis-data - {{- $claimLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.master.persistence.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.matchLabels" ( dict "customLabels" $claimLabels "context" $ ) | nindent 10 }} - app.kubernetes.io/component: replica - {{- if .Values.replica.persistence.annotations }} - annotations: {{- toYaml .Values.replica.persistence.annotations | nindent 10 }} - {{- end }} - spec: - accessModes: - {{- range .Values.replica.persistence.accessModes }} - - {{ . | quote }} - {{- end }} - resources: - requests: - storage: {{ .Values.replica.persistence.size | quote }} - {{- if .Values.replica.persistence.selector }} - selector: {{- include "common.tplvalues.render" (dict "value" .Values.replica.persistence.selector "context" $) | nindent 10 }} - {{- end }} - {{- if .Values.replica.persistence.dataSource }} - dataSource: {{- include "common.tplvalues.render" (dict "value" .Values.replica.persistence.dataSource "context" $) | nindent 10 }} - {{- end }} - {{- include "common.storage.class" (dict "persistence" .Values.replica.persistence "global" .Values.global) | nindent 8 }} - {{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/replicas/hpa.yaml b/charts/redis/redis/templates/replicas/hpa.yaml deleted file mode 100644 index 85adf72..0000000 --- a/charts/redis/redis/templates/replicas/hpa.yaml +++ /dev/null @@ -1,49 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.replica.autoscaling.enabled (not .Values.sentinel.enabled) }} -apiVersion: {{ include "common.capabilities.hpa.apiVersion" ( dict "context" $ ) }} -kind: HorizontalPodAutoscaler -metadata: - name: {{ printf "%s-replicas" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: replica - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - scaleTargetRef: - apiVersion: {{ include "common.capabilities.deployment.apiVersion" . }} - kind: StatefulSet - name: {{ printf "%s-replicas" (include "common.names.fullname" .) }} - minReplicas: {{ .Values.replica.autoscaling.minReplicas }} - maxReplicas: {{ .Values.replica.autoscaling.maxReplicas }} - metrics: - {{- if .Values.replica.autoscaling.targetCPU }} - - type: Resource - resource: - name: cpu - {{- if semverCompare "<1.23-0" (include "common.capabilities.kubeVersion" .) }} - targetAverageUtilization: {{ .Values.replica.autoscaling.targetCPU }} - {{- else }} - target: - type: Utilization - averageUtilization: {{ .Values.replica.autoscaling.targetCPU }} - {{- end }} - {{- end }} - {{- if .Values.replica.autoscaling.targetMemory }} - - type: Resource - resource: - name: memory - {{- if semverCompare "<1.23-0" (include "common.capabilities.kubeVersion" .) }} - targetAverageUtilization: {{ .Values.replica.autoscaling.targetMemory }} - {{- else }} - target: - type: Utilization - averageUtilization: {{ .Values.replica.autoscaling.targetMemory }} - {{- end }} - {{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/replicas/pdb.yaml b/charts/redis/redis/templates/replicas/pdb.yaml deleted file mode 100644 index d7b777b..0000000 --- a/charts/redis/redis/templates/replicas/pdb.yaml +++ /dev/null @@ -1,28 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- $pdb := coalesce .Values.pdb .Values.replica.pdb }} -{{- if and (eq .Values.architecture "replication") (not .Values.sentinel.enabled) $pdb.create }} -apiVersion: {{ include "common.capabilities.policy.apiVersion" . }} -kind: PodDisruptionBudget -metadata: - name: {{ printf "%s-replicas" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: replica - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- if $pdb.minAvailable }} - minAvailable: {{ $pdb.minAvailable }} - {{- end }} - {{- if or $pdb.maxUnavailable (not $pdb.minAvailable) }} - maxUnavailable: {{ $pdb.maxUnavailable | default 1 }} - {{- end }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: replica -{{- end }} diff --git a/charts/redis/redis/templates/replicas/service.yaml b/charts/redis/redis/templates/replicas/service.yaml deleted file mode 100644 index ebb2a4f..0000000 --- a/charts/redis/redis/templates/replicas/service.yaml +++ /dev/null @@ -1,59 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and (eq .Values.architecture "replication") (not .Values.sentinel.enabled) }} -apiVersion: v1 -kind: Service -metadata: - name: {{ printf "%s-replicas" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: replica - {{- if or .Values.replica.service.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.replica.service.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: {{ .Values.replica.service.type }} - {{- if or (eq .Values.replica.service.type "LoadBalancer") (eq .Values.replica.service.type "NodePort") }} - externalTrafficPolicy: {{ .Values.replica.service.externalTrafficPolicy | quote }} - {{- end }} - {{- if (semverCompare ">=1.22-0" (include "common.capabilities.kubeVersion" .)) }} - internalTrafficPolicy: {{ .Values.replica.service.internalTrafficPolicy }} - {{- end }} - {{- if and (eq .Values.replica.service.type "LoadBalancer") (not (empty .Values.replica.service.loadBalancerIP)) }} - loadBalancerIP: {{ .Values.replica.service.loadBalancerIP }} - {{- end }} - {{- if and (eq .Values.replica.service.type "LoadBalancer") .Values.replica.service.loadBalancerClass }} - loadBalancerClass: {{ .Values.replica.service.loadBalancerClass }} - {{- end }} - {{- if and (eq .Values.replica.service.type "LoadBalancer") (not (empty .Values.replica.service.loadBalancerSourceRanges)) }} - loadBalancerSourceRanges: {{ toYaml .Values.replica.service.loadBalancerSourceRanges | nindent 4 }} - {{- end }} - {{- if and .Values.replica.service.clusterIP (eq .Values.replica.service.type "ClusterIP") }} - clusterIP: {{ .Values.replica.service.clusterIP }} - {{- end }} - {{- if .Values.replica.service.sessionAffinity }} - sessionAffinity: {{ .Values.replica.service.sessionAffinity }} - {{- end }} - {{- if .Values.replica.service.sessionAffinityConfig }} - sessionAffinityConfig: {{- include "common.tplvalues.render" (dict "value" .Values.replica.service.sessionAffinityConfig "context" $) | nindent 4 }} - {{- end }} - ports: - - name: tcp-redis - port: {{ .Values.replica.service.ports.redis }} - targetPort: redis - {{- if and (or (eq .Values.replica.service.type "NodePort") (eq .Values.replica.service.type "LoadBalancer")) .Values.replica.service.nodePorts.redis}} - nodePort: {{ .Values.replica.service.nodePorts.redis}} - {{- else if eq .Values.replica.service.type "ClusterIP" }} - nodePort: null - {{- end }} - {{- if .Values.replica.service.extraPorts }} - {{- include "common.tplvalues.render" (dict "value" .Values.replica.service.extraPorts "context" $) | nindent 4 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.replica.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: replica -{{- end }} diff --git a/charts/redis/redis/templates/replicas/serviceaccount.yaml b/charts/redis/redis/templates/replicas/serviceaccount.yaml deleted file mode 100644 index 6cf3411..0000000 --- a/charts/redis/redis/templates/replicas/serviceaccount.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.replica.serviceAccount.create (eq .Values.architecture "replication") (not .Values.sentinel.enabled) }} -apiVersion: v1 -kind: ServiceAccount -automountServiceAccountToken: {{ .Values.replica.serviceAccount.automountServiceAccountToken }} -metadata: - name: {{ template "redis.replicaServiceAccountName" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if or .Values.replica.serviceAccount.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.replica.serviceAccount.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/role.yaml b/charts/redis/redis/templates/role.yaml deleted file mode 100644 index 54a2b1b..0000000 --- a/charts/redis/redis/templates/role.yaml +++ /dev/null @@ -1,34 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.rbac.create }} -apiVersion: {{ include "common.capabilities.rbac.apiVersion" . }} -kind: Role -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -rules: - {{- if and (include "common.capabilities.psp.supported" .) .Values.podSecurityPolicy.enabled }} - - apiGroups: - - '{{ template "podSecurityPolicy.apiGroup" . }}' - resources: - - 'podsecuritypolicies' - verbs: - - 'use' - resourceNames: [{{ printf "%s-master" (include "common.names.fullname" .) }}] - {{- end }} - {{- if and .Values.sentinel.enabled (or .Values.sentinel.masterService.enabled .Values.sentinel.service.createMaster) }} - - apiGroups: [""] - resources: ["pods"] - verbs: ["list", "patch"] - {{- end -}} - {{- if .Values.rbac.rules }} - {{- include "common.tplvalues.render" ( dict "value" .Values.rbac.rules "context" $ ) | nindent 2 }} - {{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/rolebinding.yaml b/charts/redis/redis/templates/rolebinding.yaml deleted file mode 100644 index a164289..0000000 --- a/charts/redis/redis/templates/rolebinding.yaml +++ /dev/null @@ -1,23 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.rbac.create }} -apiVersion: {{ include "common.capabilities.rbac.apiVersion" . }} -kind: RoleBinding -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: {{ template "common.names.fullname" . }} -subjects: - - kind: ServiceAccount - name: {{ template "redis.serviceAccountName" . }} -{{- end }} diff --git a/charts/redis/redis/templates/scripts-configmap.yaml b/charts/redis/redis/templates/scripts-configmap.yaml deleted file mode 100644 index cf85307..0000000 --- a/charts/redis/redis/templates/scripts-configmap.yaml +++ /dev/null @@ -1,791 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ printf "%s-scripts" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: -{{- if and (eq .Values.architecture "replication") .Values.sentinel.enabled }} - start-node.sh: | - #!/bin/bash - - . /opt/bitnami/scripts/libos.sh - . /opt/bitnami/scripts/liblog.sh - . /opt/bitnami/scripts/libvalidations.sh - - get_port() { - hostname="$1" - type="$2" - - port_var=$(echo "${hostname^^}_SERVICE_PORT_$type" | sed "s/-/_/g") - port=${!port_var} - - if [ -z "$port" ]; then - case $type in - "SENTINEL") - echo {{ .Values.sentinel.containerPorts.sentinel }} - ;; - "REDIS") - echo {{ .Values.master.containerPorts.redis }} - ;; - esac - else - echo $port - fi - } - - get_full_hostname() { - hostname="$1" - - {{- if .Values.useExternalDNS.enabled }} - full_hostname="${hostname}.{{- include "redis.externalDNS.suffix" . }}" - {{- else if eq .Values.sentinel.service.type "NodePort" }} - full_hostname="${hostname}.{{- include "common.names.namespace" . }}" - {{- else }} - full_hostname="${hostname}.${HEADLESS_SERVICE}" - {{- end }} - - {{- if .Values.useHostnames }} - echo "${full_hostname}" - {{- else }} - retry_count=0 - until getent hosts "${full_hostname}" | awk '{ print $1; exit }' | grep .; do - if [[ $retry_count -lt {{ .Values.nameResolutionThreshold }} ]]; then - sleep {{ .Values.nameResolutionTimeout }} - else - error "IP address for ${full_hostname} not found" - exit 1 - fi - ((retry_count++)) - done - {{- end }} - } - - REDISPORT=$(get_port "$HOSTNAME" "REDIS") - - HEADLESS_SERVICE="{{ template "common.names.fullname" . }}-headless.{{ include "common.names.namespace" . }}.svc.{{ .Values.clusterDomain }}" - - if [ -n "$REDIS_EXTERNAL_MASTER_HOST" ]; then - REDIS_SERVICE="$REDIS_EXTERNAL_MASTER_HOST" - else - REDIS_SERVICE="{{ template "common.names.fullname" . }}.{{ include "common.names.namespace" . }}.svc.{{ .Values.clusterDomain }}" - fi - - SENTINEL_SERVICE_PORT=$(get_port "{{ include "common.names.fullname" . }}" "SENTINEL") - validate_quorum() { - if is_boolean_yes "$REDIS_TLS_ENABLED"; then - quorum_info_command="{{- if and .Values.auth.enabled .Values.auth.sentinel }}REDISCLI_AUTH="\$REDIS_PASSWORD" {{ end }}redis-cli -h $REDIS_SERVICE -p $SENTINEL_SERVICE_PORT --tls --cert ${REDIS_TLS_CERT_FILE} --key ${REDIS_TLS_KEY_FILE} --cacert ${REDIS_TLS_CA_FILE} sentinel master {{ .Values.sentinel.masterSet }}" - else - quorum_info_command="{{- if and .Values.auth.enabled .Values.auth.sentinel }}REDISCLI_AUTH="\$REDIS_PASSWORD" {{ end }}redis-cli -h $REDIS_SERVICE -p $SENTINEL_SERVICE_PORT sentinel master {{ .Values.sentinel.masterSet }}" - fi - info "about to run the command: $quorum_info_command" - eval $quorum_info_command | grep -Fq "s_down" - } - - trigger_manual_failover() { - if is_boolean_yes "$REDIS_TLS_ENABLED"; then - failover_command="{{- if and .Values.auth.enabled .Values.auth.sentinel }}REDISCLI_AUTH="\$REDIS_PASSWORD" {{ end }}redis-cli -h $REDIS_SERVICE -p $SENTINEL_SERVICE_PORT --tls --cert ${REDIS_TLS_CERT_FILE} --key ${REDIS_TLS_KEY_FILE} --cacert ${REDIS_TLS_CA_FILE} sentinel failover {{ .Values.sentinel.masterSet }}" - else - failover_command="{{- if and .Values.auth.enabled .Values.auth.sentinel }}REDISCLI_AUTH="\$REDIS_PASSWORD" {{ end }}redis-cli -h $REDIS_SERVICE -p $SENTINEL_SERVICE_PORT sentinel failover {{ .Values.sentinel.masterSet }}" - fi - - info "about to run the command: $failover_command" - eval $failover_command - } - - get_sentinel_master_info() { - if is_boolean_yes "$REDIS_TLS_ENABLED"; then - sentinel_info_command="{{- if and .Values.auth.enabled .Values.auth.sentinel }}REDISCLI_AUTH="\$REDIS_PASSWORD" {{ end }}timeout {{ .Values.sentinel.getMasterTimeout }} redis-cli -h $REDIS_SERVICE -p $SENTINEL_SERVICE_PORT --tls --cert ${REDIS_TLS_CERT_FILE} --key ${REDIS_TLS_KEY_FILE} --cacert ${REDIS_TLS_CA_FILE} sentinel get-master-addr-by-name {{ .Values.sentinel.masterSet }}" - else - sentinel_info_command="{{- if and .Values.auth.enabled .Values.auth.sentinel }}REDISCLI_AUTH="\$REDIS_PASSWORD" {{ end }}timeout {{ .Values.sentinel.getMasterTimeout }} redis-cli -h $REDIS_SERVICE -p $SENTINEL_SERVICE_PORT sentinel get-master-addr-by-name {{ .Values.sentinel.masterSet }}" - fi - - info "about to run the command: $sentinel_info_command" - retry_while "eval $sentinel_info_command" 2 5 - } - - {{- if and .Values.replica.containerSecurityContext.runAsUser (eq (.Values.replica.containerSecurityContext.runAsUser | int) 0) }} - useradd redis - chown -R redis {{ .Values.replica.persistence.path }} - {{- end }} - - [[ -f $REDIS_PASSWORD_FILE ]] && export REDIS_PASSWORD="$(< "${REDIS_PASSWORD_FILE}")" - [[ -f $REDIS_MASTER_PASSWORD_FILE ]] && export REDIS_MASTER_PASSWORD="$(< "${REDIS_MASTER_PASSWORD_FILE}")" - - # check if there is a master - master_in_persisted_conf="$(get_full_hostname "$HOSTNAME")" - master_port_in_persisted_conf="$REDIS_MASTER_PORT_NUMBER" - master_in_sentinel="$(get_sentinel_master_info)" - redisRetVal=$? - - if [[ -f /opt/bitnami/redis-sentinel/etc/sentinel.conf ]]; then - master_in_persisted_conf="$(awk '/monitor/ {print $4}' /opt/bitnami/redis-sentinel/etc/sentinel.conf)" - master_port_in_persisted_conf="$(awk '/monitor/ {print $5}' /opt/bitnami/redis-sentinel/etc/sentinel.conf)" - info "Found previous master ${master_in_persisted_conf}:${master_port_in_persisted_conf} in /opt/bitnami/redis-sentinel/etc/sentinel.conf" - debug "$(cat /opt/bitnami/redis-sentinel/etc/sentinel.conf | grep monitor)" - fi - - if [[ $redisRetVal -ne 0 ]]; then - if [[ "$master_in_persisted_conf" == "$(get_full_hostname "$HOSTNAME")" ]]; then - # Case 1: No active sentinel and in previous sentinel.conf we were the master --> MASTER - info "Configuring the node as master" - export REDIS_REPLICATION_MODE="master" - else - # Case 2: No active sentinel and in previous sentinel.conf we were not master --> REPLICA - info "Configuring the node as replica" - export REDIS_REPLICATION_MODE="replica" - REDIS_MASTER_HOST=${master_in_persisted_conf} - REDIS_MASTER_PORT_NUMBER=${master_port_in_persisted_conf} - fi - else - # Fetches current master's host and port - REDIS_SENTINEL_INFO=($(get_sentinel_master_info)) - info "Current master: REDIS_SENTINEL_INFO=(${REDIS_SENTINEL_INFO[0]},${REDIS_SENTINEL_INFO[1]})" - REDIS_MASTER_HOST=${REDIS_SENTINEL_INFO[0]} - REDIS_MASTER_PORT_NUMBER=${REDIS_SENTINEL_INFO[1]} - - if [[ "$REDIS_MASTER_HOST" == "$(get_full_hostname "$HOSTNAME")" ]]; then - # Case 3: Active sentinel and master it is this node --> MASTER - info "Configuring the node as master" - export REDIS_REPLICATION_MODE="master" - else - # Case 4: Active sentinel and master is not this node --> REPLICA - info "Configuring the node as replica" - export REDIS_REPLICATION_MODE="replica" - - {{- if and .Values.sentinel.automateClusterRecovery (le (int .Values.sentinel.downAfterMilliseconds) 2000) }} - retry_count=1 - while validate_quorum - do - info "sleeping, waiting for Redis master to come up" - sleep 1s - if ! ((retry_count % 11)); then - info "Trying to manually failover" - failover_result=$(trigger_manual_failover) - - debug "Failover result: $failover_result" - fi - - ((retry_count+=1)) - done - info "Redis master is up now" - {{- end }} - fi - fi - - if [[ -n "$REDIS_EXTERNAL_MASTER_HOST" ]]; then - REDIS_MASTER_HOST="$REDIS_EXTERNAL_MASTER_HOST" - REDIS_MASTER_PORT_NUMBER="${REDIS_EXTERNAL_MASTER_PORT}" - fi - - if [[ -f /opt/bitnami/redis/mounted-etc/replica.conf ]];then - cp /opt/bitnami/redis/mounted-etc/replica.conf /opt/bitnami/redis/etc/replica.conf - fi - - if [[ -f /opt/bitnami/redis/mounted-etc/redis.conf ]];then - cp /opt/bitnami/redis/mounted-etc/redis.conf /opt/bitnami/redis/etc/redis.conf - fi - - echo "" >> /opt/bitnami/redis/etc/replica.conf - echo "replica-announce-port $REDISPORT" >> /opt/bitnami/redis/etc/replica.conf - echo "replica-announce-ip $(get_full_hostname "$HOSTNAME")" >> /opt/bitnami/redis/etc/replica.conf - - {{- if .Values.tls.enabled }} - ARGS=("--port" "0") - ARGS+=("--tls-port" "${REDIS_TLS_PORT}") - ARGS+=("--tls-cert-file" "${REDIS_TLS_CERT_FILE}") - ARGS+=("--tls-key-file" "${REDIS_TLS_KEY_FILE}") - ARGS+=("--tls-ca-cert-file" "${REDIS_TLS_CA_FILE}") - ARGS+=("--tls-auth-clients" "${REDIS_TLS_AUTH_CLIENTS}") - ARGS+=("--tls-replication" "yes") - {{- if .Values.tls.dhParamsFilename }} - ARGS+=("--tls-dh-params-file" "${REDIS_TLS_DH_PARAMS_FILE}") - {{- end }} - {{- else }} - ARGS=("--port" "${REDIS_PORT}") - {{- end }} - - if [[ "$REDIS_REPLICATION_MODE" = "slave" ]] || [[ "$REDIS_REPLICATION_MODE" = "replica" ]]; then - ARGS+=("--replicaof" "${REDIS_MASTER_HOST}" "${REDIS_MASTER_PORT_NUMBER}") - fi - - {{- if .Values.auth.enabled }} - ARGS+=("--requirepass" "${REDIS_PASSWORD}") - ARGS+=("--masterauth" "${REDIS_MASTER_PASSWORD}") - {{- else }} - ARGS+=("--protected-mode" "no") - {{- end }} - ARGS+=("--include" "/opt/bitnami/redis/etc/replica.conf") - ARGS+=("--include" "/opt/bitnami/redis/etc/redis.conf") - {{- if .Values.replica.extraFlags }} - {{- range .Values.replica.extraFlags }} - ARGS+=({{ . | quote }}) - {{- end }} - {{- end }} - - {{- if .Values.replica.preExecCmds }} - {{- .Values.replica.preExecCmds | nindent 4 }} - {{- end }} - - {{- if .Values.replica.command }} - exec {{ .Values.replica.command }} "${ARGS[@]}" - {{- else }} - exec redis-server "${ARGS[@]}" - {{- end }} - - start-sentinel.sh: | - #!/bin/bash - - . /opt/bitnami/scripts/libos.sh - . /opt/bitnami/scripts/libvalidations.sh - . /opt/bitnami/scripts/libfile.sh - - HEADLESS_SERVICE="{{ template "common.names.fullname" . }}-headless.{{ include "common.names.namespace" . }}.svc.{{ .Values.clusterDomain }}" - REDIS_SERVICE="{{ template "common.names.fullname" . }}.{{ include "common.names.namespace" . }}.svc.{{ .Values.clusterDomain }}" - - get_port() { - hostname="$1" - type="$2" - - port_var=$(echo "${hostname^^}_SERVICE_PORT_$type" | sed "s/-/_/g") - port=${!port_var} - - if [ -z "$port" ]; then - case $type in - "SENTINEL") - echo {{ .Values.sentinel.containerPorts.sentinel }} - ;; - "REDIS") - echo {{ .Values.master.containerPorts.redis }} - ;; - esac - else - echo $port - fi - } - - get_full_hostname() { - hostname="$1" - - {{- if .Values.useExternalDNS.enabled }} - full_hostname="${hostname}.{{- include "redis.externalDNS.suffix" . }}" - {{- else if eq .Values.sentinel.service.type "NodePort" }} - full_hostname="${hostname}.{{- include "common.names.namespace" . }}" - {{- else }} - full_hostname="${hostname}.${HEADLESS_SERVICE}" - {{- end }} - - {{- if .Values.useHostnames }} - echo "${full_hostname}" - {{- else }} - retry_count=0 - until getent hosts "${full_hostname}" | awk '{ print $1; exit }' | grep .; do - if [[ $retry_count -lt {{ .Values.nameResolutionThreshold }} ]]; then - sleep {{ .Values.nameResolutionTimeout }} - else - error "IP address for ${full_hostname} not found" - exit 1 - fi - ((retry_count++)) - done - {{- end }} - } - - SERVPORT=$(get_port "$HOSTNAME" "SENTINEL") - REDISPORT=$(get_port "$HOSTNAME" "REDIS") - SENTINEL_SERVICE_PORT=$(get_port "{{ include "common.names.fullname" . }}" "SENTINEL") - - sentinel_conf_set() { - local -r key="${1:?missing key}" - local value="${2:-}" - - # Sanitize inputs - value="${value//\\/\\\\}" - value="${value//&/\\&}" - value="${value//\?/\\?}" - [[ "$value" = "" ]] && value="\"$value\"" - - replace_in_file "/opt/bitnami/redis-sentinel/etc/prepare-sentinel.conf" "^#*\s*${key} .*" "${key} ${value}" false - } - sentinel_conf_add() { - echo $'\n'"$@" >> "/opt/bitnami/redis-sentinel/etc/prepare-sentinel.conf" - } - host_id() { - echo "$1" | openssl sha1 | awk '{print $2}' - } - get_sentinel_master_info() { - if is_boolean_yes "$REDIS_SENTINEL_TLS_ENABLED"; then - sentinel_info_command="{{- if and .Values.auth.enabled .Values.auth.sentinel }}REDISCLI_AUTH="\$REDIS_PASSWORD" {{ end }}timeout {{ .Values.sentinel.getMasterTimeout }} redis-cli -h $REDIS_SERVICE -p $SENTINEL_SERVICE_PORT --tls --cert ${REDIS_SENTINEL_TLS_CERT_FILE} --key ${REDIS_SENTINEL_TLS_KEY_FILE} --cacert ${REDIS_SENTINEL_TLS_CA_FILE} sentinel get-master-addr-by-name {{ .Values.sentinel.masterSet }}" - else - sentinel_info_command="{{- if and .Values.auth.enabled .Values.auth.sentinel }}REDISCLI_AUTH="\$REDIS_PASSWORD" {{ end }}timeout {{ .Values.sentinel.getMasterTimeout }} redis-cli -h $REDIS_SERVICE -p $SENTINEL_SERVICE_PORT sentinel get-master-addr-by-name {{ .Values.sentinel.masterSet }}" - fi - info "about to run the command: $sentinel_info_command" - retry_while "eval $sentinel_info_command" 2 5 - } - - [[ -f $REDIS_PASSWORD_FILE ]] && export REDIS_PASSWORD="$(< "${REDIS_PASSWORD_FILE}")" - - master_in_persisted_conf="$(get_full_hostname "$HOSTNAME")" - - if [[ -f /opt/bitnami/redis-sentinel/etc/sentinel.conf ]]; then - master_in_persisted_conf="$(awk '/monitor/ {print $4}' /opt/bitnami/redis-sentinel/etc/sentinel.conf)" - info "Found previous master $master_in_persisted_conf in /opt/bitnami/redis-sentinel/etc/sentinel.conf" - debug "$(cat /opt/bitnami/redis-sentinel/etc/sentinel.conf | grep monitor)" - fi - REDIS_SENTINEL_INFO=($(get_sentinel_master_info)) - if [ "$?" -eq "0" ]; then - # current master's host and port obtained from other Sentinel - info "printing REDIS_SENTINEL_INFO=(${REDIS_SENTINEL_INFO[0]},${REDIS_SENTINEL_INFO[1]})" - REDIS_MASTER_HOST=${REDIS_SENTINEL_INFO[0]} - REDIS_MASTER_PORT_NUMBER=${REDIS_SENTINEL_INFO[1]} - else - REDIS_MASTER_HOST="$master_in_persisted_conf" - REDIS_MASTER_PORT_NUMBER="$REDISPORT" - fi - if [[ "$REDIS_MASTER_HOST" == "$(get_full_hostname "$HOSTNAME")" ]]; then - export REDIS_REPLICATION_MODE="master" - else - export REDIS_REPLICATION_MODE="replica" - fi - - {{- if or .Values.sentinel.masterService.enabled .Values.sentinel.service.createMaster }} - if [[ "${REDIS_REPLICATION_MODE}" == "master" ]]; then - # Add isMaster label to master node for master service - echo "${REDIS_MASTER_HOST/.*}" > /etc/shared/current - fi - {{- end }} - - if [[ -n "$REDIS_EXTERNAL_MASTER_HOST" ]]; then - REDIS_MASTER_HOST="$REDIS_EXTERNAL_MASTER_HOST" - REDIS_MASTER_PORT_NUMBER="${REDIS_EXTERNAL_MASTER_PORT}" - fi - - # To prevent incomplete configuration and as the redis container accesses /opt/bitnami/redis-sentinel/etc/sentinel.conf - # as well, prepare the new config in `prepare-sentinel.conf` and move it atomically to the ultimate destination when it is complete. - cp /opt/bitnami/redis-sentinel/mounted-etc/sentinel.conf /opt/bitnami/redis-sentinel/etc/prepare-sentinel.conf - {{- if .Values.auth.enabled }} - printf "\nsentinel auth-pass %s %s" "{{ .Values.sentinel.masterSet }}" "$REDIS_PASSWORD" >> /opt/bitnami/redis-sentinel/etc/prepare-sentinel.conf - {{- if and .Values.auth.enabled .Values.auth.sentinel }} - printf "\nrequirepass %s" "$REDIS_PASSWORD" >> /opt/bitnami/redis-sentinel/etc/prepare-sentinel.conf - {{- end }} - {{- end }} - printf "\nsentinel myid %s" "$(host_id "$HOSTNAME")" >> /opt/bitnami/redis-sentinel/etc/prepare-sentinel.conf - - if [[ -z "$REDIS_MASTER_HOST" ]] || [[ -z "$REDIS_MASTER_PORT_NUMBER" ]] - then - # Prevent incorrect configuration to be written to sentinel.conf - error "Redis master host is configured incorrectly (host: $REDIS_MASTER_HOST, port: $REDIS_MASTER_PORT_NUMBER)" - exit 1 - fi - - sentinel_conf_set "sentinel monitor" "{{ .Values.sentinel.masterSet }} "$REDIS_MASTER_HOST" "$REDIS_MASTER_PORT_NUMBER" {{ .Values.sentinel.quorum }}" - - add_known_sentinel() { - hostname="$1" - ip="$2" - - if [[ -n "$hostname" && -n "$ip" && "$hostname" != "$HOSTNAME" ]]; then - sentinel_conf_add "sentinel known-sentinel {{ .Values.sentinel.masterSet }} $(get_full_hostname "$hostname") $(get_port "$hostname" "SENTINEL") $(host_id "$hostname")" - fi - } - add_known_replica() { - hostname="$1" - ip="$2" - - if [[ -n "$ip" && "$(get_full_hostname "$hostname")" != "$REDIS_MASTER_HOST" ]]; then - sentinel_conf_add "sentinel known-replica {{ .Values.sentinel.masterSet }} $(get_full_hostname "$hostname") $(get_port "$hostname" "REDIS")" - fi - } - - # Add available hosts on the network as known replicas & sentinels - for node in $(seq 0 $(({{ .Values.replica.replicaCount }}-1))); do - hostname="{{ template "common.names.fullname" . }}-node-$node" - ip="$(getent hosts "$hostname.$HEADLESS_SERVICE" | awk '{ print $1 }')" - add_known_sentinel "$hostname" "$ip" - add_known_replica "$hostname" "$ip" - done - - echo "" >> /opt/bitnami/redis-sentinel/etc/prepare-sentinel.conf - {{- if not (contains "sentinel announce-hostnames" .Values.sentinel.configuration) }} - echo "sentinel announce-hostnames yes" >> /opt/bitnami/redis-sentinel/etc/prepare-sentinel.conf - {{- end }} - {{- if not (contains "sentinel resolve-hostnames" .Values.sentinel.configuration) }} - echo "sentinel resolve-hostnames yes" >> /opt/bitnami/redis-sentinel/etc/prepare-sentinel.conf - {{- end }} - {{- if not (contains "sentinel announce-port" .Values.sentinel.configuration) }} - echo "sentinel announce-port $SERVPORT" >> /opt/bitnami/redis-sentinel/etc/prepare-sentinel.conf - {{- end }} - {{- if not (contains "sentinel announce-ip" .Values.sentinel.configuration) }} - echo "sentinel announce-ip $(get_full_hostname "$HOSTNAME")" >> /opt/bitnami/redis-sentinel/etc/prepare-sentinel.conf - {{- end }} - - {{- if .Values.tls.enabled }} - ARGS=("--port" "0") - ARGS+=("--tls-port" "${REDIS_SENTINEL_TLS_PORT_NUMBER}") - ARGS+=("--tls-cert-file" "${REDIS_SENTINEL_TLS_CERT_FILE}") - ARGS+=("--tls-key-file" "${REDIS_SENTINEL_TLS_KEY_FILE}") - ARGS+=("--tls-ca-cert-file" "${REDIS_SENTINEL_TLS_CA_FILE}") - ARGS+=("--tls-replication" "yes") - ARGS+=("--tls-auth-clients" "${REDIS_SENTINEL_TLS_AUTH_CLIENTS}") - {{- if .Values.tls.dhParamsFilename }} - ARGS+=("--tls-dh-params-file" "${REDIS_SENTINEL_TLS_DH_PARAMS_FILE}") - {{- end }} - {{- end }} - {{- if .Values.sentinel.preExecCmds }} - {{ .Values.sentinel.preExecCmds | nindent 4 }} - {{- end }} - mv /opt/bitnami/redis-sentinel/etc/prepare-sentinel.conf /opt/bitnami/redis-sentinel/etc/sentinel.conf - exec redis-server /opt/bitnami/redis-sentinel/etc/sentinel.conf {{- if .Values.tls.enabled }} "${ARGS[@]}" {{- end }} --sentinel - prestop-sentinel.sh: | - #!/bin/bash - - . /opt/bitnami/scripts/libvalidations.sh - . /opt/bitnami/scripts/libos.sh - - HEADLESS_SERVICE="{{ template "common.names.fullname" . }}-headless.{{ include "common.names.namespace" . }}.svc.{{ .Values.clusterDomain }}" - - get_full_hostname() { - hostname="$1" - - {{- if .Values.useExternalDNS.enabled }} - full_hostname="${hostname}.{{- include "redis.externalDNS.suffix" . }}" - {{- else if eq .Values.sentinel.service.type "NodePort" }} - full_hostname="${hostname}.{{- include "common.names.namespace" . }}" - {{- else }} - full_hostname="${hostname}.${HEADLESS_SERVICE}" - {{- end }} - - {{- if .Values.useHostnames }} - echo "${full_hostname}" - {{- else }} - retry_count=0 - until getent hosts "${full_hostname}" | awk '{ print $1; exit }' | grep .; do - if [[ $retry_count -lt {{ .Values.nameResolutionThreshold }} ]]; then - sleep {{ .Values.nameResolutionTimeout }} - else - error "IP address for ${full_hostname} not found" - exit 1 - fi - ((retry_count++)) - done - {{- end }} - } - - run_sentinel_command() { - if is_boolean_yes "$REDIS_SENTINEL_TLS_ENABLED"; then - redis-cli -h "$REDIS_SERVICE" -p "$REDIS_SENTINEL_TLS_PORT_NUMBER" --tls --cert "$REDIS_SENTINEL_TLS_CERT_FILE" --key "$REDIS_SENTINEL_TLS_KEY_FILE" --cacert "$REDIS_SENTINEL_TLS_CA_FILE" sentinel "$@" - else - redis-cli -h "$REDIS_SERVICE" -p "$REDIS_SENTINEL_PORT" sentinel "$@" - fi - } - sentinel_failover_finished() { - REDIS_SENTINEL_INFO=($(run_sentinel_command get-master-addr-by-name "{{ .Values.sentinel.masterSet }}")) - REDIS_MASTER_HOST="${REDIS_SENTINEL_INFO[0]}" - [[ "$REDIS_MASTER_HOST" != "$(get_full_hostname $HOSTNAME)" ]] - } - - REDIS_SERVICE="{{ include "common.names.fullname" . }}.{{ include "common.names.namespace" . }}.svc.{{ .Values.clusterDomain }}" - - {{ if .Values.auth.sentinel -}} - # redis-cli automatically consumes credentials from the REDISCLI_AUTH variable - [[ -n "$REDIS_PASSWORD" ]] && export REDISCLI_AUTH="$REDIS_PASSWORD" - [[ -f "$REDIS_PASSWORD_FILE" ]] && export REDISCLI_AUTH="$(< "${REDIS_PASSWORD_FILE}")" - {{- end }} - - if ! sentinel_failover_finished; then - echo "I am the master pod and you are stopping me. Starting sentinel failover" - if retry_while "sentinel_failover_finished" "{{ sub .Values.sentinel.terminationGracePeriodSeconds 10 }}" 1; then - echo "Master has been successfuly failed over to a different pod." - exit 0 - else - echo "Master failover failed" - exit 1 - fi - else - exit 0 - fi - prestop-redis.sh: | - #!/bin/bash - - . /opt/bitnami/scripts/libvalidations.sh - . /opt/bitnami/scripts/libos.sh - - run_redis_command() { - if is_boolean_yes "$REDIS_TLS_ENABLED"; then - redis-cli -h 127.0.0.1 -p "$REDIS_TLS_PORT" --tls --cert "$REDIS_TLS_CERT_FILE" --key "$REDIS_TLS_KEY_FILE" --cacert "$REDIS_TLS_CA_FILE" "$@" - else - redis-cli -h 127.0.0.1 -p "$REDIS_PORT" "$@" - fi - } - is_master() { - REDIS_ROLE=$(run_redis_command role | head -1) - [[ "$REDIS_ROLE" == "master" ]] - } - - HEADLESS_SERVICE="{{ template "common.names.fullname" . }}-headless.{{- include "common.names.namespace" . }}.svc.{{ .Values.clusterDomain }}" - - get_full_hostname() { - hostname="$1" - - {{- if .Values.useExternalDNS.enabled }} - full_hostname="${hostname}.{{- include "redis.externalDNS.suffix" . }}" - {{- else if eq .Values.sentinel.service.type "NodePort" }} - full_hostname="${hostname}.{{- include "common.names.namespace" . }}" - {{- else }} - full_hostname="${hostname}.${HEADLESS_SERVICE}" - {{- end }} - - {{- if .Values.useHostnames }} - echo "${full_hostname}" - {{- else }} - retry_count=0 - until getent hosts "${full_hostname}" | awk '{ print $1; exit }' | grep .; do - if [[ $retry_count -lt {{ .Values.nameResolutionThreshold }} ]]; then - sleep {{ .Values.nameResolutionTimeout }} - else - error "IP address for ${full_hostname} not found" - exit 1 - fi - ((retry_count++)) - done - {{- end }} - } - - run_sentinel_command() { - if is_boolean_yes "$REDIS_SENTINEL_TLS_ENABLED"; then - {{ .Values.auth.sentinel | ternary "" "env -u REDISCLI_AUTH " -}} redis-cli -h "$REDIS_SERVICE" -p "$REDIS_SENTINEL_TLS_PORT_NUMBER" --tls --cert "$REDIS_SENTINEL_TLS_CERT_FILE" --key "$REDIS_SENTINEL_TLS_KEY_FILE" --cacert "$REDIS_SENTINEL_TLS_CA_FILE" sentinel "$@" - else - {{ .Values.auth.sentinel | ternary "" "env -u REDISCLI_AUTH " -}} redis-cli -h "$REDIS_SERVICE" -p "$REDIS_SENTINEL_PORT" sentinel "$@" - fi - } - sentinel_failover_finished() { - REDIS_SENTINEL_INFO=($(run_sentinel_command get-master-addr-by-name "{{ .Values.sentinel.masterSet }}")) - REDIS_MASTER_HOST="${REDIS_SENTINEL_INFO[0]}" - [[ "$REDIS_MASTER_HOST" != "$(get_full_hostname $HOSTNAME)" ]] - } - - REDIS_SERVICE="{{ include "common.names.fullname" . }}.{{ include "common.names.namespace" . }}.svc.{{ .Values.clusterDomain }}" - - # redis-cli automatically consumes credentials from the REDISCLI_AUTH variable - [[ -n "$REDIS_PASSWORD" ]] && export REDISCLI_AUTH="$REDIS_PASSWORD" - [[ -f "$REDIS_PASSWORD_FILE" ]] && export REDISCLI_AUTH="$(< "${REDIS_PASSWORD_FILE}")" - - - if is_master && ! sentinel_failover_finished; then - echo "I am the master pod and you are stopping me. Pausing client connections." - # Pausing client write connections to avoid data loss - run_redis_command CLIENT PAUSE "{{ mul (add 2 (sub .Values.sentinel.terminationGracePeriodSeconds 10)) 1000 }}" WRITE - - echo "Issuing failover" - # if I am the master, issue a command to failover once - run_sentinel_command failover "{{ .Values.sentinel.masterSet }}" - - {{- if .Values.sentinel.redisShutdownWaitFailover }} - echo "Waiting for sentinel to complete failover for up to {{ sub .Values.sentinel.terminationGracePeriodSeconds 10 }}s" - retry_while "sentinel_failover_finished" "{{ sub .Values.sentinel.terminationGracePeriodSeconds 10 }}" 1 - {{- end }} - else - exit 0 - fi - - {{- if or .Values.sentinel.masterService.enabled .Values.sentinel.service.createMaster }} - push-master-label.sh: | - #!/bin/bash - # https://download.redis.io/redis-stable/sentinel.conf - - echo "${6/.*}" > /etc/shared/current - echo "${4/.*}" > /etc/shared/previous - {{- end }} -{{- else }} - start-master.sh: | - #!/bin/bash - - [[ -f $REDIS_PASSWORD_FILE ]] && export REDIS_PASSWORD="$(< "${REDIS_PASSWORD_FILE}")" - {{- if and .Values.master.containerSecurityContext.runAsUser (eq (.Values.master.containerSecurityContext.runAsUser | int) 0) }} - useradd redis - chown -R redis {{ .Values.master.persistence.path }} - {{- end }} - if [[ -f /opt/bitnami/redis/mounted-etc/master.conf ]];then - cp /opt/bitnami/redis/mounted-etc/master.conf /opt/bitnami/redis/etc/master.conf - fi - if [[ -f /opt/bitnami/redis/mounted-etc/redis.conf ]];then - cp /opt/bitnami/redis/mounted-etc/redis.conf /opt/bitnami/redis/etc/redis.conf - fi - {{- if .Values.tls.enabled }} - ARGS=("--port" "0") - ARGS+=("--tls-port" "${REDIS_TLS_PORT}") - ARGS+=("--tls-cert-file" "${REDIS_TLS_CERT_FILE}") - ARGS+=("--tls-key-file" "${REDIS_TLS_KEY_FILE}") - ARGS+=("--tls-ca-cert-file" "${REDIS_TLS_CA_FILE}") - ARGS+=("--tls-auth-clients" "${REDIS_TLS_AUTH_CLIENTS}") - {{- if .Values.tls.dhParamsFilename }} - ARGS+=("--tls-dh-params-file" "${REDIS_TLS_DH_PARAMS_FILE}") - {{- end }} - {{- else }} - ARGS=("--port" "${REDIS_PORT}") - {{- end }} - {{- if .Values.auth.enabled }} - ARGS+=("--requirepass" "${REDIS_PASSWORD}") - ARGS+=("--masterauth" "${REDIS_PASSWORD}") - {{- else }} - ARGS+=("--protected-mode" "no") - {{- end }} - ARGS+=("--include" "/opt/bitnami/redis/etc/redis.conf") - ARGS+=("--include" "/opt/bitnami/redis/etc/master.conf") - {{- if .Values.master.extraFlags }} - {{- range .Values.master.extraFlags }} - ARGS+=({{ . | quote }}) - {{- end }} - {{- end }} - {{- if .Values.master.preExecCmds }} - {{ .Values.master.preExecCmds | nindent 4 }} - {{- end }} - {{- if .Values.master.command }} - exec {{ .Values.master.command }} "${ARGS[@]}" - {{- else }} - exec redis-server "${ARGS[@]}" - {{- end }} - {{- if eq .Values.architecture "replication" }} - start-replica.sh: | - #!/bin/bash - - get_port() { - hostname="$1" - type="$2" - - port_var=$(echo "${hostname^^}_SERVICE_PORT_$type" | sed "s/-/_/g") - port=${!port_var} - - if [ -z "$port" ]; then - case $type in - "SENTINEL") - echo {{ .Values.sentinel.containerPorts.sentinel }} - ;; - "REDIS") - echo {{ .Values.master.containerPorts.redis }} - ;; - esac - else - echo $port - fi - } - - get_full_hostname() { - hostname="$1" - - {{- if .Values.useExternalDNS.enabled }} - full_hostname="${hostname}.{{- include "redis.externalDNS.suffix" . }}" - {{- else if eq .Values.sentinel.service.type "NodePort" }} - full_hostname="${hostname}.{{- include "common.names.namespace" . }}" - {{- else }} - full_hostname="${hostname}.${HEADLESS_SERVICE}" - {{- end }} - - {{- if .Values.useHostnames }} - echo "${full_hostname}" - {{- else }} - retry_count=0 - until getent hosts "${full_hostname}" | awk '{ print $1; exit }' | grep .; do - if [[ $retry_count -lt {{ .Values.nameResolutionThreshold }} ]]; then - sleep {{ .Values.nameResolutionTimeout }} - else - error "IP address for ${full_hostname} not found" - exit 1 - fi - ((retry_count++)) - done - {{- end }} - } - - REDISPORT=$(get_port "$HOSTNAME" "REDIS") - HEADLESS_SERVICE="{{ template "common.names.fullname" . }}-headless.{{ include "common.names.namespace" . }}.svc.{{ .Values.clusterDomain }}" - - [[ -f $REDIS_PASSWORD_FILE ]] && export REDIS_PASSWORD="$(< "${REDIS_PASSWORD_FILE}")" - [[ -f $REDIS_MASTER_PASSWORD_FILE ]] && export REDIS_MASTER_PASSWORD="$(< "${REDIS_MASTER_PASSWORD_FILE}")" - {{- if and .Values.replica.containerSecurityContext.runAsUser (eq (.Values.replica.containerSecurityContext.runAsUser | int) 0) }} - useradd redis - chown -R redis {{ .Values.replica.persistence.path }} - {{- end }} - if [[ -f /opt/bitnami/redis/mounted-etc/replica.conf ]];then - cp /opt/bitnami/redis/mounted-etc/replica.conf /opt/bitnami/redis/etc/replica.conf - fi - if [[ -f /opt/bitnami/redis/mounted-etc/redis.conf ]];then - cp /opt/bitnami/redis/mounted-etc/redis.conf /opt/bitnami/redis/etc/redis.conf - fi - - echo "" >> /opt/bitnami/redis/etc/replica.conf - echo "replica-announce-port $REDISPORT" >> /opt/bitnami/redis/etc/replica.conf - echo "replica-announce-ip $(get_full_hostname "$HOSTNAME")" >> /opt/bitnami/redis/etc/replica.conf - - {{- if .Values.tls.enabled }} - ARGS=("--port" "0") - ARGS+=("--tls-port" "${REDIS_TLS_PORT}") - ARGS+=("--tls-cert-file" "${REDIS_TLS_CERT_FILE}") - ARGS+=("--tls-key-file" "${REDIS_TLS_KEY_FILE}") - ARGS+=("--tls-ca-cert-file" "${REDIS_TLS_CA_FILE}") - ARGS+=("--tls-auth-clients" "${REDIS_TLS_AUTH_CLIENTS}") - ARGS+=("--tls-replication" "yes") - {{- if .Values.tls.dhParamsFilename }} - ARGS+=("--tls-dh-params-file" "${REDIS_TLS_DH_PARAMS_FILE}") - {{- end }} - {{- else }} - ARGS=("--port" "${REDIS_PORT}") - {{- end }} - ARGS+=("--replicaof" "${REDIS_MASTER_HOST}" "${REDIS_MASTER_PORT_NUMBER}") - {{- if .Values.auth.enabled }} - ARGS+=("--requirepass" "${REDIS_PASSWORD}") - ARGS+=("--masterauth" "${REDIS_MASTER_PASSWORD}") - {{- else }} - ARGS+=("--protected-mode" "no") - {{- end }} - ARGS+=("--include" "/opt/bitnami/redis/etc/redis.conf") - ARGS+=("--include" "/opt/bitnami/redis/etc/replica.conf") - {{- if .Values.replica.extraFlags }} - {{- range .Values.replica.extraFlags }} - ARGS+=({{ . | quote }}) - {{- end }} - {{- end }} - {{- if .Values.replica.preExecCmds }} - {{ .Values.replica.preExecCmds | nindent 4 }} - {{- end }} - {{- if .Values.replica.command }} - exec {{ .Values.replica.command }} "${ARGS[@]}" - {{- else }} - exec redis-server "${ARGS[@]}" - {{- end }} - {{- end }} -{{- end }} ---- -{{- if or .Values.sentinel.masterService.enabled .Values.sentinel.service.createMaster }} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ printf "%s-kubectl-scripts" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: - update-master-label.sh: | - #!/bin/bash - while true; do - while [ ! -f "/etc/shared/current" ]; do - sleep 1 - done - echo "new master elected, updating label(s)..." - kubectl label pod --field-selector metadata.name="$(< "/etc/shared/current")" isMaster="true" --overwrite - if [ -f /etc/shared/previous ]; then - kubectl label pod --field-selector metadata.name="$(< "/etc/shared/previous")" isMaster="false" --overwrite - fi - rm "/etc/shared/current" "/etc/shared/previous" - done -{{- end }} diff --git a/charts/redis/redis/templates/secret-svcbind.yaml b/charts/redis/redis/templates/secret-svcbind.yaml deleted file mode 100644 index d3c74ff..0000000 --- a/charts/redis/redis/templates/secret-svcbind.yaml +++ /dev/null @@ -1,37 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if .Values.serviceBindings.enabled }} -{{- $host := include "common.names.fullname" . }} -{{- if not .Values.sentinel.enabled }} -{{- $host = printf "%s-master" (include "common.names.fullname" .) }} -{{- end }} -{{- $port := print .Values.master.service.ports.redis }} -{{- if .Values.sentinel.enabled }} -{{- $port = print .Values.sentinel.service.ports.redis }} -{{- end }} -{{- $password := include "redis.password" . }} -apiVersion: v1 -kind: Secret -metadata: - name: {{ include "common.names.fullname" . }}-svcbind - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: servicebinding.io/redis -data: - provider: {{ print "bitnami" | b64enc | quote }} - type: {{ print "redis" | b64enc | quote }} - host: {{ print $host | b64enc | quote }} - port: {{ print $port | b64enc | quote }} - password: {{ print $password | b64enc | quote }} - {{- if $password }} - uri: {{ printf "redis://:%s@%s:%s" $password $host $port | b64enc | quote }} - {{- else }} - uri: {{ printf "redis://%s:%s" $host $port | b64enc | quote }} - {{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/secret.yaml b/charts/redis/redis/templates/secret.yaml deleted file mode 100644 index ec69fe2..0000000 --- a/charts/redis/redis/templates/secret.yaml +++ /dev/null @@ -1,25 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.auth.enabled (not .Values.auth.existingSecret) (or .Values.auth.usePasswordFileFromSecret (not .Values.auth.usePasswordFiles)) -}} -apiVersion: v1 -kind: Secret -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if or .Values.secretAnnotations .Values.commonAnnotations }} - annotations: - {{- if .Values.secretAnnotations }} - {{- include "common.tplvalues.render" ( dict "value" .Values.secretAnnotations "context" $ ) | nindent 4 }} - {{- end }} - {{- if .Values.commonAnnotations }} - {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} - {{- end }} -type: Opaque -data: - redis-password: {{ include "redis.password" . | b64enc | quote }} -{{- end -}} diff --git a/charts/redis/redis/templates/sentinel/hpa.yaml b/charts/redis/redis/templates/sentinel/hpa.yaml deleted file mode 100644 index 54ec485..0000000 --- a/charts/redis/redis/templates/sentinel/hpa.yaml +++ /dev/null @@ -1,49 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.replica.autoscaling.enabled .Values.sentinel.enabled }} -apiVersion: {{ include "common.capabilities.hpa.apiVersion" ( dict "context" $ ) }} -kind: HorizontalPodAutoscaler -metadata: - name: {{ printf "%s-node" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: replica - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - scaleTargetRef: - apiVersion: {{ include "common.capabilities.deployment.apiVersion" . }} - kind: StatefulSet - name: {{ printf "%s-node" (include "common.names.fullname" .) }} - minReplicas: {{ .Values.replica.autoscaling.minReplicas }} - maxReplicas: {{ .Values.replica.autoscaling.maxReplicas }} - metrics: - {{- if .Values.replica.autoscaling.targetMemory }} - - type: Resource - resource: - name: memory - {{- if semverCompare "<1.23-0" (include "common.capabilities.kubeVersion" .) }} - targetAverageUtilization: {{ .Values.replica.autoscaling.targetMemory }} - {{- else }} - target: - type: Utilization - averageUtilization: {{ .Values.replica.autoscaling.targetMemory }} - {{- end }} - {{- end }} - {{- if .Values.replica.autoscaling.targetCPU }} - - type: Resource - resource: - name: cpu - {{- if semverCompare "<1.23-0" (include "common.capabilities.kubeVersion" .) }} - targetAverageUtilization: {{ .Values.replica.autoscaling.targetCPU }} - {{- else }} - target: - type: Utilization - averageUtilization: {{ .Values.replica.autoscaling.targetCPU }} - {{- end }} - {{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/sentinel/node-services.yaml b/charts/redis/redis/templates/sentinel/node-services.yaml deleted file mode 100644 index 30ccad5..0000000 --- a/charts/redis/redis/templates/sentinel/node-services.yaml +++ /dev/null @@ -1,67 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and (eq .Values.architecture "replication") .Values.sentinel.enabled (eq .Values.sentinel.service.type "NodePort") (or .Release.IsUpgrade .Values.sentinel.service.nodePorts.redis ) }} - -{{- range $i := until (int .Values.replica.replicaCount) }} - -{{ $portsmap := (lookup "v1" "ConfigMap" (include "common.names.namespace" $) (printf "%s-%s" ( include "common.names.fullname" $ ) "ports-configmap")).data }} - -{{ $sentinelport := 0}} -{{ $redisport := 0}} -{{- if $portsmap }} -{{ $sentinelport = index $portsmap (printf "%s-node-%s-%s" (include "common.names.fullname" $) (toString $i) "sentinel") }} -{{ $redisport = index $portsmap (printf "%s-node-%s-%s" (include "common.names.fullname" $) (toString $i) "redis") }} -{{- else }} -{{- end }} -apiVersion: v1 -kind: Service -metadata: - name: {{ template "common.names.fullname" $ }}-node-{{ $i }} - namespace: {{ include "common.names.namespace" $ | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" $.Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: node - {{- if or $.Values.commonAnnotations $.Values.sentinel.service.annotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list $.Values.sentinel.service.annotations $.Values.commonAnnotations ) "context" $ ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: NodePort - ports: - - name: sentinel - {{- if $.Values.sentinel.service.nodePorts.sentinel }} - nodePort: {{ (add $.Values.sentinel.service.nodePorts.sentinel $i 1) }} - port: {{ (add $.Values.sentinel.service.nodePorts.sentinel $i 1) }} - {{- else }} - nodePort: {{ $sentinelport }} - port: {{ $sentinelport }} - {{- end }} - protocol: TCP - targetPort: {{ $.Values.sentinel.containerPorts.sentinel }} - - name: redis - {{- if $.Values.sentinel.service.nodePorts.redis }} - nodePort: {{ (add $.Values.sentinel.service.nodePorts.redis $i 1) }} - port: {{ (add $.Values.sentinel.service.nodePorts.redis $i 1) }} - {{- else }} - nodePort: {{ $redisport }} - port: {{ $redisport }} - {{- end }} - protocol: TCP - targetPort: {{ $.Values.replica.containerPorts.redis }} - - name: sentinel-internal - nodePort: null - port: {{ $.Values.sentinel.containerPorts.sentinel }} - protocol: TCP - targetPort: {{ $.Values.sentinel.containerPorts.sentinel }} - - name: redis-internal - nodePort: null - port: {{ $.Values.replica.containerPorts.redis }} - protocol: TCP - targetPort: {{ $.Values.replica.containerPorts.redis }} - selector: - statefulset.kubernetes.io/pod-name: {{ template "common.names.fullname" $ }}-node-{{ $i }} ---- -{{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/sentinel/pdb.yaml b/charts/redis/redis/templates/sentinel/pdb.yaml deleted file mode 100644 index 32ddad6..0000000 --- a/charts/redis/redis/templates/sentinel/pdb.yaml +++ /dev/null @@ -1,27 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} -{{- $pdb := coalesce .Values.pdb .Values.replica.pdb }} -{{- if and (eq .Values.architecture "replication") .Values.sentinel.enabled $pdb.create }} -apiVersion: {{ include "common.capabilities.policy.apiVersion" . }} -kind: PodDisruptionBudget -metadata: - name: {{ printf "%s-node" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: node - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - {{- if $pdb.minAvailable }} - minAvailable: {{ $pdb.minAvailable }} - {{- end }} - {{- if or $pdb.maxUnavailable (not $pdb.minAvailable) }} - maxUnavailable: {{ $pdb.maxUnavailable | default 1 }} - {{- end }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: node -{{- end }} diff --git a/charts/redis/redis/templates/sentinel/ports-configmap.yaml b/charts/redis/redis/templates/sentinel/ports-configmap.yaml deleted file mode 100644 index d55f01a..0000000 --- a/charts/redis/redis/templates/sentinel/ports-configmap.yaml +++ /dev/null @@ -1,102 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and (eq .Values.architecture "replication") .Values.sentinel.enabled (eq .Values.sentinel.service.type "NodePort") (not .Values.sentinel.service.nodePorts.redis ) }} -{{- /* create a list to keep track of ports we choose to use */}} -{{ $chosenports := (list ) }} - -{{- /* Get list of all used nodeports */}} -{{ $usedports := (list ) }} -{{- range $index, $service := (lookup "v1" "Service" "" "").items }} - {{- range.spec.ports }} - {{- if .nodePort }} - {{- $usedports = (append $usedports .nodePort) }} - {{- end }} - {{- end }} -{{- end }} - -{{- /* -comments that start with # are rendered in the output when you debug, so you can less and search for them -Vars in the comment will be rendered out, so you can check their value this way. -https://helm.sh/docs/chart_best_practices/templates/#comments-yaml-comments-vs-template-comments - -remove the template comments and leave the yaml comments to help debug -*/}} - -{{- /* Sort the list */}} -{{ $usedports = $usedports | sortAlpha }} -#usedports {{ $usedports }} - -{{- /* How many nodeports per service do we want to create, except for the main service which is always two */}} -{{ $numberofPortsPerNodeService := 2 }} - -{{- /* for every nodeport we want, loop though the used ports to get an unused port */}} -{{- range $j := until (int (add (mul (int .Values.replica.replicaCount) $numberofPortsPerNodeService) 2)) }} - {{- /* #j={{ $j }} */}} - {{- $nodeport := (add $j 30000) }} - {{- $nodeportfound := false }} - {{- range $i := $usedports }} - {{- /* #i={{ $i }} - #nodeport={{ $nodeport }} - #usedports={{ $usedports }} */}} - {{- if and (has (toString $nodeport) $usedports) (eq $nodeportfound false) }} - {{- /* nodeport conflicts with in use */}} - {{- $nodeport = (add $nodeport 1) }} - {{- else if and ( has $nodeport $chosenports) (eq $nodeportfound false) }} - {{- /* nodeport already chosen, try another */}} - {{- $nodeport = (add $nodeport 1) }} - {{- else if (eq $nodeportfound false) }} - {{- /* nodeport free to use: not already claimed and not in use */}} - {{- /* select nodeport, and place into usedports */}} - {{- $chosenports = (append $chosenports $nodeport) }} - {{- $nodeportfound = true }} - {{- else }} - {{- /* nodeport has already been chosen and locked in, just work through the rest of the list to get to the next nodeport selection */}} - {{- end }} - {{- end }} - {{- if (eq $nodeportfound false) }} - {{- $chosenports = (append $chosenports $nodeport) }} - {{- end }} - -{{- end }} - -{{- /* print the usedports and chosenports for debugging */}} -#usedports {{ $usedports }} -#chosenports {{ $chosenports }}}} - ---- -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ template "common.names.fullname" . }}-ports-configmap - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: - {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -data: -{{ $portsmap := (lookup "v1" "ConfigMap" (include "common.names.namespace" .) (printf "%s-%s" ( include "common.names.fullname" . ) "ports-configmap")).data }} -{{- if $portsmap }} -{{- /* configmap already exists, do not install again */ -}} - {{- range $name, $value := $portsmap }} - "{{ $name }}": "{{ $value }}" - {{- end }} -{{- else }} -{{- /* configmap being set for first time */ -}} - {{- range $index, $port := $chosenports }} - {{- $nodenumber := (floor (div $index 2)) }} - {{- if (eq $index 0) }} - "{{ template "common.names.fullname" $ }}-sentinel": "{{ $port }}" - {{- else if (eq $index 1) }} - "{{ template "common.names.fullname" $ }}-redis": "{{ $port }}" - {{- else if (eq (mod $index 2) 0) }} - "{{ template "common.names.fullname" $ }}-node-{{ (sub $nodenumber 1) }}-sentinel": "{{ $port }}" - {{- else if (eq (mod $index 2) 1) }} - "{{ template "common.names.fullname" $ }}-node-{{ (sub $nodenumber 1) }}-redis": "{{ $port }}" - {{- end }} - {{- end }} -{{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/sentinel/service.yaml b/charts/redis/redis/templates/sentinel/service.yaml deleted file mode 100644 index 9530bde..0000000 --- a/charts/redis/redis/templates/sentinel/service.yaml +++ /dev/null @@ -1,160 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if or .Release.IsUpgrade (ne .Values.sentinel.service.type "NodePort") .Values.sentinel.service.nodePorts.redis -}} -{{- if and (eq .Values.architecture "replication") .Values.sentinel.enabled }} -{{ $portsmap := (lookup "v1" "ConfigMap" (include "common.names.namespace" .) (printf "%s-%s" ( include "common.names.fullname" . ) "ports-configmap")).data }} - -{{ $sentinelport := 0}} -{{ $redisport := 0}} -{{- if $portsmap }} -{{ $sentinelport = index $portsmap (printf "%s-%s" (include "common.names.fullname" $) "sentinel") }} -{{ $redisport = index $portsmap (printf "%s-%s" (include "common.names.fullname" $) "redis") }} -{{- else }} -{{- end }} - -apiVersion: v1 -kind: Service -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: node - {{- if or .Values.sentinel.service.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.sentinel.service.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: {{ .Values.sentinel.service.type }} - {{- if or (eq .Values.sentinel.service.type "LoadBalancer") (eq .Values.sentinel.service.type "NodePort") }} - externalTrafficPolicy: {{ .Values.sentinel.service.externalTrafficPolicy | quote }} - {{- end }} - {{- if and (eq .Values.sentinel.service.type "LoadBalancer") (not (empty .Values.sentinel.service.loadBalancerIP)) }} - loadBalancerIP: {{ .Values.sentinel.service.loadBalancerIP }} - {{- end }} - {{- if and (eq .Values.sentinel.service.type "LoadBalancer") .Values.sentinel.service.loadBalancerClass }} - loadBalancerClass: {{ .Values.sentinel.service.loadBalancerClass }} - {{- end }} - {{- if and (eq .Values.sentinel.service.type "LoadBalancer") (not (empty .Values.sentinel.service.loadBalancerSourceRanges)) }} - loadBalancerSourceRanges: {{ toYaml .Values.sentinel.service.loadBalancerSourceRanges | nindent 4 }} - {{- end }} - {{- if and .Values.sentinel.service.clusterIP (eq .Values.sentinel.service.type "ClusterIP") }} - clusterIP: {{ .Values.sentinel.service.clusterIP }} - {{- end }} - {{- if .Values.sentinel.service.sessionAffinity }} - sessionAffinity: {{ .Values.sentinel.service.sessionAffinity }} - {{- end }} - {{- if .Values.sentinel.service.sessionAffinityConfig }} - sessionAffinityConfig: {{- include "common.tplvalues.render" (dict "value" .Values.sentinel.service.sessionAffinityConfig "context" $) | nindent 4 }} - {{- end }} - ports: - - name: tcp-redis - {{- if and (or (eq .Values.sentinel.service.type "NodePort") (eq .Values.sentinel.service.type "LoadBalancer")) .Values.sentinel.service.nodePorts.redis }} - port: {{ .Values.sentinel.service.nodePorts.redis }} - {{- else if eq .Values.sentinel.service.type "NodePort" }} - port: {{ $redisport }} - {{- else}} - port: {{ .Values.sentinel.service.ports.redis }} - {{- end }} - targetPort: {{ .Values.replica.containerPorts.redis }} - {{- if and (or (eq .Values.sentinel.service.type "NodePort") (eq .Values.sentinel.service.type "LoadBalancer")) .Values.sentinel.service.nodePorts.redis }} - nodePort: {{ .Values.sentinel.service.nodePorts.redis }} - {{- else if eq .Values.sentinel.service.type "ClusterIP" }} - nodePort: null - {{- else if eq .Values.sentinel.service.type "NodePort" }} - nodePort: {{ $redisport }} - {{- end }} - - name: tcp-sentinel - {{- if and (or (eq .Values.sentinel.service.type "NodePort") (eq .Values.sentinel.service.type "LoadBalancer")) .Values.sentinel.service.nodePorts.sentinel }} - port: {{ .Values.sentinel.service.nodePorts.sentinel }} - {{- else if eq .Values.sentinel.service.type "NodePort" }} - port: {{ $sentinelport }} - {{- else }} - port: {{ .Values.sentinel.service.ports.sentinel }} - {{- end }} - targetPort: {{ .Values.sentinel.containerPorts.sentinel }} - {{- if and (or (eq .Values.sentinel.service.type "NodePort") (eq .Values.sentinel.service.type "LoadBalancer")) .Values.sentinel.service.nodePorts.sentinel }} - nodePort: {{ .Values.sentinel.service.nodePorts.sentinel }} - {{- else if eq .Values.sentinel.service.type "ClusterIP" }} - nodePort: null - {{- else if eq .Values.sentinel.service.type "NodePort" }} - nodePort: {{ $sentinelport }} - {{- end }} - {{- if eq .Values.sentinel.service.type "NodePort" }} - - name: sentinel-internal - nodePort: null - port: {{ .Values.sentinel.containerPorts.sentinel }} - protocol: TCP - targetPort: {{ .Values.sentinel.containerPorts.sentinel }} - - name: redis-internal - nodePort: null - port: {{ .Values.replica.containerPorts.redis }} - protocol: TCP - targetPort: {{ .Values.replica.containerPorts.redis }} - {{- end }} - {{- if .Values.sentinel.service.extraPorts }} - {{- include "common.tplvalues.render" (dict "value" .Values.sentinel.service.extraPorts "context" $) | nindent 4 }} - {{- end }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.replica.podLabels .Values.commonLabels ) "context" . ) }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: node - -{{- $masterServiceConfig := ternary .Values.sentinel.masterService .Values.sentinel.service .Values.sentinel.masterService.enabled -}} -{{- if and .Values.sentinel.enabled (or .Values.sentinel.masterService.enabled .Values.sentinel.service.createMaster) }} ---- -apiVersion: v1 -kind: Service -metadata: - name: "{{ template "common.names.fullname" . }}-master" - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: node - {{- if or $masterServiceConfig.annotations .Values.commonAnnotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list ($masterServiceConfig.annotations) .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - type: {{ $masterServiceConfig.type }} - {{- if or (eq $masterServiceConfig.type "LoadBalancer") (eq $masterServiceConfig.type "NodePort") }} - externalTrafficPolicy: {{ $masterServiceConfig.externalTrafficPolicy | quote }} - {{- end }} - {{- if and (eq $masterServiceConfig.type "LoadBalancer") (not (empty ($masterServiceConfig.loadBalancerIP))) }} - loadBalancerIP: {{ $masterServiceConfig.loadBalancerIP }} - {{- end }} - {{- if and (eq $masterServiceConfig.type "LoadBalancer") (not (empty ($masterServiceConfig.loadBalancerClass))) }} - loadBalancerClass: {{ $masterServiceConfig.loadBalancerClass }} - {{- end }} - {{- if and (eq $masterServiceConfig.type "LoadBalancer") (not (empty ($masterServiceConfig.loadBalancerSourceRanges))) }} - loadBalancerSourceRanges: {{ toYaml ($masterServiceConfig.loadBalancerSourceRanges) | nindent 4 }} - {{- end }} - {{- if and (eq $masterServiceConfig.type "ClusterIP") (not (empty ($masterServiceConfig.clusterIP))) }} - clusterIP: {{ $masterServiceConfig.clusterIP }} - {{- end }} - sessionAffinity: {{ $masterServiceConfig.sessionAffinity }} - {{- if $masterServiceConfig.sessionAffinityConfig }} - sessionAffinityConfig: {{- include "common.tplvalues.render" (dict "value" ($masterServiceConfig.sessionAffinityConfig) "context" $) | nindent 4 }} - {{- end }} - ports: - - name: tcp-redis - {{- if and (or (eq $masterServiceConfig.type "NodePort") (eq $masterServiceConfig.type "LoadBalancer")) ($masterServiceConfig.nodePorts.redis) }} - port: {{ $masterServiceConfig.nodePorts.redis }} - {{- else if eq $masterServiceConfig.type "NodePort" }} - port: {{ $redisport }} - {{- else }} - port: {{ $masterServiceConfig.ports.redis }} - {{- end }} - targetPort: {{ .Values.replica.containerPorts.redis }} - {{- if and (or (eq $masterServiceConfig.type "NodePort") (eq $masterServiceConfig.type "LoadBalancer")) ($masterServiceConfig.nodePorts.redis) }} - nodePort: {{ $masterServiceConfig.nodePorts.redis }} - {{- else if eq $masterServiceConfig.type "ClusterIP" }} - nodePort: null - {{- else if eq $masterServiceConfig.type "NodePort" }} - nodePort: {{ $redisport }} - {{- end }} - selector: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 4 }} - isMaster: "true" -{{- end }} -{{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/sentinel/statefulset.yaml b/charts/redis/redis/templates/sentinel/statefulset.yaml deleted file mode 100644 index 59cc292..0000000 --- a/charts/redis/redis/templates/sentinel/statefulset.yaml +++ /dev/null @@ -1,831 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if or .Release.IsUpgrade (ne .Values.sentinel.service.type "NodePort") .Values.sentinel.service.nodePorts.redis -}} -{{- if and (eq .Values.architecture "replication") .Values.sentinel.enabled }} -apiVersion: {{ include "common.capabilities.statefulset.apiVersion" . }} -kind: StatefulSet -metadata: - name: {{ printf "%s-node" (include "common.names.fullname" .) }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - app.kubernetes.io/component: node - {{- if or .Values.commonAnnotations .Values.sentinel.annotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.sentinel.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -spec: - replicas: {{ .Values.replica.replicaCount }} - revisionHistoryLimit: {{ .Values.replica.revisionHistoryLimit }} - {{- $podLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.replica.podLabels .Values.commonLabels ) "context" . ) }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" $podLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: node - serviceName: {{ printf "%s-headless" (include "common.names.fullname" .) }} - {{- if .Values.replica.updateStrategy }} - updateStrategy: {{- toYaml .Values.replica.updateStrategy | nindent 4 }} - {{- end }} - {{- if and .Values.replica.minReadySeconds (semverCompare ">= 1.23-0" (include "common.capabilities.kubeVersion" .)) }} - minReadySeconds: {{ .Values.replica.minReadySeconds }} - {{- end }} - {{- if .Values.replica.podManagementPolicy }} - podManagementPolicy: {{ .Values.replica.podManagementPolicy | quote }} - {{- end }} - template: - metadata: - labels: {{- include "common.labels.standard" ( dict "customLabels" $podLabels "context" $ ) | nindent 8 }} - app.kubernetes.io/component: node - {{- if and .Values.metrics.enabled .Values.metrics.podLabels }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.podLabels "context" $ ) | nindent 8 }} - {{- end }} - annotations: - {{- if (include "redis.createConfigmap" .) }} - checksum/configmap: {{ pick ( include (print $.Template.BasePath "/configmap.yaml") . | fromYaml ) "data" | toYaml | sha256sum }} - {{- end }} - checksum/health: {{ pick ( include (print $.Template.BasePath "/health-configmap.yaml") . | fromYaml ) "data" | toYaml | sha256sum }} - checksum/scripts: {{ pick ( include (print $.Template.BasePath "/scripts-configmap.yaml") . | fromYaml ) "data" | toYaml | sha256sum }} - checksum/secret: {{ pick ( include (print $.Template.BasePath "/secret.yaml") . | fromYaml ) "data" | toYaml | sha256sum }} - {{- if .Values.replica.podAnnotations }} - {{- include "common.tplvalues.render" ( dict "value" .Values.replica.podAnnotations "context" $ ) | nindent 8 }} - {{- end }} - {{- if and .Values.metrics.enabled .Values.metrics.podAnnotations }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.podAnnotations "context" $ ) | nindent 8 }} - {{- end }} - spec: - {{- include "redis.imagePullSecrets" . | nindent 6 }} - automountServiceAccountToken: {{ .Values.replica.automountServiceAccountToken }} - {{- if .Values.replica.hostAliases }} - hostAliases: {{- include "common.tplvalues.render" (dict "value" .Values.replica.hostAliases "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.replica.podSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.replica.podSecurityContext "context" $) | nindent 8 }} - {{- end }} - serviceAccountName: {{ template "redis.serviceAccountName" . }} - {{- if .Values.replica.priorityClassName }} - priorityClassName: {{ .Values.replica.priorityClassName | quote }} - {{- end }} - {{- if .Values.replica.affinity }} - affinity: {{- include "common.tplvalues.render" (dict "value" .Values.replica.affinity "context" $) | nindent 8 }} - {{- else }} - affinity: - podAffinity: {{- include "common.affinities.pods" (dict "type" .Values.replica.podAffinityPreset "component" "node" "customLabels" $podLabels "context" $) | nindent 10 }} - podAntiAffinity: {{- include "common.affinities.pods" (dict "type" .Values.replica.podAntiAffinityPreset "component" "node" "customLabels" $podLabels "context" $) | nindent 10 }} - nodeAffinity: {{- include "common.affinities.nodes" (dict "type" .Values.replica.nodeAffinityPreset.type "key" .Values.replica.nodeAffinityPreset.key "values" .Values.replica.nodeAffinityPreset.values) | nindent 10 }} - {{- end }} - {{- if .Values.replica.nodeSelector }} - nodeSelector: {{- include "common.tplvalues.render" (dict "value" .Values.replica.nodeSelector "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.replica.tolerations }} - tolerations: {{- include "common.tplvalues.render" (dict "value" .Values.replica.tolerations "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.replica.topologySpreadConstraints }} - topologySpreadConstraints: {{- include "common.tplvalues.render" (dict "value" .Values.replica.topologySpreadConstraints "context" $) | nindent 8 }} - {{- end }} - {{- if .Values.replica.shareProcessNamespace }} - shareProcessNamespace: {{ .Values.replica.shareProcessNamespace }} - {{- end }} - {{- if .Values.replica.schedulerName }} - schedulerName: {{ .Values.replica.schedulerName | quote }} - {{- end }} - {{- if .Values.replica.dnsPolicy }} - dnsPolicy: {{ .Values.replica.dnsPolicy }} - {{- end }} - {{- if .Values.replica.dnsConfig }} - dnsConfig: {{- include "common.tplvalues.render" (dict "value" .Values.replica.dnsConfig "context" $) | nindent 8 }} - {{- end }} - enableServiceLinks: {{ .Values.sentinel.enableServiceLinks }} - terminationGracePeriodSeconds: {{ .Values.sentinel.terminationGracePeriodSeconds }} - containers: - - name: redis - image: {{ template "redis.image" . }} - imagePullPolicy: {{ .Values.image.pullPolicy | quote }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.replica.lifecycleHooks }} - lifecycle: {{- include "common.tplvalues.render" (dict "value" .Values.replica.lifecycleHooks "context" $) | nindent 12 }} - {{- else }} - lifecycle: - preStop: - exec: - command: - - /bin/bash - - -c - - /opt/bitnami/scripts/start-scripts/prestop-redis.sh - {{- end }} - {{- end }} - {{- if .Values.replica.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.replica.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - {{- else if .Values.replica.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.replica.command "context" $) | nindent 12 }} - {{- else }} - command: - - /bin/bash - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- else if .Values.replica.args }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.replica.args "context" $) | nindent 12 }} - {{- else }} - args: - - -c - - /opt/bitnami/scripts/start-scripts/start-node.sh - {{- end }} - env: - - name: BITNAMI_DEBUG - value: {{ ternary "true" "false" (or .Values.image.debug .Values.diagnosticMode.enabled) | quote }} - - name: REDIS_MASTER_PORT_NUMBER - value: {{ .Values.replica.containerPorts.redis | quote }} - - name: ALLOW_EMPTY_PASSWORD - value: {{ ternary "no" "yes" .Values.auth.enabled | quote }} - {{- if .Values.auth.enabled }} - {{- if .Values.auth.usePasswordFiles }} - - name: REDIS_PASSWORD_FILE - value: "/opt/bitnami/redis/secrets/redis-password" - - name: REDIS_MASTER_PASSWORD_FILE - value: "/opt/bitnami/redis/secrets/redis-password" - {{- else }} - - name: REDIS_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "redis.secretName" . }} - key: {{ template "redis.secretPasswordKey" . }} - - name: REDIS_MASTER_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "redis.secretName" . }} - key: {{ template "redis.secretPasswordKey" . }} - {{- end }} - {{- end }} - - name: REDIS_TLS_ENABLED - value: {{ ternary "yes" "no" .Values.tls.enabled | quote }} - {{- if .Values.tls.enabled }} - - name: REDIS_TLS_PORT - value: {{ .Values.replica.containerPorts.redis | quote }} - - name: REDIS_TLS_AUTH_CLIENTS - value: {{ ternary "yes" "no" .Values.tls.authClients | quote }} - - name: REDIS_TLS_CERT_FILE - value: {{ template "redis.tlsCert" . }} - - name: REDIS_TLS_KEY_FILE - value: {{ template "redis.tlsCertKey" . }} - - name: REDIS_TLS_CA_FILE - value: {{ template "redis.tlsCACert" . }} - {{- if .Values.tls.dhParamsFilename }} - - name: REDIS_TLS_DH_PARAMS_FILE - value: {{ template "redis.tlsDHParams" . }} - {{- end }} - {{- else }} - - name: REDIS_PORT - value: {{ .Values.replica.containerPorts.redis | quote }} - {{- end }} - - name: REDIS_SENTINEL_TLS_ENABLED - value: {{ ternary "yes" "no" .Values.tls.enabled | quote }} - {{- if .Values.tls.enabled }} - - name: REDIS_SENTINEL_TLS_PORT_NUMBER - value: {{ .Values.sentinel.containerPorts.sentinel | quote }} - - name: REDIS_SENTINEL_TLS_AUTH_CLIENTS - value: {{ ternary "yes" "no" .Values.tls.authClients | quote }} - - name: REDIS_SENTINEL_TLS_CERT_FILE - value: {{ template "redis.tlsCert" . }} - - name: REDIS_SENTINEL_TLS_KEY_FILE - value: {{ template "redis.tlsCertKey" . }} - - name: REDIS_SENTINEL_TLS_CA_FILE - value: {{ template "redis.tlsCACert" . }} - {{- if .Values.tls.dhParamsFilename }} - - name: REDIS_SENTINEL_TLS_DH_PARAMS_FILE - value: {{ template "redis.tlsDHParams" . }} - {{- end }} - {{- else }} - - name: REDIS_SENTINEL_PORT - value: {{ .Values.sentinel.containerPorts.sentinel | quote }} - {{- end }} - - name: REDIS_DATA_DIR - value: {{ .Values.replica.persistence.path }} - {{- if .Values.replica.externalMaster.enabled }} - - name: REDIS_EXTERNAL_MASTER_HOST - value: {{ .Values.replica.externalMaster.host | quote }} - - name: REDIS_EXTERNAL_MASTER_PORT - value: {{ .Values.replica.externalMaster.port | quote }} - {{- end }} - {{- if .Values.replica.extraEnvVars }} - {{- include "common.tplvalues.render" ( dict "value" .Values.replica.extraEnvVars "context" $ ) | nindent 12 }} - {{- end }} - {{- if or .Values.replica.extraEnvVarsCM .Values.replica.extraEnvVarsSecret }} - envFrom: - {{- if .Values.replica.extraEnvVarsCM }} - - configMapRef: - name: {{ .Values.replica.extraEnvVarsCM }} - {{- end }} - {{- if .Values.replica.extraEnvVarsSecret }} - - secretRef: - name: {{ .Values.replica.extraEnvVarsSecret }} - {{- end }} - {{- end }} - ports: - - name: redis - containerPort: {{ .Values.replica.containerPorts.redis }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.replica.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.replica.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.replica.startupProbe.enabled }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.replica.startupProbe "enabled") "context" $) | nindent 12 }} - exec: - command: - - sh - - -c - - /health/ping_liveness_local.sh {{ .Values.replica.livenessProbe.timeoutSeconds }} - {{- end }} - {{- if .Values.replica.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.replica.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.replica.livenessProbe.enabled }} - livenessProbe: - initialDelaySeconds: {{ .Values.replica.livenessProbe.initialDelaySeconds }} - periodSeconds: {{ .Values.replica.livenessProbe.periodSeconds }} - timeoutSeconds: {{ .Values.replica.livenessProbe.timeoutSeconds }} - successThreshold: {{ .Values.replica.livenessProbe.successThreshold }} - failureThreshold: {{ .Values.replica.livenessProbe.failureThreshold }} - exec: - command: - - sh - - -c - - /health/ping_liveness_local.sh {{ .Values.replica.livenessProbe.timeoutSeconds }} - {{- end }} - {{- if .Values.replica.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.replica.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.replica.readinessProbe.enabled }} - readinessProbe: - initialDelaySeconds: {{ .Values.replica.readinessProbe.initialDelaySeconds }} - periodSeconds: {{ .Values.replica.readinessProbe.periodSeconds }} - timeoutSeconds: {{ .Values.replica.readinessProbe.timeoutSeconds }} - successThreshold: {{ .Values.replica.readinessProbe.successThreshold }} - failureThreshold: {{ .Values.replica.readinessProbe.failureThreshold }} - exec: - command: - - sh - - -c - - /health/ping_readiness_local.sh {{ .Values.replica.readinessProbe.timeoutSeconds }} - {{- end }} - {{- end }} - {{- if .Values.replica.resources }} - resources: {{- toYaml .Values.replica.resources | nindent 12 }} - {{- else if ne .Values.replica.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.replica.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: start-scripts - mountPath: /opt/bitnami/scripts/start-scripts - - name: health - mountPath: /health - - name: sentinel-data - mountPath: /opt/bitnami/redis-sentinel/etc - {{- if .Values.auth.usePasswordFiles }} - - name: redis-password - mountPath: /opt/bitnami/redis/secrets/ - {{- end }} - - name: redis-data - mountPath: {{ .Values.replica.persistence.path }} - {{- if .Values.replica.persistence.subPath }} - subPath: {{ .Values.replica.persistence.subPath }} - {{- else if .Values.replica.persistence.subPathExpr }} - subPathExpr: {{ .Values.replica.persistence.subPathExpr }} - {{- end }} - - name: config - mountPath: /opt/bitnami/redis/mounted-etc - - name: empty-dir - mountPath: /opt/bitnami/redis/etc - subPath: app-conf-dir - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - {{- if .Values.tls.enabled }} - - name: redis-certificates - mountPath: /opt/bitnami/redis/certs - readOnly: true - {{- end }} - {{- if .Values.replica.extraVolumeMounts }} - {{- include "common.tplvalues.render" ( dict "value" .Values.replica.extraVolumeMounts "context" $ ) | nindent 12 }} - {{- end }} - - name: sentinel - image: {{ template "redis.sentinel.image" . }} - imagePullPolicy: {{ .Values.sentinel.image.pullPolicy | quote }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.sentinel.lifecycleHooks }} - lifecycle: {{- include "common.tplvalues.render" (dict "value" .Values.sentinel.lifecycleHooks "context" $) | nindent 12 }} - {{- else }} - lifecycle: - preStop: - exec: - command: - - /bin/bash - - -c - - /opt/bitnami/scripts/start-scripts/prestop-sentinel.sh - {{- end }} - {{- end }} - {{- if .Values.sentinel.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.sentinel.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - {{- else if .Values.sentinel.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.sentinel.command "context" $) | nindent 12 }} - {{- else }} - command: - - /bin/bash - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- else if .Values.sentinel.args }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.sentinel.args "context" $) | nindent 12 }} - {{- else }} - args: - - -c - - /opt/bitnami/scripts/start-scripts/start-sentinel.sh - {{- end }} - env: - - name: BITNAMI_DEBUG - value: {{ ternary "true" "false" (or .Values.sentinel.image.debug .Values.diagnosticMode.enabled) | quote }} - {{- if .Values.auth.enabled }} - {{- if .Values.auth.usePasswordFiles }} - - name: REDIS_PASSWORD_FILE - value: "/opt/bitnami/redis/secrets/redis-password" - {{- else }} - - name: REDIS_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "redis.secretName" . }} - key: {{ template "redis.secretPasswordKey" . }} - {{- end }} - {{- else }} - - name: ALLOW_EMPTY_PASSWORD - value: "yes" - {{- end }} - - name: REDIS_SENTINEL_TLS_ENABLED - value: {{ ternary "yes" "no" .Values.tls.enabled | quote }} - {{- if .Values.tls.enabled }} - - name: REDIS_SENTINEL_TLS_PORT_NUMBER - value: {{ .Values.sentinel.containerPorts.sentinel | quote }} - - name: REDIS_SENTINEL_TLS_AUTH_CLIENTS - value: {{ ternary "yes" "no" .Values.tls.authClients | quote }} - - name: REDIS_SENTINEL_TLS_CERT_FILE - value: {{ template "redis.tlsCert" . }} - - name: REDIS_SENTINEL_TLS_KEY_FILE - value: {{ template "redis.tlsCertKey" . }} - - name: REDIS_SENTINEL_TLS_CA_FILE - value: {{ template "redis.tlsCACert" . }} - {{- if .Values.tls.dhParamsFilename }} - - name: REDIS_SENTINEL_TLS_DH_PARAMS_FILE - value: {{ template "redis.tlsDHParams" . }} - {{- end }} - {{- else }} - - name: REDIS_SENTINEL_PORT - value: {{ .Values.sentinel.containerPorts.sentinel | quote }} - {{- end }} - {{- if .Values.sentinel.externalMaster.enabled }} - - name: REDIS_EXTERNAL_MASTER_HOST - value: {{ .Values.sentinel.externalMaster.host | quote }} - - name: REDIS_EXTERNAL_MASTER_PORT - value: {{ .Values.sentinel.externalMaster.port | quote }} - {{- end }} - {{- if .Values.sentinel.extraEnvVars }} - {{- include "common.tplvalues.render" ( dict "value" .Values.sentinel.extraEnvVars "context" $ ) | nindent 12 }} - {{- end }} - {{- if or .Values.sentinel.extraEnvVarsCM .Values.sentinel.extraEnvVarsSecret }} - envFrom: - {{- if .Values.sentinel.extraEnvVarsCM }} - - configMapRef: - name: {{ .Values.sentinel.extraEnvVarsCM }} - {{- end }} - {{- if .Values.sentinel.extraEnvVarsSecret }} - - secretRef: - name: {{ .Values.sentinel.extraEnvVarsSecret }} - {{- end }} - {{- end }} - ports: - - name: redis-sentinel - containerPort: {{ .Values.sentinel.containerPorts.sentinel }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.sentinel.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.sentinel.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.sentinel.startupProbe.enabled }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.sentinel.startupProbe "enabled") "context" $) | nindent 12 }} - exec: - command: - - sh - - -c - - /health/ping_sentinel.sh {{ .Values.sentinel.livenessProbe.timeoutSeconds }} - {{- end }} - {{- if .Values.sentinel.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.sentinel.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.sentinel.livenessProbe.enabled }} - livenessProbe: - initialDelaySeconds: {{ .Values.sentinel.livenessProbe.initialDelaySeconds }} - periodSeconds: {{ .Values.sentinel.livenessProbe.periodSeconds }} - timeoutSeconds: {{ .Values.sentinel.livenessProbe.timeoutSeconds }} - successThreshold: {{ .Values.sentinel.livenessProbe.successThreshold }} - failureThreshold: {{ .Values.sentinel.livenessProbe.failureThreshold }} - exec: - command: - - sh - - -c - - /health/ping_sentinel.sh {{ .Values.sentinel.livenessProbe.timeoutSeconds }} - {{- end }} - {{- end }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.sentinel.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.sentinel.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.sentinel.readinessProbe.enabled }} - readinessProbe: - initialDelaySeconds: {{ .Values.sentinel.readinessProbe.initialDelaySeconds }} - periodSeconds: {{ .Values.sentinel.readinessProbe.periodSeconds }} - timeoutSeconds: {{ .Values.sentinel.readinessProbe.timeoutSeconds }} - successThreshold: {{ .Values.sentinel.readinessProbe.successThreshold }} - failureThreshold: {{ .Values.sentinel.readinessProbe.failureThreshold }} - exec: - command: - - sh - - -c - - /health/ping_sentinel.sh {{ .Values.sentinel.readinessProbe.timeoutSeconds }} - {{- end }} - {{- end }} - {{- if .Values.sentinel.resources }} - resources: {{- toYaml .Values.sentinel.resources | nindent 12 }} - {{- else if ne .Values.sentinel.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.sentinel.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: start-scripts - mountPath: /opt/bitnami/scripts/start-scripts - - name: health - mountPath: /health - {{- if or .Values.sentinel.masterService.enabled .Values.sentinel.service.createMaster}} - - name: kubectl-shared - mountPath: /etc/shared - {{- end }} - - name: sentinel-data - mountPath: /opt/bitnami/redis-sentinel/etc - {{- if .Values.auth.usePasswordFiles }} - - name: redis-password - mountPath: /opt/bitnami/redis/secrets/ - {{- end }} - - name: redis-data - mountPath: {{ .Values.replica.persistence.path }} - {{- if .Values.replica.persistence.subPath }} - subPath: {{ .Values.replica.persistence.subPath }} - {{- else if .Values.replica.persistence.subPathExpr }} - subPathExpr: {{ .Values.replica.persistence.subPathExpr }} - {{- end }} - - name: config - mountPath: /opt/bitnami/redis-sentinel/mounted-etc - {{- if .Values.tls.enabled }} - - name: redis-certificates - mountPath: /opt/bitnami/redis/certs - readOnly: true - {{- end }} - {{- if .Values.sentinel.extraVolumeMounts }} - {{- include "common.tplvalues.render" ( dict "value" .Values.sentinel.extraVolumeMounts "context" $ ) | nindent 12 }} - {{- end }} - {{- if .Values.metrics.enabled }} - - name: metrics - image: {{ template "redis.metrics.image" . }} - imagePullPolicy: {{ .Values.metrics.image.pullPolicy | quote }} - {{- if .Values.metrics.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.metrics.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.command "context" $) | nindent 12 }} - {{- else }} - command: - - /bin/bash - - -c - - | - if [[ -f '/secrets/redis-password' ]]; then - export REDIS_PASSWORD=$(cat /secrets/redis-password) - fi - redis_exporter{{- range $key, $value := .Values.metrics.extraArgs }} --{{ $key }}={{ $value }}{{- end }} - {{- end }} - {{- if .Values.diagnosticMode.enabled }} - args: {{- include "common.tplvalues.render" (dict "value" .Values.diagnosticMode.args "context" $) | nindent 12 }} - {{- end }} - env: - - name: REDIS_ALIAS - value: {{ template "common.names.fullname" . }} - - name: REDIS_EXPORTER_WEB_LISTEN_ADDRESS - value: {{ printf ":%v" .Values.metrics.containerPorts.http }} - {{- if .Values.auth.enabled }} - - name: REDIS_USER - value: default - {{- if (not .Values.auth.usePasswordFiles) }} - - name: REDIS_PASSWORD - valueFrom: - secretKeyRef: - name: {{ template "redis.secretName" . }} - key: {{ template "redis.secretPasswordKey" . }} - {{- end }} - {{- end }} - {{- if .Values.tls.enabled }} - - name: REDIS_ADDR - value: rediss://{{ .Values.metrics.redisTargetHost }}:{{ .Values.replica.containerPorts.redis }} - {{- if .Values.tls.authClients }} - - name: REDIS_EXPORTER_TLS_CLIENT_KEY_FILE - value: {{ template "redis.tlsCertKey" . }} - - name: REDIS_EXPORTER_TLS_CLIENT_CERT_FILE - value: {{ template "redis.tlsCert" . }} - {{- end }} - - name: REDIS_EXPORTER_TLS_CA_CERT_FILE - value: {{ template "redis.tlsCACert" . }} - {{- end }} - {{- if .Values.metrics.extraEnvVars }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.extraEnvVars "context" $) | nindent 12 }} - {{- end }} - ports: - - name: metrics - containerPort: {{ .Values.metrics.containerPorts.http }} - {{- if not .Values.diagnosticMode.enabled }} - {{- if .Values.metrics.customStartupProbe }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customStartupProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.startupProbe.enabled }} - startupProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.startupProbe "enabled") "context" $) | nindent 12 }} - tcpSocket: - port: metrics - {{- end }} - {{- if .Values.metrics.customLivenessProbe }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customLivenessProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.livenessProbe.enabled }} - livenessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.livenessProbe "enabled") "context" $) | nindent 12 }} - tcpSocket: - port: metrics - {{- end }} - {{- if .Values.metrics.customReadinessProbe }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.metrics.customReadinessProbe "context" $) | nindent 12 }} - {{- else if .Values.metrics.readinessProbe.enabled }} - readinessProbe: {{- include "common.tplvalues.render" (dict "value" (omit .Values.metrics.readinessProbe "enabled") "context" $) | nindent 12 }} - httpGet: - path: / - port: metrics - {{- end }} - {{- end }} - {{- if .Values.metrics.resources }} - resources: {{- toYaml .Values.metrics.resources | nindent 12 }} - {{- else if ne .Values.metrics.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.metrics.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - {{- if .Values.auth.usePasswordFiles }} - - name: redis-password - mountPath: /secrets/ - {{- end }} - {{- if .Values.tls.enabled }} - - name: redis-certificates - mountPath: /opt/bitnami/redis/certs - readOnly: true - {{- end }} - {{- if .Values.metrics.extraVolumeMounts }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.extraVolumeMounts "context" $ ) | nindent 12 }} - {{- end }} - {{- end }} - {{- if or .Values.sentinel.masterService.enabled .Values.sentinel.service.createMaster }} - - name: kubectl-shared - image: {{ template "redis.kubectl.image" . }} - imagePullPolicy: {{ .Values.kubectl.image.pullPolicy | quote }} - command: {{- toYaml .Values.kubectl.command | nindent 12 }} - {{- if .Values.kubectl.containerSecurityContext.enabled }} - securityContext: {{- include "common.compatibility.renderSecurityContext" (dict "secContext" .Values.kubectl.containerSecurityContext "context" $) | nindent 12 }} - {{- end }} - volumeMounts: - - name: kubectl-shared - mountPath: /etc/shared - - name: kubectl-scripts - mountPath: /opt/bitnami/scripts/kubectl-scripts - {{- if .Values.kubectl.resources }} - resources: {{- toYaml .Values.kubectl.resources | nindent 12 }} - {{- end }} - {{- end }} - {{- if .Values.replica.sidecars }} - {{- include "common.tplvalues.render" (dict "value" .Values.replica.sidecars "context" $) | nindent 8 }} - {{- end }} - {{- $needsVolumePermissions := and .Values.volumePermissions.enabled .Values.replica.persistence.enabled .Values.replica.podSecurityContext.enabled .Values.replica.containerSecurityContext.enabled }} - {{- if or .Values.replica.initContainers $needsVolumePermissions .Values.sysctl.enabled }} - initContainers: - {{- if .Values.replica.initContainers }} - {{- include "common.tplvalues.render" (dict "value" .Values.replica.initContainers "context" $) | nindent 8 }} - {{- end }} - {{- if $needsVolumePermissions }} - - name: volume-permissions - image: {{ include "redis.volumePermissions.image" . }} - imagePullPolicy: {{ .Values.volumePermissions.image.pullPolicy | quote }} - command: - - /bin/bash - - -ec - - | - {{- if eq ( toString ( .Values.volumePermissions.containerSecurityContext.runAsUser )) "auto" }} - chown -R `id -u`:`id -G | cut -d " " -f2` {{ .Values.replica.persistence.path }} - {{- else }} - chown -R {{ .Values.replica.containerSecurityContext.runAsUser }}:{{ .Values.replica.podSecurityContext.fsGroup }} {{ .Values.replica.persistence.path }} - {{- end }} - {{- if eq ( toString ( .Values.volumePermissions.containerSecurityContext.runAsUser )) "auto" }} - securityContext: {{- omit .Values.volumePermissions.containerSecurityContext "runAsUser" | toYaml | nindent 12 }} - {{- else }} - securityContext: {{- .Values.volumePermissions.containerSecurityContext | toYaml | nindent 12 }} - {{- end }} - {{- if .Values.volumePermissions.resources }} - resources: {{- toYaml .Values.volumePermissions.resources | nindent 12 }} - {{- else if ne .Values.volumePermissions.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.volumePermissions.resourcesPreset) | nindent 12 }} - {{- end }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: redis-data - mountPath: {{ .Values.replica.persistence.path }} - {{- if .Values.replica.persistence.subPath }} - subPath: {{ .Values.replica.persistence.subPath }} - {{- else if .Values.replica.persistence.subPathExpr }} - subPathExpr: {{ .Values.replica.persistence.subPathExpr }} - {{- end }} - {{- end }} - {{- if .Values.sysctl.enabled }} - - name: init-sysctl - image: {{ include "redis.sysctl.image" . }} - imagePullPolicy: {{ default "" .Values.sysctl.image.pullPolicy | quote }} - securityContext: - privileged: true - runAsUser: 0 - {{- if .Values.sysctl.command }} - command: {{- include "common.tplvalues.render" (dict "value" .Values.sysctl.command "context" $) | nindent 12 }} - {{- end }} - {{- if .Values.sysctl.resources }} - resources: {{- toYaml .Values.sysctl.resources | nindent 12 }} - {{- else if ne .Values.sysctl.resourcesPreset "none" }} - resources: {{- include "common.resources.preset" (dict "type" .Values.sysctl.resourcesPreset) | nindent 12 }} - {{- end }} - {{- if .Values.sysctl.mountHostSys }} - volumeMounts: - - name: empty-dir - mountPath: /tmp - subPath: tmp-dir - - name: host-sys - mountPath: /host-sys - {{- end }} - {{- end }} - {{- end }} - volumes: - - name: start-scripts - configMap: - name: {{ printf "%s-scripts" (include "common.names.fullname" .) }} - defaultMode: 0755 - - name: health - configMap: - name: {{ printf "%s-health" (include "common.names.fullname" .) }} - defaultMode: 0755 - {{- if or .Values.sentinel.masterService.enabled .Values.sentinel.service.createMaster}} - - name: kubectl-shared - emptyDir: {} - - name: kubectl-scripts - configMap: - name: {{ printf "%s-kubectl-scripts" (include "common.names.fullname" .) }} - defaultMode: 0755 - {{- end }} - {{- if .Values.auth.usePasswordFiles }} - - name: redis-password - {{ if .Values.auth.usePasswordFileFromSecret }} - secret: - secretName: {{ template "redis.secretName" . }} - items: - - key: {{ template "redis.secretPasswordKey" . }} - path: redis-password - {{- else }} - emptyDir: {} - {{- end }} - {{- end }} - - name: config - configMap: - name: {{ include "redis.configmapName" . }} - {{- if .Values.sysctl.mountHostSys }} - - name: host-sys - hostPath: - path: /sys - {{- end }} - {{- if not .Values.sentinel.persistence.enabled }} - - name: sentinel-data - {{- if or .Values.sentinel.persistence.medium .Values.sentinel.persistence.sizeLimit }} - emptyDir: - {{- if .Values.sentinel.persistence.medium }} - medium: {{ .Values.sentinel.persistence.medium | quote }} - {{- end }} - {{- if .Values.sentinel.persistence.sizeLimit }} - sizeLimit: {{ .Values.sentinel.persistence.sizeLimit | quote }} - {{- end }} - {{- else }} - emptyDir: {} - {{- end }} - {{- end }} - - name: empty-dir - {{- if or .Values.sentinel.persistence.medium .Values.sentinel.persistence.sizeLimit }} - emptyDir: - {{- if .Values.sentinel.persistence.medium }} - medium: {{ .Values.sentinel.persistence.medium | quote }} - {{- end }} - {{- if .Values.sentinel.persistence.sizeLimit }} - sizeLimit: {{ .Values.sentinel.persistence.sizeLimit | quote }} - {{- end }} - {{- else }} - emptyDir: {} - {{- end }} - {{- if .Values.replica.extraVolumes }} - {{- include "common.tplvalues.render" ( dict "value" .Values.replica.extraVolumes "context" $ ) | nindent 8 }} - {{- end }} - {{- if .Values.metrics.extraVolumes }} - {{- include "common.tplvalues.render" ( dict "value" .Values.metrics.extraVolumes "context" $ ) | nindent 8 }} - {{- end }} - {{- if .Values.sentinel.extraVolumes }} - {{- include "common.tplvalues.render" ( dict "value" .Values.sentinel.extraVolumes "context" $ ) | nindent 8 }} - {{- end }} - {{- if .Values.tls.enabled }} - - name: redis-certificates - secret: - secretName: {{ include "redis.tlsSecretName" . }} - defaultMode: 256 - {{- end }} - {{- if not .Values.replica.persistence.enabled }} - - name: redis-data - {{- if or .Values.replica.persistence.medium .Values.replica.persistence.sizeLimit }} - emptyDir: - {{- if .Values.replica.persistence.medium }} - medium: {{ .Values.replica.persistence.medium | quote }} - {{- end }} - {{- if .Values.replica.persistence.sizeLimit }} - sizeLimit: {{ .Values.replica.persistence.sizeLimit | quote }} - {{- end }} - {{- else }} - emptyDir: {} - {{- end }} - {{- else if .Values.replica.persistence.existingClaim }} - - name: redis-data - persistentVolumeClaim: - claimName: {{ printf "%s" (tpl .Values.replica.persistence.existingClaim .) }} - {{- else }} - {{- if .Values.sentinel.persistentVolumeClaimRetentionPolicy.enabled }} - persistentVolumeClaimRetentionPolicy: - whenDeleted: {{ .Values.sentinel.persistentVolumeClaimRetentionPolicy.whenDeleted }} - whenScaled: {{ .Values.sentinel.persistentVolumeClaimRetentionPolicy.whenScaled }} - {{- end }} - volumeClaimTemplates: - - apiVersion: v1 - kind: PersistentVolumeClaim - metadata: - name: redis-data - labels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 10 }} - app.kubernetes.io/component: node - {{- if .Values.replica.persistence.annotations }} - annotations: {{- toYaml .Values.replica.persistence.annotations | nindent 10 }} - {{- end }} - spec: - accessModes: - {{- range .Values.replica.persistence.accessModes }} - - {{ . | quote }} - {{- end }} - resources: - requests: - storage: {{ .Values.replica.persistence.size | quote }} - {{- if .Values.replica.persistence.selector }} - selector: {{- include "common.tplvalues.render" ( dict "value" .Values.replica.persistence.selector "context" $) | nindent 10 }} - {{- end }} - {{- include "common.storage.class" (dict "persistence" .Values.replica.persistence "global" .Values.global) | nindent 8 }} - {{- if .Values.sentinel.persistence.enabled }} - - metadata: - name: sentinel-data - {{- $claimLabels := include "common.tplvalues.merge" ( dict "values" ( list .Values.sentinel.persistence.labels .Values.commonLabels ) "context" . ) }} - labels: {{- include "common.labels.matchLabels" ( dict "customLabels" $claimLabels "context" $ ) | nindent 10 }} - app.kubernetes.io/component: node - {{- if .Values.sentinel.persistence.annotations }} - annotations: {{- toYaml .Values.sentinel.persistence.annotations | nindent 10 }} - {{- end }} - spec: - accessModes: - {{- range .Values.sentinel.persistence.accessModes }} - - {{ . | quote }} - {{- end }} - resources: - requests: - storage: {{ .Values.sentinel.persistence.size | quote }} - {{- if .Values.sentinel.persistence.selector }} - selector: {{- include "common.tplvalues.render" ( dict "value" .Values.sentinel.persistence.selector "context" $) | nindent 10 }} - {{- end }} - {{- if .Values.sentinel.persistence.dataSource }} - dataSource: {{- include "common.tplvalues.render" (dict "value" .Values.sentinel.persistence.dataSource "context" $) | nindent 10 }} - {{- end }} - {{- include "common.storage.class" (dict "persistence" .Values.sentinel.persistence "global" .Values.global) | nindent 8 }} - {{- end }} - {{- end }} -{{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/serviceaccount.yaml b/charts/redis/redis/templates/serviceaccount.yaml deleted file mode 100644 index 0b7d39d..0000000 --- a/charts/redis/redis/templates/serviceaccount.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.serviceAccount.create .Values.sentinel.enabled }} -apiVersion: v1 -kind: ServiceAccount -automountServiceAccountToken: {{ .Values.serviceAccount.automountServiceAccountToken }} -metadata: - name: {{ template "redis.serviceAccountName" . }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if or .Values.commonAnnotations .Values.serviceAccount.annotations }} - {{- $annotations := include "common.tplvalues.merge" ( dict "values" ( list .Values.serviceAccount.annotations .Values.commonAnnotations ) "context" . ) }} - annotations: {{- include "common.tplvalues.render" ( dict "value" $annotations "context" $) | nindent 4 }} - {{- end }} -{{- end }} diff --git a/charts/redis/redis/templates/servicemonitor.yaml b/charts/redis/redis/templates/servicemonitor.yaml deleted file mode 100644 index e5ffd90..0000000 --- a/charts/redis/redis/templates/servicemonitor.yaml +++ /dev/null @@ -1,82 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if and .Values.metrics.enabled .Values.metrics.serviceMonitor.enabled }} -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ template "common.names.fullname" . }} - namespace: {{ default (include "common.names.namespace" .) .Values.metrics.serviceMonitor.namespace | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.metrics.serviceMonitor.additionalLabels }} - {{- include "common.tplvalues.render" (dict "value" .Values.metrics.serviceMonitor.additionalLabels "context" $) | nindent 4 }} - {{- end }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -spec: - endpoints: - - port: {{ .Values.metrics.serviceMonitor.port }} - {{- if .Values.metrics.serviceMonitor.interval }} - interval: {{ .Values.metrics.serviceMonitor.interval }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.scrapeTimeout }} - scrapeTimeout: {{ .Values.metrics.serviceMonitor.scrapeTimeout }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.honorLabels }} - honorLabels: {{ .Values.metrics.serviceMonitor.honorLabels }} - {{- end }} - {{- with concat .Values.metrics.serviceMonitor.relabelings .Values.metrics.serviceMonitor.relabellings }} - relabelings: {{- toYaml . | nindent 6 }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.metricRelabelings }} - metricRelabelings: {{- toYaml .Values.metrics.serviceMonitor.metricRelabelings | nindent 6 }} - {{- end }} - {{- range .Values.metrics.serviceMonitor.additionalEndpoints }} - - port: {{ .port }} - {{- if .interval }} - interval: {{ .interval }} - {{- end }} - {{- if .scrapeTimeout }} - scrapeTimeout: {{ .scrapeTimeout }} - {{- end }} - {{- if .honorLabels }} - honorLabels: {{ .honorLabels }} - {{- end }} - {{- with concat .Values.metrics.serviceMonitor.relabelings .Values.metrics.serviceMonitor.relabellings }} - relabelings: {{- toYaml . | nindent 6 }} - {{- end }} - {{- if .metricRelabelings }} - metricRelabelings: {{- toYaml .metricRelabelings | nindent 6 }} - {{- end }} - {{- if .path }} - path: {{ .path }} - {{- end }} - {{- if .params }} - params: - {{- range $key, $value := .params }} - {{ $key }}: - {{- range $value }} - - {{ . | quote }} - {{- end }} - {{- end }} - {{- end }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.podTargetLabels }} - podTargetLabels: {{- toYaml .Values.metrics.serviceMonitor.podTargetLabels | nindent 4 }} - {{- end }} - {{- with .Values.metrics.serviceMonitor.sampleLimit }} - sampleLimit: {{ . }} - {{- end }} - {{- with .Values.metrics.serviceMonitor.targetLimit }} - targetLimit: {{ . }} - {{- end }} - namespaceSelector: - matchNames: - - {{ include "common.names.namespace" . | quote }} - selector: - matchLabels: {{- include "common.labels.matchLabels" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 6 }} - app.kubernetes.io/component: metrics -{{- end }} diff --git a/charts/redis/redis/templates/tls-secret.yaml b/charts/redis/redis/templates/tls-secret.yaml deleted file mode 100644 index aa1c1a7..0000000 --- a/charts/redis/redis/templates/tls-secret.yaml +++ /dev/null @@ -1,31 +0,0 @@ -{{- /* -Copyright Broadcom, Inc. All Rights Reserved. -SPDX-License-Identifier: APACHE-2.0 -*/}} - -{{- if (include "redis.createTlsSecret" .) }} -{{- $secretName := printf "%s-crt" (include "common.names.fullname" .) }} -{{- $ca := genCA "redis-ca" 365 }} -{{- $releaseNamespace := (include "common.names.namespace" .) }} -{{- $clusterDomain := .Values.clusterDomain }} -{{- $fullname := include "common.names.fullname" . }} -{{- $serviceName := include "common.names.fullname" . }} -{{- $headlessServiceName := printf "%s-headless" (include "common.names.fullname" .) }} -{{- $masterServiceName := printf "%s-master" (include "common.names.fullname" .) }} -{{- $altNames := list (printf "*.%s.%s.svc.%s" $serviceName $releaseNamespace $clusterDomain) (printf "%s.%s.svc.%s" $masterServiceName $releaseNamespace $clusterDomain) (printf "*.%s.%s.svc.%s" $masterServiceName $releaseNamespace $clusterDomain) (printf "*.%s.%s.svc.%s" $headlessServiceName $releaseNamespace $clusterDomain) (printf "%s.%s.svc.%s" $headlessServiceName $releaseNamespace $clusterDomain) "127.0.0.1" "localhost" $fullname }} -{{- $cert := genSignedCert $fullname nil $altNames 365 $ca }} -apiVersion: v1 -kind: Secret -metadata: - name: {{ $secretName }} - namespace: {{ include "common.names.namespace" . | quote }} - labels: {{- include "common.labels.standard" ( dict "customLabels" .Values.commonLabels "context" $ ) | nindent 4 }} - {{- if .Values.commonAnnotations }} - annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} - {{- end }} -type: kubernetes.io/tls -data: - tls.crt: {{ include "common.secrets.lookup" (dict "secret" $secretName "key" "tls.crt" "defaultValue" $cert.Cert "context" $) }} - tls.key: {{ include "common.secrets.lookup" (dict "secret" $secretName "key" "tls.key" "defaultValue" $cert.Key "context" $) }} - ca.crt: {{ include "common.secrets.lookup" (dict "secret" $secretName "key" "ca.crt" "defaultValue" $ca.Cert "context" $) }} -{{- end }} diff --git a/charts/redis/redis/values.schema.json b/charts/redis/redis/values.schema.json deleted file mode 100644 index e9feba6..0000000 --- a/charts/redis/redis/values.schema.json +++ /dev/null @@ -1,163 +0,0 @@ -{ - "$schema": "http://json-schema.org/schema#", - "type": "object", - "properties": { - "architecture": { - "type": "string", - "title": "Redis architecture", - "form": true, - "description": "Allowed values: `standalone` or `replication`", - "enum": ["standalone", "replication"] - }, - "auth": { - "type": "object", - "title": "Authentication configuration", - "form": true, - "properties": { - "enabled": { - "type": "boolean", - "form": true, - "title": "Use password authentication" - }, - "password": { - "type": "string", - "title": "Redis password", - "form": true, - "description": "Defaults to a random 10-character alphanumeric string if not set", - "hidden": { - "value": false, - "path": "auth/enabled" - } - } - } - }, - "master": { - "type": "object", - "title": "Master replicas settings", - "form": true, - "properties": { - "kind": { - "type": "string", - "title": "Workload Kind", - "form": true, - "description": "Allowed values: `Deployment`, `StatefulSet` or `DaemonSet`", - "enum": ["Deployment", "StatefulSet", "DaemonSet"] - }, - "persistence": { - "type": "object", - "title": "Persistence for master replicas", - "form": true, - "properties": { - "enabled": { - "type": "boolean", - "form": true, - "title": "Enable persistence", - "description": "Enable persistence using Persistent Volume Claims" - }, - "size": { - "type": "string", - "title": "Persistent Volume Size", - "form": true, - "render": "slider", - "sliderMin": 1, - "sliderMax": 100, - "sliderUnit": "Gi", - "hidden": { - "value": false, - "path": "master/persistence/enabled" - } - } - } - } - } - }, - "replica": { - "type": "object", - "title": "Redis replicas settings", - "form": true, - "hidden": { - "value": "standalone", - "path": "architecture" - }, - "properties": { - "kind": { - "type": "string", - "title": "Workload Kind", - "form": true, - "description": "Allowed values: `DaemonSet` or `StatefulSet`", - "enum": ["DaemonSet", "StatefulSet"] - }, - "replicaCount": { - "type": "integer", - "form": true, - "title": "Number of Redis replicas" - }, - "persistence": { - "type": "object", - "title": "Persistence for Redis replicas", - "form": true, - "properties": { - "enabled": { - "type": "boolean", - "form": true, - "title": "Enable persistence", - "description": "Enable persistence using Persistent Volume Claims" - }, - "size": { - "type": "string", - "title": "Persistent Volume Size", - "form": true, - "render": "slider", - "sliderMin": 1, - "sliderMax": 100, - "sliderUnit": "Gi", - "hidden": { - "value": false, - "path": "replica/persistence/enabled" - } - } - } - } - } - }, - "volumePermissions": { - "type": "object", - "properties": { - "enabled": { - "type": "boolean", - "form": true, - "title": "Enable Init Containers", - "description": "Use an init container to set required folder permissions on the data volume before mounting it in the final destination" - } - } - }, - "metrics": { - "type": "object", - "form": true, - "title": "Prometheus metrics details", - "properties": { - "enabled": { - "type": "boolean", - "title": "Create Prometheus metrics exporter", - "description": "Create a side-car container to expose Prometheus metrics", - "form": true - }, - "serviceMonitor": { - "type": "object", - "properties": { - "enabled": { - "type": "boolean", - "title": "Create Prometheus Operator ServiceMonitor", - "description": "Create a ServiceMonitor to track metrics using Prometheus Operator", - "form": true, - "hidden": { - "value": false, - "path": "metrics/enabled" - } - } - } - } - } - } - } -} diff --git a/charts/redis/redis/values.yaml b/charts/redis/redis/values.yaml deleted file mode 100644 index de2fc53..0000000 --- a/charts/redis/redis/values.yaml +++ /dev/null @@ -1,2237 +0,0 @@ -# Copyright Broadcom, Inc. All Rights Reserved. -# SPDX-License-Identifier: APACHE-2.0 - -## @section Global parameters -## Global Docker image parameters -## Please, note that this will override the image parameters, including dependencies, configured to use the global value -## Current available global Docker image parameters: imageRegistry, imagePullSecrets and storageClass -## - -## @param global.imageRegistry Global Docker image registry -## @param global.imagePullSecrets Global Docker registry secret names as an array -## @param global.defaultStorageClass Global default StorageClass for Persistent Volume(s) -## @param global.storageClass DEPRECATED: use global.defaultStorageClass instead -## @param global.redis.password Global Redis® password (overrides `auth.password`) -## -global: - imageRegistry: "" - ## E.g. - ## imagePullSecrets: - ## - myRegistryKeySecretName - ## - imagePullSecrets: [] - defaultStorageClass: "" - storageClass: "" - redis: - password: "" - ## Compatibility adaptations for Kubernetes platforms - ## - compatibility: - ## Compatibility adaptations for Openshift - ## - openshift: - ## @param global.compatibility.openshift.adaptSecurityContext Adapt the securityContext sections of the deployment to make them compatible with Openshift restricted-v2 SCC: remove runAsUser, runAsGroup and fsGroup and let the platform use their allowed default IDs. Possible values: auto (apply if the detected running cluster is Openshift), force (perform the adaptation always), disabled (do not perform adaptation) - ## - adaptSecurityContext: auto -## @section Common parameters -## - -## @param kubeVersion Override Kubernetes version -## -kubeVersion: "" -## @param nameOverride String to partially override common.names.fullname -## -nameOverride: "" -## @param fullnameOverride String to fully override common.names.fullname -## -fullnameOverride: "" -## @param namespaceOverride String to fully override common.names.namespace -## -namespaceOverride: "" -## @param commonLabels Labels to add to all deployed objects -## -commonLabels: {} -## @param commonAnnotations Annotations to add to all deployed objects -## -commonAnnotations: {} -## @param secretAnnotations Annotations to add to secret -## -secretAnnotations: {} -## @param clusterDomain Kubernetes cluster domain name -## -clusterDomain: cluster.local -## @param extraDeploy Array of extra objects to deploy with the release -## -extraDeploy: [] -## @param useHostnames Use hostnames internally when announcing replication. If false, the hostname will be resolved to an IP address -## -useHostnames: true -## @param nameResolutionThreshold Failure threshold for internal hostnames resolution -## -nameResolutionThreshold: 5 -## @param nameResolutionTimeout Timeout seconds between probes for internal hostnames resolution -## -nameResolutionTimeout: 5 -## Enable diagnostic mode in the deployment -## -diagnosticMode: - ## @param diagnosticMode.enabled Enable diagnostic mode (all probes will be disabled and the command will be overridden) - ## - enabled: false - ## @param diagnosticMode.command Command to override all containers in the deployment - ## - command: - - sleep - ## @param diagnosticMode.args Args to override all containers in the deployment - ## - args: - - infinity -## @section Redis® Image parameters -## - -## Bitnami Redis® image -## ref: https://hub.docker.com/r/bitnami/redis/tags/ -## @param image.registry [default: REGISTRY_NAME] Redis® image registry -## @param image.repository [default: REPOSITORY_NAME/redis] Redis® image repository -## @skip image.tag Redis® image tag (immutable tags are recommended) -## @param image.digest Redis® image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag -## @param image.pullPolicy Redis® image pull policy -## @param image.pullSecrets Redis® image pull secrets -## @param image.debug Enable image debug mode -## -image: - registry: docker.io - repository: bitnami/redis - tag: 7.4.0-debian-12-r0 - digest: "" - ## Specify a imagePullPolicy - ## Defaults to 'Always' if image tag is 'latest', else set to 'IfNotPresent' - ## ref: https://kubernetes.io/docs/concepts/containers/images/#pre-pulled-images - ## - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets. - ## Secrets must be manually created in the namespace. - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## e.g: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## Enable debug mode - ## - debug: false -## @section Redis® common configuration parameters -## https://github.com/bitnami/containers/tree/main/bitnami/redis#configuration -## - -## @param architecture Redis® architecture. Allowed values: `standalone` or `replication` -## -architecture: replication -## Redis® Authentication parameters -## ref: https://github.com/bitnami/containers/tree/main/bitnami/redis#setting-the-server-password-on-first-run -## -auth: - ## @param auth.enabled Enable password authentication - ## - enabled: true - ## @param auth.sentinel Enable password authentication on sentinels too - ## - sentinel: true - ## @param auth.password Redis® password - ## Defaults to a random 10-character alphanumeric string if not set - ## - password: "" - ## @param auth.existingSecret The name of an existing secret with Redis® credentials - ## NOTE: When it's set, the previous `auth.password` parameter is ignored - ## - existingSecret: "" - ## @param auth.existingSecretPasswordKey Password key to be retrieved from existing secret - ## NOTE: ignored unless `auth.existingSecret` parameter is set - ## - existingSecretPasswordKey: "" - ## @param auth.usePasswordFiles Mount credentials as files instead of using an environment variable - ## - usePasswordFiles: false - ## @param auth.usePasswordFileFromSecret Mount password file from secret - ## - usePasswordFileFromSecret: true -## @param commonConfiguration [string] Common configuration to be added into the ConfigMap -## ref: https://redis.io/topics/config -## -commonConfiguration: |- - # Enable AOF https://redis.io/topics/persistence#append-only-file - appendonly yes - # Disable RDB persistence, AOF persistence already enabled. - save "" -## @param existingConfigmap The name of an existing ConfigMap with your custom configuration for Redis® nodes -## -existingConfigmap: "" -## @section Redis® master configuration parameters -## -master: - ## @param master.count Number of Redis® master instances to deploy (experimental, requires additional configuration) - ## - count: 1 - ## @param master.revisionHistoryLimit The number of old history to retain to allow rollback - ## NOTE: Explicitly setting this field to 0, will result in cleaning up all the history, breaking ability to rollback - revisionHistoryLimit: 10 - ## @param master.configuration Configuration for Redis® master nodes - ## ref: https://redis.io/topics/config - ## - configuration: "" - ## @param master.disableCommands Array with Redis® commands to disable on master nodes - ## Commands will be completely disabled by renaming each to an empty string. - ## ref: https://redis.io/topics/security#disabling-of-specific-commands - ## - disableCommands: - - FLUSHDB - - FLUSHALL - ## @param master.command Override default container command (useful when using custom images) - ## - command: [] - ## @param master.args Override default container args (useful when using custom images) - ## - args: [] - ## @param master.enableServiceLinks Whether information about services should be injected into pod's environment variable - ## - enableServiceLinks: true - ## @param master.preExecCmds Additional commands to run prior to starting Redis® master - ## - preExecCmds: [] - ## @param master.extraFlags Array with additional command line flags for Redis® master - ## e.g: - ## extraFlags: - ## - "--maxmemory-policy volatile-ttl" - ## - "--repl-backlog-size 1024mb" - ## - extraFlags: [] - ## @param master.extraEnvVars Array with extra environment variables to add to Redis® master nodes - ## e.g: - ## extraEnvVars: - ## - name: FOO - ## value: "bar" - ## - extraEnvVars: [] - ## @param master.extraEnvVarsCM Name of existing ConfigMap containing extra env vars for Redis® master nodes - ## - extraEnvVarsCM: "" - ## @param master.extraEnvVarsSecret Name of existing Secret containing extra env vars for Redis® master nodes - ## - extraEnvVarsSecret: "" - ## @param master.containerPorts.redis Container port to open on Redis® master nodes - ## - containerPorts: - redis: 6379 - ## Configure extra options for Redis® containers' liveness and readiness probes - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes - ## @param master.startupProbe.enabled Enable startupProbe on Redis® master nodes - ## @param master.startupProbe.initialDelaySeconds Initial delay seconds for startupProbe - ## @param master.startupProbe.periodSeconds Period seconds for startupProbe - ## @param master.startupProbe.timeoutSeconds Timeout seconds for startupProbe - ## @param master.startupProbe.failureThreshold Failure threshold for startupProbe - ## @param master.startupProbe.successThreshold Success threshold for startupProbe - ## - startupProbe: - enabled: false - initialDelaySeconds: 20 - periodSeconds: 5 - timeoutSeconds: 5 - successThreshold: 1 - failureThreshold: 5 - ## @param master.livenessProbe.enabled Enable livenessProbe on Redis® master nodes - ## @param master.livenessProbe.initialDelaySeconds Initial delay seconds for livenessProbe - ## @param master.livenessProbe.periodSeconds Period seconds for livenessProbe - ## @param master.livenessProbe.timeoutSeconds Timeout seconds for livenessProbe - ## @param master.livenessProbe.failureThreshold Failure threshold for livenessProbe - ## @param master.livenessProbe.successThreshold Success threshold for livenessProbe - ## - livenessProbe: - enabled: true - initialDelaySeconds: 20 - periodSeconds: 5 - timeoutSeconds: 5 - successThreshold: 1 - failureThreshold: 5 - ## @param master.readinessProbe.enabled Enable readinessProbe on Redis® master nodes - ## @param master.readinessProbe.initialDelaySeconds Initial delay seconds for readinessProbe - ## @param master.readinessProbe.periodSeconds Period seconds for readinessProbe - ## @param master.readinessProbe.timeoutSeconds Timeout seconds for readinessProbe - ## @param master.readinessProbe.failureThreshold Failure threshold for readinessProbe - ## @param master.readinessProbe.successThreshold Success threshold for readinessProbe - ## - readinessProbe: - enabled: true - initialDelaySeconds: 20 - periodSeconds: 5 - timeoutSeconds: 1 - successThreshold: 1 - failureThreshold: 5 - ## @param master.customStartupProbe Custom startupProbe that overrides the default one - ## - customStartupProbe: {} - ## @param master.customLivenessProbe Custom livenessProbe that overrides the default one - ## - customLivenessProbe: {} - ## @param master.customReadinessProbe Custom readinessProbe that overrides the default one - ## - customReadinessProbe: {} - ## Redis® master resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## @param master.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if master.resources is set (master.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "nano" - ## @param master.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} - ## Configure Pods Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-pod - ## @param master.podSecurityContext.enabled Enabled Redis® master pods' Security Context - ## @param master.podSecurityContext.fsGroupChangePolicy Set filesystem group change policy - ## @param master.podSecurityContext.sysctls Set kernel settings using the sysctl interface - ## @param master.podSecurityContext.supplementalGroups Set filesystem extra groups - ## @param master.podSecurityContext.fsGroup Set Redis® master pod's Security Context fsGroup - ## - podSecurityContext: - enabled: true - fsGroupChangePolicy: Always - sysctls: [] - supplementalGroups: [] - fsGroup: 1001 - ## Configure Container Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-pod - ## @param master.containerSecurityContext.enabled Enabled Redis® master containers' Security Context - ## @param master.containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container - ## @param master.containerSecurityContext.runAsUser Set Redis® master containers' Security Context runAsUser - ## @param master.containerSecurityContext.runAsGroup Set Redis® master containers' Security Context runAsGroup - ## @param master.containerSecurityContext.runAsNonRoot Set Redis® master containers' Security Context runAsNonRoot - ## @param master.containerSecurityContext.allowPrivilegeEscalation Is it possible to escalate Redis® pod(s) privileges - ## @param master.containerSecurityContext.readOnlyRootFilesystem Set container's Security Context read-only root filesystem - ## @param master.containerSecurityContext.seccompProfile.type Set Redis® master containers' Security Context seccompProfile - ## @param master.containerSecurityContext.capabilities.drop Set Redis® master containers' Security Context capabilities to drop - ## - containerSecurityContext: - enabled: true - seLinuxOptions: {} - runAsUser: 1001 - runAsGroup: 1001 - runAsNonRoot: true - allowPrivilegeEscalation: false - readOnlyRootFilesystem: true - seccompProfile: - type: RuntimeDefault - capabilities: - drop: ["ALL"] - ## @param master.kind Use either Deployment, StatefulSet (default) or DaemonSet - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/ - ## - kind: StatefulSet - ## @param master.schedulerName Alternate scheduler for Redis® master pods - ## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ - ## - schedulerName: "" - ## @param master.updateStrategy.type Redis® master statefulset strategy type - ## @skip master.updateStrategy.rollingUpdate - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#update-strategies - ## - updateStrategy: - ## StrategyType - ## Can be set to RollingUpdate, OnDelete (statefulset), Recreate (deployment) - ## - type: RollingUpdate - ## @param master.minReadySeconds How many seconds a pod needs to be ready before killing the next, during update - ## - minReadySeconds: 0 - ## @param master.priorityClassName Redis® master pods' priorityClassName - ## - priorityClassName: "" - ## @param master.automountServiceAccountToken Mount Service Account token in pod - ## - automountServiceAccountToken: false - ## @param master.hostAliases Redis® master pods host aliases - ## https://kubernetes.io/docs/concepts/services-networking/add-entries-to-pod-etc-hosts-with-host-aliases/ - ## - hostAliases: [] - ## @param master.podLabels Extra labels for Redis® master pods - ## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/ - ## - podLabels: {} - ## @param master.podAnnotations Annotations for Redis® master pods - ## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ - ## - podAnnotations: {} - ## @param master.shareProcessNamespace Share a single process namespace between all of the containers in Redis® master pods - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/share-process-namespace/ - ## - shareProcessNamespace: false - ## @param master.podAffinityPreset Pod affinity preset. Ignored if `master.affinity` is set. Allowed values: `soft` or `hard` - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity - ## - podAffinityPreset: "" - ## @param master.podAntiAffinityPreset Pod anti-affinity preset. Ignored if `master.affinity` is set. Allowed values: `soft` or `hard` - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity - ## - podAntiAffinityPreset: soft - ## Node master.affinity preset - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity - ## - nodeAffinityPreset: - ## @param master.nodeAffinityPreset.type Node affinity preset type. Ignored if `master.affinity` is set. Allowed values: `soft` or `hard` - ## - type: "" - ## @param master.nodeAffinityPreset.key Node label key to match. Ignored if `master.affinity` is set - ## - key: "" - ## @param master.nodeAffinityPreset.values Node label values to match. Ignored if `master.affinity` is set - ## E.g. - ## values: - ## - e2e-az1 - ## - e2e-az2 - ## - values: [] - ## @param master.affinity Affinity for Redis® master pods assignment - ## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity - ## NOTE: `master.podAffinityPreset`, `master.podAntiAffinityPreset`, and `master.nodeAffinityPreset` will be ignored when it's set - ## - affinity: {} - ## @param master.nodeSelector Node labels for Redis® master pods assignment - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/ - ## - nodeSelector: {} - ## @param master.tolerations Tolerations for Redis® master pods assignment - ## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ - ## - tolerations: [] - ## @param master.topologySpreadConstraints Spread Constraints for Redis® master pod assignment - ## ref: https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/ - ## E.g. - ## topologySpreadConstraints: - ## - maxSkew: 1 - ## topologyKey: node - ## whenUnsatisfiable: DoNotSchedule - ## - topologySpreadConstraints: [] - ## @param master.dnsPolicy DNS Policy for Redis® master pod - ## ref: https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/ - ## E.g. - ## dnsPolicy: ClusterFirst - ## - dnsPolicy: "" - ## @param master.dnsConfig DNS Configuration for Redis® master pod - ## ref: https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/ - ## E.g. - ## dnsConfig: - ## options: - ## - name: ndots - ## value: "4" - ## - name: single-request-reopen - ## - dnsConfig: {} - ## @param master.lifecycleHooks for the Redis® master container(s) to automate configuration before or after startup - ## - lifecycleHooks: {} - ## @param master.extraVolumes Optionally specify extra list of additional volumes for the Redis® master pod(s) - ## - extraVolumes: [] - ## @param master.extraVolumeMounts Optionally specify extra list of additional volumeMounts for the Redis® master container(s) - ## - extraVolumeMounts: [] - ## @param master.sidecars Add additional sidecar containers to the Redis® master pod(s) - ## e.g: - ## sidecars: - ## - name: your-image-name - ## image: your-image - ## imagePullPolicy: Always - ## ports: - ## - name: portname - ## containerPort: 1234 - ## - sidecars: [] - ## @param master.initContainers Add additional init containers to the Redis® master pod(s) - ## ref: https://kubernetes.io/docs/concepts/workloads/pods/init-containers/ - ## e.g: - ## initContainers: - ## - name: your-image-name - ## image: your-image - ## imagePullPolicy: Always - ## command: ['sh', '-c', 'echo "hello world"'] - ## - initContainers: [] - ## Persistence parameters - ## ref: https://kubernetes.io/docs/concepts/storage/persistent-volumes/ - ## - persistence: - ## @param master.persistence.enabled Enable persistence on Redis® master nodes using Persistent Volume Claims - ## - enabled: true - ## @param master.persistence.medium Provide a medium for `emptyDir` volumes. - ## - medium: "" - ## @param master.persistence.sizeLimit Set this to enable a size limit for `emptyDir` volumes. - ## - sizeLimit: "" - ## @param master.persistence.path The path the volume will be mounted at on Redis® master containers - ## NOTE: Useful when using different Redis® images - ## - path: /data - ## @param master.persistence.subPath The subdirectory of the volume to mount on Redis® master containers - ## NOTE: Useful in dev environments - ## - subPath: "" - ## @param master.persistence.subPathExpr Used to construct the subPath subdirectory of the volume to mount on Redis® master containers - ## - subPathExpr: "" - ## @param master.persistence.storageClass Persistent Volume storage class - ## If defined, storageClassName: - ## If set to "-", storageClassName: "", which disables dynamic provisioning - ## If undefined (the default) or set to null, no storageClassName spec is set, choosing the default provisioner - ## - storageClass: "" - ## @param master.persistence.accessModes Persistent Volume access modes - ## - accessModes: - - ReadWriteOnce - ## @param master.persistence.size Persistent Volume size - ## - size: 8Gi - ## @param master.persistence.annotations Additional custom annotations for the PVC - ## - annotations: {} - ## @param master.persistence.labels Additional custom labels for the PVC - ## - labels: {} - ## @param master.persistence.selector Additional labels to match for the PVC - ## e.g: - ## selector: - ## matchLabels: - ## app: my-app - ## - selector: {} - ## @param master.persistence.dataSource Custom PVC data source - ## - dataSource: {} - ## @param master.persistence.existingClaim Use a existing PVC which must be created manually before bound - ## NOTE: requires master.persistence.enabled: true - ## - existingClaim: "" - ## persistentVolumeClaimRetentionPolicy - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#persistentvolumeclaim-retention - ## @param master.persistentVolumeClaimRetentionPolicy.enabled Controls if and how PVCs are deleted during the lifecycle of a StatefulSet - ## @param master.persistentVolumeClaimRetentionPolicy.whenScaled Volume retention behavior when the replica count of the StatefulSet is reduced - ## @param master.persistentVolumeClaimRetentionPolicy.whenDeleted Volume retention behavior that applies when the StatefulSet is deleted - ## - persistentVolumeClaimRetentionPolicy: - enabled: false - whenScaled: Retain - whenDeleted: Retain - ## Redis® master service parameters - ## - service: - ## @param master.service.type Redis® master service type - ## - type: ClusterIP - ## @param master.service.portNames.redis Redis® master service port name - ## - portNames: - redis: "tcp-redis" - ## @param master.service.ports.redis Redis® master service port - ## - ports: - redis: 6379 - ## @param master.service.nodePorts.redis Node port for Redis® master - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport - ## NOTE: choose port between <30000-32767> - ## - nodePorts: - redis: "" - ## @param master.service.externalTrafficPolicy Redis® master service external traffic policy - ## ref: https://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer/#preserving-the-client-source-ip - ## - externalTrafficPolicy: Cluster - ## @param master.service.extraPorts Extra ports to expose (normally used with the `sidecar` value) - ## - extraPorts: [] - ## @param master.service.internalTrafficPolicy Redis® master service internal traffic policy (requires Kubernetes v1.22 or greater to be usable) - ## ref: https://kubernetes.io/docs/concepts/services-networking/service-traffic-policy/ - ## - internalTrafficPolicy: Cluster - ## @param master.service.clusterIP Redis® master service Cluster IP - ## - clusterIP: "" - ## @param master.service.loadBalancerIP Redis® master service Load Balancer IP - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#internal-load-balancer - ## - loadBalancerIP: "" - ## @param master.service.loadBalancerClass master service Load Balancer class if service type is `LoadBalancer` (optional, cloud specific) - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-loadbalancer - ## - loadBalancerClass: "" - ## @param master.service.loadBalancerSourceRanges Redis® master service Load Balancer sources - ## https://kubernetes.io/docs/tasks/access-application-cluster/configure-cloud-provider-firewall/#restrict-access-for-loadbalancer-service - ## e.g. - ## loadBalancerSourceRanges: - ## - 10.10.10.0/24 - ## - loadBalancerSourceRanges: [] - ## @param master.service.externalIPs Redis® master service External IPs - ## https://kubernetes.io/docs/concepts/services-networking/service/#external-ips - ## e.g. - ## externalIPs: - ## - 10.10.10.1 - ## - 201.22.30.1 - ## - externalIPs: [] - ## @param master.service.annotations Additional custom annotations for Redis® master service - ## - annotations: {} - ## @param master.service.sessionAffinity Session Affinity for Kubernetes service, can be "None" or "ClientIP" - ## If "ClientIP", consecutive client requests will be directed to the same Pod - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies - ## - sessionAffinity: None - ## @param master.service.sessionAffinityConfig Additional settings for the sessionAffinity - ## sessionAffinityConfig: - ## clientIP: - ## timeoutSeconds: 300 - ## - sessionAffinityConfig: {} - ## @param master.terminationGracePeriodSeconds Integer setting the termination grace period for the redis-master pods - ## - terminationGracePeriodSeconds: 30 - ## ServiceAccount configuration - ## - serviceAccount: - ## @param master.serviceAccount.create Specifies whether a ServiceAccount should be created - ## - create: true - ## @param master.serviceAccount.name The name of the ServiceAccount to use. - ## If not set and create is true, a name is generated using the common.names.fullname template - ## - name: "" - ## @param master.serviceAccount.automountServiceAccountToken Whether to auto mount the service account token - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/#use-the-default-service-account-to-access-the-api-server - ## - automountServiceAccountToken: false - ## @param master.serviceAccount.annotations Additional custom annotations for the ServiceAccount - ## - annotations: {} - ## Pod Disruption Budget configuration - ## ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb - ## @param master.pdb.create Enable/disable a Pod Disruption Budget creation - ## @param master.pdb.minAvailable [object] Minimum number/percentage of pods that should remain scheduled - ## @param master.pdb.maxUnavailable [object] Maximum number/percentage of pods that may be made unavailable. Defaults to `1` if both `master.pdb.minAvailable` and `master.pdb.maxUnavailable` are empty. - ## - pdb: - create: true - minAvailable: "" - maxUnavailable: "" -## @section Redis® replicas configuration parameters -## -replica: - ## @param replica.kind Use either DaemonSet or StatefulSet (default) - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/ - ## - kind: StatefulSet - ## @param replica.replicaCount Number of Redis® replicas to deploy - ## - replicaCount: 3 - ## @param replica.revisionHistoryLimit The number of old history to retain to allow rollback - ## NOTE: Explicitly setting this field to 0, will result in cleaning up all the history, breaking ability to rollback - revisionHistoryLimit: 10 - ## @param replica.configuration Configuration for Redis® replicas nodes - ## ref: https://redis.io/topics/config - ## - configuration: "" - ## @param replica.disableCommands Array with Redis® commands to disable on replicas nodes - ## Commands will be completely disabled by renaming each to an empty string. - ## ref: https://redis.io/topics/security#disabling-of-specific-commands - ## - disableCommands: - - FLUSHDB - - FLUSHALL - ## @param replica.command Override default container command (useful when using custom images) - ## - command: [] - ## @param replica.args Override default container args (useful when using custom images) - ## - args: [] - ## @param replica.enableServiceLinks Whether information about services should be injected into pod's environment variable - ## - enableServiceLinks: true - ## @param replica.preExecCmds Additional commands to run prior to starting Redis® replicas - ## - preExecCmds: [] - ## @param replica.extraFlags Array with additional command line flags for Redis® replicas - ## e.g: - ## extraFlags: - ## - "--maxmemory-policy volatile-ttl" - ## - "--repl-backlog-size 1024mb" - ## - extraFlags: [] - ## @param replica.extraEnvVars Array with extra environment variables to add to Redis® replicas nodes - ## e.g: - ## extraEnvVars: - ## - name: FOO - ## value: "bar" - ## - extraEnvVars: [] - ## @param replica.extraEnvVarsCM Name of existing ConfigMap containing extra env vars for Redis® replicas nodes - ## - extraEnvVarsCM: "" - ## @param replica.extraEnvVarsSecret Name of existing Secret containing extra env vars for Redis® replicas nodes - ## - extraEnvVarsSecret: "" - ## @param replica.externalMaster.enabled Use external master for bootstrapping - ## @param replica.externalMaster.host External master host to bootstrap from - ## @param replica.externalMaster.port Port for Redis service external master host - ## - externalMaster: - enabled: false - host: "" - port: 6379 - ## @param replica.containerPorts.redis Container port to open on Redis® replicas nodes - ## - containerPorts: - redis: 6379 - ## Configure extra options for Redis® containers' liveness and readiness probes - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes - ## @param replica.startupProbe.enabled Enable startupProbe on Redis® replicas nodes - ## @param replica.startupProbe.initialDelaySeconds Initial delay seconds for startupProbe - ## @param replica.startupProbe.periodSeconds Period seconds for startupProbe - ## @param replica.startupProbe.timeoutSeconds Timeout seconds for startupProbe - ## @param replica.startupProbe.failureThreshold Failure threshold for startupProbe - ## @param replica.startupProbe.successThreshold Success threshold for startupProbe - ## - startupProbe: - enabled: true - initialDelaySeconds: 10 - periodSeconds: 10 - timeoutSeconds: 5 - successThreshold: 1 - failureThreshold: 22 - ## @param replica.livenessProbe.enabled Enable livenessProbe on Redis® replicas nodes - ## @param replica.livenessProbe.initialDelaySeconds Initial delay seconds for livenessProbe - ## @param replica.livenessProbe.periodSeconds Period seconds for livenessProbe - ## @param replica.livenessProbe.timeoutSeconds Timeout seconds for livenessProbe - ## @param replica.livenessProbe.failureThreshold Failure threshold for livenessProbe - ## @param replica.livenessProbe.successThreshold Success threshold for livenessProbe - ## - livenessProbe: - enabled: true - initialDelaySeconds: 20 - periodSeconds: 5 - timeoutSeconds: 5 - successThreshold: 1 - failureThreshold: 5 - ## @param replica.readinessProbe.enabled Enable readinessProbe on Redis® replicas nodes - ## @param replica.readinessProbe.initialDelaySeconds Initial delay seconds for readinessProbe - ## @param replica.readinessProbe.periodSeconds Period seconds for readinessProbe - ## @param replica.readinessProbe.timeoutSeconds Timeout seconds for readinessProbe - ## @param replica.readinessProbe.failureThreshold Failure threshold for readinessProbe - ## @param replica.readinessProbe.successThreshold Success threshold for readinessProbe - ## - readinessProbe: - enabled: true - initialDelaySeconds: 20 - periodSeconds: 5 - timeoutSeconds: 1 - successThreshold: 1 - failureThreshold: 5 - ## @param replica.customStartupProbe Custom startupProbe that overrides the default one - ## - customStartupProbe: {} - ## @param replica.customLivenessProbe Custom livenessProbe that overrides the default one - ## - customLivenessProbe: {} - ## @param replica.customReadinessProbe Custom readinessProbe that overrides the default one - ## - customReadinessProbe: {} - ## Redis® replicas resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## @param replica.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if replica.resources is set (replica.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "nano" - ## @param replica.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} - ## Configure Pods Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-pod - ## @param replica.podSecurityContext.enabled Enabled Redis® replicas pods' Security Context - ## @param replica.podSecurityContext.fsGroupChangePolicy Set filesystem group change policy - ## @param replica.podSecurityContext.sysctls Set kernel settings using the sysctl interface - ## @param replica.podSecurityContext.supplementalGroups Set filesystem extra groups - ## @param replica.podSecurityContext.fsGroup Set Redis® replicas pod's Security Context fsGroup - ## - podSecurityContext: - enabled: true - fsGroupChangePolicy: Always - sysctls: [] - supplementalGroups: [] - fsGroup: 1001 - ## Configure Container Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-pod - ## @param replica.containerSecurityContext.enabled Enabled Redis® replicas containers' Security Context - ## @param replica.containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container - ## @param replica.containerSecurityContext.runAsUser Set Redis® replicas containers' Security Context runAsUser - ## @param replica.containerSecurityContext.runAsGroup Set Redis® replicas containers' Security Context runAsGroup - ## @param replica.containerSecurityContext.runAsNonRoot Set Redis® replicas containers' Security Context runAsNonRoot - ## @param replica.containerSecurityContext.allowPrivilegeEscalation Set Redis® replicas pod's Security Context allowPrivilegeEscalation - ## @param replica.containerSecurityContext.readOnlyRootFilesystem Set container's Security Context read-only root filesystem - ## @param replica.containerSecurityContext.seccompProfile.type Set Redis® replicas containers' Security Context seccompProfile - ## @param replica.containerSecurityContext.capabilities.drop Set Redis® replicas containers' Security Context capabilities to drop - ## - containerSecurityContext: - enabled: true - seLinuxOptions: {} - runAsUser: 1001 - runAsGroup: 1001 - runAsNonRoot: true - allowPrivilegeEscalation: false - readOnlyRootFilesystem: true - seccompProfile: - type: RuntimeDefault - capabilities: - drop: ["ALL"] - ## @param replica.schedulerName Alternate scheduler for Redis® replicas pods - ## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ - ## - schedulerName: "" - ## @param replica.updateStrategy.type Redis® replicas statefulset strategy type - ## @skip replica.updateStrategy.rollingUpdate - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#update-strategies - ## - updateStrategy: - ## StrategyType - ## Can be set to RollingUpdate, OnDelete (statefulset), Recreate (deployment) - ## - type: RollingUpdate - ## @param replica.minReadySeconds How many seconds a pod needs to be ready before killing the next, during update - ## - minReadySeconds: 0 - ## @param replica.priorityClassName Redis® replicas pods' priorityClassName - ## - priorityClassName: "" - ## @param replica.podManagementPolicy podManagementPolicy to manage scaling operation of %%MAIN_CONTAINER_NAME%% pods - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#pod-management-policies - ## - podManagementPolicy: "" - ## @param replica.automountServiceAccountToken Mount Service Account token in pod - ## - automountServiceAccountToken: false - ## @param replica.hostAliases Redis® replicas pods host aliases - ## https://kubernetes.io/docs/concepts/services-networking/add-entries-to-pod-etc-hosts-with-host-aliases/ - ## - hostAliases: [] - ## @param replica.podLabels Extra labels for Redis® replicas pods - ## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/ - ## - podLabels: {} - ## @param replica.podAnnotations Annotations for Redis® replicas pods - ## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ - ## - podAnnotations: {} - ## @param replica.shareProcessNamespace Share a single process namespace between all of the containers in Redis® replicas pods - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/share-process-namespace/ - ## - shareProcessNamespace: false - ## @param replica.podAffinityPreset Pod affinity preset. Ignored if `replica.affinity` is set. Allowed values: `soft` or `hard` - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity - ## - podAffinityPreset: "" - ## @param replica.podAntiAffinityPreset Pod anti-affinity preset. Ignored if `replica.affinity` is set. Allowed values: `soft` or `hard` - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity - ## - podAntiAffinityPreset: soft - ## Node affinity preset - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity - ## - nodeAffinityPreset: - ## @param replica.nodeAffinityPreset.type Node affinity preset type. Ignored if `replica.affinity` is set. Allowed values: `soft` or `hard` - ## - type: "" - ## @param replica.nodeAffinityPreset.key Node label key to match. Ignored if `replica.affinity` is set - ## - key: "" - ## @param replica.nodeAffinityPreset.values Node label values to match. Ignored if `replica.affinity` is set - ## E.g. - ## values: - ## - e2e-az1 - ## - e2e-az2 - ## - values: [] - ## @param replica.affinity Affinity for Redis® replicas pods assignment - ## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity - ## NOTE: `replica.podAffinityPreset`, `replica.podAntiAffinityPreset`, and `replica.nodeAffinityPreset` will be ignored when it's set - ## - affinity: {} - ## @param replica.nodeSelector Node labels for Redis® replicas pods assignment - ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/ - ## - nodeSelector: {} - ## @param replica.tolerations Tolerations for Redis® replicas pods assignment - ## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ - ## - tolerations: [] - ## @param replica.topologySpreadConstraints Spread Constraints for Redis® replicas pod assignment - ## ref: https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/ - ## E.g. - ## topologySpreadConstraints: - ## - maxSkew: 1 - ## topologyKey: node - ## whenUnsatisfiable: DoNotSchedule - ## - topologySpreadConstraints: [] - ## @param replica.dnsPolicy DNS Policy for Redis® replica pods - ## ref: https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/ - ## E.g. - ## dnsPolicy: ClusterFirst - ## - dnsPolicy: "" - ## @param replica.dnsConfig DNS Configuration for Redis® replica pods - ## ref: https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/ - ## E.g. - ## dnsConfig: - ## options: - ## - name: ndots - ## value: "4" - ## - name: single-request-reopen - ## - dnsConfig: {} - ## @param replica.lifecycleHooks for the Redis® replica container(s) to automate configuration before or after startup - ## - lifecycleHooks: {} - ## @param replica.extraVolumes Optionally specify extra list of additional volumes for the Redis® replicas pod(s) - ## - extraVolumes: [] - ## @param replica.extraVolumeMounts Optionally specify extra list of additional volumeMounts for the Redis® replicas container(s) - ## - extraVolumeMounts: [] - ## @param replica.sidecars Add additional sidecar containers to the Redis® replicas pod(s) - ## e.g: - ## sidecars: - ## - name: your-image-name - ## image: your-image - ## imagePullPolicy: Always - ## ports: - ## - name: portname - ## containerPort: 1234 - ## - sidecars: [] - ## @param replica.initContainers Add additional init containers to the Redis® replicas pod(s) - ## ref: https://kubernetes.io/docs/concepts/workloads/pods/init-containers/ - ## e.g: - ## initContainers: - ## - name: your-image-name - ## image: your-image - ## imagePullPolicy: Always - ## command: ['sh', '-c', 'echo "hello world"'] - ## - initContainers: [] - ## Persistence Parameters - ## ref: https://kubernetes.io/docs/concepts/storage/persistent-volumes/ - ## - persistence: - ## @param replica.persistence.enabled Enable persistence on Redis® replicas nodes using Persistent Volume Claims - ## - enabled: true - ## @param replica.persistence.medium Provide a medium for `emptyDir` volumes. - ## - medium: "" - ## @param replica.persistence.sizeLimit Set this to enable a size limit for `emptyDir` volumes. - ## - sizeLimit: "" - ## @param replica.persistence.path The path the volume will be mounted at on Redis® replicas containers - ## NOTE: Useful when using different Redis® images - ## - path: /data - ## @param replica.persistence.subPath The subdirectory of the volume to mount on Redis® replicas containers - ## NOTE: Useful in dev environments - ## - subPath: "" - ## @param replica.persistence.subPathExpr Used to construct the subPath subdirectory of the volume to mount on Redis® replicas containers - ## - subPathExpr: "" - ## @param replica.persistence.storageClass Persistent Volume storage class - ## If defined, storageClassName: - ## If set to "-", storageClassName: "", which disables dynamic provisioning - ## If undefined (the default) or set to null, no storageClassName spec is set, choosing the default provisioner - ## - storageClass: "" - ## @param replica.persistence.accessModes Persistent Volume access modes - ## - accessModes: - - ReadWriteOnce - ## @param replica.persistence.size Persistent Volume size - ## - size: 8Gi - ## @param replica.persistence.annotations Additional custom annotations for the PVC - ## - annotations: {} - ## @param replica.persistence.labels Additional custom labels for the PVC - ## - labels: {} - ## @param replica.persistence.selector Additional labels to match for the PVC - ## e.g: - ## selector: - ## matchLabels: - ## app: my-app - ## - selector: {} - ## @param replica.persistence.dataSource Custom PVC data source - ## - dataSource: {} - ## @param replica.persistence.existingClaim Use a existing PVC which must be created manually before bound - ## NOTE: requires replica.persistence.enabled: true - ## - existingClaim: "" - ## persistentVolumeClaimRetentionPolicy - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#persistentvolumeclaim-retention - ## @param replica.persistentVolumeClaimRetentionPolicy.enabled Controls if and how PVCs are deleted during the lifecycle of a StatefulSet - ## @param replica.persistentVolumeClaimRetentionPolicy.whenScaled Volume retention behavior when the replica count of the StatefulSet is reduced - ## @param replica.persistentVolumeClaimRetentionPolicy.whenDeleted Volume retention behavior that applies when the StatefulSet is deleted - ## - persistentVolumeClaimRetentionPolicy: - enabled: false - whenScaled: Retain - whenDeleted: Retain - ## Redis® replicas service parameters - ## - service: - ## @param replica.service.type Redis® replicas service type - ## - type: ClusterIP - ## @param replica.service.ports.redis Redis® replicas service port - ## - ports: - redis: 6379 - ## @param replica.service.nodePorts.redis Node port for Redis® replicas - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport - ## NOTE: choose port between <30000-32767> - ## - nodePorts: - redis: "" - ## @param replica.service.externalTrafficPolicy Redis® replicas service external traffic policy - ## ref: https://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer/#preserving-the-client-source-ip - ## - externalTrafficPolicy: Cluster - ## @param replica.service.internalTrafficPolicy Redis® replicas service internal traffic policy (requires Kubernetes v1.22 or greater to be usable) - ## ref: https://kubernetes.io/docs/concepts/services-networking/service-traffic-policy/ - ## - internalTrafficPolicy: Cluster - ## @param replica.service.extraPorts Extra ports to expose (normally used with the `sidecar` value) - ## - extraPorts: [] - ## @param replica.service.clusterIP Redis® replicas service Cluster IP - ## - clusterIP: "" - ## @param replica.service.loadBalancerIP Redis® replicas service Load Balancer IP - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#internal-load-balancer - ## - loadBalancerIP: "" - ## @param replica.service.loadBalancerClass replicas service Load Balancer class if service type is `LoadBalancer` (optional, cloud specific) - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-loadbalancer - ## - loadBalancerClass: "" - ## @param replica.service.loadBalancerSourceRanges Redis® replicas service Load Balancer sources - ## https://kubernetes.io/docs/tasks/access-application-cluster/configure-cloud-provider-firewall/#restrict-access-for-loadbalancer-service - ## e.g. - ## loadBalancerSourceRanges: - ## - 10.10.10.0/24 - ## - loadBalancerSourceRanges: [] - ## @param replica.service.annotations Additional custom annotations for Redis® replicas service - ## - annotations: {} - ## @param replica.service.sessionAffinity Session Affinity for Kubernetes service, can be "None" or "ClientIP" - ## If "ClientIP", consecutive client requests will be directed to the same Pod - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies - ## - sessionAffinity: None - ## @param replica.service.sessionAffinityConfig Additional settings for the sessionAffinity - ## sessionAffinityConfig: - ## clientIP: - ## timeoutSeconds: 300 - ## - sessionAffinityConfig: {} - ## @param replica.terminationGracePeriodSeconds Integer setting the termination grace period for the redis-replicas pods - ## - terminationGracePeriodSeconds: 30 - ## Autoscaling configuration - ## - autoscaling: - ## @param replica.autoscaling.enabled Enable replica autoscaling settings - ## - enabled: false - ## @param replica.autoscaling.minReplicas Minimum replicas for the pod autoscaling - ## - minReplicas: 1 - ## @param replica.autoscaling.maxReplicas Maximum replicas for the pod autoscaling - ## - maxReplicas: 11 - ## @param replica.autoscaling.targetCPU Percentage of CPU to consider when autoscaling - ## - targetCPU: "" - ## @param replica.autoscaling.targetMemory Percentage of Memory to consider when autoscaling - ## - targetMemory: "" - ## ServiceAccount configuration - ## - serviceAccount: - ## @param replica.serviceAccount.create Specifies whether a ServiceAccount should be created - ## - create: true - ## @param replica.serviceAccount.name The name of the ServiceAccount to use. - ## If not set and create is true, a name is generated using the common.names.fullname template - ## - name: "" - ## @param replica.serviceAccount.automountServiceAccountToken Whether to auto mount the service account token - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/#use-the-default-service-account-to-access-the-api-server - ## - automountServiceAccountToken: false - ## @param replica.serviceAccount.annotations Additional custom annotations for the ServiceAccount - ## - annotations: {} - ## Pod Disruption Budget configuration - ## ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb - ## @param replica.pdb.create Enable/disable a Pod Disruption Budget creation - ## @param replica.pdb.minAvailable [object] Minimum number/percentage of pods that should remain scheduled - ## @param replica.pdb.maxUnavailable [object] Maximum number/percentage of pods that may be made unavailable. Defaults to `1` if both `replica.pdb.minAvailable` and `replica.pdb.maxUnavailable` are empty. - ## - pdb: - create: true - minAvailable: "" - maxUnavailable: "" -## @section Redis® Sentinel configuration parameters -## - -sentinel: - ## @param sentinel.enabled Use Redis® Sentinel on Redis® pods. - ## IMPORTANT: this will disable the master and replicas services and - ## create a single Redis® service exposing both the Redis and Sentinel ports - ## - enabled: false - ## Bitnami Redis® Sentinel image version - ## ref: https://hub.docker.com/r/bitnami/redis-sentinel/tags/ - ## @param sentinel.image.registry [default: REGISTRY_NAME] Redis® Sentinel image registry - ## @param sentinel.image.repository [default: REPOSITORY_NAME/redis-sentinel] Redis® Sentinel image repository - ## @skip sentinel.image.tag Redis® Sentinel image tag (immutable tags are recommended) - ## @param sentinel.image.digest Redis® Sentinel image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag - ## @param sentinel.image.pullPolicy Redis® Sentinel image pull policy - ## @param sentinel.image.pullSecrets Redis® Sentinel image pull secrets - ## @param sentinel.image.debug Enable image debug mode - ## - image: - registry: docker.io - repository: bitnami/redis-sentinel - tag: 7.4.0-debian-12-r0 - digest: "" - ## Specify a imagePullPolicy - ## Defaults to 'Always' if image tag is 'latest', else set to 'IfNotPresent' - ## ref: https://kubernetes.io/docs/concepts/containers/images/#pre-pulled-images - ## - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets. - ## Secrets must be manually created in the namespace. - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## e.g: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## Enable debug mode - ## - debug: false - ## @param sentinel.annotations Additional custom annotations for Redis® Sentinel resource - ## - annotations: {} - ## @param sentinel.masterSet Master set name - ## - masterSet: mymaster - ## @param sentinel.quorum Sentinel Quorum - ## - quorum: 2 - ## @param sentinel.getMasterTimeout Amount of time to allow before get_sentinel_master_info() times out. - ## - getMasterTimeout: 90 - ## @param sentinel.automateClusterRecovery Automate cluster recovery in cases where the last replica is not considered a good replica and Sentinel won't automatically failover to it. - ## This also prevents any new replica from starting until the last remaining replica is elected as master to guarantee that it is the one to be elected by Sentinel, and not a newly started replica with no data. - ## NOTE: This feature requires a "downAfterMilliseconds" value less or equal to 2000. - ## - automateClusterRecovery: false - ## @param sentinel.redisShutdownWaitFailover Whether the Redis® master container waits for the failover at shutdown (in addition to the Redis® Sentinel container). - ## - redisShutdownWaitFailover: true - ## Sentinel timing restrictions - ## @param sentinel.downAfterMilliseconds Timeout for detecting a Redis® node is down - ## @param sentinel.failoverTimeout Timeout for performing a election failover - ## - downAfterMilliseconds: 60000 - failoverTimeout: 180000 - ## @param sentinel.parallelSyncs Number of replicas that can be reconfigured in parallel to use the new master after a failover - ## - parallelSyncs: 1 - ## @param sentinel.configuration Configuration for Redis® Sentinel nodes - ## ref: https://redis.io/topics/sentinel - ## - configuration: "" - ## @param sentinel.command Override default container command (useful when using custom images) - ## - command: [] - ## @param sentinel.args Override default container args (useful when using custom images) - ## - args: [] - ## @param sentinel.enableServiceLinks Whether information about services should be injected into pod's environment variable - ## - enableServiceLinks: true - ## @param sentinel.preExecCmds Additional commands to run prior to starting Redis® Sentinel - ## - preExecCmds: [] - ## @param sentinel.extraEnvVars Array with extra environment variables to add to Redis® Sentinel nodes - ## e.g: - ## extraEnvVars: - ## - name: FOO - ## value: "bar" - ## - extraEnvVars: [] - ## @param sentinel.extraEnvVarsCM Name of existing ConfigMap containing extra env vars for Redis® Sentinel nodes - ## - extraEnvVarsCM: "" - ## @param sentinel.extraEnvVarsSecret Name of existing Secret containing extra env vars for Redis® Sentinel nodes - ## - extraEnvVarsSecret: "" - ## @param sentinel.externalMaster.enabled Use external master for bootstrapping - ## @param sentinel.externalMaster.host External master host to bootstrap from - ## @param sentinel.externalMaster.port Port for Redis service external master host - ## - externalMaster: - enabled: false - host: "" - port: 6379 - ## @param sentinel.containerPorts.sentinel Container port to open on Redis® Sentinel nodes - ## - containerPorts: - sentinel: 26379 - ## Configure extra options for Redis® containers' liveness and readiness probes - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes - ## @param sentinel.startupProbe.enabled Enable startupProbe on Redis® Sentinel nodes - ## @param sentinel.startupProbe.initialDelaySeconds Initial delay seconds for startupProbe - ## @param sentinel.startupProbe.periodSeconds Period seconds for startupProbe - ## @param sentinel.startupProbe.timeoutSeconds Timeout seconds for startupProbe - ## @param sentinel.startupProbe.failureThreshold Failure threshold for startupProbe - ## @param sentinel.startupProbe.successThreshold Success threshold for startupProbe - ## - startupProbe: - enabled: true - initialDelaySeconds: 10 - periodSeconds: 10 - timeoutSeconds: 5 - successThreshold: 1 - failureThreshold: 22 - ## @param sentinel.livenessProbe.enabled Enable livenessProbe on Redis® Sentinel nodes - ## @param sentinel.livenessProbe.initialDelaySeconds Initial delay seconds for livenessProbe - ## @param sentinel.livenessProbe.periodSeconds Period seconds for livenessProbe - ## @param sentinel.livenessProbe.timeoutSeconds Timeout seconds for livenessProbe - ## @param sentinel.livenessProbe.failureThreshold Failure threshold for livenessProbe - ## @param sentinel.livenessProbe.successThreshold Success threshold for livenessProbe - ## - livenessProbe: - enabled: true - initialDelaySeconds: 20 - periodSeconds: 10 - timeoutSeconds: 5 - successThreshold: 1 - failureThreshold: 6 - ## @param sentinel.readinessProbe.enabled Enable readinessProbe on Redis® Sentinel nodes - ## @param sentinel.readinessProbe.initialDelaySeconds Initial delay seconds for readinessProbe - ## @param sentinel.readinessProbe.periodSeconds Period seconds for readinessProbe - ## @param sentinel.readinessProbe.timeoutSeconds Timeout seconds for readinessProbe - ## @param sentinel.readinessProbe.failureThreshold Failure threshold for readinessProbe - ## @param sentinel.readinessProbe.successThreshold Success threshold for readinessProbe - ## - readinessProbe: - enabled: true - initialDelaySeconds: 20 - periodSeconds: 5 - timeoutSeconds: 1 - successThreshold: 1 - failureThreshold: 6 - ## @param sentinel.customStartupProbe Custom startupProbe that overrides the default one - ## - customStartupProbe: {} - ## @param sentinel.customLivenessProbe Custom livenessProbe that overrides the default one - ## - customLivenessProbe: {} - ## @param sentinel.customReadinessProbe Custom readinessProbe that overrides the default one - ## - customReadinessProbe: {} - ## Persistence parameters - ## ref: https://kubernetes.io/docs/concepts/storage/persistent-volumes/ - ## - persistence: - ## @param sentinel.persistence.enabled Enable persistence on Redis® sentinel nodes using Persistent Volume Claims (Experimental) - ## - enabled: false - ## @param sentinel.persistence.storageClass Persistent Volume storage class - ## If defined, storageClassName: - ## If set to "-", storageClassName: "", which disables dynamic provisioning - ## If undefined (the default) or set to null, no storageClassName spec is set, choosing the default provisioner - ## - storageClass: "" - ## @param sentinel.persistence.accessModes Persistent Volume access modes - ## - accessModes: - - ReadWriteOnce - ## @param sentinel.persistence.size Persistent Volume size - ## - size: 100Mi - ## @param sentinel.persistence.annotations Additional custom annotations for the PVC - ## - annotations: {} - ## @param sentinel.persistence.labels Additional custom labels for the PVC - ## - labels: {} - ## @param sentinel.persistence.selector Additional labels to match for the PVC - ## e.g: - ## selector: - ## matchLabels: - ## app: my-app - ## - selector: {} - ## @param sentinel.persistence.dataSource Custom PVC data source - ## - dataSource: {} - ## @param sentinel.persistence.medium Provide a medium for `emptyDir` volumes. - ## - medium: "" - ## @param sentinel.persistence.sizeLimit Set this to enable a size limit for `emptyDir` volumes. - ## - sizeLimit: "" - ## persistentVolumeClaimRetentionPolicy - ## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#persistentvolumeclaim-retention - ## @param sentinel.persistentVolumeClaimRetentionPolicy.enabled Controls if and how PVCs are deleted during the lifecycle of a StatefulSet - ## @param sentinel.persistentVolumeClaimRetentionPolicy.whenScaled Volume retention behavior when the replica count of the StatefulSet is reduced - ## @param sentinel.persistentVolumeClaimRetentionPolicy.whenDeleted Volume retention behavior that applies when the StatefulSet is deleted - ## - persistentVolumeClaimRetentionPolicy: - enabled: false - whenScaled: Retain - whenDeleted: Retain - ## Redis® Sentinel resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## @param sentinel.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if sentinel.resources is set (sentinel.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "nano" - ## @param sentinel.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} - ## Configure Container Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-pod - ## @param sentinel.containerSecurityContext.enabled Enabled Redis® Sentinel containers' Security Context - ## @param sentinel.containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container - ## @param sentinel.containerSecurityContext.runAsUser Set Redis® Sentinel containers' Security Context runAsUser - ## @param sentinel.containerSecurityContext.runAsGroup Set Redis® Sentinel containers' Security Context runAsGroup - ## @param sentinel.containerSecurityContext.runAsNonRoot Set Redis® Sentinel containers' Security Context runAsNonRoot - ## @param sentinel.containerSecurityContext.readOnlyRootFilesystem Set container's Security Context read-only root filesystem - ## @param sentinel.containerSecurityContext.allowPrivilegeEscalation Set Redis® Sentinel containers' Security Context allowPrivilegeEscalation - ## @param sentinel.containerSecurityContext.seccompProfile.type Set Redis® Sentinel containers' Security Context seccompProfile - ## @param sentinel.containerSecurityContext.capabilities.drop Set Redis® Sentinel containers' Security Context capabilities to drop - ## - containerSecurityContext: - enabled: true - seLinuxOptions: {} - runAsUser: 1001 - runAsGroup: 1001 - runAsNonRoot: true - allowPrivilegeEscalation: false - readOnlyRootFilesystem: true - seccompProfile: - type: RuntimeDefault - capabilities: - drop: ["ALL"] - ## @param sentinel.lifecycleHooks for the Redis® sentinel container(s) to automate configuration before or after startup - ## - lifecycleHooks: {} - ## @param sentinel.extraVolumes Optionally specify extra list of additional volumes for the Redis® Sentinel - ## - extraVolumes: [] - ## @param sentinel.extraVolumeMounts Optionally specify extra list of additional volumeMounts for the Redis® Sentinel container(s) - ## - extraVolumeMounts: [] - ## Redis® Sentinel service parameters - ## Note: values passed in this section also configure the master service, unless the sentinel.masterService is explicitly overridden. - service: - ## @param sentinel.service.type Redis® Sentinel service type - ## - type: ClusterIP - ## @param sentinel.service.ports.redis Redis® service port for Redis® - ## @param sentinel.service.ports.sentinel Redis® service port for Redis® Sentinel - ## - ports: - redis: 6379 - sentinel: 26379 - ## @param sentinel.service.nodePorts.redis Node port for Redis® - ## @param sentinel.service.nodePorts.sentinel Node port for Sentinel - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport - ## NOTE: choose port between <30000-32767> - ## NOTE: By leaving these values blank, they will be generated by ports-configmap - ## If setting manually, please leave at least replica.replicaCount + 1 in between sentinel.service.nodePorts.redis and sentinel.service.nodePorts.sentinel to take into account the ports that will be created while incrementing that base port - ## - nodePorts: - redis: "" - sentinel: "" - ## @param sentinel.service.externalTrafficPolicy Redis® Sentinel service external traffic policy - ## ref: https://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer/#preserving-the-client-source-ip - ## - externalTrafficPolicy: Cluster - ## @param sentinel.service.extraPorts Extra ports to expose (normally used with the `sidecar` value) - ## - extraPorts: [] - ## @param sentinel.service.clusterIP Redis® Sentinel service Cluster IP - ## - clusterIP: "" - ## @param sentinel.service.createMaster Enable master service pointing to the current master (experimental) - ## NOTE: rbac.create need to be set to true - ## - createMaster: false - - ## @param sentinel.service.loadBalancerIP Redis® Sentinel service Load Balancer IP - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#internal-load-balancer - ## - loadBalancerIP: "" - ## @param sentinel.service.loadBalancerClass sentinel service Load Balancer class if service type is `LoadBalancer` (optional, cloud specific) - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-loadbalancer - ## - loadBalancerClass: "" - ## @param sentinel.service.loadBalancerSourceRanges Redis® Sentinel service Load Balancer sources - ## https://kubernetes.io/docs/tasks/access-application-cluster/configure-cloud-provider-firewall/#restrict-access-for-loadbalancer-service - ## e.g. - ## loadBalancerSourceRanges: - ## - 10.10.10.0/24 - ## - loadBalancerSourceRanges: [] - ## @param sentinel.service.annotations Additional custom annotations for Redis® Sentinel service - ## - annotations: {} - ## @param sentinel.service.sessionAffinity Session Affinity for Kubernetes service, can be "None" or "ClientIP" - ## If "ClientIP", consecutive client requests will be directed to the same Pod - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies - ## - sessionAffinity: None - ## @param sentinel.service.sessionAffinityConfig Additional settings for the sessionAffinity - ## sessionAffinityConfig: - ## clientIP: - ## timeoutSeconds: 300 - ## - sessionAffinityConfig: {} - ## Headless service properties - ## - headless: - ## @param sentinel.service.headless.annotations Annotations for the headless service. - ## - annotations: {} - - ## Redis® master service parameters - ## - masterService: - ## @param sentinel.masterService.enabled Enable master service pointing to the current master (experimental) - ## NOTE: rbac.create need to be set to true - ## - enabled: false - ## @param sentinel.masterService.type Redis® Sentinel master service type - ## - type: ClusterIP - ## @param sentinel.masterService.ports.redis Redis® service port for Redis® - ## - ports: - redis: 6379 - ## @param sentinel.masterService.nodePorts.redis Node port for Redis® - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport - ## NOTE: choose port between <30000-32767> - ## NOTE: By leaving these values blank, they will be generated by ports-configmap - ## If setting manually, please leave at least replica.replicaCount + 1 in between sentinel.service.nodePorts.redis and sentinel.service.nodePorts.sentinel to take into account the ports that will be created while incrementing that base port - ## - nodePorts: - redis: "" - ## @param sentinel.masterService.externalTrafficPolicy Redis® master service external traffic policy - ## ref: https://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer/#preserving-the-client-source-ip - ## - externalTrafficPolicy: "" - ## @param sentinel.masterService.extraPorts Extra ports to expose (normally used with the `sidecar` value) - ## - extraPorts: [] - ## @param sentinel.masterService.clusterIP Redis® master service Cluster IP - ## - clusterIP: "" - ## @param sentinel.masterService.loadBalancerIP Redis® master service Load Balancer IP - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#internal-load-balancer - ## - loadBalancerIP: "" - ## @param sentinel.masterService.loadBalancerClass master service Load Balancer class if service type is `LoadBalancer` (optional, cloud specific) - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-loadbalancer - ## - loadBalancerClass: "" - ## @param sentinel.masterService.loadBalancerSourceRanges Redis® master service Load Balancer sources - ## https://kubernetes.io/docs/tasks/access-application-cluster/configure-cloud-provider-firewall/#restrict-access-for-loadbalancer-service - ## e.g. - ## loadBalancerSourceRanges: - ## - 10.10.10.0/24 - ## - loadBalancerSourceRanges: [] - ## @param sentinel.masterService.annotations Additional custom annotations for Redis® master service - ## - annotations: {} - ## @param sentinel.masterService.sessionAffinity Session Affinity for Kubernetes service, can be "None" or "ClientIP" - ## If "ClientIP", consecutive client requests will be directed to the same Pod - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies - ## - sessionAffinity: None - ## @param sentinel.masterService.sessionAffinityConfig Additional settings for the sessionAffinity - ## sessionAffinityConfig: - ## clientIP: - ## timeoutSeconds: 300 - ## - sessionAffinityConfig: {} - ## @param sentinel.terminationGracePeriodSeconds Integer setting the termination grace period for the redis-node pods - ## - terminationGracePeriodSeconds: 30 -## @section Other Parameters -## - -## @param serviceBindings.enabled Create secret for service binding (Experimental) -## Ref: https://servicebinding.io/service-provider/ -## -serviceBindings: - enabled: false -## Network Policy configuration -## ref: https://kubernetes.io/docs/concepts/services-networking/network-policies/ -## -networkPolicy: - ## @param networkPolicy.enabled Enable creation of NetworkPolicy resources - ## - enabled: true - ## @param networkPolicy.allowExternal Don't require client label for connections - ## When set to false, only pods with the correct client label will have network access to the ports - ## Redis® is listening on. When true, Redis® will accept connections from any source - ## (with the correct destination port). - ## - allowExternal: true - ## @param networkPolicy.allowExternalEgress Allow the pod to access any range of port and all destinations. - ## - allowExternalEgress: true - ## @param networkPolicy.extraIngress Add extra ingress rules to the NetworkPolicy - ## e.g: - ## extraIngress: - ## - ports: - ## - port: 1234 - ## from: - ## - podSelector: - ## - matchLabels: - ## - role: frontend - ## - podSelector: - ## - matchExpressions: - ## - key: role - ## operator: In - ## values: - ## - frontend - ## - extraIngress: [] - ## @param networkPolicy.extraEgress Add extra egress rules to the NetworkPolicy - ## e.g: - ## extraEgress: - ## - ports: - ## - port: 1234 - ## to: - ## - podSelector: - ## - matchLabels: - ## - role: frontend - ## - podSelector: - ## - matchExpressions: - ## - key: role - ## operator: In - ## values: - ## - frontend - ## - extraEgress: [] - ## @param networkPolicy.ingressNSMatchLabels Labels to match to allow traffic from other namespaces - ## @param networkPolicy.ingressNSPodMatchLabels Pod labels to match to allow traffic from other namespaces - ## - ingressNSMatchLabels: {} - ingressNSPodMatchLabels: {} - metrics: - ## @param networkPolicy.metrics.allowExternal Don't require client label for connections for metrics endpoint - ## When set to false, only pods with the correct client label will have network access to the metrics port - ## - allowExternal: true - ## @param networkPolicy.metrics.ingressNSMatchLabels Labels to match to allow traffic from other namespaces to metrics endpoint - ## @param networkPolicy.metrics.ingressNSPodMatchLabels Pod labels to match to allow traffic from other namespaces to metrics endpoint - ## - ingressNSMatchLabels: {} - ingressNSPodMatchLabels: {} -## PodSecurityPolicy configuration -## ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/ -## -podSecurityPolicy: - ## @param podSecurityPolicy.create Whether to create a PodSecurityPolicy. WARNING: PodSecurityPolicy is deprecated in Kubernetes v1.21 or later, unavailable in v1.25 or later - ## - create: false - ## @param podSecurityPolicy.enabled Enable PodSecurityPolicy's RBAC rules - ## - enabled: false -## RBAC configuration -## -rbac: - ## @param rbac.create Specifies whether RBAC resources should be created - ## - create: false - ## @param rbac.rules Custom RBAC rules to set - ## e.g: - ## rules: - ## - apiGroups: - ## - "" - ## resources: - ## - pods - ## verbs: - ## - get - ## - list - ## - rules: [] -## ServiceAccount configuration -## -serviceAccount: - ## @param serviceAccount.create Specifies whether a ServiceAccount should be created - ## - create: true - ## @param serviceAccount.name The name of the ServiceAccount to use. - ## If not set and create is true, a name is generated using the common.names.fullname template - ## - name: "" - ## @param serviceAccount.automountServiceAccountToken Whether to auto mount the service account token - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/#use-the-default-service-account-to-access-the-api-server - ## - automountServiceAccountToken: false - ## @param serviceAccount.annotations Additional custom annotations for the ServiceAccount - ## - annotations: {} -## Redis® Pod Disruption Budget configuration -## ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb/ -## @param pdb DEPRECATED Please use `master.pdb` and `replica.pdb` values instead -## -pdb: {} -## TLS configuration -## -tls: - ## @param tls.enabled Enable TLS traffic - ## - enabled: false - ## @param tls.authClients Require clients to authenticate - ## - authClients: true - ## @param tls.autoGenerated Enable autogenerated certificates - ## - autoGenerated: false - ## @param tls.existingSecret The name of the existing secret that contains the TLS certificates - ## - existingSecret: "" - ## @param tls.certificatesSecret DEPRECATED. Use existingSecret instead. - ## - certificatesSecret: "" - ## @param tls.certFilename Certificate filename - ## - certFilename: "" - ## @param tls.certKeyFilename Certificate Key filename - ## - certKeyFilename: "" - ## @param tls.certCAFilename CA Certificate filename - ## - certCAFilename: "" - ## @param tls.dhParamsFilename File containing DH params (in order to support DH based ciphers) - ## - dhParamsFilename: "" -## @section Metrics Parameters -## -metrics: - ## @param metrics.enabled Start a sidecar prometheus exporter to expose Redis® metrics - ## - enabled: false - ## Bitnami Redis® Exporter image - ## ref: https://hub.docker.com/r/bitnami/redis-exporter/tags/ - ## @param metrics.image.registry [default: REGISTRY_NAME] Redis® Exporter image registry - ## @param metrics.image.repository [default: REPOSITORY_NAME/redis-exporter] Redis® Exporter image repository - ## @skip metrics.image.tag Redis® Exporter image tag (immutable tags are recommended) - ## @param metrics.image.digest Redis® Exporter image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag - ## @param metrics.image.pullPolicy Redis® Exporter image pull policy - ## @param metrics.image.pullSecrets Redis® Exporter image pull secrets - ## - image: - registry: docker.io - repository: bitnami/redis-exporter - tag: 1.62.0-debian-12-r3 - digest: "" - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets. - ## Secrets must be manually created in the namespace. - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## e.g: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## @param metrics.containerPorts.http Metrics HTTP container port - ## - containerPorts: - http: 9121 - ## Configure extra options for Redis® containers' liveness, readiness & startup probes - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/ - ## @param metrics.startupProbe.enabled Enable startupProbe on Redis® replicas nodes - ## @param metrics.startupProbe.initialDelaySeconds Initial delay seconds for startupProbe - ## @param metrics.startupProbe.periodSeconds Period seconds for startupProbe - ## @param metrics.startupProbe.timeoutSeconds Timeout seconds for startupProbe - ## @param metrics.startupProbe.failureThreshold Failure threshold for startupProbe - ## @param metrics.startupProbe.successThreshold Success threshold for startupProbe - ## - startupProbe: - enabled: false - initialDelaySeconds: 10 - periodSeconds: 10 - timeoutSeconds: 5 - successThreshold: 1 - failureThreshold: 5 - ## @param metrics.livenessProbe.enabled Enable livenessProbe on Redis® replicas nodes - ## @param metrics.livenessProbe.initialDelaySeconds Initial delay seconds for livenessProbe - ## @param metrics.livenessProbe.periodSeconds Period seconds for livenessProbe - ## @param metrics.livenessProbe.timeoutSeconds Timeout seconds for livenessProbe - ## @param metrics.livenessProbe.failureThreshold Failure threshold for livenessProbe - ## @param metrics.livenessProbe.successThreshold Success threshold for livenessProbe - ## - livenessProbe: - enabled: true - initialDelaySeconds: 10 - periodSeconds: 10 - timeoutSeconds: 5 - successThreshold: 1 - failureThreshold: 5 - ## @param metrics.readinessProbe.enabled Enable readinessProbe on Redis® replicas nodes - ## @param metrics.readinessProbe.initialDelaySeconds Initial delay seconds for readinessProbe - ## @param metrics.readinessProbe.periodSeconds Period seconds for readinessProbe - ## @param metrics.readinessProbe.timeoutSeconds Timeout seconds for readinessProbe - ## @param metrics.readinessProbe.failureThreshold Failure threshold for readinessProbe - ## @param metrics.readinessProbe.successThreshold Success threshold for readinessProbe - ## - readinessProbe: - enabled: true - initialDelaySeconds: 5 - periodSeconds: 10 - timeoutSeconds: 1 - successThreshold: 1 - failureThreshold: 3 - ## @param metrics.customStartupProbe Custom startupProbe that overrides the default one - ## - customStartupProbe: {} - ## @param metrics.customLivenessProbe Custom livenessProbe that overrides the default one - ## - customLivenessProbe: {} - ## @param metrics.customReadinessProbe Custom readinessProbe that overrides the default one - ## - customReadinessProbe: {} - ## @param metrics.command Override default metrics container init command (useful when using custom images) - ## - command: [] - ## @param metrics.redisTargetHost A way to specify an alternative Redis® hostname - ## Useful for certificate CN/SAN matching - ## - redisTargetHost: "localhost" - ## @param metrics.extraArgs Extra arguments for Redis® exporter, for example: - ## e.g.: - ## extraArgs: - ## check-keys: myKey,myOtherKey - ## - extraArgs: {} - ## @param metrics.extraEnvVars Array with extra environment variables to add to Redis® exporter - ## e.g: - ## extraEnvVars: - ## - name: FOO - ## value: "bar" - ## - extraEnvVars: [] - ## Configure Container Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-pod - ## @param metrics.containerSecurityContext.enabled Enabled Redis® exporter containers' Security Context - ## @param metrics.containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container - ## @param metrics.containerSecurityContext.runAsUser Set Redis® exporter containers' Security Context runAsUser - ## @param metrics.containerSecurityContext.runAsGroup Set Redis® exporter containers' Security Context runAsGroup - ## @param metrics.containerSecurityContext.runAsNonRoot Set Redis® exporter containers' Security Context runAsNonRoot - ## @param metrics.containerSecurityContext.allowPrivilegeEscalation Set Redis® exporter containers' Security Context allowPrivilegeEscalation - ## @param metrics.containerSecurityContext.readOnlyRootFilesystem Set container's Security Context read-only root filesystem - ## @param metrics.containerSecurityContext.seccompProfile.type Set Redis® exporter containers' Security Context seccompProfile - ## @param metrics.containerSecurityContext.capabilities.drop Set Redis® exporter containers' Security Context capabilities to drop - ## - containerSecurityContext: - enabled: true - seLinuxOptions: {} - runAsUser: 1001 - runAsGroup: 1001 - runAsNonRoot: true - allowPrivilegeEscalation: false - readOnlyRootFilesystem: true - seccompProfile: - type: RuntimeDefault - capabilities: - drop: ["ALL"] - ## @param metrics.extraVolumes Optionally specify extra list of additional volumes for the Redis® metrics sidecar - ## - extraVolumes: [] - ## @param metrics.extraVolumeMounts Optionally specify extra list of additional volumeMounts for the Redis® metrics sidecar - ## - extraVolumeMounts: [] - ## Redis® exporter resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## @param metrics.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if metrics.resources is set (metrics.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "nano" - ## @param metrics.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} - ## @param metrics.podLabels Extra labels for Redis® exporter pods - ## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/ - ## - podLabels: {} - ## @param metrics.podAnnotations [object] Annotations for Redis® exporter pods - ## ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ - ## - podAnnotations: - prometheus.io/scrape: "true" - prometheus.io/port: "9121" - ## Redis® exporter service parameters - ## - service: - ## @param metrics.service.enabled Create Service resource(s) for scraping metrics using PrometheusOperator ServiceMonitor, can be disabled when using a PodMonitor - ## - enabled: true - ## @param metrics.service.type Redis® exporter service type - ## - type: ClusterIP - ## @param metrics.service.ports.http Redis® exporter service port - ## - ports: - http: 9121 - ## @param metrics.service.externalTrafficPolicy Redis® exporter service external traffic policy - ## ref: https://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer/#preserving-the-client-source-ip - ## - externalTrafficPolicy: Cluster - ## @param metrics.service.extraPorts Extra ports to expose (normally used with the `sidecar` value) - ## - extraPorts: [] - ## @param metrics.service.loadBalancerIP Redis® exporter service Load Balancer IP - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#internal-load-balancer - ## - loadBalancerIP: "" - ## @param metrics.service.loadBalancerClass exporter service Load Balancer class if service type is `LoadBalancer` (optional, cloud specific) - ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-loadbalancer - ## - loadBalancerClass: "" - ## @param metrics.service.loadBalancerSourceRanges Redis® exporter service Load Balancer sources - ## https://kubernetes.io/docs/tasks/access-application-cluster/configure-cloud-provider-firewall/#restrict-access-for-loadbalancer-service - ## e.g. - ## loadBalancerSourceRanges: - ## - 10.10.10.0/24 - ## - loadBalancerSourceRanges: [] - ## @param metrics.service.annotations Additional custom annotations for Redis® exporter service - ## - annotations: {} - ## @param metrics.service.clusterIP Redis® exporter service Cluster IP - ## - clusterIP: "" - ## Prometheus Service Monitor - ## ref: https://github.com/coreos/prometheus-operator - ## https://github.com/coreos/prometheus-operator/blob/master/Documentation/api.md#endpoint - ## - serviceMonitor: - ## @param metrics.serviceMonitor.port the service port to scrape metrics from - ## - port: http-metrics - ## @param metrics.serviceMonitor.enabled Create ServiceMonitor resource(s) for scraping metrics using PrometheusOperator - ## - enabled: false - ## @param metrics.serviceMonitor.namespace The namespace in which the ServiceMonitor will be created - ## - namespace: "" - ## @param metrics.serviceMonitor.interval The interval at which metrics should be scraped - ## - interval: 30s - ## @param metrics.serviceMonitor.scrapeTimeout The timeout after which the scrape is ended - ## - scrapeTimeout: "" - ## @param metrics.serviceMonitor.relabelings Metrics RelabelConfigs to apply to samples before scraping. - ## - relabelings: [] - ## @skip metrics.serviceMonitor.relabellings DEPRECATED: Use `metrics.serviceMonitor.relabelings` instead. - ## - relabellings: [] - ## @param metrics.serviceMonitor.metricRelabelings Metrics RelabelConfigs to apply to samples before ingestion. - ## - metricRelabelings: [] - ## @param metrics.serviceMonitor.honorLabels Specify honorLabels parameter to add the scrape endpoint - ## - honorLabels: false - ## @param metrics.serviceMonitor.additionalLabels Additional labels that can be used so ServiceMonitor resource(s) can be discovered by Prometheus - ## - additionalLabels: {} - ## @param metrics.serviceMonitor.podTargetLabels Labels from the Kubernetes pod to be transferred to the created metrics - ## - podTargetLabels: [] - ## @param metrics.serviceMonitor.sampleLimit Limit of how many samples should be scraped from every Pod - ## - sampleLimit: false - ## @param metrics.serviceMonitor.targetLimit Limit of how many targets should be scraped - ## - targetLimit: false - ## @param metrics.serviceMonitor.additionalEndpoints Additional endpoints to scrape (e.g sentinel) - ## - additionalEndpoints: [] - # uncomment in order to scrape sentinel metrics, also to in order distinguish between Sentinel and Redis container metrics - # add metricRelabelings with label like app=redis to main redis pod-monitor port - # - interval: "30s" - # path: "/scrape" - # port: "metrics" - # params: - # target: ["localhost:26379"] - # metricRelabelings: - # - targetLabel: "app" - # replacement: "sentinel" - ## Prometheus Pod Monitor - ## ref: https://github.com/coreos/prometheus-operator - ## https://github.com/coreos/prometheus-operator/blob/master/Documentation/api.md#podmonitor - ## - podMonitor: - ## @param metrics.podMonitor.port the pod port to scrape metrics from - ## - port: metrics - ## @param metrics.podMonitor.enabled Create PodMonitor resource(s) for scraping metrics using PrometheusOperator - ## - enabled: false - ## @param metrics.podMonitor.namespace The namespace in which the PodMonitor will be created - ## - namespace: "" - ## @param metrics.podMonitor.interval The interval at which metrics should be scraped - ## - interval: 30s - ## @param metrics.podMonitor.scrapeTimeout The timeout after which the scrape is ended - ## - scrapeTimeout: "" - ## @param metrics.podMonitor.relabelings Metrics RelabelConfigs to apply to samples before scraping. - ## - relabelings: [] - ## @skip metrics.podMonitor.relabellings DEPRECATED: Use `metrics.podMonitor.relabelings` instead. - ## - relabellings: [] - ## @param metrics.podMonitor.metricRelabelings Metrics RelabelConfigs to apply to samples before ingestion. - ## - metricRelabelings: [] - # - targetLabel: "app" - # replacement: "redis" - ## @param metrics.podMonitor.honorLabels Specify honorLabels parameter to add the scrape endpoint - ## - honorLabels: false - ## @param metrics.podMonitor.additionalLabels Additional labels that can be used so PodMonitor resource(s) can be discovered by Prometheus - ## - additionalLabels: {} - ## @param metrics.podMonitor.podTargetLabels Labels from the Kubernetes pod to be transferred to the created metrics - ## - podTargetLabels: [] - ## @param metrics.podMonitor.sampleLimit Limit of how many samples should be scraped from every Pod - ## - sampleLimit: false - ## @param metrics.podMonitor.targetLimit Limit of how many targets should be scraped - ## - targetLimit: false - ## @param metrics.podMonitor.additionalEndpoints Additional endpoints to scrape (e.g sentinel) - ## - additionalEndpoints: [] - # - interval: "30s" - # path: "/scrape" - # port: "metrics" - # params: - # target: ["localhost:26379"] - # metricRelabelings: - # - targetLabel: "app" - # replacement: "sentinel" - ## Custom PrometheusRule to be defined - ## ref: https://github.com/coreos/prometheus-operator#customresourcedefinitions - ## - prometheusRule: - ## @param metrics.prometheusRule.enabled Create a custom prometheusRule Resource for scraping metrics using PrometheusOperator - ## - enabled: false - ## @param metrics.prometheusRule.namespace The namespace in which the prometheusRule will be created - ## - namespace: "" - ## @param metrics.prometheusRule.additionalLabels Additional labels for the prometheusRule - ## - additionalLabels: {} - ## @param metrics.prometheusRule.rules Custom Prometheus rules - ## e.g: - ## rules: - ## - alert: RedisDown - ## expr: redis_up{service="{{ template "common.names.fullname" . }}-metrics"} == 0 - ## for: 2m - ## labels: - ## severity: error - ## annotations: - ## summary: Redis® instance {{ "{{ $labels.instance }}" }} down - ## description: Redis® instance {{ "{{ $labels.instance }}" }} is down - ## - alert: RedisMemoryHigh - ## expr: > - ## redis_memory_used_bytes{service="{{ template "common.names.fullname" . }}-metrics"} * 100 - ## / - ## redis_memory_max_bytes{service="{{ template "common.names.fullname" . }}-metrics"} - ## > 90 - ## for: 2m - ## labels: - ## severity: error - ## annotations: - ## summary: Redis® instance {{ "{{ $labels.instance }}" }} is using too much memory - ## description: | - ## Redis® instance {{ "{{ $labels.instance }}" }} is using {{ "{{ $value }}" }}% of its available memory. - ## - alert: RedisKeyEviction - ## expr: | - ## increase(redis_evicted_keys_total{service="{{ template "common.names.fullname" . }}-metrics"}[5m]) > 0 - ## for: 1s - ## labels: - ## severity: error - ## annotations: - ## summary: Redis® instance {{ "{{ $labels.instance }}" }} has evicted keys - ## description: | - ## Redis® instance {{ "{{ $labels.instance }}" }} has evicted {{ "{{ $value }}" }} keys in the last 5 minutes. - ## - rules: [] -## @section Init Container Parameters -## - -## 'volumePermissions' init container parameters -## Changes the owner and group of the persistent volume mount point to runAsUser:fsGroup values -## based on the *podSecurityContext/*containerSecurityContext parameters -## -volumePermissions: - ## @param volumePermissions.enabled Enable init container that changes the owner/group of the PV mount point to `runAsUser:fsGroup` - ## - enabled: false - ## OS Shell + Utility image - ## ref: https://hub.docker.com/r/bitnami/os-shell/tags/ - ## @param volumePermissions.image.registry [default: REGISTRY_NAME] OS Shell + Utility image registry - ## @param volumePermissions.image.repository [default: REPOSITORY_NAME/os-shell] OS Shell + Utility image repository - ## @skip volumePermissions.image.tag OS Shell + Utility image tag (immutable tags are recommended) - ## @param volumePermissions.image.digest OS Shell + Utility image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag - ## @param volumePermissions.image.pullPolicy OS Shell + Utility image pull policy - ## @param volumePermissions.image.pullSecrets OS Shell + Utility image pull secrets - ## - image: - registry: docker.io - repository: bitnami/os-shell - tag: 12-debian-12-r27 - digest: "" - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets. - ## Secrets must be manually created in the namespace. - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## e.g: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## Init container's resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## @param volumePermissions.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if volumePermissions.resources is set (volumePermissions.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "nano" - ## @param volumePermissions.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} - ## Init container Container Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container - ## @param volumePermissions.containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container - ## @param volumePermissions.containerSecurityContext.runAsUser Set init container's Security Context runAsUser - ## NOTE: when runAsUser is set to special value "auto", init container will try to chown the - ## data folder to auto-determined user&group, using commands: `id -u`:`id -G | cut -d" " -f2` - ## "auto" is especially useful for OpenShift which has scc with dynamic user ids (and 0 is not allowed) - ## - containerSecurityContext: - seLinuxOptions: {} - runAsUser: 0 - -## Kubectl InitContainer -## used by Sentinel to update the isMaster label on the Redis(TM) pods -## -kubectl: - ## Bitnami Kubectl image version - ## ref: https://hub.docker.com/r/bitnami/kubectl/tags/ - ## @param kubectl.image.registry [default: REGISTRY_NAME] Kubectl image registry - ## @param kubectl.image.repository [default: REPOSITORY_NAME/kubectl] Kubectl image repository - ## @skip kubectl.image.tag Kubectl image tag (immutable tags are recommended), by default, using the current version - ## @param kubectl.image.digest Kubectl image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag - ## @param kubectl.image.pullPolicy Kubectl image pull policy - ## @param kubectl.image.pullSecrets Kubectl pull secrets - ## - image: - registry: docker.io - repository: bitnami/kubectl - tag: 1.30.3-debian-12-r5 - digest: "" - ## Specify a imagePullPolicy - ## Defaults to 'Always' if image tag is 'latest', else set to 'IfNotPresent' - ## ref: https://kubernetes.io/docs/concepts/containers/images/#pre-pulled-images - ## - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets. - ## Secrets must be manually created in the namespace. - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## e.g: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## @param kubectl.command kubectl command to execute - ## - command: ["/opt/bitnami/scripts/kubectl-scripts/update-master-label.sh"] - ## Configure Container Security Context - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-pod - ## @param kubectl.containerSecurityContext.enabled Enabled kubectl containers' Security Context - ## @param kubectl.containerSecurityContext.seLinuxOptions [object,nullable] Set SELinux options in container - ## @param kubectl.containerSecurityContext.runAsUser Set kubectl containers' Security Context runAsUser - ## @param kubectl.containerSecurityContext.runAsGroup Set kubectl containers' Security Context runAsGroup - ## @param kubectl.containerSecurityContext.runAsNonRoot Set kubectl containers' Security Context runAsNonRoot - ## @param kubectl.containerSecurityContext.allowPrivilegeEscalation Set kubectl containers' Security Context allowPrivilegeEscalation - ## @param kubectl.containerSecurityContext.readOnlyRootFilesystem Set container's Security Context read-only root filesystem - ## @param kubectl.containerSecurityContext.seccompProfile.type Set kubectl containers' Security Context seccompProfile - ## @param kubectl.containerSecurityContext.capabilities.drop Set kubectl containers' Security Context capabilities to drop - ## - containerSecurityContext: - enabled: true - seLinuxOptions: {} - runAsUser: 1001 - runAsGroup: 1001 - runAsNonRoot: true - allowPrivilegeEscalation: false - readOnlyRootFilesystem: true - seccompProfile: - type: RuntimeDefault - capabilities: - drop: ["ALL"] - ## Bitnami Kubectl resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## @param kubectl.resources.limits The resources limits for the kubectl containers - ## @param kubectl.resources.requests The requested resources for the kubectl containers - ## - resources: - limits: {} - requests: {} - -## init-sysctl container parameters -## used to perform sysctl operation to modify Kernel settings (needed sometimes to avoid warnings) -## -sysctl: - ## @param sysctl.enabled Enable init container to modify Kernel settings - ## - enabled: false - ## OS Shell + Utility image - ## ref: https://hub.docker.com/r/bitnami/os-shell/tags/ - ## @param sysctl.image.registry [default: REGISTRY_NAME] OS Shell + Utility image registry - ## @param sysctl.image.repository [default: REPOSITORY_NAME/os-shell] OS Shell + Utility image repository - ## @skip sysctl.image.tag OS Shell + Utility image tag (immutable tags are recommended) - ## @param sysctl.image.digest OS Shell + Utility image digest in the way sha256:aa.... Please note this parameter, if set, will override the tag - ## @param sysctl.image.pullPolicy OS Shell + Utility image pull policy - ## @param sysctl.image.pullSecrets OS Shell + Utility image pull secrets - ## - image: - registry: docker.io - repository: bitnami/os-shell - tag: 12-debian-12-r27 - digest: "" - pullPolicy: IfNotPresent - ## Optionally specify an array of imagePullSecrets. - ## Secrets must be manually created in the namespace. - ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ - ## e.g: - ## pullSecrets: - ## - myRegistryKeySecretName - ## - pullSecrets: [] - ## @param sysctl.command Override default init-sysctl container command (useful when using custom images) - ## - command: [] - ## @param sysctl.mountHostSys Mount the host `/sys` folder to `/host-sys` - ## - mountHostSys: false - ## Init container's resource requests and limits - ## ref: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/ - ## @param sysctl.resourcesPreset Set container resources according to one common preset (allowed values: none, nano, micro, small, medium, large, xlarge, 2xlarge). This is ignored if sysctl.resources is set (sysctl.resources is recommended for production). - ## More information: https://github.com/bitnami/charts/blob/main/bitnami/common/templates/_resources.tpl#L15 - ## - resourcesPreset: "nano" - ## @param sysctl.resources Set container requests and limits for different resources like CPU or memory (essential for production workloads) - ## Example: - ## resources: - ## requests: - ## cpu: 2 - ## memory: 512Mi - ## limits: - ## cpu: 3 - ## memory: 1024Mi - ## - resources: {} -## @section useExternalDNS Parameters -## -## @param useExternalDNS.enabled Enable various syntax that would enable external-dns to work. Note this requires a working installation of `external-dns` to be usable. -## @param useExternalDNS.additionalAnnotations Extra annotations to be utilized when `external-dns` is enabled. -## @param useExternalDNS.annotationKey The annotation key utilized when `external-dns` is enabled. Setting this to `false` will disable annotations. -## @param useExternalDNS.suffix The DNS suffix utilized when `external-dns` is enabled. Note that we prepend the suffix with the full name of the release. -## -useExternalDNS: - enabled: false - suffix: "" - annotationKey: external-dns.alpha.kubernetes.io/ - additionalAnnotations: {} diff --git a/charts/redis/values-overrides.yaml b/charts/redis/values-overrides.yaml deleted file mode 100644 index 41f29e1..0000000 --- a/charts/redis/values-overrides.yaml +++ /dev/null @@ -1,8 +0,0 @@ -architecture: standalone -auth: - enabled: false -master: - replicaCount: 1 -fullnameOverride: redis -image: - debug: true \ No newline at end of file diff --git a/charts/traefik/traefik/.helmignore b/charts/traefik/traefik/.helmignore deleted file mode 100644 index 9c42ddd..0000000 --- a/charts/traefik/traefik/.helmignore +++ /dev/null @@ -1,2 +0,0 @@ -tests/ -crds/kustomization.yaml diff --git a/charts/traefik/traefik/Changelog.md b/charts/traefik/traefik/Changelog.md deleted file mode 100644 index be5ee51..0000000 --- a/charts/traefik/traefik/Changelog.md +++ /dev/null @@ -1,10505 +0,0 @@ -# Change Log - -## 33.0.0 ![AppVersion: v3.2.0](https://img.shields.io/static/v1?label=AppVersion&message=v3.2.0&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-10-30 - -* fix: 🐛 http3 with internal service -* fix: use correct children indentation for logs.access.filters -* fix(schema): :bug: targetPort can also be a string -* fix(certificateResolvers)!: :boom: :bug: use same syntax in Chart and in Traefik -* fix(Traefik)!: :boom: set 8080 as default port for `traefik` entrypoint -* fix(Traefik Hub): RBAC for distributedAcme -* fix(Kubernetes Ingress)!: :boom: :sparkles: enable publishedService by default -* fix(Gateway API): :bug: add missing required RBAC for v3.2 with experimental Channel -* fix(Env Variables)!: allow extending env without overwrite -* feat(deps): update traefik docker tag to v3.2.0 -* feat(deps): update traefik docker tag to v3.1.6 -* feat(Traefik): ✨ support Gateway API statusAddress -* feat(Traefik Proxy): CRDs for v3.2+ -* feat(Gateway API): :sparkles: standard install CRD v1.2.0 -* feat(Gateway API): :sparkles: add infrastructure in the values -* chore: allow TRACE log level -* chore(release): 🚀 publish v33.0.0 -* Update topology spread constraints comments - -**Upgrade Notes** - -There are multiple breaking changes in this release: - -1. The default port of `traefik` entrypoint has changed from `9000` to `8080`, just like the Traefik Proxy default port - * You _may_ have to update probes accordingly (or set this port back to 9000) -2. `publishedService` is enabled by default on Ingress provider - * You _can_ disable it, if needed -3. The `POD_NAME` and `POD_NAMESPACE` environment variables are now set by default, without values. - * It is no longer necessary to add them in values and so, it can be removed from user values. -4. In _values_, **certResolvers** specific syntax has been reworked to align with Traefik Proxy syntax. - * PR [#1214](https://github.com/traefik/traefik-helm-chart/pull/1214) contains a complete before / after example on how to update _values_ -5. Traefik Proxy 3.2 supports Gateway API v1.2 - * The CRDs of this version comes with Gateway API CRD v1.2 of standard channel. - * The CRDs needs to be updated, as documented in the README. - * It is recommended to check that other software using Gateway API on your cluster are compatible - -:information_source: A separate helm chart, just for CRDs, is being considered for a future release. See PR [#1123](https://github.com/traefik/traefik-helm-chart/pull/1223) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 73371f3..be89b00 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -95,7 +95,7 @@ deployment: - # postStart: - # httpGet: - # path: /ping -- # port: 9000 -+ # port: 8080 - # host: localhost - # scheme: HTTP - # -- Set a runtimeClassName on pod -@@ -138,6 +138,8 @@ gateway: - namespace: "" - # -- Additional gateway annotations (e.g. for cert-manager.io/issuer) - annotations: {} -+ # -- [Infrastructure](https://kubernetes.io/blog/2023/11/28/gateway-api-ga/#gateway-infrastructure-labels) -+ infrastructure: {} - # -- Define listeners - listeners: - web: -@@ -283,10 +285,11 @@ providers: # @schema additionalProperties: false - namespaces: [] - # IP used for Kubernetes Ingress endpoints - publishedService: -- enabled: false -- # Published Kubernetes Service to copy status from. Format: namespace/servicename -- # By default this Traefik service -- # pathOverride: "" -+ # -- Enable [publishedService](https://doc.traefik.io/traefik/providers/kubernetes-ingress/#publishedservice) -+ enabled: true -+ # -- Override path of Kubernetes Service used to copy status from. Format: namespace/servicename. -+ # Default to Service deployed with this Chart. -+ pathOverride: "" - # -- Defines whether to use Native Kubernetes load-balancing mode by default. - nativeLBByDefault: false - -@@ -300,6 +303,15 @@ providers: # @schema additionalProperties: false - namespaces: [] - # -- A label selector can be defined to filter on specific GatewayClass objects only. - labelselector: "" -+ statusAddress: -+ # -- This IP will get copied to the Gateway status.addresses, and currently only supports one IP value (IPv4 or IPv6). -+ ip: "" -+ # -- This Hostname will get copied to the Gateway status.addresses. -+ hostname: "" -+ # -- The Kubernetes service to copy status addresses from. When using third parties tools like External-DNS, this option can be used to copy the service loadbalancer.status (containing the service's endpoints IPs) to the gateways. Default to Service of this Chart. -+ service: -+ name: "{{ (include \"traefik.fullname\" .) }}" -+ namespace: "{{ .Release.Namespace }}" - - file: - # -- Create a file provider -@@ -335,8 +347,8 @@ logs: - # -- Set [logs format](https://doc.traefik.io/traefik/observability/logs/#format) - format: # @schema enum:["common", "json", null]; type:[string, null]; default: "common" - # By default, the level is set to INFO. -- # -- Alternative logging levels are DEBUG, PANIC, FATAL, ERROR, WARN, and INFO. -- level: "INFO" # @schema enum:[INFO,WARN,ERROR,FATAL,PANIC,DEBUG]; default: "INFO" -+ # -- Alternative logging levels are TRACE, DEBUG, INFO, WARN, ERROR, FATAL, and PANIC. -+ level: "INFO" # @schema enum:[TRACE,DEBUG,INFO,WARN,ERROR,FATAL,PANIC]; default: "INFO" - # -- To write the logs into a log file, use the filePath option. - filePath: "" - # -- When set to true and format is common, it disables the colorized output. -@@ -350,10 +362,13 @@ logs: - # -- Set [bufferingSize](https://doc.traefik.io/traefik/observability/access-logs/#bufferingsize) - bufferingSize: # @schema type:[integer, null] - # -- Set [filtering](https://docs.traefik.io/observability/access-logs/#filtering) -- filters: {} -- statuscodes: "" -- retryattempts: false -- minduration: "" -+ filters: # @schema additionalProperties: false -+ # -- Set statusCodes, to limit the access logs to requests with a status codes in the specified range -+ statuscodes: "" -+ # -- Set retryAttempts, to keep the access logs when at least one retry has happened -+ retryattempts: false -+ # -- Set minDuration, to keep access logs when requests take longer than the specified duration -+ minduration: "" - # -- Enables accessLogs for internal resources. Default: false. - addInternals: false - fields: -@@ -566,24 +581,16 @@ additionalArguments: [] - # - "--providers.kubernetesingress.ingressclass=traefik-internal" - # - "--log.level=DEBUG" - --# -- Environment variables to be passed to Traefik's binary -+# -- Additional Environment variables to be passed to Traefik's binary - # @default -- See _values.yaml_ --env: --- name: POD_NAME -- valueFrom: -- fieldRef: -- fieldPath: metadata.name --- name: POD_NAMESPACE -- valueFrom: -- fieldRef: -- fieldPath: metadata.namespace -+env: [] - - # -- Environment variables to be passed to Traefik's binary from configMaps or secrets - envFrom: [] - - ports: - traefik: -- port: 9000 -+ port: 8080 - # -- Use hostPort if set. - hostPort: # @schema type:[integer, null]; minimum:0 - # -- Use hostIP if set. If not set, Kubernetes will default to 0.0.0.0, which -@@ -601,7 +608,7 @@ ports: - expose: - default: false - # -- The exposed port for this service -- exposedPort: 9000 -+ exposedPort: 8080 - # -- The port protocol (TCP/UDP) - protocol: TCP - web: -@@ -614,7 +621,7 @@ ports: - default: true - exposedPort: 80 - ## -- Different target traefik port on the cluster, useful for IP type LB -- targetPort: # @schema type:[integer, null]; minimum:0 -+ targetPort: # @schema type:[string, integer, null]; minimum:0 - # The port protocol (TCP/UDP) - protocol: TCP - # -- See [upstream documentation](https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport) -@@ -653,7 +660,7 @@ ports: - default: true - exposedPort: 443 - ## -- Different target traefik port on the cluster, useful for IP type LB -- targetPort: # @schema type:[integer, null]; minimum:0 -+ targetPort: # @schema type:[string, integer, null]; minimum:0 - ## -- The port protocol (TCP/UDP) - protocol: TCP - # -- See [upstream documentation](https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport) -@@ -780,8 +787,8 @@ autoscaling: - - persistence: - # -- Enable persistence using Persistent Volume Claims -- # ref: http://kubernetes.io/docs/user-guide/persistent-volumes/ -- # It can be used to store TLS certificates, see `storage` in certResolvers -+ # ref: http://kubernetes.io/docs/user-guide/persistent-volumes/. -+ # It can be used to store TLS certificates along with `certificatesResolvers..acme.storage` option - enabled: false - name: data - existingClaim: "" -@@ -797,7 +804,7 @@ persistence: - # -- Certificates resolvers configuration. - # Ref: https://doc.traefik.io/traefik/https/acme/#certificate-resolvers - # See EXAMPLES.md for more details. --certResolvers: {} -+certificatesResolvers: {} - - # -- If hostNetwork is true, runs traefik in the host network namespace - # To prevent unschedulabel pods due to port collisions, if hostNetwork=true -@@ -860,7 +867,7 @@ topologySpreadConstraints: [] - # on nodes where no other traefik pods are scheduled. - # - labelSelector: - # matchLabels: --# app: '{{ template "traefik.name" . }}' -+# app.kubernetes.io/name: '{{ template "traefik.name" . }}' - # maxSkew: 1 - # topologyKey: kubernetes.io/hostname - # whenUnsatisfiable: DoNotSchedule -``` - -## 32.1.0 ![AppVersion: v3.1.5](https://img.shields.io/static/v1?label=AppVersion&message=v3.1.5&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-10-04 - -* fix: :bug: set disableIngressClassLookup until 3.1.4 -* feat(deps): update traefik docker tag to v3.1.5 -* feat(Traefik Proxy): update rbac following v3.2 migration guide -* chore(release): 🚀 publish v32.1.0 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index f36a9dd..73371f3 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -809,9 +809,9 @@ hostNetwork: false - rbac: # @schema additionalProperties: false - enabled: true - # When set to true: -- # 1. Use `Role` and `RoleBinding` instead of `ClusterRole` and `ClusterRoleBinding`. -- # 2. Set `disableIngressClassLookup` on Kubernetes Ingress providers with Traefik Proxy v3 until v3.1.1 -- # 3. Set `disableClusterScopeResources` on Kubernetes Ingress and CRD providers with Traefik Proxy v3.1.2+ -+ # 1. It switches respectively the use of `ClusterRole` and `ClusterRoleBinding` to `Role` and `RoleBinding`. -+ # 2. It adds `disableIngressClassLookup` on Kubernetes Ingress with Traefik Proxy v3 until v3.1.4 -+ # 3. It adds `disableClusterScopeResources` on Ingress and CRD (Kubernetes) providers with Traefik Proxy v3.1.2+ - # **NOTE**: `IngressClass`, `NodePortLB` and **Gateway** provider cannot be used with namespaced RBAC. - # See [upstream documentation](https://doc.traefik.io/traefik/providers/kubernetes-ingress/#disableclusterscoperesources) for more details. - namespaced: false - -## 32.0.0 ![AppVersion: v3.1.4](https://img.shields.io/static/v1?label=AppVersion&message=v3.1.4&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-09-27 - -* chore(release): :rocket: publish 32.0.0 -* fix: replace `CLF` with `common` in `values.yaml` -* feat(Traefik Hub): add APIPlans and APIBundles CRDs - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 51dec67..f36a9dd 100644 -index d5173dc..f36a9dd 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -345,7 +345,7 @@ logs: - # -- To enable access logs - enabled: false - # -- Set [access log format](https://doc.traefik.io/traefik/observability/access-logs/#format) -- format: # @schema enum:["CLF", "json", null]; type:[string, null]; default: "CLF" -+ format: # @schema enum:["common", "json", null]; type:[string, null]; default: "common" - # filePath: "/var/log/traefik/access.log - # -- Set [bufferingSize](https://doc.traefik.io/traefik/observability/access-logs/#bufferingsize) - bufferingSize: # @schema type:[integer, null] -@@ -911,35 +911,34 @@ hub: - # -- Certificate of the WebHook admission server. Default: "hub-agent-cert". - secretName: "" - -- ratelimit: -- redis: -- # -- Enable Redis Cluster. Default: true. -- cluster: # @schema type:[boolean, null] -- # -- Database used to store information. Default: "0". -- database: # @schema type:[string, null] -- # -- Endpoints of the Redis instances to connect to. Default: "". -- endpoints: "" -- # -- The username to use when connecting to Redis endpoints. Default: "". -+ redis: -+ # -- Enable Redis Cluster. Default: true. -+ cluster: # @schema type:[boolean, null] -+ # -- Database used to store information. Default: "0". -+ database: # @schema type:[string, null] -+ # -- Endpoints of the Redis instances to connect to. Default: "". -+ endpoints: "" -+ # -- The username to use when connecting to Redis endpoints. Default: "". -+ username: "" -+ # -- The password to use when connecting to Redis endpoints. Default: "". -+ password: "" -+ sentinel: -+ # -- Name of the set of main nodes to use for main selection. Required when using Sentinel. Default: "". -+ masterset: "" -+ # -- Username to use for sentinel authentication (can be different from endpoint username). Default: "". - username: "" -- # -- The password to use when connecting to Redis endpoints. Default: "". -+ # -- Password to use for sentinel authentication (can be different from endpoint password). Default: "". - password: "" -- sentinel: -- # -- Name of the set of main nodes to use for main selection. Required when using Sentinel. Default: "". -- masterset: "" -- # -- Username to use for sentinel authentication (can be different from endpoint username). Default: "". -- username: "" -- # -- Password to use for sentinel authentication (can be different from endpoint password). Default: "". -- password: "" -- # -- Timeout applied on connection with redis. Default: "0s". -- timeout: "" -- tls: -- # -- Path to the certificate authority used for the secured connection. -- ca: "" -- # -- Path to the public certificate used for the secure connection. -- cert: "" -- # -- Path to the private key used for the secure connection. -- key: "" -- # -- When insecureSkipVerify is set to true, the TLS connection accepts any certificate presented by the server. Default: false. -- insecureSkipVerify: false -+ # -- Timeout applied on connection with redis. Default: "0s". -+ timeout: "" -+ tls: -+ # -- Path to the certificate authority used for the secured connection. -+ ca: "" -+ # -- Path to the public certificate used for the secure connection. -+ cert: "" -+ # -- Path to the private key used for the secure connection. -+ key: "" -+ # -- When insecureSkipVerify is set to true, the TLS connection accepts any certificate presented by the server. Default: false. -+ insecureSkipVerify: false - # Enable export of errors logs to the platform. Default: true. - sendlogs: # @schema type:[boolean, null] -``` - -## 32.0.0-rc1 ![AppVersion: v3.1.4](https://img.shields.io/static/v1?label=AppVersion&message=v3.1.4&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-09-20 - -* feat(Traefik Hub): add APIPlans and APIBundles CRDs -* chore(release): 🚀 publish 32.0.0-rc1 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index d5173dc..51dec67 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -911,35 +911,34 @@ hub: - # -- Certificate of the WebHook admission server. Default: "hub-agent-cert". - secretName: "" - -- ratelimit: -- redis: -- # -- Enable Redis Cluster. Default: true. -- cluster: # @schema type:[boolean, null] -- # -- Database used to store information. Default: "0". -- database: # @schema type:[string, null] -- # -- Endpoints of the Redis instances to connect to. Default: "". -- endpoints: "" -- # -- The username to use when connecting to Redis endpoints. Default: "". -+ redis: -+ # -- Enable Redis Cluster. Default: true. -+ cluster: # @schema type:[boolean, null] -+ # -- Database used to store information. Default: "0". -+ database: # @schema type:[string, null] -+ # -- Endpoints of the Redis instances to connect to. Default: "". -+ endpoints: "" -+ # -- The username to use when connecting to Redis endpoints. Default: "". -+ username: "" -+ # -- The password to use when connecting to Redis endpoints. Default: "". -+ password: "" -+ sentinel: -+ # -- Name of the set of main nodes to use for main selection. Required when using Sentinel. Default: "". -+ masterset: "" -+ # -- Username to use for sentinel authentication (can be different from endpoint username). Default: "". - username: "" -- # -- The password to use when connecting to Redis endpoints. Default: "". -+ # -- Password to use for sentinel authentication (can be different from endpoint password). Default: "". - password: "" -- sentinel: -- # -- Name of the set of main nodes to use for main selection. Required when using Sentinel. Default: "". -- masterset: "" -- # -- Username to use for sentinel authentication (can be different from endpoint username). Default: "". -- username: "" -- # -- Password to use for sentinel authentication (can be different from endpoint password). Default: "". -- password: "" -- # -- Timeout applied on connection with redis. Default: "0s". -- timeout: "" -- tls: -- # -- Path to the certificate authority used for the secured connection. -- ca: "" -- # -- Path to the public certificate used for the secure connection. -- cert: "" -- # -- Path to the private key used for the secure connection. -- key: "" -- # -- When insecureSkipVerify is set to true, the TLS connection accepts any certificate presented by the server. Default: false. -- insecureSkipVerify: false -+ # -- Timeout applied on connection with redis. Default: "0s". -+ timeout: "" -+ tls: -+ # -- Path to the certificate authority used for the secured connection. -+ ca: "" -+ # -- Path to the public certificate used for the secure connection. -+ cert: "" -+ # -- Path to the private key used for the secure connection. -+ key: "" -+ # -- When insecureSkipVerify is set to true, the TLS connection accepts any certificate presented by the server. Default: false. -+ insecureSkipVerify: false - # Enable export of errors logs to the platform. Default: true. - sendlogs: # @schema type:[boolean, null] -``` - -## 31.1.1 ![AppVersion: v3.1.4](https://img.shields.io/static/v1?label=AppVersion&message=v3.1.4&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-09-20 - -* fix: 🐛 updateStrategy behavior -* feat(deps): update traefik docker tag to v3.1.4 -* chore(release): 🚀 publish v31.1.1 - -## 31.1.0 ![AppVersion: v3.1.3](https://img.shields.io/static/v1?label=AppVersion&message=v3.1.3&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-09-18 - -* fix: 🐛 update CRD to v3.1 -* feat: ✨ input validation using schema -* feat: ✨ add AllowACMEByPass and improve schema/doc on ports values -* feat: add new webhooks and removes unnecessary ones -* feat(deps): update traefik docker tag to v3.1.3 -* chore(release): 🚀 publish v31.1.0 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 2232d9e..1b9d0fd 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -2,13 +2,13 @@ - # This is a YAML-formatted file. - # Declare variables to be passed into templates - --image: -+image: # @schema additionalProperties: false - # -- Traefik image host registry - registry: docker.io - # -- Traefik image repository - repository: traefik - # -- defaults to appVersion -- tag: -+ tag: # @schema type:[string, null] - # -- Traefik image pull policy - pullPolicy: IfNotPresent - -@@ -23,27 +23,27 @@ deployment: - # -- Number of pods of the deployment (only applies when kind == Deployment) - replicas: 1 - # -- Number of old history to retain to allow rollback (If not set, default Kubernetes value is set to 10) -- # revisionHistoryLimit: 1 -+ revisionHistoryLimit: # @schema type:[integer, null];minimum:0 - # -- Amount of time (in seconds) before Kubernetes will send the SIGKILL signal if Traefik does not shut down - terminationGracePeriodSeconds: 60 - # -- The minimum number of seconds Traefik needs to be up and running before the DaemonSet/Deployment controller considers it available - minReadySeconds: 0 -- ## Override the liveness/readiness port. This is useful to integrate traefik -+ ## -- Override the liveness/readiness port. This is useful to integrate traefik - ## with an external Load Balancer that performs healthchecks. - ## Default: ports.traefik.port -- # healthchecksPort: 9000 -- ## Override the liveness/readiness host. Useful for getting ping to respond on non-default entryPoint. -+ healthchecksPort: # @schema type:[integer, null];minimum:0 -+ ## -- Override the liveness/readiness host. Useful for getting ping to respond on non-default entryPoint. - ## Default: ports.traefik.hostIP if set, otherwise Pod IP -- # healthchecksHost: localhost -- ## Override the liveness/readiness scheme. Useful for getting ping to -+ healthchecksHost: "" -+ ## -- Override the liveness/readiness scheme. Useful for getting ping to - ## respond on websecure entryPoint. -- # healthchecksScheme: HTTPS -- ## Override the readiness path. -+ healthchecksScheme: # @schema enum:[HTTP, HTTPS, null]; type:[string, null]; default: HTTP -+ ## -- Override the readiness path. - ## Default: /ping -- # readinessPath: /ping -- # Override the liveness path. -+ readinessPath: "" -+ # -- Override the liveness path. - # Default: /ping -- # livenessPath: /ping -+ livenessPath: "" - # -- Additional deployment annotations (e.g. for jaeger-operator sidecar injection) - annotations: {} - # -- Additional deployment labels (e.g. for filtering deployment by custom labels) -@@ -80,7 +80,7 @@ deployment: - # -- Use process namespace sharing - shareProcessNamespace: false - # -- Custom pod DNS policy. Apply if `hostNetwork: true` -- # dnsPolicy: ClusterFirstWithHostNet -+ dnsPolicy: "" - # -- Custom pod [DNS config](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.30/#poddnsconfig-v1-core) - dnsConfig: {} - # -- Custom [host aliases](https://kubernetes.io/docs/tasks/network/customize-hosts-file-for-pods/) -@@ -99,24 +99,24 @@ deployment: - # host: localhost - # scheme: HTTP - # -- Set a runtimeClassName on pod -- runtimeClassName: -+ runtimeClassName: "" - - # -- [Pod Disruption Budget](https://kubernetes.io/docs/reference/kubernetes-api/policy-resources/pod-disruption-budget-v1/) --podDisruptionBudget: -- enabled: -- maxUnavailable: -- minAvailable: -+podDisruptionBudget: # @schema additionalProperties: false -+ enabled: false -+ maxUnavailable: # @schema type:[string, integer, null];minimum:0 -+ minAvailable: # @schema type:[string, integer, null];minimum:0 - - # -- Create a default IngressClass for Traefik --ingressClass: -+ingressClass: # @schema additionalProperties: false - enabled: true - isDefaultClass: true -- # name: my-custom-class -+ name: "" - --core: -+core: # @schema additionalProperties: false - # -- Can be used to use globally v2 router syntax - # See https://doc.traefik.io/traefik/v3.0/migration/v2-to-v3/#new-v3-syntax-notable-changes -- defaultRuleSyntax: -+ defaultRuleSyntax: "" - - # Traefik experimental features - experimental: -@@ -133,11 +133,11 @@ gateway: - # -- When providers.kubernetesGateway.enabled, deploy a default gateway - enabled: true - # -- Set a custom name to gateway -- name: -+ name: "" - # -- By default, Gateway is created in the same `Namespace` than Traefik. -- namespace: -+ namespace: "" - # -- Additional gateway annotations (e.g. for cert-manager.io/issuer) -- annotations: -+ annotations: {} - # -- Define listeners - listeners: - web: -@@ -145,11 +145,11 @@ gateway: - # The port must match a port declared in ports section. - port: 8000 - # -- Optional hostname. See [Hostname](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.Hostname) -- hostname: -+ hostname: "" - # Specify expected protocol on this listener. See [ProtocolType](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.ProtocolType) - protocol: HTTP - # -- Routes are restricted to namespace of the gateway [by default](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.FromNamespaces -- namespacePolicy: -+ namespacePolicy: # @schema type:[string, null] - # websecure listener is disabled by default because certificateRefs needs to be added, - # or you may specify TLS protocol with Passthrough mode and add "--providers.kubernetesGateway.experimentalChannel=true" in additionalArguments section. - # websecure: -@@ -167,13 +167,13 @@ gateway: - # # -- TLS behavior for the TLS session initiated by the client. See [TLSModeType](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.TLSModeType). - # mode: - --gatewayClass: -+gatewayClass: # @schema additionalProperties: false - # -- When providers.kubernetesGateway.enabled and gateway.enabled, deploy a default gatewayClass - enabled: true - # -- Set a custom name to GatewayClass -- name: -+ name: "" - # -- Additional gatewayClass labels (e.g. for filtering gateway objects by custom labels) -- labels: -+ labels: {} - - ingressRoute: - dashboard: -@@ -218,14 +218,14 @@ ingressRoute: - # -- TLS options (e.g. secret containing certificate) - tls: {} - --updateStrategy: -+updateStrategy: # @schema additionalProperties: false - # -- Customize updateStrategy: RollingUpdate or OnDelete - type: RollingUpdate - rollingUpdate: -- maxUnavailable: 0 -- maxSurge: 1 -+ maxUnavailable: 0 # @schema type:[integer, string, null] -+ maxSurge: 1 # @schema type:[integer, string, null] - --readinessProbe: -+readinessProbe: # @schema additionalProperties: false - # -- The number of consecutive failures allowed before considering the probe as failed. - failureThreshold: 1 - # -- The number of seconds to wait before starting the first probe. -@@ -236,7 +236,7 @@ readinessProbe: - successThreshold: 1 - # -- The number of seconds to wait for a probe response before considering it as failed. - timeoutSeconds: 2 --livenessProbe: -+livenessProbe: # @schema additionalProperties: false - # -- The number of consecutive failures allowed before considering the probe as failed. - failureThreshold: 3 - # -- The number of seconds to wait before starting the first probe. -@@ -249,9 +249,9 @@ livenessProbe: - timeoutSeconds: 2 - - # -- Define [Startup Probe](https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/#define-startup-probes) --startupProbe: -+startupProbe: {} - --providers: -+providers: # @schema additionalProperties: false - kubernetesCRD: - # -- Load Kubernetes IngressRoute provider - enabled: true -@@ -262,12 +262,12 @@ providers: - # -- Allows to return 503 when there is no endpoints available - allowEmptyServices: true - # -- When the parameter is set, only resources containing an annotation with the same value are processed. Otherwise, resources missing the annotation, having an empty value, or the value traefik are processed. It will also set required annotation on Dashboard and Healthcheck IngressRoute when enabled. -- ingressClass: -+ ingressClass: "" - # labelSelector: environment=production,method=traefik - # -- Array of namespaces to watch. If left empty, Traefik watches all namespaces. - namespaces: [] - # -- Defines whether to use Native Kubernetes load-balancing mode by default. -- nativeLBByDefault: -+ nativeLBByDefault: false - - kubernetesIngress: - # -- Load Kubernetes Ingress provider -@@ -277,7 +277,7 @@ providers: - # -- Allows to return 503 when there is no endpoints available - allowEmptyServices: true - # -- When ingressClass is set, only Ingresses containing an annotation with the same value are processed. Otherwise, Ingresses missing the annotation, having an empty value, or the value traefik are processed. -- ingressClass: -+ ingressClass: # @schema type:[string, null] - # labelSelector: environment=production,method=traefik - # -- Array of namespaces to watch. If left empty, Traefik watches all namespaces. - namespaces: [] -@@ -288,7 +288,7 @@ providers: - # By default this Traefik service - # pathOverride: "" - # -- Defines whether to use Native Kubernetes load-balancing mode by default. -- nativeLBByDefault: -+ nativeLBByDefault: false - - kubernetesGateway: - # -- Enable Traefik Gateway provider for Gateway API -@@ -299,7 +299,7 @@ providers: - # -- Array of namespaces to watch. If left empty, Traefik watches all namespaces. - namespaces: [] - # -- A label selector can be defined to filter on specific GatewayClass objects only. -- labelselector: -+ labelselector: "" - - file: - # -- Create a file provider -@@ -307,7 +307,7 @@ providers: - # -- Allows Traefik to automatically watch for file changes - watch: true - # -- File content (YAML format, go template supported) (see https://doc.traefik.io/traefik/providers/file/) -- content: -+ content: "" - - # -- Add volumes to the traefik pod. The volume name will be passed to tpl. - # This can be used to mount a cert pair or a configmap that holds a config.toml file. -@@ -333,90 +333,88 @@ additionalVolumeMounts: [] - logs: - general: - # -- Set [logs format](https://doc.traefik.io/traefik/observability/logs/#format) -- # @default common -- format: -+ format: # @schema enum:["common", "json", null]; type:[string, null]; default: "common" - # By default, the level is set to INFO. - # -- Alternative logging levels are DEBUG, PANIC, FATAL, ERROR, WARN, and INFO. -- level: INFO -- # -- # filePath: "/var/log/traefik/traefik.log -- # noColor: true -+ level: "INFO" # @schema enum:[INFO,WARN,ERROR,FATAL,PANIC,DEBUG]; default: "INFO" -+ # -- To write the logs into a log file, use the filePath option. -+ filePath: "" -+ # -- When set to true and format is common, it disables the colorized output. -+ noColor: false - access: - # -- To enable access logs - enabled: false - # -- Set [access log format](https://doc.traefik.io/traefik/observability/access-logs/#format) -- format: -+ format: # @schema enum:["CLF", "json", null]; type:[string, null]; default: "CLF" - # filePath: "/var/log/traefik/access.log - # -- Set [bufferingSize](https://doc.traefik.io/traefik/observability/access-logs/#bufferingsize) -- bufferingSize: -+ bufferingSize: # @schema type:[integer, null] - # -- Set [filtering](https://docs.traefik.io/observability/access-logs/#filtering) - filters: {} -- # statuscodes: "200,300-302" -- # retryattempts: true -- # minduration: 10ms -+ statuscodes: "" -+ retryattempts: false -+ minduration: "" - # -- Enables accessLogs for internal resources. Default: false. -- addInternals: -+ addInternals: false - fields: - general: -- # -- Available modes: keep, drop, redact. -- defaultmode: keep -+ # -- Set default mode for fields.names -+ defaultmode: keep # @schema enum:[keep, drop, redact]; default: keep - # -- Names of the fields to limit. - names: {} -- ## Examples: -- # ClientUsername: drop - # -- [Limit logged fields or headers](https://doc.traefik.io/traefik/observability/access-logs/#limiting-the-fieldsincluding-headers) - headers: -- # -- Available modes: keep, drop, redact. -- defaultmode: drop -+ # -- Set default mode for fields.headers -+ defaultmode: drop # @schema enum:[keep, drop, redact]; default: drop - names: {} - - metrics: - ## -- Enable metrics for internal resources. Default: false -- addInternals: -+ addInternals: false - - ## -- Prometheus is enabled by default. - ## -- It can be disabled by setting "prometheus: null" - prometheus: - # -- Entry point used to expose metrics. - entryPoint: metrics -- ## Enable metrics on entry points. Default=true -- # addEntryPointsLabels: false -- ## Enable metrics on routers. Default=false -- # addRoutersLabels: true -- ## Enable metrics on services. Default=true -- # addServicesLabels: false -+ ## Enable metrics on entry points. Default: true -+ addEntryPointsLabels: # @schema type:[boolean, null] -+ ## Enable metrics on routers. Default: false -+ addRoutersLabels: # @schema type:[boolean, null] -+ ## Enable metrics on services. Default: true -+ addServicesLabels: # @schema type:[boolean, null] - ## Buckets for latency metrics. Default="0.1,0.3,1.2,5.0" -- # buckets: "0.5,1.0,2.5" -+ buckets: "" - ## When manualRouting is true, it disables the default internal router in - ## order to allow creating a custom router for prometheus@internal service. -- # manualRouting: true -+ manualRouting: false - service: - # -- Create a dedicated metrics service to use with ServiceMonitor -- enabled: -- labels: -- annotations: -+ enabled: false -+ labels: {} -+ annotations: {} - # -- When set to true, it won't check if Prometheus Operator CRDs are deployed -- disableAPICheck: -+ disableAPICheck: # @schema type:[boolean, null] - serviceMonitor: - # -- Enable optional CR for Prometheus Operator. See EXAMPLES.md for more details. - enabled: false -- metricRelabelings: -- relabelings: -- jobLabel: -- interval: -- honorLabels: -- scrapeTimeout: -- honorTimestamps: -- enableHttp2: -- followRedirects: -- additionalLabels: -- namespace: -- namespaceSelector: -+ metricRelabelings: [] -+ relabelings: [] -+ jobLabel: "" -+ interval: "" -+ honorLabels: false -+ scrapeTimeout: "" -+ honorTimestamps: false -+ enableHttp2: false -+ followRedirects: false -+ additionalLabels: {} -+ namespace: "" -+ namespaceSelector: {} - prometheusRule: - # -- Enable optional CR for Prometheus Operator. See EXAMPLES.md for more details. - enabled: false -- additionalLabels: -- namespace: -+ additionalLabels: {} -+ namespace: "" - - # datadog: - # ## Address instructs exporter to send metrics to datadog-agent at this address. -@@ -469,55 +467,55 @@ metrics: - # -- Set to true in order to enable the OpenTelemetry metrics - enabled: false - # -- Enable metrics on entry points. Default: true -- addEntryPointsLabels: -+ addEntryPointsLabels: # @schema type:[boolean, null] - # -- Enable metrics on routers. Default: false -- addRoutersLabels: -+ addRoutersLabels: # @schema type:[boolean, null] - # -- Enable metrics on services. Default: true -- addServicesLabels: -+ addServicesLabels: # @schema type:[boolean, null] - # -- Explicit boundaries for Histogram data points. Default: [.005, .01, .025, .05, .1, .25, .5, 1, 2.5, 5, 10] -- explicitBoundaries: -+ explicitBoundaries: [] - # -- Interval at which metrics are sent to the OpenTelemetry Collector. Default: 10s -- pushInterval: -+ pushInterval: "" - http: - # -- Set to true in order to send metrics to the OpenTelemetry Collector using HTTP. - enabled: false - # -- Format: ://:. Default: http://localhost:4318/v1/metrics -- endpoint: -+ endpoint: "" - # -- Additional headers sent with metrics by the reporter to the OpenTelemetry Collector. -- headers: -+ headers: {} - ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. - tls: - # -- The path to the certificate authority, it defaults to the system bundle. -- ca: -+ ca: "" - # -- The path to the public certificate. When using this option, setting the key option is required. -- cert: -+ cert: "" - # -- The path to the private key. When using this option, setting the cert option is required. -- key: -+ key: "" - # -- When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. -- insecureSkipVerify: -+ insecureSkipVerify: # @schema type:[boolean, null] - grpc: - # -- Set to true in order to send metrics to the OpenTelemetry Collector using gRPC - enabled: false - # -- Format: ://:. Default: http://localhost:4318/v1/metrics -- endpoint: -+ endpoint: "" - # -- Allows reporter to send metrics to the OpenTelemetry Collector without using a secured protocol. -- insecure: -+ insecure: false - ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. - tls: - # -- The path to the certificate authority, it defaults to the system bundle. -- ca: -+ ca: "" - # -- The path to the public certificate. When using this option, setting the key option is required. -- cert: -+ cert: "" - # -- The path to the private key. When using this option, setting the cert option is required. -- key: -+ key: "" - # -- When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. -- insecureSkipVerify: -+ insecureSkipVerify: false - - ## Tracing - # -- https://doc.traefik.io/traefik/observability/tracing/overview/ --tracing: -+tracing: # @schema additionalProperties: false - # -- Enables tracing for internal resources. Default: false. -- addInternals: -+ addInternals: false - otlp: - # -- See https://doc.traefik.io/traefik/v3.0/observability/tracing/opentelemetry/ - enabled: false -@@ -525,36 +523,36 @@ tracing: - # -- Set to true in order to send metrics to the OpenTelemetry Collector using HTTP. - enabled: false - # -- Format: ://:. Default: http://localhost:4318/v1/metrics -- endpoint: -+ endpoint: "" - # -- Additional headers sent with metrics by the reporter to the OpenTelemetry Collector. -- headers: -+ headers: {} - ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. - tls: - # -- The path to the certificate authority, it defaults to the system bundle. -- ca: -+ ca: "" - # -- The path to the public certificate. When using this option, setting the key option is required. -- cert: -+ cert: "" - # -- The path to the private key. When using this option, setting the cert option is required. -- key: -+ key: "" - # -- When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. -- insecureSkipVerify: -+ insecureSkipVerify: false - grpc: - # -- Set to true in order to send metrics to the OpenTelemetry Collector using gRPC - enabled: false - # -- Format: ://:. Default: http://localhost:4318/v1/metrics -- endpoint: -+ endpoint: "" - # -- Allows reporter to send metrics to the OpenTelemetry Collector without using a secured protocol. -- insecure: -+ insecure: false - ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. - tls: - # -- The path to the certificate authority, it defaults to the system bundle. -- ca: -+ ca: "" - # -- The path to the public certificate. When using this option, setting the key option is required. -- cert: -+ cert: "" - # -- The path to the private key. When using this option, setting the cert option is required. -- key: -+ key: "" - # -- When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. -- insecureSkipVerify: -+ insecureSkipVerify: false - - # -- Global command arguments to be passed to all traefik's pods - globalArguments: -@@ -587,13 +585,12 @@ ports: - traefik: - port: 9000 - # -- Use hostPort if set. -- # hostPort: 9000 -- # -+ hostPort: # @schema type:[integer, null]; minimum:0 - # -- Use hostIP if set. If not set, Kubernetes will default to 0.0.0.0, which - # means it's listening on all your interfaces and all your IPs. You may want - # to set this value if you need traefik to listen on specific interface - # only. -- # hostIP: 192.168.100.10 -+ hostIP: # @schema type:[string, null] - - # Defines whether the port is exposed if service.type is LoadBalancer or - # NodePort. -@@ -617,112 +614,93 @@ ports: - default: true - exposedPort: 80 - ## -- Different target traefik port on the cluster, useful for IP type LB -- # targetPort: 80 -+ targetPort: # @schema type:[integer, null]; minimum:0 - # The port protocol (TCP/UDP) - protocol: TCP -- # -- Use nodeport if set. This is useful if you have configured Traefik in a -- # LoadBalancer. -- # nodePort: 32080 -+ # -- See [upstream documentation](https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport) -+ nodePort: # @schema type:[integer, null]; minimum:0 - # Port Redirections - # Added in 2.2, you can make permanent redirects via entrypoints. - # https://docs.traefik.io/routing/entrypoints/#redirection -- # redirectTo: -- # port: websecure -- # (Optional) -- # priority: 10 -- # permanent: true -- # -- # -- Trust forwarded headers information (X-Forwarded-*). -- # forwardedHeaders: -- # trustedIPs: [] -- # insecure: false -- # -- # -- Enable the Proxy Protocol header parsing for the entry point -- # proxyProtocol: -- # trustedIPs: [] -- # insecure: false -- # -+ redirectTo: {} -+ forwardedHeaders: -+ # -- Trust forwarded headers information (X-Forwarded-*). -+ trustedIPs: [] -+ insecure: false -+ proxyProtocol: -+ # -- Enable the Proxy Protocol header parsing for the entry point -+ trustedIPs: [] -+ insecure: false - # -- Set transport settings for the entrypoint; see also - # https://doc.traefik.io/traefik/routing/entrypoints/#transport - transport: - respondingTimeouts: -- readTimeout: -- writeTimeout: -- idleTimeout: -+ readTimeout: # @schema type:[string, integer, null] -+ writeTimeout: # @schema type:[string, integer, null] -+ idleTimeout: # @schema type:[string, integer, null] - lifeCycle: -- requestAcceptGraceTimeout: -- graceTimeOut: -- keepAliveMaxRequests: -- keepAliveMaxTime: -+ requestAcceptGraceTimeout: # @schema type:[string, integer, null] -+ graceTimeOut: # @schema type:[string, integer, null] -+ keepAliveMaxRequests: # @schema type:[integer, null]; minimum:0 -+ keepAliveMaxTime: # @schema type:[string, integer, null] - websecure: - ## -- Enable this entrypoint as a default entrypoint. When a service doesn't explicitly set an entrypoint it will only use this entrypoint. - # asDefault: true - port: 8443 -- # hostPort: 8443 -- # containerPort: 8443 -+ hostPort: # @schema type:[integer, null]; minimum:0 -+ containerPort: # @schema type:[integer, null]; minimum:0 - expose: - default: true - exposedPort: 443 - ## -- Different target traefik port on the cluster, useful for IP type LB -- # targetPort: 80 -+ targetPort: # @schema type:[integer, null]; minimum:0 - ## -- The port protocol (TCP/UDP) - protocol: TCP -- # nodePort: 32443 -- ## -- Specify an application protocol. This may be used as a hint for a Layer 7 load balancer. -- # appProtocol: https -- # -- ## -- Enable HTTP/3 on the entrypoint -- ## Enabling it will also enable http3 experimental feature -- ## https://doc.traefik.io/traefik/routing/entrypoints/#http3 -- ## There are known limitations when trying to listen on same ports for -- ## TCP & UDP (Http3). There is a workaround in this chart using dual Service. -- ## https://github.com/kubernetes/kubernetes/issues/47249#issuecomment-587960741 -+ # -- See [upstream documentation](https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport) -+ nodePort: # @schema type:[integer, null]; minimum:0 -+ # -- See [upstream documentation](https://kubernetes.io/docs/concepts/services-networking/service/#application-protocol) -+ appProtocol: # @schema type:[string, null] -+ # -- See [upstream documentation](https://doc.traefik.io/traefik/routing/entrypoints/#allowacmebypass) -+ allowACMEByPass: false - http3: -+ ## -- Enable HTTP/3 on the entrypoint -+ ## Enabling it will also enable http3 experimental feature -+ ## https://doc.traefik.io/traefik/routing/entrypoints/#http3 -+ ## There are known limitations when trying to listen on same ports for -+ ## TCP & UDP (Http3). There is a workaround in this chart using dual Service. -+ ## https://github.com/kubernetes/kubernetes/issues/47249#issuecomment-587960741 - enabled: false -- # advertisedPort: 4443 -- # -- # -- Trust forwarded headers information (X-Forwarded-*). -- # forwardedHeaders: -- # trustedIPs: [] -- # insecure: false -- # -- # -- Enable the Proxy Protocol header parsing for the entry point -- # proxyProtocol: -- # trustedIPs: [] -- # insecure: false -- # -- # -- Set transport settings for the entrypoint; see also -- # https://doc.traefik.io/traefik/routing/entrypoints/#transport -+ advertisedPort: # @schema type:[integer, null]; minimum:0 -+ forwardedHeaders: -+ # -- Trust forwarded headers information (X-Forwarded-*). -+ trustedIPs: [] -+ insecure: false -+ proxyProtocol: -+ # -- Enable the Proxy Protocol header parsing for the entry point -+ trustedIPs: [] -+ insecure: false -+ # -- See [upstream documentation](https://doc.traefik.io/traefik/routing/entrypoints/#transport) - transport: - respondingTimeouts: -- readTimeout: -- writeTimeout: -- idleTimeout: -+ readTimeout: # @schema type:[string, integer, null] -+ writeTimeout: # @schema type:[string, integer, null] -+ idleTimeout: # @schema type:[string, integer, null] - lifeCycle: -- requestAcceptGraceTimeout: -- graceTimeOut: -- keepAliveMaxRequests: -- keepAliveMaxTime: -- # -- ## Set TLS at the entrypoint -- ## https://doc.traefik.io/traefik/routing/entrypoints/#tls -+ requestAcceptGraceTimeout: # @schema type:[string, integer, null] -+ graceTimeOut: # @schema type:[string, integer, null] -+ keepAliveMaxRequests: # @schema type:[integer, null]; minimum:0 -+ keepAliveMaxTime: # @schema type:[string, integer, null] -+ # -- See [upstream documentation](https://doc.traefik.io/traefik/routing/entrypoints/#tls) - tls: - enabled: true -- # this is the name of a TLSOption definition - options: "" - certResolver: "" - domains: [] -- # - main: example.com -- # sans: -- # - foo.example.com -- # - bar.example.com -- # - # -- One can apply Middlewares on an entrypoint - # https://doc.traefik.io/traefik/middlewares/overview/ - # https://doc.traefik.io/traefik/routing/entrypoints/#middlewares - # -- /!\ It introduces here a link between your static configuration and your dynamic configuration /!\ - # It follows the provider naming convention: https://doc.traefik.io/traefik/providers/overview/#provider-namespace -- # middlewares: - # - namespace-name1@kubernetescrd - # - namespace-name2@kubernetescrd - middlewares: [] -@@ -730,10 +708,6 @@ ports: - # -- When using hostNetwork, use another port to avoid conflict with node exporter: - # https://github.com/prometheus/prometheus/wiki/Default-port-allocations - port: 9100 -- # hostPort: 9100 -- # Defines whether the port is exposed if service.type is LoadBalancer or -- # NodePort. -- # - # -- You may not want to expose the metrics port on production deployments. - # If you want to access it from outside your cluster, - # use `kubectl port-forward` or create a secure ingress -@@ -810,15 +784,15 @@ persistence: - # It can be used to store TLS certificates, see `storage` in certResolvers - enabled: false - name: data -- # existingClaim: "" -+ existingClaim: "" - accessMode: ReadWriteOnce - size: 128Mi -- # storageClass: "" -- # volumeName: "" -+ storageClass: "" -+ volumeName: "" - path: /data - annotations: {} - # -- Only mount a subpath of the Volume into the pod -- # subPath: "" -+ subPath: "" - - # -- Certificates resolvers configuration. - # Ref: https://doc.traefik.io/traefik/https/acme/#certificate-resolvers -@@ -832,7 +806,7 @@ certResolvers: {} - hostNetwork: false - - # -- Whether Role Based Access Control objects like roles and rolebindings should be created --rbac: -+rbac: # @schema additionalProperties: false - enabled: true - # When set to true: - # 1. Use `Role` and `RoleBinding` instead of `ClusterRole` and `ClusterRoleBinding`. -@@ -843,7 +817,7 @@ rbac: - namespaced: false - # Enable user-facing roles - # https://kubernetes.io/docs/reference/access-authn-authz/rbac/#user-facing-roles -- # aggregateTo: [ "admin" ] -+ aggregateTo: [] - # List of Kubernetes secrets that are accessible for Traefik. If empty, then access is granted to every secret. - secretResourceNames: [] - -@@ -852,7 +826,7 @@ podSecurityPolicy: - enabled: false - - # -- The service account the pods will use to interact with the Kubernetes API --serviceAccount: -+serviceAccount: # @schema additionalProperties: false - # If set, an existing service account is used - # If not set, a service account is created automatically using the fullname template - name: "" -@@ -918,54 +892,54 @@ extraObjects: [] - - # -- This field override the default Release Namespace for Helm. - # It will not affect optional CRDs such as `ServiceMonitor` and `PrometheusRules` --namespaceOverride: -+namespaceOverride: "" - - ## -- This field override the default app.kubernetes.io/instance label for all Objects. --instanceLabelOverride: -+instanceLabelOverride: "" - - # Traefik Hub configuration. See https://doc.traefik.io/traefik-hub/ - hub: - # -- Name of `Secret` with key 'token' set to a valid license token. - # It enables API Gateway. -- token: -+ token: "" - apimanagement: - # -- Set to true in order to enable API Management. Requires a valid license token. -- enabled: -+ enabled: false - admission: - # -- WebHook admission server listen address. Default: "0.0.0.0:9943". -- listenAddr: -+ listenAddr: "" - # -- Certificate of the WebHook admission server. Default: "hub-agent-cert". -- secretName: -+ secretName: "" - - ratelimit: - redis: - # -- Enable Redis Cluster. Default: true. -- cluster: -+ cluster: # @schema type:[boolean, null] - # -- Database used to store information. Default: "0". -- database: -+ database: # @schema type:[string, null] - # -- Endpoints of the Redis instances to connect to. Default: "". -- endpoints: -+ endpoints: "" - # -- The username to use when connecting to Redis endpoints. Default: "". -- username: -+ username: "" - # -- The password to use when connecting to Redis endpoints. Default: "". -- password: -+ password: "" - sentinel: - # -- Name of the set of main nodes to use for main selection. Required when using Sentinel. Default: "". -- masterset: -+ masterset: "" - # -- Username to use for sentinel authentication (can be different from endpoint username). Default: "". -- username: -+ username: "" - # -- Password to use for sentinel authentication (can be different from endpoint password). Default: "". -- password: -+ password: "" - # -- Timeout applied on connection with redis. Default: "0s". -- timeout: -+ timeout: "" - tls: - # -- Path to the certificate authority used for the secured connection. -- ca: -+ ca: "" - # -- Path to the public certificate used for the secure connection. -- cert: -+ cert: "" - # -- Path to the private key used for the secure connection. -- key: -+ key: "" - # -- When insecureSkipVerify is set to true, the TLS connection accepts any certificate presented by the server. Default: false. -- insecureSkipVerify: -+ insecureSkipVerify: false - # Enable export of errors logs to the platform. Default: true. -- sendlogs: -+ sendlogs: # @schema type:[boolean, null] -``` - -## 31.0.0 ![AppVersion: v3.1.2](https://img.shields.io/static/v1?label=AppVersion&message=v3.1.2&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-09-03 - -* fix(Traefik Hub): update CRDs to v1.5.0 -* fix(HTTP3): split udp and tcp Service when service.single is false -* fix!: 🐛 set allowEmptyServices to true by default -* feat(Traefik Hub): update CRDs to v1.7.0 -* chore(release): 🚀 publish v31.0.0 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 78eeacf..2232d9e 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -260,7 +260,7 @@ providers: - # -- Allows to reference ExternalName services in IngressRoute - allowExternalNameServices: false - # -- Allows to return 503 when there is no endpoints available -- allowEmptyServices: false -+ allowEmptyServices: true - # -- When the parameter is set, only resources containing an annotation with the same value are processed. Otherwise, resources missing the annotation, having an empty value, or the value traefik are processed. It will also set required annotation on Dashboard and Healthcheck IngressRoute when enabled. - ingressClass: - # labelSelector: environment=production,method=traefik -@@ -275,7 +275,7 @@ providers: - # -- Allows to reference ExternalName services in Ingress - allowExternalNameServices: false - # -- Allows to return 503 when there is no endpoints available -- allowEmptyServices: false -+ allowEmptyServices: true - # -- When ingressClass is set, only Ingresses containing an annotation with the same value are processed. Otherwise, Ingresses missing the annotation, having an empty value, or the value traefik are processed. - ingressClass: - # labelSelector: environment=production,method=traefik -``` - -## 30.1.0 ![AppVersion: v3.1.2](https://img.shields.io/static/v1?label=AppVersion&message=v3.1.2&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-08-14 - -* fix: disable default HTTPS listener for gateway -* fix(Gateway API): wildcard support in hostname -* fix(Gateway API): use Standard channel by default -* feat: ✨ rework namespaced RBAC with `disableClusterScopeResources` -* chore(release): 🚀 publish v30.1.0 -* chore(deps): update traefik docker tag to v3.1.2 -* chore(deps): update traefik docker tag to v3.1.1 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 83b6d98..78eeacf 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -150,20 +150,22 @@ gateway: - protocol: HTTP - # -- Routes are restricted to namespace of the gateway [by default](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.FromNamespaces - namespacePolicy: -- websecure: -- # -- Port is the network port. Multiple listeners may use the same port, subject to the Listener compatibility rules. -- # The port must match a port declared in ports section. -- port: 8443 -- # -- Optional hostname. See [Hostname](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.Hostname) -- hostname: -- # Specify expected protocol on this listener See [ProtocolType](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.ProtocolType) -- protocol: HTTPS -- # -- Routes are restricted to namespace of the gateway [by default](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.FromNamespaces) -- namespacePolicy: -- # -- Add certificates for TLS or HTTPS protocols. See [GatewayTLSConfig](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io%2fv1.GatewayTLSConfig) -- certificateRefs: -- # -- TLS behavior for the TLS session initiated by the client. See [TLSModeType](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.TLSModeType). -- mode: -+ # websecure listener is disabled by default because certificateRefs needs to be added, -+ # or you may specify TLS protocol with Passthrough mode and add "--providers.kubernetesGateway.experimentalChannel=true" in additionalArguments section. -+ # websecure: -+ # # -- Port is the network port. Multiple listeners may use the same port, subject to the Listener compatibility rules. -+ # # The port must match a port declared in ports section. -+ # port: 8443 -+ # # -- Optional hostname. See [Hostname](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.Hostname) -+ # hostname: -+ # # Specify expected protocol on this listener See [ProtocolType](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.ProtocolType) -+ # protocol: HTTPS -+ # # -- Routes are restricted to namespace of the gateway [by default](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.FromNamespaces) -+ # namespacePolicy: -+ # # -- Add certificates for TLS or HTTPS protocols. See [GatewayTLSConfig](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io%2fv1.GatewayTLSConfig) -+ # certificateRefs: -+ # # -- TLS behavior for the TLS session initiated by the client. See [TLSModeType](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.TLSModeType). -+ # mode: - - gatewayClass: - # -- When providers.kubernetesGateway.enabled and gateway.enabled, deploy a default gatewayClass -@@ -279,10 +281,6 @@ providers: - # labelSelector: environment=production,method=traefik - # -- Array of namespaces to watch. If left empty, Traefik watches all namespaces. - namespaces: [] -- # - "default" -- # Disable cluster IngressClass Lookup - Requires Traefik V3. -- # When combined with rbac.namespaced: true, ClusterRole will not be created and ingresses must use kubernetes.io/ingress.class annotation instead of spec.ingressClassName. -- disableIngressClassLookup: false - # IP used for Kubernetes Ingress endpoints - publishedService: - enabled: false -@@ -836,9 +834,12 @@ hostNetwork: false - # -- Whether Role Based Access Control objects like roles and rolebindings should be created - rbac: - enabled: true -- # If set to false, installs ClusterRole and ClusterRoleBinding so Traefik can be used across namespaces. -- # If set to true, installs Role and RoleBinding instead of ClusterRole/ClusterRoleBinding. Providers will only watch target namespace. -- # When combined with providers.kubernetesIngress.disableIngressClassLookup: true and Traefik V3, ClusterRole to watch IngressClass is also disabled. -+ # When set to true: -+ # 1. Use `Role` and `RoleBinding` instead of `ClusterRole` and `ClusterRoleBinding`. -+ # 2. Set `disableIngressClassLookup` on Kubernetes Ingress providers with Traefik Proxy v3 until v3.1.1 -+ # 3. Set `disableClusterScopeResources` on Kubernetes Ingress and CRD providers with Traefik Proxy v3.1.2+ -+ # **NOTE**: `IngressClass`, `NodePortLB` and **Gateway** provider cannot be used with namespaced RBAC. -+ # See [upstream documentation](https://doc.traefik.io/traefik/providers/kubernetes-ingress/#disableclusterscoperesources) for more details. - namespaced: false - # Enable user-facing roles - # https://kubernetes.io/docs/reference/access-authn-authz/rbac/#user-facing-roles -``` - -## 30.0.2 ![AppVersion: v3.1.0](https://img.shields.io/static/v1?label=AppVersion&message=v3.1.0&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-07-30 - -* fix(Traefik Hub): missing RBACs for Traefik Hub -* chore(release): 🚀 publish v30.0.2 - -## 30.0.1 ![AppVersion: v3.1.0](https://img.shields.io/static/v1?label=AppVersion&message=v3.1.0&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-07-29 - -* fix(Traefik Hub): support new RBACs for upcoming traefik hub release -* fix(Traefik Hub): RBACs missing with API Gateway -* feat: :release: v30.0.1 - -## 30.0.0 ![AppVersion: v3.1.0](https://img.shields.io/static/v1?label=AppVersion&message=v3.1.0&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-07-24 - -* fix: 🐛 ingressroute default name -* fix: namespaced RBACs hub api gateway -* fix: can't set gateway name -* fix(Gateway API): provide expected roles when using namespaced RBAC -* fix(Gateway API)!: revamp Gateway implementation -* feat: ✨ display release name and image full path in installation notes -* feat: use single ingressRoute template -* feat: handle log filePath and noColor -* chore(release): 🚀 publish v30.0.0 -* chore(deps): update traefik docker tag to v3.1.0 - -**Upgrade Notes** - -There is a breaking upgrade on how to configure Gateway with _values_. -This release supports Traefik Proxy v3.0 **and** v3.1. - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index c8bfd5b..83b6d98 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -134,14 +134,36 @@ gateway: - enabled: true - # -- Set a custom name to gateway - name: -- # -- Routes are restricted to namespace of the gateway [by default](https://gateway-api.sigs.k8s.io/references/spec/#gateway.networking.k8s.io/v1beta1.FromNamespaces) -- namespacePolicy: -- # -- See [GatewayTLSConfig](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io%2fv1.GatewayTLSConfig) -- certificateRefs: - # -- By default, Gateway is created in the same `Namespace` than Traefik. - namespace: - # -- Additional gateway annotations (e.g. for cert-manager.io/issuer) - annotations: -+ # -- Define listeners -+ listeners: -+ web: -+ # -- Port is the network port. Multiple listeners may use the same port, subject to the Listener compatibility rules. -+ # The port must match a port declared in ports section. -+ port: 8000 -+ # -- Optional hostname. See [Hostname](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.Hostname) -+ hostname: -+ # Specify expected protocol on this listener. See [ProtocolType](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.ProtocolType) -+ protocol: HTTP -+ # -- Routes are restricted to namespace of the gateway [by default](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.FromNamespaces -+ namespacePolicy: -+ websecure: -+ # -- Port is the network port. Multiple listeners may use the same port, subject to the Listener compatibility rules. -+ # The port must match a port declared in ports section. -+ port: 8443 -+ # -- Optional hostname. See [Hostname](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.Hostname) -+ hostname: -+ # Specify expected protocol on this listener See [ProtocolType](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.ProtocolType) -+ protocol: HTTPS -+ # -- Routes are restricted to namespace of the gateway [by default](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.FromNamespaces) -+ namespacePolicy: -+ # -- Add certificates for TLS or HTTPS protocols. See [GatewayTLSConfig](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io%2fv1.GatewayTLSConfig) -+ certificateRefs: -+ # -- TLS behavior for the TLS session initiated by the client. See [TLSModeType](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.TLSModeType). -+ mode: - - gatewayClass: - # -- When providers.kubernetesGateway.enabled and gateway.enabled, deploy a default gatewayClass -@@ -161,6 +183,10 @@ ingressRoute: - labels: {} - # -- The router match rule used for the dashboard ingressRoute - matchRule: PathPrefix(`/dashboard`) || PathPrefix(`/api`) -+ # -- The internal service used for the dashboard ingressRoute -+ services: -+ - name: api@internal -+ kind: TraefikService - # -- Specify the allowed entrypoints to use for the dashboard ingress route, (e.g. traefik, web, websecure). - # By default, it's using traefik entrypoint, which is not exposed. - # /!\ Do not expose your dashboard without any protection over the internet /!\ -@@ -178,6 +204,10 @@ ingressRoute: - labels: {} - # -- The router match rule used for the healthcheck ingressRoute - matchRule: PathPrefix(`/ping`) -+ # -- The internal service used for the healthcheck ingressRoute -+ services: -+ - name: ping@internal -+ kind: TraefikService - # -- Specify the allowed entrypoints to use for the healthcheck ingress route, (e.g. traefik, web, websecure). - # By default, it's using traefik entrypoint, which is not exposed. - entryPoints: ["traefik"] -@@ -307,9 +337,12 @@ logs: - # -- Set [logs format](https://doc.traefik.io/traefik/observability/logs/#format) - # @default common - format: -- # By default, the level is set to ERROR. -+ # By default, the level is set to INFO. - # -- Alternative logging levels are DEBUG, PANIC, FATAL, ERROR, WARN, and INFO. - level: INFO -+ # -+ # filePath: "/var/log/traefik/traefik.log -+ # noColor: true - access: - # -- To enable access logs - enabled: false -``` - - -## 29.0.1 ![AppVersion: v3.0.4](https://img.shields.io/static/v1?label=AppVersion&message=v3.0.4&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-07-09 - -* fix: semverCompare failing on some legitimate tags -* fix: RBACs for hub and disabled namespaced RBACs -* chore(release): 🚀 publish v29.0.1 -* chore(deps): update jnorwood/helm-docs docker tag to v1.14.0 - -## 29.0.0 ![AppVersion: v3.0.4](https://img.shields.io/static/v1?label=AppVersion&message=v3.0.4&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Upgrade Notes** - -This is a major breaking upgrade. [Migration guide](https://doc.traefik.io/traefik/v3.1/migration/v3/#v30-to-v31) from v3.0 to v3.1rc has been applied on this chart. - -This release supports both Traefik Proxy v3.0.x and v3.1rc. - -It comes with those breaking changes: - -- Far better support on Gateway API v1.1: Gateway, GatewayClass, CRDs & RBAC (#1107) -- Many changes on CRDs & RBAC (#1072 & #1108) -- Refactor on Prometheus Operator support. Values has changed (#1114) -- Dashboard `IngressRoute` is now disabled by default (#1111) - -CRDs needs to be upgraded: `kubectl apply --server-side --force-conflicts -k https://github.com/traefik/traefik-helm-chart/traefik/crds/` - -**Release date:** 2024-07-05 - -* fix: 🐛 improve error message on additional service without ports -* fix: allow multiples values in the `secretResourceNames` slice -* fix(rbac)!: nodes API permissions for Traefik v3.1+ -* fix(dashboard): Only set ingressClass annotation when kubernetesCRD provider is listening for it -* fix!: prometheus operator settings -* feat: ✨ update CRDs & RBAC for Traefik Proxy -* feat: ✨ migrate to endpointslices rbac -* feat: allow to set hostAliases for traefik pod -* feat(providers): add nativeLBByDefault support -* feat(providers)!: improve kubernetesGateway and Gateway API support -* feat(dashboard)!: dashboard `IngressRoute` should be disabled by default -* docs: fix typos and broken link -* chore: update CRDs to v1.5.0 -* chore: update CRDs to v1.4.0 -* chore(release): publish v29.0.0 -* chore(deps): update traefik docker tag to v3.0.4 -* chore(deps): update traefik docker tag to v3.0.3 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index e440dcf..c8bfd5b 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -8,7 +8,7 @@ image: - # -- Traefik image repository - repository: traefik - # -- defaults to appVersion -- tag: "" -+ tag: - # -- Traefik image pull policy - pullPolicy: IfNotPresent - -@@ -81,19 +81,12 @@ deployment: - shareProcessNamespace: false - # -- Custom pod DNS policy. Apply if `hostNetwork: true` - # dnsPolicy: ClusterFirstWithHostNet -+ # -- Custom pod [DNS config](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.30/#poddnsconfig-v1-core) - dnsConfig: {} -- # nameservers: -- # - 192.0.2.1 # this is an example -- # searches: -- # - ns1.svc.cluster-domain.example -- # - my.dns.search.suffix -- # options: -- # - name: ndots -- # value: "2" -- # - name: edns0 -- # -- Additional imagePullSecrets -+ # -- Custom [host aliases](https://kubernetes.io/docs/tasks/network/customize-hosts-file-for-pods/) -+ hostAliases: [] -+ # -- Pull secret for fetching traefik container image - imagePullSecrets: [] -- # - name: myRegistryKeySecretName - # -- Pod lifecycle actions - lifecycle: {} - # preStop: -@@ -135,24 +128,33 @@ experimental: - kubernetesGateway: - # -- Enable traefik experimental GatewayClass CRD - enabled: false -- ## Routes are restricted to namespace of the gateway by default. -- ## https://gateway-api.sigs.k8s.io/references/spec/#gateway.networking.k8s.io/v1beta1.FromNamespaces -- # namespacePolicy: All -- # certificate: -- # group: "core" -- # kind: "Secret" -- # name: "mysecret" -- # -- By default, Gateway would be created to the Namespace you are deploying Traefik to. -- # You may create that Gateway in another namespace, setting its name below: -- # namespace: default -- # Additional gateway annotations (e.g. for cert-manager.io/issuer) -- # annotations: -- # cert-manager.io/issuer: letsencrypt -+ -+gateway: -+ # -- When providers.kubernetesGateway.enabled, deploy a default gateway -+ enabled: true -+ # -- Set a custom name to gateway -+ name: -+ # -- Routes are restricted to namespace of the gateway [by default](https://gateway-api.sigs.k8s.io/references/spec/#gateway.networking.k8s.io/v1beta1.FromNamespaces) -+ namespacePolicy: -+ # -- See [GatewayTLSConfig](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io%2fv1.GatewayTLSConfig) -+ certificateRefs: -+ # -- By default, Gateway is created in the same `Namespace` than Traefik. -+ namespace: -+ # -- Additional gateway annotations (e.g. for cert-manager.io/issuer) -+ annotations: -+ -+gatewayClass: -+ # -- When providers.kubernetesGateway.enabled and gateway.enabled, deploy a default gatewayClass -+ enabled: true -+ # -- Set a custom name to GatewayClass -+ name: -+ # -- Additional gatewayClass labels (e.g. for filtering gateway objects by custom labels) -+ labels: - - ingressRoute: - dashboard: - # -- Create an IngressRoute for the dashboard -- enabled: true -+ enabled: false - # -- Additional ingressRoute annotations (e.g. for kubernetes.io/ingress.class) - annotations: {} - # -- Additional ingressRoute labels (e.g. for filtering IngressRoute by custom labels) -@@ -227,11 +229,13 @@ providers: - allowExternalNameServices: false - # -- Allows to return 503 when there is no endpoints available - allowEmptyServices: false -- # ingressClass: traefik-internal -+ # -- When the parameter is set, only resources containing an annotation with the same value are processed. Otherwise, resources missing the annotation, having an empty value, or the value traefik are processed. It will also set required annotation on Dashboard and Healthcheck IngressRoute when enabled. -+ ingressClass: - # labelSelector: environment=production,method=traefik - # -- Array of namespaces to watch. If left empty, Traefik watches all namespaces. - namespaces: [] -- # - "default" -+ # -- Defines whether to use Native Kubernetes load-balancing mode by default. -+ nativeLBByDefault: - - kubernetesIngress: - # -- Load Kubernetes Ingress provider -@@ -240,7 +244,8 @@ providers: - allowExternalNameServices: false - # -- Allows to return 503 when there is no endpoints available - allowEmptyServices: false -- # ingressClass: traefik-internal -+ # -- When ingressClass is set, only Ingresses containing an annotation with the same value are processed. Otherwise, Ingresses missing the annotation, having an empty value, or the value traefik are processed. -+ ingressClass: - # labelSelector: environment=production,method=traefik - # -- Array of namespaces to watch. If left empty, Traefik watches all namespaces. - namespaces: [] -@@ -254,6 +259,19 @@ providers: - # Published Kubernetes Service to copy status from. Format: namespace/servicename - # By default this Traefik service - # pathOverride: "" -+ # -- Defines whether to use Native Kubernetes load-balancing mode by default. -+ nativeLBByDefault: -+ -+ kubernetesGateway: -+ # -- Enable Traefik Gateway provider for Gateway API -+ enabled: false -+ # -- Toggles support for the Experimental Channel resources (Gateway API release channels documentation). -+ # This option currently enables support for TCPRoute and TLSRoute. -+ experimentalChannel: false -+ # -- Array of namespaces to watch. If left empty, Traefik watches all namespaces. -+ namespaces: [] -+ # -- A label selector can be defined to filter on specific GatewayClass objects only. -+ labelselector: - - file: - # -- Create a file provider -@@ -341,6 +359,34 @@ metrics: - ## When manualRouting is true, it disables the default internal router in - ## order to allow creating a custom router for prometheus@internal service. - # manualRouting: true -+ service: -+ # -- Create a dedicated metrics service to use with ServiceMonitor -+ enabled: -+ labels: -+ annotations: -+ # -- When set to true, it won't check if Prometheus Operator CRDs are deployed -+ disableAPICheck: -+ serviceMonitor: -+ # -- Enable optional CR for Prometheus Operator. See EXAMPLES.md for more details. -+ enabled: false -+ metricRelabelings: -+ relabelings: -+ jobLabel: -+ interval: -+ honorLabels: -+ scrapeTimeout: -+ honorTimestamps: -+ enableHttp2: -+ followRedirects: -+ additionalLabels: -+ namespace: -+ namespaceSelector: -+ prometheusRule: -+ # -- Enable optional CR for Prometheus Operator. See EXAMPLES.md for more details. -+ enabled: false -+ additionalLabels: -+ namespace: -+ - # datadog: - # ## Address instructs exporter to send metrics to datadog-agent at this address. - # address: "127.0.0.1:8125" -@@ -436,55 +482,6 @@ metrics: - # -- When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. - insecureSkipVerify: - -- ## -- enable optional CRDs for Prometheus Operator -- ## -- ## Create a dedicated metrics service for use with ServiceMonitor -- # service: -- # enabled: false -- # labels: {} -- # annotations: {} -- ## When set to true, it won't check if Prometheus Operator CRDs are deployed -- # disableAPICheck: false -- # serviceMonitor: -- # metricRelabelings: [] -- # - sourceLabels: [__name__] -- # separator: ; -- # regex: ^fluentd_output_status_buffer_(oldest|newest)_.+ -- # replacement: $1 -- # action: drop -- # relabelings: [] -- # - sourceLabels: [__meta_kubernetes_pod_node_name] -- # separator: ; -- # regex: ^(.*)$ -- # targetLabel: nodename -- # replacement: $1 -- # action: replace -- # jobLabel: traefik -- # interval: 30s -- # honorLabels: true -- # # (Optional) -- # # scrapeTimeout: 5s -- # # honorTimestamps: true -- # # enableHttp2: true -- # # followRedirects: true -- # # additionalLabels: -- # # foo: bar -- # # namespace: "another-namespace" -- # # namespaceSelector: {} -- # prometheusRule: -- # additionalLabels: {} -- # namespace: "another-namespace" -- # rules: -- # - alert: TraefikDown -- # expr: up{job="traefik"} == 0 -- # for: 5m -- # labels: -- # context: traefik -- # severity: warning -- # annotations: -- # summary: "Traefik Down" -- # description: "{{ $labels.pod }} on {{ $labels.nodename }} is down" -- - ## Tracing - # -- https://doc.traefik.io/traefik/observability/tracing/overview/ - tracing: -``` - -## 28.3.0 ![AppVersion: v3.0.2](https://img.shields.io/static/v1?label=AppVersion&message=v3.0.2&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-06-14 - -* fix: 🐛 namespaced rbac when kubernetesIngress provider is disabled -* fix: 🐛 add divisor: '1' to GOMAXPROCS and GOMEMLIMIT -* fix(security): 🐛 🔒️ mount service account token on pod level -* fix(Traefik Hub): remove obsolete CRD -* fix(Traefik Hub): remove namespace in mutating webhook -* feat: allow setting permanent on redirectTo -* chore(release): publish v28.3.0 -* chore(deps): update traefik docker tag to v3.0.2 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index c558c78..e440dcf 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -602,6 +602,7 @@ ports: - # port: websecure - # (Optional) - # priority: 10 -+ # permanent: true - # - # -- Trust forwarded headers information (X-Forwarded-*). - # forwardedHeaders: -``` - -## 28.2.0 ![AppVersion: v3.0.1](https://img.shields.io/static/v1?label=AppVersion&message=v3.0.1&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-05-28 - -* fix(IngressClass): provides annotation on IngressRoutes when it's enabled -* feat: ✨ simplify values and provide more examples -* feat: add deletecollection right on secrets -* chore(release): 🚀 publish v28.2.0 -* chore(deps): update traefik docker tag to v3.0.1 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 2fd9282..c558c78 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1,4 +1,7 @@ - # Default values for Traefik -+# This is a YAML-formatted file. -+# Declare variables to be passed into templates -+ - image: - # -- Traefik image host registry - registry: docker.io -@@ -12,9 +15,6 @@ image: - # -- Add additional label to all resources - commonLabels: {} - --# --# Configure the deployment --# - deployment: - # -- Enable deployment - enabled: true -@@ -74,10 +74,6 @@ deployment: - # - name: volume-permissions - # image: busybox:latest - # command: ["sh", "-c", "touch /data/acme.json; chmod -v 600 /data/acme.json"] -- # securityContext: -- # runAsNonRoot: true -- # runAsGroup: 65532 -- # runAsUser: 65532 - # volumeMounts: - # - name: data - # mountPath: /data -@@ -112,13 +108,11 @@ deployment: - # -- Set a runtimeClassName on pod - runtimeClassName: - --# -- Pod disruption budget -+# -- [Pod Disruption Budget](https://kubernetes.io/docs/reference/kubernetes-api/policy-resources/pod-disruption-budget-v1/) - podDisruptionBudget: -- enabled: false -- # maxUnavailable: 1 -- # maxUnavailable: 33% -- # minAvailable: 0 -- # minAvailable: 25% -+ enabled: -+ maxUnavailable: -+ minAvailable: - - # -- Create a default IngressClass for Traefik - ingressClass: -@@ -155,7 +149,6 @@ experimental: - # annotations: - # cert-manager.io/issuer: letsencrypt - --## Create an IngressRoute for the dashboard - ingressRoute: - dashboard: - # -- Create an IngressRoute for the dashboard -@@ -221,15 +214,7 @@ livenessProbe: - # -- The number of seconds to wait for a probe response before considering it as failed. - timeoutSeconds: 2 - --# -- Define Startup Probe for container: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/#define-startup-probes --# eg. --# `startupProbe: --# exec: --# command: --# - mycommand --# - foo --# initialDelaySeconds: 5 --# periodSeconds: 5` -+# -- Define [Startup Probe](https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/#define-startup-probes) - startupProbe: - - providers: -@@ -276,18 +261,8 @@ providers: - # -- Allows Traefik to automatically watch for file changes - watch: true - # -- File content (YAML format, go template supported) (see https://doc.traefik.io/traefik/providers/file/) -- content: "" -- # http: -- # routers: -- # router0: -- # entryPoints: -- # - web -- # middlewares: -- # - my-basic-auth -- # service: service-foo -- # rule: Path(`/foo`) -+ content: - --# - # -- Add volumes to the traefik pod. The volume name will be passed to tpl. - # This can be used to mount a cert pair or a configmap that holds a config.toml file. - # After the volume has been mounted, add the configs into traefik by using the `additionalArguments` list below, eg: -@@ -311,26 +286,21 @@ additionalVolumeMounts: [] - - logs: - general: -- # -- By default, the logs use a text format (common), but you can -- # also ask for the json format in the format option -- # format: json -+ # -- Set [logs format](https://doc.traefik.io/traefik/observability/logs/#format) -+ # @default common -+ format: - # By default, the level is set to ERROR. - # -- Alternative logging levels are DEBUG, PANIC, FATAL, ERROR, WARN, and INFO. - level: INFO - access: - # -- To enable access logs - enabled: false -- ## By default, logs are written using the Common Log Format (CLF) on stdout. -- ## To write logs in JSON, use json in the format option. -- ## If the given format is unsupported, the default (CLF) is used instead. -- # format: json -+ # -- Set [access log format](https://doc.traefik.io/traefik/observability/access-logs/#format) -+ format: - # filePath: "/var/log/traefik/access.log -- ## To write the logs in an asynchronous fashion, specify a bufferingSize option. -- ## This option represents the number of log lines Traefik will keep in memory before writing -- ## them to the selected output. In some cases, this option can greatly help performances. -- # bufferingSize: 100 -- ## Filtering -- # -- https://docs.traefik.io/observability/access-logs/#filtering -+ # -- Set [bufferingSize](https://doc.traefik.io/traefik/observability/access-logs/#bufferingsize) -+ bufferingSize: -+ # -- Set [filtering](https://docs.traefik.io/observability/access-logs/#filtering) - filters: {} - # statuscodes: "200,300-302" - # retryattempts: true -@@ -345,15 +315,11 @@ logs: - names: {} - ## Examples: - # ClientUsername: drop -+ # -- [Limit logged fields or headers](https://doc.traefik.io/traefik/observability/access-logs/#limiting-the-fieldsincluding-headers) - headers: - # -- Available modes: keep, drop, redact. - defaultmode: drop -- # -- Names of the headers to limit. - names: {} -- ## Examples: -- # User-Agent: redact -- # Authorization: drop -- # Content-Type: keep - - metrics: - ## -- Enable metrics for internal resources. Default: false -@@ -567,16 +533,15 @@ globalArguments: - - "--global.checknewversion" - - "--global.sendanonymoususage" - --# --# Configure Traefik static configuration - # -- Additional arguments to be passed at Traefik's binary --# All available options available on https://docs.traefik.io/reference/static-configuration/cli/ --## Use curly braces to pass values: `helm install --set="additionalArguments={--providers.kubernetesingress.ingressclass=traefik-internal,--log.level=DEBUG}"` -+# See [CLI Reference](https://docs.traefik.io/reference/static-configuration/cli/) -+# Use curly braces to pass values: `helm install --set="additionalArguments={--providers.kubernetesingress.ingressclass=traefik-internal,--log.level=DEBUG}"` - additionalArguments: [] - # - "--providers.kubernetesingress.ingressclass=traefik-internal" - # - "--log.level=DEBUG" - - # -- Environment variables to be passed to Traefik's binary -+# @default -- See _values.yaml_ - env: - - name: POD_NAME - valueFrom: -@@ -586,25 +551,9 @@ env: - valueFrom: - fieldRef: - fieldPath: metadata.namespace --# - name: SOME_VAR --# value: some-var-value --# - name: SOME_VAR_FROM_CONFIG_MAP --# valueFrom: --# configMapRef: --# name: configmap-name --# key: config-key --# - name: SOME_SECRET --# valueFrom: --# secretKeyRef: --# name: secret-name --# key: secret-key - - # -- Environment variables to be passed to Traefik's binary from configMaps or secrets - envFrom: [] --# - configMapRef: --# name: config-map-name --# - secretRef: --# name: secret-name - - ports: - traefik: -@@ -766,28 +715,12 @@ ports: - # -- The port protocol (TCP/UDP) - protocol: TCP - --# -- TLS Options are created as TLSOption CRDs --# https://doc.traefik.io/traefik/https/tls/#tls-options -+# -- TLS Options are created as [TLSOption CRDs](https://doc.traefik.io/traefik/https/tls/#tls-options) - # When using `labelSelector`, you'll need to set labels on tlsOption accordingly. --# Example: --# tlsOptions: --# default: --# labels: {} --# sniStrict: true --# custom-options: --# labels: {} --# curvePreferences: --# - CurveP521 --# - CurveP384 -+# See EXAMPLE.md for details. - tlsOptions: {} - --# -- TLS Store are created as TLSStore CRDs. This is useful if you want to set a default certificate --# https://doc.traefik.io/traefik/https/tls/#default-certificate --# Example: --# tlsStore: --# default: --# defaultCertificate: --# secretName: tls-cert -+# -- TLS Store are created as [TLSStore CRDs](https://doc.traefik.io/traefik/https/tls/#default-certificate). This is useful if you want to set a default certificate. See EXAMPLE.md for details. - tlsStore: {} - - service: -@@ -839,29 +772,8 @@ service: - - autoscaling: - # -- Create HorizontalPodAutoscaler object. -+ # See EXAMPLES.md for more details. - enabled: false --# minReplicas: 1 --# maxReplicas: 10 --# metrics: --# - type: Resource --# resource: --# name: cpu --# target: --# type: Utilization --# averageUtilization: 60 --# - type: Resource --# resource: --# name: memory --# target: --# type: Utilization --# averageUtilization: 60 --# behavior: --# scaleDown: --# stabilizationWindowSeconds: 300 --# policies: --# - type: Pods --# value: 1 --# periodSeconds: 60 - - persistence: - # -- Enable persistence using Persistent Volume Claims -@@ -879,27 +791,10 @@ persistence: - # -- Only mount a subpath of the Volume into the pod - # subPath: "" - --# -- Certificates resolvers configuration -+# -- Certificates resolvers configuration. -+# Ref: https://doc.traefik.io/traefik/https/acme/#certificate-resolvers -+# See EXAMPLES.md for more details. - certResolvers: {} --# letsencrypt: --# # for challenge options cf. https://doc.traefik.io/traefik/https/acme/ --# email: email@example.com --# dnsChallenge: --# # also add the provider's required configuration under env --# # or expand then from secrets/configmaps with envfrom --# # cf. https://doc.traefik.io/traefik/https/acme/#providers --# provider: digitalocean --# # add futher options for the dns challenge as needed --# # cf. https://doc.traefik.io/traefik/https/acme/#dnschallenge --# delayBeforeCheck: 30 --# resolvers: --# - 1.1.1.1 --# - 8.8.8.8 --# tlsChallenge: true --# httpChallenge: --# entryPoint: "web" --# # It has to match the path with a persistent volume --# storage: /data/acme.json - - # -- If hostNetwork is true, runs traefik in the host network namespace - # To prevent unschedulabel pods due to port collisions, if hostNetwork=true -@@ -933,14 +828,8 @@ serviceAccount: - # -- Additional serviceAccount annotations (e.g. for oidc authentication) - serviceAccountAnnotations: {} - --# -- The resources parameter defines CPU and memory requirements and limits for Traefik's containers. -+# -- [Resources](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/) for `traefik` container. - resources: {} --# requests: --# cpu: "100m" --# memory: "50Mi" --# limits: --# cpu: "300m" --# memory: "150Mi" - - # -- This example pod anti-affinity forces the scheduler to put traefik pods - # -- on nodes where no other traefik pods are scheduled. -@@ -970,30 +859,22 @@ topologySpreadConstraints: [] - # topologyKey: kubernetes.io/hostname - # whenUnsatisfiable: DoNotSchedule - --# -- Pods can have priority. --# -- Priority indicates the importance of a Pod relative to other Pods. -+# -- [Pod Priority and Preemption](https://kubernetes.io/docs/concepts/scheduling-eviction/pod-priority-preemption/) - priorityClassName: "" - --# -- Set the container security context --# -- To run the container with ports below 1024 this will need to be adjusted to run as root -+# -- [SecurityContext](https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/pod-v1/#security-context-1) -+# @default -- See _values.yaml_ - securityContext: -+ allowPrivilegeEscalation: false - capabilities: - drop: [ALL] - readOnlyRootFilesystem: true -- allowPrivilegeEscalation: false - -+# -- [Pod Security Context](https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/pod-v1/#security-context) -+# @default -- See _values.yaml_ - podSecurityContext: -- # /!\ When setting fsGroup, Kubernetes will recursively change ownership and -- # permissions for the contents of each volume to match the fsGroup. This can -- # be an issue when storing sensitive content like TLS Certificates /!\ -- # fsGroup: 65532 -- # -- Specifies the policy for changing ownership and permissions of volume contents to match the fsGroup. -- fsGroupChangePolicy: "OnRootMismatch" -- # -- The ID of the group for all containers in the pod to run as. - runAsGroup: 65532 -- # -- Specifies whether the containers should run as a non-root user. - runAsNonRoot: true -- # -- The ID of the user for all containers in the pod to run as. - runAsUser: 65532 - - # -@@ -1003,16 +884,16 @@ podSecurityContext: - # See #595 for more details and traefik/tests/values/extra.yaml for example. - extraObjects: [] - --# This will override the default Release Namespace for Helm. -+# -- This field override the default Release Namespace for Helm. - # It will not affect optional CRDs such as `ServiceMonitor` and `PrometheusRules` --# namespaceOverride: traefik --# --## -- This will override the default app.kubernetes.io/instance label for all Objects. --# instanceLabelOverride: traefik -+namespaceOverride: -+ -+## -- This field override the default app.kubernetes.io/instance label for all Objects. -+instanceLabelOverride: - --# -- Traefik Hub configuration. See https://doc.traefik.io/traefik-hub/ -+# Traefik Hub configuration. See https://doc.traefik.io/traefik-hub/ - hub: -- # Name of Secret with key 'token' set to a valid license token. -+ # -- Name of `Secret` with key 'token' set to a valid license token. - # It enables API Gateway. - token: - apimanagement: -``` - -## 28.1.0 ![AppVersion: v3.0.0](https://img.shields.io/static/v1?label=AppVersion&message=v3.0.0&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -* fix(Traefik Hub): do not deploy mutating webhook when enabling only API Gateway -* feat(Traefik Hub): use Traefik Proxy otlp config -* chore: 🔧 update Traefik Hub CRD to v1.3.3 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 70297f6..2fd9282 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1010,3 +1010,49 @@ - ## -- This will override the default app.kubernetes.io/instance label for all Objects. - # instanceLabelOverride: traefik - -+# -- Traefik Hub configuration. See https://doc.traefik.io/traefik-hub/ -+hub: -+ # Name of Secret with key 'token' set to a valid license token. -+ # It enables API Gateway. -+ token: -+ apimanagement: -+ # -- Set to true in order to enable API Management. Requires a valid license token. -+ enabled: -+ admission: -+ # -- WebHook admission server listen address. Default: "0.0.0.0:9943". -+ listenAddr: -+ # -- Certificate of the WebHook admission server. Default: "hub-agent-cert". -+ secretName: -+ -+ ratelimit: -+ redis: -+ # -- Enable Redis Cluster. Default: true. -+ cluster: -+ # -- Database used to store information. Default: "0". -+ database: -+ # -- Endpoints of the Redis instances to connect to. Default: "". -+ endpoints: -+ # -- The username to use when connecting to Redis endpoints. Default: "". -+ username: -+ # -- The password to use when connecting to Redis endpoints. Default: "". -+ password: -+ sentinel: -+ # -- Name of the set of main nodes to use for main selection. Required when using Sentinel. Default: "". -+ masterset: -+ # -- Username to use for sentinel authentication (can be different from endpoint username). Default: "". -+ username: -+ # -- Password to use for sentinel authentication (can be different from endpoint password). Default: "". -+ password: -+ # -- Timeout applied on connection with redis. Default: "0s". -+ timeout: -+ tls: -+ # -- Path to the certificate authority used for the secured connection. -+ ca: -+ # -- Path to the public certificate used for the secure connection. -+ cert: -+ # -- Path to the private key used for the secure connection. -+ key: -+ # -- When insecureSkipVerify is set to true, the TLS connection accepts any certificate presented by the server. Default: false. -+ insecureSkipVerify: -+ # Enable export of errors logs to the platform. Default: true. -+ sendlogs: -``` - -## 28.1.0-beta.3 ![AppVersion: v3.0.0](https://img.shields.io/static/v1?label=AppVersion&message=v3.0.0&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-05-03 - -* chore: 🔧 update Traefik Hub CRD to v1.3.2 -* chore(release): 🚀 publish v28.1.0-beta.3 - -## 28.1.0-beta.2 ![AppVersion: v3.0.0](https://img.shields.io/static/v1?label=AppVersion&message=v3.0.0&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-05-02 - -* fix: 🐛 refine Traefik Hub support -* chore(release): 🚀 publish v28.1.0-beta.2 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index ce0a7a3..70297f6 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1015,13 +1015,15 @@ hub: - # Name of Secret with key 'token' set to a valid license token. - # It enables API Gateway. - token: -- admission: -- # -- WebHook admission server listen address. Default: "0.0.0.0:9943". -- listenAddr: -- # -- Certificate of the WebHook admission server. Default: "hub-agent-cert". -- secretName: -- # -- Set to true in order to enable API Management. Requires a valid license token. - apimanagement: -+ # -- Set to true in order to enable API Management. Requires a valid license token. -+ enabled: -+ admission: -+ # -- WebHook admission server listen address. Default: "0.0.0.0:9943". -+ listenAddr: -+ # -- Certificate of the WebHook admission server. Default: "hub-agent-cert". -+ secretName: -+ - metrics: - opentelemetry: - # -- Set to true to enable OpenTelemetry metrics exporter of Traefik Hub. -``` - -## 28.1.0-beta.1 ![AppVersion: v3.0.0](https://img.shields.io/static/v1?label=AppVersion&message=v3.0.0&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-04-30 - -* feat: :rocket: add initial support for Traefik Hub Api Gateway -* chore(release): 🚀 publish v28.1.0-beta.1 - -## 28.0.0 ![AppVersion: v3.0.0](https://img.shields.io/static/v1?label=AppVersion&message=v3.0.0&color=success&logo=) ![Kubernetes: >=1.22.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.22.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-04-30 - -* style: 🎨 consistent capitalization on `--entryPoints` CLI flag -* fix: 🐛 only expose http3 port on service when TCP variant is exposed -* fix: 🐛 logs filters on status codes -* feat: ✨ add support of `experimental-v3.0` unstable version -* feat: ability to override liveness and readiness probe paths -* feat(ports): add transport options -* chore(release): publish v28.0.0 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index c0d72d8..2bff10d 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -38,6 +38,12 @@ deployment: - ## Override the liveness/readiness scheme. Useful for getting ping to - ## respond on websecure entryPoint. - # healthchecksScheme: HTTPS -+ ## Override the readiness path. -+ ## Default: /ping -+ # readinessPath: /ping -+ # Override the liveness path. -+ # Default: /ping -+ # livenessPath: /ping - # -- Additional deployment annotations (e.g. for jaeger-operator sidecar injection) - annotations: {} - # -- Additional deployment labels (e.g. for filtering deployment by custom labels) -@@ -648,15 +654,28 @@ ports: - # (Optional) - # priority: 10 - # -- # Trust forwarded headers information (X-Forwarded-*). -+ # -- Trust forwarded headers information (X-Forwarded-*). - # forwardedHeaders: - # trustedIPs: [] - # insecure: false - # -- # Enable the Proxy Protocol header parsing for the entry point -+ # -- Enable the Proxy Protocol header parsing for the entry point - # proxyProtocol: - # trustedIPs: [] - # insecure: false -+ # -+ # -- Set transport settings for the entrypoint; see also -+ # https://doc.traefik.io/traefik/routing/entrypoints/#transport -+ transport: -+ respondingTimeouts: -+ readTimeout: -+ writeTimeout: -+ idleTimeout: -+ lifeCycle: -+ requestAcceptGraceTimeout: -+ graceTimeOut: -+ keepAliveMaxRequests: -+ keepAliveMaxTime: - websecure: - ## -- Enable this entrypoint as a default entrypoint. When a service doesn't explicitly set an entrypoint it will only use this entrypoint. - # asDefault: true -@@ -684,16 +703,29 @@ ports: - enabled: false - # advertisedPort: 4443 - # -- ## -- Trust forwarded headers information (X-Forwarded-*). -+ # -- Trust forwarded headers information (X-Forwarded-*). - # forwardedHeaders: - # trustedIPs: [] - # insecure: false - # -- ## -- Enable the Proxy Protocol header parsing for the entry point -+ # -- Enable the Proxy Protocol header parsing for the entry point - # proxyProtocol: - # trustedIPs: [] - # insecure: false - # -+ # -- Set transport settings for the entrypoint; see also -+ # https://doc.traefik.io/traefik/routing/entrypoints/#transport -+ transport: -+ respondingTimeouts: -+ readTimeout: -+ writeTimeout: -+ idleTimeout: -+ lifeCycle: -+ requestAcceptGraceTimeout: -+ graceTimeOut: -+ keepAliveMaxRequests: -+ keepAliveMaxTime: -+ # - ## Set TLS at the entrypoint - ## https://doc.traefik.io/traefik/routing/entrypoints/#tls - tls: -``` - -## 28.0.0-rc1 ![AppVersion: v3.0.0-rc5](https://img.shields.io/static/v1?label=AppVersion&message=v3.0.0-rc5&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-04-17 - -**Upgrade Notes** - -This is a major breaking upgrade. [Migration guide](https://doc.traefik.io/traefik/v3.0/migration/v2-to-v3/) have been applied on the chart. - -It needs a Kubernetes v1.22 or higher. -All CRDs using _API Group_ `traefik.containo.us` are not supported anymore in Traefik Proxy v3 - -CRDs needs to be upgraded: `kubectl apply --server-side --force-conflicts -k https://github.com/traefik/traefik-helm-chart/traefik/crds/` - -After upgrade, CRDs with _API Group_ `traefik.containo.us` can be removed: - -```shell -kubectl delete crds \ - ingressroutes.traefik.containo.us \ - ingressroutetcps.traefik.containo.us \ - ingressrouteudps.traefik.containo.us \ - middlewares.traefik.containo.us \ - middlewaretcps.traefik.containo.us \ - serverstransports.traefik.containo.us \ - tlsoptions.traefik.containo.us \ - tlsstores.traefik.containo.us \ - traefikservices.traefik.containo.us -``` - -**Changes** - -* feat(podtemplate): set GOMEMLIMIT, GOMAXPROCS when limits are defined -* feat: ✨ fail gracefully when required port number is not set -* feat!: :boom: initial support of Traefik Proxy v3 -* docs: 📚️ improve EXAMPLES on acme resolver -* chore(release): 🚀 publish v28 rc1 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index cd9fb6e..c0d72d8 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -120,12 +120,13 @@ ingressClass: - isDefaultClass: true - # name: my-custom-class - -+core: -+ # -- Can be used to use globally v2 router syntax -+ # See https://doc.traefik.io/traefik/v3.0/migration/v2-to-v3/#new-v3-syntax-notable-changes -+ defaultRuleSyntax: -+ - # Traefik experimental features - experimental: -- # This value is no longer used, set the image.tag to a semver higher than 3.0, e.g. "v3.0.0-beta3" -- # v3: -- # -- Enable traefik version 3 -- - # -- Enable traefik experimental plugins - plugins: {} - # demo: -@@ -309,7 +310,7 @@ logs: - # format: json - # By default, the level is set to ERROR. - # -- Alternative logging levels are DEBUG, PANIC, FATAL, ERROR, WARN, and INFO. -- level: ERROR -+ level: INFO - access: - # -- To enable access logs - enabled: false -@@ -328,6 +329,8 @@ logs: - # statuscodes: "200,300-302" - # retryattempts: true - # minduration: 10ms -+ # -- Enables accessLogs for internal resources. Default: false. -+ addInternals: - fields: - general: - # -- Available modes: keep, drop, redact. -@@ -347,6 +350,9 @@ logs: - # Content-Type: keep - - metrics: -+ ## -- Enable metrics for internal resources. Default: false -+ addInternals: -+ - ## -- Prometheus is enabled by default. - ## -- It can be disabled by setting "prometheus: null" - prometheus: -@@ -376,31 +382,6 @@ metrics: - # # addRoutersLabels: true - # ## Enable metrics on services. Default=true - # # addServicesLabels: false -- # influxdb: -- # ## Address instructs exporter to send metrics to influxdb at this address. -- # address: localhost:8089 -- # ## InfluxDB's address protocol (udp or http). Default="udp" -- # protocol: udp -- # ## InfluxDB database used when protocol is http. Default="" -- # # database: "" -- # ## InfluxDB retention policy used when protocol is http. Default="" -- # # retentionPolicy: "" -- # ## InfluxDB username (only with http). Default="" -- # # username: "" -- # ## InfluxDB password (only with http). Default="" -- # # password: "" -- # ## The interval used by the exporter to push metrics to influxdb. Default=10s -- # # pushInterval: 30s -- # ## Additional labels (influxdb tags) on all metrics. -- # # additionalLabels: -- # # env: production -- # # foo: bar -- # ## Enable metrics on entry points. Default=true -- # # addEntryPointsLabels: false -- # ## Enable metrics on routers. Default=false -- # # addRoutersLabels: true -- # ## Enable metrics on services. Default=true -- # # addServicesLabels: false - # influxdb2: - # ## Address instructs exporter to send metrics to influxdb v2 at this address. - # address: localhost:8086 -@@ -435,43 +416,53 @@ metrics: - # # addRoutersLabels: true - # ## Enable metrics on services. Default=true - # # addServicesLabels: false -- # openTelemetry: -- # ## Address of the OpenTelemetry Collector to send metrics to. -- # address: "localhost:4318" -- # ## Enable metrics on entry points. -- # addEntryPointsLabels: true -- # ## Enable metrics on routers. -- # addRoutersLabels: true -- # ## Enable metrics on services. -- # addServicesLabels: true -- # ## Explicit boundaries for Histogram data points. -- # explicitBoundaries: -- # - "0.1" -- # - "0.3" -- # - "1.2" -- # - "5.0" -- # ## Additional headers sent with metrics by the reporter to the OpenTelemetry Collector. -- # headers: -- # foo: bar -- # test: test -- # ## Allows reporter to send metrics to the OpenTelemetry Collector without using a secured protocol. -- # insecure: true -- # ## Interval at which metrics are sent to the OpenTelemetry Collector. -- # pushInterval: 10s -- # ## Allows to override the default URL path used for sending metrics. This option has no effect when using gRPC transport. -- # path: /foo/v1/traces -- # ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. -- # tls: -- # ## The path to the certificate authority, it defaults to the system bundle. -- # ca: path/to/ca.crt -- # ## The path to the public certificate. When using this option, setting the key option is required. -- # cert: path/to/foo.cert -- # ## The path to the private key. When using this option, setting the cert option is required. -- # key: path/to/key.key -- # ## If set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. -- # insecureSkipVerify: true -- # ## This instructs the reporter to send metrics to the OpenTelemetry Collector using gRPC. -- # grpc: true -+ otlp: -+ # -- Set to true in order to enable the OpenTelemetry metrics -+ enabled: false -+ # -- Enable metrics on entry points. Default: true -+ addEntryPointsLabels: -+ # -- Enable metrics on routers. Default: false -+ addRoutersLabels: -+ # -- Enable metrics on services. Default: true -+ addServicesLabels: -+ # -- Explicit boundaries for Histogram data points. Default: [.005, .01, .025, .05, .1, .25, .5, 1, 2.5, 5, 10] -+ explicitBoundaries: -+ # -- Interval at which metrics are sent to the OpenTelemetry Collector. Default: 10s -+ pushInterval: -+ http: -+ # -- Set to true in order to send metrics to the OpenTelemetry Collector using HTTP. -+ enabled: false -+ # -- Format: ://:. Default: http://localhost:4318/v1/metrics -+ endpoint: -+ # -- Additional headers sent with metrics by the reporter to the OpenTelemetry Collector. -+ headers: -+ ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. -+ tls: -+ # -- The path to the certificate authority, it defaults to the system bundle. -+ ca: -+ # -- The path to the public certificate. When using this option, setting the key option is required. -+ cert: -+ # -- The path to the private key. When using this option, setting the cert option is required. -+ key: -+ # -- When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. -+ insecureSkipVerify: -+ grpc: -+ # -- Set to true in order to send metrics to the OpenTelemetry Collector using gRPC -+ enabled: false -+ # -- Format: ://:. Default: http://localhost:4318/v1/metrics -+ endpoint: -+ # -- Allows reporter to send metrics to the OpenTelemetry Collector without using a secured protocol. -+ insecure: -+ ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. -+ tls: -+ # -- The path to the certificate authority, it defaults to the system bundle. -+ ca: -+ # -- The path to the public certificate. When using this option, setting the key option is required. -+ cert: -+ # -- The path to the private key. When using this option, setting the cert option is required. -+ key: -+ # -- When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. -+ insecureSkipVerify: - - ## -- enable optional CRDs for Prometheus Operator - ## -@@ -524,51 +515,46 @@ metrics: - - ## Tracing - # -- https://doc.traefik.io/traefik/observability/tracing/overview/ --tracing: {} --# openTelemetry: # traefik v3+ only --# grpc: true --# insecure: true --# address: localhost:4317 --# instana: --# localAgentHost: 127.0.0.1 --# localAgentPort: 42699 --# logLevel: info --# enableAutoProfile: true --# datadog: --# localAgentHostPort: 127.0.0.1:8126 --# debug: false --# globalTag: "" --# prioritySampling: false --# jaeger: --# samplingServerURL: http://localhost:5778/sampling --# samplingType: const --# samplingParam: 1.0 --# localAgentHostPort: 127.0.0.1:6831 --# gen128Bit: false --# propagation: jaeger --# traceContextHeaderName: uber-trace-id --# disableAttemptReconnecting: true --# collector: --# endpoint: "" --# user: "" --# password: "" --# zipkin: --# httpEndpoint: http://localhost:9411/api/v2/spans --# sameSpan: false --# id128Bit: true --# sampleRate: 1.0 --# haystack: --# localAgentHost: 127.0.0.1 --# localAgentPort: 35000 --# globalTag: "" --# traceIDHeaderName: "" --# parentIDHeaderName: "" --# spanIDHeaderName: "" --# baggagePrefixHeaderName: "" --# elastic: --# serverURL: http://localhost:8200 --# secretToken: "" --# serviceEnvironment: "" -+tracing: -+ # -- Enables tracing for internal resources. Default: false. -+ addInternals: -+ otlp: -+ # -- See https://doc.traefik.io/traefik/v3.0/observability/tracing/opentelemetry/ -+ enabled: false -+ http: -+ # -- Set to true in order to send metrics to the OpenTelemetry Collector using HTTP. -+ enabled: false -+ # -- Format: ://:. Default: http://localhost:4318/v1/metrics -+ endpoint: -+ # -- Additional headers sent with metrics by the reporter to the OpenTelemetry Collector. -+ headers: -+ ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. -+ tls: -+ # -- The path to the certificate authority, it defaults to the system bundle. -+ ca: -+ # -- The path to the public certificate. When using this option, setting the key option is required. -+ cert: -+ # -- The path to the private key. When using this option, setting the cert option is required. -+ key: -+ # -- When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. -+ insecureSkipVerify: -+ grpc: -+ # -- Set to true in order to send metrics to the OpenTelemetry Collector using gRPC -+ enabled: false -+ # -- Format: ://:. Default: http://localhost:4318/v1/metrics -+ endpoint: -+ # -- Allows reporter to send metrics to the OpenTelemetry Collector without using a secured protocol. -+ insecure: -+ ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. -+ tls: -+ # -- The path to the certificate authority, it defaults to the system bundle. -+ ca: -+ # -- The path to the public certificate. When using this option, setting the key option is required. -+ cert: -+ # -- The path to the private key. When using this option, setting the cert option is required. -+ key: -+ # -- When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. -+ insecureSkipVerify: - - # -- Global command arguments to be passed to all traefik's pods - globalArguments: -@@ -756,7 +742,6 @@ ports: - # default: - # labels: {} - # sniStrict: true --# preferServerCipherSuites: true - # custom-options: - # labels: {} - # curvePreferences: -``` - -## 27.0.0 ![AppVersion: v2.11.0](https://img.shields.io/static/v1?label=AppVersion&message=v2.11.0&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-04-02 - -**Upgrade notes** - -Custom services and port exposure have been redesigned, requiring the following changes: -- if you were overriding port exposure behavior using the `expose` or `exposeInternal` flags, you should replace them with a service name to boolean mapping, i.e. replace this: - -```yaml -ports: - web: - expose: false - exposeInternal: true -``` - -with this: - -```yaml -ports: - web: - expose: - default: false - internal: true -``` - -- if you were previously using the `service.internal` value, you should migrate the values to the `service.additionalServices.internal` value instead; this should yield the same results, but make sure to carefully check for any changes! - -**Changes** - -* fix: remove null annotations on dashboard `IngressRoute` -* fix(rbac): do not create clusterrole for namespace deployment on Traefik v3 -* feat: restrict access to secrets -* feat!: :boom: refactor custom services and port exposure -* chore(release): 🚀 publish v27.0.0 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index dbd078f..363871d 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -250,6 +250,9 @@ providers: - # -- Array of namespaces to watch. If left empty, Traefik watches all namespaces. - namespaces: [] - # - "default" -+ # Disable cluster IngressClass Lookup - Requires Traefik V3. -+ # When combined with rbac.namespaced: true, ClusterRole will not be created and ingresses must use kubernetes.io/ingress.class annotation instead of spec.ingressClassName. -+ disableIngressClassLookup: false - # IP used for Kubernetes Ingress endpoints - publishedService: - enabled: false -@@ -626,22 +629,20 @@ ports: - # -- You SHOULD NOT expose the traefik port on production deployments. - # If you want to access it from outside your cluster, - # use `kubectl port-forward` or create a secure ingress -- expose: false -+ expose: -+ default: false - # -- The exposed port for this service - exposedPort: 9000 - # -- The port protocol (TCP/UDP) - protocol: TCP -- # -- Defines whether the port is exposed on the internal service; -- # note that ports exposed on the default service are exposed on the internal -- # service by default as well. -- exposeInternal: false - web: - ## -- Enable this entrypoint as a default entrypoint. When a service doesn't explicitly set an entrypoint it will only use this entrypoint. - # asDefault: true - port: 8000 - # hostPort: 8000 - # containerPort: 8000 -- expose: true -+ expose: -+ default: true - exposedPort: 80 - ## -- Different target traefik port on the cluster, useful for IP type LB - # targetPort: 80 -@@ -650,10 +651,6 @@ ports: - # -- Use nodeport if set. This is useful if you have configured Traefik in a - # LoadBalancer. - # nodePort: 32080 -- # -- Defines whether the port is exposed on the internal service; -- # note that ports exposed on the default service are exposed on the internal -- # service by default as well. -- exposeInternal: false - # Port Redirections - # Added in 2.2, you can make permanent redirects via entrypoints. - # https://docs.traefik.io/routing/entrypoints/#redirection -@@ -677,17 +674,14 @@ ports: - port: 8443 - # hostPort: 8443 - # containerPort: 8443 -- expose: true -+ expose: -+ default: true - exposedPort: 443 - ## -- Different target traefik port on the cluster, useful for IP type LB - # targetPort: 80 - ## -- The port protocol (TCP/UDP) - protocol: TCP - # nodePort: 32443 -- # -- Defines whether the port is exposed on the internal service; -- # note that ports exposed on the default service are exposed on the internal -- # service by default as well. -- exposeInternal: false - ## -- Specify an application protocol. This may be used as a hint for a Layer 7 load balancer. - # appProtocol: https - # -@@ -744,15 +738,12 @@ ports: - # -- You may not want to expose the metrics port on production deployments. - # If you want to access it from outside your cluster, - # use `kubectl port-forward` or create a secure ingress -- expose: false -+ expose: -+ default: false - # -- The exposed port for this service - exposedPort: 9100 - # -- The port protocol (TCP/UDP) - protocol: TCP -- # -- Defines whether the port is exposed on the internal service; -- # note that ports exposed on the default service are exposed on the internal -- # service by default as well. -- exposeInternal: false - - # -- TLS Options are created as TLSOption CRDs - # https://doc.traefik.io/traefik/https/tls/#tls-options -@@ -814,6 +805,7 @@ service: - # - IPv4 - # - IPv6 - ## -+ additionalServices: {} - ## -- An additional and optional internal Service. - ## Same parameters as external Service - # internal: -@@ -899,11 +891,14 @@ hostNetwork: false - rbac: - enabled: true - # If set to false, installs ClusterRole and ClusterRoleBinding so Traefik can be used across namespaces. -- # If set to true, installs Role and RoleBinding. Providers will only watch target namespace. -+ # If set to true, installs Role and RoleBinding instead of ClusterRole/ClusterRoleBinding. Providers will only watch target namespace. -+ # When combined with providers.kubernetesIngress.disableIngressClassLookup: true and Traefik V3, ClusterRole to watch IngressClass is also disabled. - namespaced: false - # Enable user-facing roles - # https://kubernetes.io/docs/reference/access-authn-authz/rbac/#user-facing-roles - # aggregateTo: [ "admin" ] -+ # List of Kubernetes secrets that are accessible for Traefik. If empty, then access is granted to every secret. -+ secretResourceNames: [] - - # -- Enable to create a PodSecurityPolicy and assign it to the Service Account via RoleBinding or ClusterRoleBinding - podSecurityPolicy: -``` - -## 26.1.0 ![AppVersion: v2.11.0](https://img.shields.io/static/v1?label=AppVersion&message=v2.11.0&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2024-02-19 - -* fix: 🐛 set runtimeClassName at pod level -* fix: 🐛 missing quote on experimental plugin args -* fix: update traefik v3 serverstransporttcps CRD -* feat: set runtimeClassName on pod spec -* feat: create v1 Gateway and GatewayClass Version for Traefik v3 -* feat: allow exposure of ports on internal service only -* doc: fix invalid suggestion on TLSOption (#996) -* chore: 🔧 update maintainers -* chore: 🔧 promote jnoordsij to Traefik Helm Chart maintainer -* chore(release): 🚀 publish v26.1.0 -* chore(deps): update traefik docker tag to v2.11.0 -* chore(deps): update traefik docker tag to v2.10.7 -* chore(crds): update definitions for traefik v2.11 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index f9dac91..dbd078f 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -100,6 +100,8 @@ deployment: - # port: 9000 - # host: localhost - # scheme: HTTP -+ # -- Set a runtimeClassName on pod -+ runtimeClassName: - - # -- Pod disruption budget - podDisruptionBudget: -@@ -629,6 +631,10 @@ ports: - exposedPort: 9000 - # -- The port protocol (TCP/UDP) - protocol: TCP -+ # -- Defines whether the port is exposed on the internal service; -+ # note that ports exposed on the default service are exposed on the internal -+ # service by default as well. -+ exposeInternal: false - web: - ## -- Enable this entrypoint as a default entrypoint. When a service doesn't explicitly set an entrypoint it will only use this entrypoint. - # asDefault: true -@@ -644,6 +650,10 @@ ports: - # -- Use nodeport if set. This is useful if you have configured Traefik in a - # LoadBalancer. - # nodePort: 32080 -+ # -- Defines whether the port is exposed on the internal service; -+ # note that ports exposed on the default service are exposed on the internal -+ # service by default as well. -+ exposeInternal: false - # Port Redirections - # Added in 2.2, you can make permanent redirects via entrypoints. - # https://docs.traefik.io/routing/entrypoints/#redirection -@@ -674,6 +684,10 @@ ports: - ## -- The port protocol (TCP/UDP) - protocol: TCP - # nodePort: 32443 -+ # -- Defines whether the port is exposed on the internal service; -+ # note that ports exposed on the default service are exposed on the internal -+ # service by default as well. -+ exposeInternal: false - ## -- Specify an application protocol. This may be used as a hint for a Layer 7 load balancer. - # appProtocol: https - # -@@ -735,6 +749,10 @@ ports: - exposedPort: 9100 - # -- The port protocol (TCP/UDP) - protocol: TCP -+ # -- Defines whether the port is exposed on the internal service; -+ # note that ports exposed on the default service are exposed on the internal -+ # service by default as well. -+ exposeInternal: false - - # -- TLS Options are created as TLSOption CRDs - # https://doc.traefik.io/traefik/https/tls/#tls-options -@@ -745,7 +763,7 @@ ports: - # labels: {} - # sniStrict: true - # preferServerCipherSuites: true --# customOptions: -+# custom-options: - # labels: {} - # curvePreferences: - # - CurveP521 -@@ -796,7 +814,7 @@ service: - # - IPv4 - # - IPv6 - ## -- ## -- An additionnal and optional internal Service. -+ ## -- An additional and optional internal Service. - ## Same parameters as external Service - # internal: - # type: ClusterIP -``` - -## 26.0.0 ![AppVersion: v2.10.6](https://img.shields.io/static/v1?label=AppVersion&message=v2.10.6&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-12-05 - -* fix: 🐛 improve confusing suggested value on openTelemetry.grpc -* fix: 🐛 declare http3 udp port, with or without hostport -* feat: 💥 deployment.podannotations support interpolation with tpl -* feat: allow update of namespace policy for websecure listener -* feat: allow defining startupProbe -* feat: add file provider -* feat: :boom: unify plugin import between traefik and this chart -* chore(release): 🚀 publish v26 -* chore(deps): update traefik docker tag to v2.10.6 -* Release namespace for Prometheus Operator resources - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 71e377e..f9dac91 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -40,6 +40,7 @@ deployment: - # -- Additional deployment labels (e.g. for filtering deployment by custom labels) - labels: {} - # -- Additional pod annotations (e.g. for mesh injection or prometheus scraping) -+ # It supports templating. One can set it with values like traefik/name: '{{ template "traefik.name" . }}' - podAnnotations: {} - # -- Additional Pod labels (e.g. for filtering Pod by custom labels) - podLabels: {} -@@ -119,10 +120,12 @@ experimental: - # This value is no longer used, set the image.tag to a semver higher than 3.0, e.g. "v3.0.0-beta3" - # v3: - # -- Enable traefik version 3 -- # enabled: false -- plugins: -- # -- Enable traefik experimental plugins -- enabled: false -+ -+ # -- Enable traefik experimental plugins -+ plugins: {} -+ # demo: -+ # moduleName: github.com/traefik/plugindemo -+ # version: v0.2.1 - kubernetesGateway: - # -- Enable traefik experimental GatewayClass CRD - enabled: false -@@ -206,6 +209,17 @@ livenessProbe: - # -- The number of seconds to wait for a probe response before considering it as failed. - timeoutSeconds: 2 - -+# -- Define Startup Probe for container: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/#define-startup-probes -+# eg. -+# `startupProbe: -+# exec: -+# command: -+# - mycommand -+# - foo -+# initialDelaySeconds: 5 -+# periodSeconds: 5` -+startupProbe: -+ - providers: - kubernetesCRD: - # -- Load Kubernetes IngressRoute provider -@@ -241,6 +255,23 @@ providers: - # By default this Traefik service - # pathOverride: "" - -+ file: -+ # -- Create a file provider -+ enabled: false -+ # -- Allows Traefik to automatically watch for file changes -+ watch: true -+ # -- File content (YAML format, go template supported) (see https://doc.traefik.io/traefik/providers/file/) -+ content: "" -+ # http: -+ # routers: -+ # router0: -+ # entryPoints: -+ # - web -+ # middlewares: -+ # - my-basic-auth -+ # service: service-foo -+ # rule: Path(`/foo`) -+ - # - # -- Add volumes to the traefik pod. The volume name will be passed to tpl. - # This can be used to mount a cert pair or a configmap that holds a config.toml file. -@@ -487,7 +518,7 @@ metrics: - # -- https://doc.traefik.io/traefik/observability/tracing/overview/ - tracing: {} - # openTelemetry: # traefik v3+ only --# grpc: {} -+# grpc: true - # insecure: true - # address: localhost:4317 - # instana: -``` - -## 25.0.0 ![AppVersion: v2.10.5](https://img.shields.io/static/v1?label=AppVersion&message=v2.10.5&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-10-23 - -* revert: "fix: 🐛 remove old CRDs using traefik.containo.us" -* fix: 🐛 remove old CRDs using traefik.containo.us -* fix: disable ClusterRole and ClusterRoleBinding when not needed -* fix: detect correctly v3 version when using sha in `image.tag` -* fix: allow updateStrategy.rollingUpdate.maxUnavailable to be passed in as an int or string -* fix: add missing separator in crds -* fix: add Prometheus scraping annotations only if serviceMonitor not created -* feat: ✨ add healthcheck ingressRoute -* feat: :boom: support http redirections and http challenges with cert-manager -* feat: :boom: rework and allow update of namespace policy for Gateway -* docs: Fix typo in the default values file -* chore: remove label whitespace at TLSOption -* chore(release): publish v25.0.0 -* chore(deps): update traefik docker tag to v2.10.5 -* chore(deps): update docker.io/helmunittest/helm-unittest docker tag to v3.12.3 -* chore(ci): 🔧 👷 add e2e test when releasing - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index aeec85c..71e377e 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -45,60 +45,60 @@ deployment: - podLabels: {} - # -- Additional containers (e.g. for metric offloading sidecars) - additionalContainers: [] -- # https://docs.datadoghq.com/developers/dogstatsd/unix_socket/?tab=host -- # - name: socat-proxy -- # image: alpine/socat:1.0.5 -- # args: ["-s", "-u", "udp-recv:8125", "unix-sendto:/socket/socket"] -- # volumeMounts: -- # - name: dsdsocket -- # mountPath: /socket -+ # https://docs.datadoghq.com/developers/dogstatsd/unix_socket/?tab=host -+ # - name: socat-proxy -+ # image: alpine/socat:1.0.5 -+ # args: ["-s", "-u", "udp-recv:8125", "unix-sendto:/socket/socket"] -+ # volumeMounts: -+ # - name: dsdsocket -+ # mountPath: /socket - # -- Additional volumes available for use with initContainers and additionalContainers - additionalVolumes: [] -- # - name: dsdsocket -- # hostPath: -- # path: /var/run/statsd-exporter -+ # - name: dsdsocket -+ # hostPath: -+ # path: /var/run/statsd-exporter - # -- Additional initContainers (e.g. for setting file permission as shown below) - initContainers: [] -- # The "volume-permissions" init container is required if you run into permission issues. -- # Related issue: https://github.com/traefik/traefik-helm-chart/issues/396 -- # - name: volume-permissions -- # image: busybox:latest -- # command: ["sh", "-c", "touch /data/acme.json; chmod -v 600 /data/acme.json"] -- # securityContext: -- # runAsNonRoot: true -- # runAsGroup: 65532 -- # runAsUser: 65532 -- # volumeMounts: -- # - name: data -- # mountPath: /data -+ # The "volume-permissions" init container is required if you run into permission issues. -+ # Related issue: https://github.com/traefik/traefik-helm-chart/issues/396 -+ # - name: volume-permissions -+ # image: busybox:latest -+ # command: ["sh", "-c", "touch /data/acme.json; chmod -v 600 /data/acme.json"] -+ # securityContext: -+ # runAsNonRoot: true -+ # runAsGroup: 65532 -+ # runAsUser: 65532 -+ # volumeMounts: -+ # - name: data -+ # mountPath: /data - # -- Use process namespace sharing - shareProcessNamespace: false - # -- Custom pod DNS policy. Apply if `hostNetwork: true` - # dnsPolicy: ClusterFirstWithHostNet - dnsConfig: {} -- # nameservers: -- # - 192.0.2.1 # this is an example -- # searches: -- # - ns1.svc.cluster-domain.example -- # - my.dns.search.suffix -- # options: -- # - name: ndots -- # value: "2" -- # - name: edns0 -+ # nameservers: -+ # - 192.0.2.1 # this is an example -+ # searches: -+ # - ns1.svc.cluster-domain.example -+ # - my.dns.search.suffix -+ # options: -+ # - name: ndots -+ # value: "2" -+ # - name: edns0 - # -- Additional imagePullSecrets - imagePullSecrets: [] -- # - name: myRegistryKeySecretName -+ # - name: myRegistryKeySecretName - # -- Pod lifecycle actions - lifecycle: {} -- # preStop: -- # exec: -- # command: ["/bin/sh", "-c", "sleep 40"] -- # postStart: -- # httpGet: -- # path: /ping -- # port: 9000 -- # host: localhost -- # scheme: HTTP -+ # preStop: -+ # exec: -+ # command: ["/bin/sh", "-c", "sleep 40"] -+ # postStart: -+ # httpGet: -+ # path: /ping -+ # port: 9000 -+ # host: localhost -+ # scheme: HTTP - - # -- Pod disruption budget - podDisruptionBudget: -@@ -116,9 +116,9 @@ ingressClass: - - # Traefik experimental features - experimental: -- #This value is no longer used, set the image.tag to a semver higher than 3.0, e.g. "v3.0.0-beta3" -- #v3: -- # -- Enable traefik version 3 -+ # This value is no longer used, set the image.tag to a semver higher than 3.0, e.g. "v3.0.0-beta3" -+ # v3: -+ # -- Enable traefik version 3 - # enabled: false - plugins: - # -- Enable traefik experimental plugins -@@ -126,9 +126,9 @@ experimental: - kubernetesGateway: - # -- Enable traefik experimental GatewayClass CRD - enabled: false -- gateway: -- # -- Enable traefik regular kubernetes gateway -- enabled: true -+ ## Routes are restricted to namespace of the gateway by default. -+ ## https://gateway-api.sigs.k8s.io/references/spec/#gateway.networking.k8s.io/v1beta1.FromNamespaces -+ # namespacePolicy: All - # certificate: - # group: "core" - # kind: "Secret" -@@ -159,6 +159,22 @@ ingressRoute: - middlewares: [] - # -- TLS options (e.g. secret containing certificate) - tls: {} -+ healthcheck: -+ # -- Create an IngressRoute for the healthcheck probe -+ enabled: false -+ # -- Additional ingressRoute annotations (e.g. for kubernetes.io/ingress.class) -+ annotations: {} -+ # -- Additional ingressRoute labels (e.g. for filtering IngressRoute by custom labels) -+ labels: {} -+ # -- The router match rule used for the healthcheck ingressRoute -+ matchRule: PathPrefix(`/ping`) -+ # -- Specify the allowed entrypoints to use for the healthcheck ingress route, (e.g. traefik, web, websecure). -+ # By default, it's using traefik entrypoint, which is not exposed. -+ entryPoints: ["traefik"] -+ # -- Additional ingressRoute middlewares (e.g. for authentication) -+ middlewares: [] -+ # -- TLS options (e.g. secret containing certificate) -+ tls: {} - - updateStrategy: - # -- Customize updateStrategy: RollingUpdate or OnDelete -@@ -204,10 +220,10 @@ providers: - # labelSelector: environment=production,method=traefik - # -- Array of namespaces to watch. If left empty, Traefik watches all namespaces. - namespaces: [] -- # - "default" -+ # - "default" - - kubernetesIngress: -- # -- Load Kubernetes IngressRoute provider -+ # -- Load Kubernetes Ingress provider - enabled: true - # -- Allows to reference ExternalName services in Ingress - allowExternalNameServices: false -@@ -217,7 +233,7 @@ providers: - # labelSelector: environment=production,method=traefik - # -- Array of namespaces to watch. If left empty, Traefik watches all namespaces. - namespaces: [] -- # - "default" -+ # - "default" - # IP used for Kubernetes Ingress endpoints - publishedService: - enabled: false -@@ -243,9 +259,9 @@ volumes: [] - - # -- Additional volumeMounts to add to the Traefik container - additionalVolumeMounts: [] -- # -- For instance when using a logshipper for access logs -- # - name: traefik-logs -- # mountPath: /var/log/traefik -+# -- For instance when using a logshipper for access logs -+# - name: traefik-logs -+# mountPath: /var/log/traefik - - logs: - general: -@@ -270,26 +286,26 @@ logs: - ## Filtering - # -- https://docs.traefik.io/observability/access-logs/#filtering - filters: {} -- # statuscodes: "200,300-302" -- # retryattempts: true -- # minduration: 10ms -+ # statuscodes: "200,300-302" -+ # retryattempts: true -+ # minduration: 10ms - fields: - general: - # -- Available modes: keep, drop, redact. - defaultmode: keep - # -- Names of the fields to limit. - names: {} -- ## Examples: -- # ClientUsername: drop -+ ## Examples: -+ # ClientUsername: drop - headers: - # -- Available modes: keep, drop, redact. - defaultmode: drop - # -- Names of the headers to limit. - names: {} -- ## Examples: -- # User-Agent: redact -- # Authorization: drop -- # Content-Type: keep -+ ## Examples: -+ # User-Agent: redact -+ # Authorization: drop -+ # Content-Type: keep - - metrics: - ## -- Prometheus is enabled by default. -@@ -308,118 +324,118 @@ metrics: - ## When manualRouting is true, it disables the default internal router in - ## order to allow creating a custom router for prometheus@internal service. - # manualRouting: true --# datadog: --# ## Address instructs exporter to send metrics to datadog-agent at this address. --# address: "127.0.0.1:8125" --# ## The interval used by the exporter to push metrics to datadog-agent. Default=10s --# # pushInterval: 30s --# ## The prefix to use for metrics collection. Default="traefik" --# # prefix: traefik --# ## Enable metrics on entry points. Default=true --# # addEntryPointsLabels: false --# ## Enable metrics on routers. Default=false --# # addRoutersLabels: true --# ## Enable metrics on services. Default=true --# # addServicesLabels: false --# influxdb: --# ## Address instructs exporter to send metrics to influxdb at this address. --# address: localhost:8089 --# ## InfluxDB's address protocol (udp or http). Default="udp" --# protocol: udp --# ## InfluxDB database used when protocol is http. Default="" --# # database: "" --# ## InfluxDB retention policy used when protocol is http. Default="" --# # retentionPolicy: "" --# ## InfluxDB username (only with http). Default="" --# # username: "" --# ## InfluxDB password (only with http). Default="" --# # password: "" --# ## The interval used by the exporter to push metrics to influxdb. Default=10s --# # pushInterval: 30s --# ## Additional labels (influxdb tags) on all metrics. --# # additionalLabels: --# # env: production --# # foo: bar --# ## Enable metrics on entry points. Default=true --# # addEntryPointsLabels: false --# ## Enable metrics on routers. Default=false --# # addRoutersLabels: true --# ## Enable metrics on services. Default=true --# # addServicesLabels: false --# influxdb2: --# ## Address instructs exporter to send metrics to influxdb v2 at this address. --# address: localhost:8086 --# ## Token with which to connect to InfluxDB v2. --# token: xxx --# ## Organisation where metrics will be stored. --# org: "" --# ## Bucket where metrics will be stored. --# bucket: "" --# ## The interval used by the exporter to push metrics to influxdb. Default=10s --# # pushInterval: 30s --# ## Additional labels (influxdb tags) on all metrics. --# # additionalLabels: --# # env: production --# # foo: bar --# ## Enable metrics on entry points. Default=true --# # addEntryPointsLabels: false --# ## Enable metrics on routers. Default=false --# # addRoutersLabels: true --# ## Enable metrics on services. Default=true --# # addServicesLabels: false --# statsd: --# ## Address instructs exporter to send metrics to statsd at this address. --# address: localhost:8125 --# ## The interval used by the exporter to push metrics to influxdb. Default=10s --# # pushInterval: 30s --# ## The prefix to use for metrics collection. Default="traefik" --# # prefix: traefik --# ## Enable metrics on entry points. Default=true --# # addEntryPointsLabels: false --# ## Enable metrics on routers. Default=false --# # addRoutersLabels: true --# ## Enable metrics on services. Default=true --# # addServicesLabels: false --# openTelemetry: --# ## Address of the OpenTelemetry Collector to send metrics to. --# address: "localhost:4318" --# ## Enable metrics on entry points. --# addEntryPointsLabels: true --# ## Enable metrics on routers. --# addRoutersLabels: true --# ## Enable metrics on services. --# addServicesLabels: true --# ## Explicit boundaries for Histogram data points. --# explicitBoundaries: --# - "0.1" --# - "0.3" --# - "1.2" --# - "5.0" --# ## Additional headers sent with metrics by the reporter to the OpenTelemetry Collector. --# headers: --# foo: bar --# test: test --# ## Allows reporter to send metrics to the OpenTelemetry Collector without using a secured protocol. --# insecure: true --# ## Interval at which metrics are sent to the OpenTelemetry Collector. --# pushInterval: 10s --# ## Allows to override the default URL path used for sending metrics. This option has no effect when using gRPC transport. --# path: /foo/v1/traces --# ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. --# tls: --# ## The path to the certificate authority, it defaults to the system bundle. --# ca: path/to/ca.crt --# ## The path to the public certificate. When using this option, setting the key option is required. --# cert: path/to/foo.cert --# ## The path to the private key. When using this option, setting the cert option is required. --# key: path/to/key.key --# ## If set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. --# insecureSkipVerify: true --# ## This instructs the reporter to send metrics to the OpenTelemetry Collector using gRPC. --# grpc: true -- --## -- enable optional CRDs for Prometheus Operator --## -+ # datadog: -+ # ## Address instructs exporter to send metrics to datadog-agent at this address. -+ # address: "127.0.0.1:8125" -+ # ## The interval used by the exporter to push metrics to datadog-agent. Default=10s -+ # # pushInterval: 30s -+ # ## The prefix to use for metrics collection. Default="traefik" -+ # # prefix: traefik -+ # ## Enable metrics on entry points. Default=true -+ # # addEntryPointsLabels: false -+ # ## Enable metrics on routers. Default=false -+ # # addRoutersLabels: true -+ # ## Enable metrics on services. Default=true -+ # # addServicesLabels: false -+ # influxdb: -+ # ## Address instructs exporter to send metrics to influxdb at this address. -+ # address: localhost:8089 -+ # ## InfluxDB's address protocol (udp or http). Default="udp" -+ # protocol: udp -+ # ## InfluxDB database used when protocol is http. Default="" -+ # # database: "" -+ # ## InfluxDB retention policy used when protocol is http. Default="" -+ # # retentionPolicy: "" -+ # ## InfluxDB username (only with http). Default="" -+ # # username: "" -+ # ## InfluxDB password (only with http). Default="" -+ # # password: "" -+ # ## The interval used by the exporter to push metrics to influxdb. Default=10s -+ # # pushInterval: 30s -+ # ## Additional labels (influxdb tags) on all metrics. -+ # # additionalLabels: -+ # # env: production -+ # # foo: bar -+ # ## Enable metrics on entry points. Default=true -+ # # addEntryPointsLabels: false -+ # ## Enable metrics on routers. Default=false -+ # # addRoutersLabels: true -+ # ## Enable metrics on services. Default=true -+ # # addServicesLabels: false -+ # influxdb2: -+ # ## Address instructs exporter to send metrics to influxdb v2 at this address. -+ # address: localhost:8086 -+ # ## Token with which to connect to InfluxDB v2. -+ # token: xxx -+ # ## Organisation where metrics will be stored. -+ # org: "" -+ # ## Bucket where metrics will be stored. -+ # bucket: "" -+ # ## The interval used by the exporter to push metrics to influxdb. Default=10s -+ # # pushInterval: 30s -+ # ## Additional labels (influxdb tags) on all metrics. -+ # # additionalLabels: -+ # # env: production -+ # # foo: bar -+ # ## Enable metrics on entry points. Default=true -+ # # addEntryPointsLabels: false -+ # ## Enable metrics on routers. Default=false -+ # # addRoutersLabels: true -+ # ## Enable metrics on services. Default=true -+ # # addServicesLabels: false -+ # statsd: -+ # ## Address instructs exporter to send metrics to statsd at this address. -+ # address: localhost:8125 -+ # ## The interval used by the exporter to push metrics to influxdb. Default=10s -+ # # pushInterval: 30s -+ # ## The prefix to use for metrics collection. Default="traefik" -+ # # prefix: traefik -+ # ## Enable metrics on entry points. Default=true -+ # # addEntryPointsLabels: false -+ # ## Enable metrics on routers. Default=false -+ # # addRoutersLabels: true -+ # ## Enable metrics on services. Default=true -+ # # addServicesLabels: false -+ # openTelemetry: -+ # ## Address of the OpenTelemetry Collector to send metrics to. -+ # address: "localhost:4318" -+ # ## Enable metrics on entry points. -+ # addEntryPointsLabels: true -+ # ## Enable metrics on routers. -+ # addRoutersLabels: true -+ # ## Enable metrics on services. -+ # addServicesLabels: true -+ # ## Explicit boundaries for Histogram data points. -+ # explicitBoundaries: -+ # - "0.1" -+ # - "0.3" -+ # - "1.2" -+ # - "5.0" -+ # ## Additional headers sent with metrics by the reporter to the OpenTelemetry Collector. -+ # headers: -+ # foo: bar -+ # test: test -+ # ## Allows reporter to send metrics to the OpenTelemetry Collector without using a secured protocol. -+ # insecure: true -+ # ## Interval at which metrics are sent to the OpenTelemetry Collector. -+ # pushInterval: 10s -+ # ## Allows to override the default URL path used for sending metrics. This option has no effect when using gRPC transport. -+ # path: /foo/v1/traces -+ # ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. -+ # tls: -+ # ## The path to the certificate authority, it defaults to the system bundle. -+ # ca: path/to/ca.crt -+ # ## The path to the public certificate. When using this option, setting the key option is required. -+ # cert: path/to/foo.cert -+ # ## The path to the private key. When using this option, setting the cert option is required. -+ # key: path/to/key.key -+ # ## If set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. -+ # insecureSkipVerify: true -+ # ## This instructs the reporter to send metrics to the OpenTelemetry Collector using gRPC. -+ # grpc: true -+ -+ ## -- enable optional CRDs for Prometheus Operator -+ ## - ## Create a dedicated metrics service for use with ServiceMonitor - # service: - # enabled: false -@@ -470,55 +486,55 @@ metrics: - ## Tracing - # -- https://doc.traefik.io/traefik/observability/tracing/overview/ - tracing: {} -- # openTelemetry: # traefik v3+ only -- # grpc: {} -- # insecure: true -- # address: localhost:4317 -- # instana: -- # localAgentHost: 127.0.0.1 -- # localAgentPort: 42699 -- # logLevel: info -- # enableAutoProfile: true -- # datadog: -- # localAgentHostPort: 127.0.0.1:8126 -- # debug: false -- # globalTag: "" -- # prioritySampling: false -- # jaeger: -- # samplingServerURL: http://localhost:5778/sampling -- # samplingType: const -- # samplingParam: 1.0 -- # localAgentHostPort: 127.0.0.1:6831 -- # gen128Bit: false -- # propagation: jaeger -- # traceContextHeaderName: uber-trace-id -- # disableAttemptReconnecting: true -- # collector: -- # endpoint: "" -- # user: "" -- # password: "" -- # zipkin: -- # httpEndpoint: http://localhost:9411/api/v2/spans -- # sameSpan: false -- # id128Bit: true -- # sampleRate: 1.0 -- # haystack: -- # localAgentHost: 127.0.0.1 -- # localAgentPort: 35000 -- # globalTag: "" -- # traceIDHeaderName: "" -- # parentIDHeaderName: "" -- # spanIDHeaderName: "" -- # baggagePrefixHeaderName: "" -- # elastic: -- # serverURL: http://localhost:8200 -- # secretToken: "" -- # serviceEnvironment: "" -+# openTelemetry: # traefik v3+ only -+# grpc: {} -+# insecure: true -+# address: localhost:4317 -+# instana: -+# localAgentHost: 127.0.0.1 -+# localAgentPort: 42699 -+# logLevel: info -+# enableAutoProfile: true -+# datadog: -+# localAgentHostPort: 127.0.0.1:8126 -+# debug: false -+# globalTag: "" -+# prioritySampling: false -+# jaeger: -+# samplingServerURL: http://localhost:5778/sampling -+# samplingType: const -+# samplingParam: 1.0 -+# localAgentHostPort: 127.0.0.1:6831 -+# gen128Bit: false -+# propagation: jaeger -+# traceContextHeaderName: uber-trace-id -+# disableAttemptReconnecting: true -+# collector: -+# endpoint: "" -+# user: "" -+# password: "" -+# zipkin: -+# httpEndpoint: http://localhost:9411/api/v2/spans -+# sameSpan: false -+# id128Bit: true -+# sampleRate: 1.0 -+# haystack: -+# localAgentHost: 127.0.0.1 -+# localAgentPort: 35000 -+# globalTag: "" -+# traceIDHeaderName: "" -+# parentIDHeaderName: "" -+# spanIDHeaderName: "" -+# baggagePrefixHeaderName: "" -+# elastic: -+# serverURL: http://localhost:8200 -+# secretToken: "" -+# serviceEnvironment: "" - - # -- Global command arguments to be passed to all traefik's pods - globalArguments: -- - "--global.checknewversion" -- - "--global.sendanonymoususage" -+- "--global.checknewversion" -+- "--global.sendanonymoususage" - - # - # Configure Traefik static configuration -@@ -531,14 +547,14 @@ additionalArguments: [] - - # -- Environment variables to be passed to Traefik's binary - env: -- - name: POD_NAME -- valueFrom: -- fieldRef: -- fieldPath: metadata.name -- - name: POD_NAMESPACE -- valueFrom: -- fieldRef: -- fieldPath: metadata.namespace -+- name: POD_NAME -+ valueFrom: -+ fieldRef: -+ fieldPath: metadata.name -+- name: POD_NAMESPACE -+ valueFrom: -+ fieldRef: -+ fieldPath: metadata.namespace - # - name: SOME_VAR - # value: some-var-value - # - name: SOME_VAR_FROM_CONFIG_MAP -@@ -600,7 +616,10 @@ ports: - # Port Redirections - # Added in 2.2, you can make permanent redirects via entrypoints. - # https://docs.traefik.io/routing/entrypoints/#redirection -- # redirectTo: websecure -+ # redirectTo: -+ # port: websecure -+ # (Optional) -+ # priority: 10 - # - # Trust forwarded headers information (X-Forwarded-*). - # forwardedHeaders: -@@ -638,14 +657,14 @@ ports: - # advertisedPort: 4443 - # - ## -- Trust forwarded headers information (X-Forwarded-*). -- #forwardedHeaders: -- # trustedIPs: [] -- # insecure: false -+ # forwardedHeaders: -+ # trustedIPs: [] -+ # insecure: false - # - ## -- Enable the Proxy Protocol header parsing for the entry point -- #proxyProtocol: -- # trustedIPs: [] -- # insecure: false -+ # proxyProtocol: -+ # trustedIPs: [] -+ # insecure: false - # - ## Set TLS at the entrypoint - ## https://doc.traefik.io/traefik/routing/entrypoints/#tls -@@ -728,16 +747,16 @@ service: - # -- Additional entries here will be added to the service spec. - # -- Cannot contain type, selector or ports entries. - spec: {} -- # externalTrafficPolicy: Cluster -- # loadBalancerIP: "1.2.3.4" -- # clusterIP: "2.3.4.5" -+ # externalTrafficPolicy: Cluster -+ # loadBalancerIP: "1.2.3.4" -+ # clusterIP: "2.3.4.5" - loadBalancerSourceRanges: [] -- # - 192.168.0.1/32 -- # - 172.16.0.0/16 -+ # - 192.168.0.1/32 -+ # - 172.16.0.0/16 - ## -- Class of the load balancer implementation - # loadBalancerClass: service.k8s.aws/nlb - externalIPs: [] -- # - 1.2.3.4 -+ # - 1.2.3.4 - ## One of SingleStack, PreferDualStack, or RequireDualStack. - # ipFamilyPolicy: SingleStack - ## List of IP families (e.g. IPv4 and/or IPv6). -@@ -789,7 +808,7 @@ persistence: - # It can be used to store TLS certificates, see `storage` in certResolvers - enabled: false - name: data --# existingClaim: "" -+ # existingClaim: "" - accessMode: ReadWriteOnce - size: 128Mi - # storageClass: "" -@@ -852,12 +871,12 @@ serviceAccountAnnotations: {} - - # -- The resources parameter defines CPU and memory requirements and limits for Traefik's containers. - resources: {} -- # requests: -- # cpu: "100m" -- # memory: "50Mi" -- # limits: -- # cpu: "300m" -- # memory: "150Mi" -+# requests: -+# cpu: "100m" -+# memory: "50Mi" -+# limits: -+# cpu: "300m" -+# memory: "150Mi" - - # -- This example pod anti-affinity forces the scheduler to put traefik pods - # -- on nodes where no other traefik pods are scheduled. -``` - -## 24.0.0 ![AppVersion: v2.10.4](https://img.shields.io/static/v1?label=AppVersion&message=v2.10.4&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-08-10 - -* fix: 💥 BREAKING CHANGE on healthchecks and traefik port -* fix: tracing.opentelemetry.tls is optional for all values -* fix: http3 support broken when advertisedPort set -* feat: multi namespace RBAC manifests -* chore(tests): 🔧 fix typo on tracing test -* chore(release): 🚀 publish v24.0.0 -* chore(deps): update docker.io/helmunittest/helm-unittest docker tag to v3.12.2 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 947ba56..aeec85c 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -28,6 +28,13 @@ deployment: - terminationGracePeriodSeconds: 60 - # -- The minimum number of seconds Traefik needs to be up and running before the DaemonSet/Deployment controller considers it available - minReadySeconds: 0 -+ ## Override the liveness/readiness port. This is useful to integrate traefik -+ ## with an external Load Balancer that performs healthchecks. -+ ## Default: ports.traefik.port -+ # healthchecksPort: 9000 -+ ## Override the liveness/readiness scheme. Useful for getting ping to -+ ## respond on websecure entryPoint. -+ # healthchecksScheme: HTTPS - # -- Additional deployment annotations (e.g. for jaeger-operator sidecar injection) - annotations: {} - # -- Additional deployment labels (e.g. for filtering deployment by custom labels) -@@ -112,7 +119,7 @@ experimental: - #This value is no longer used, set the image.tag to a semver higher than 3.0, e.g. "v3.0.0-beta3" - #v3: - # -- Enable traefik version 3 -- # enabled: false -+ # enabled: false - plugins: - # -- Enable traefik experimental plugins - enabled: false -@@ -564,15 +571,6 @@ ports: - # only. - # hostIP: 192.168.100.10 - -- # Override the liveness/readiness port. This is useful to integrate traefik -- # with an external Load Balancer that performs healthchecks. -- # Default: ports.traefik.port -- # healthchecksPort: 9000 -- -- # Override the liveness/readiness scheme. Useful for getting ping to -- # respond on websecure entryPoint. -- # healthchecksScheme: HTTPS -- - # Defines whether the port is exposed if service.type is LoadBalancer or - # NodePort. - # -@@ -877,7 +875,7 @@ affinity: {} - nodeSelector: {} - # -- Tolerations allow the scheduler to schedule pods with matching taints. - tolerations: [] --# -- You can use topology spread constraints to control -+# -- You can use topology spread constraints to control - # how Pods are spread across your cluster among failure-domains. - topologySpreadConstraints: [] - # This example topologySpreadConstraints forces the scheduler to put traefik pods -``` - -## 23.2.0 ![AppVersion: v2.10.4](https://img.shields.io/static/v1?label=AppVersion&message=v2.10.4&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-07-27 - -* ⬆️ Upgrade traefik Docker tag to v2.10.3 -* release: :rocket: publish v23.2.0 -* fix: 🐛 update traefik.containo.us CRDs to v2.10 -* fix: 🐛 traefik or metrics port can be disabled -* fix: ingressclass name should be customizable (#864) -* feat: ✨ add support for traefik v3.0.0-beta3 and openTelemetry -* feat: disable allowPrivilegeEscalation -* feat: add pod_name as default in values.yaml -* chore(tests): 🔧 use more accurate asserts on refactor'd isNull test -* chore(deps): update traefik docker tag to v2.10.4 -* chore(deps): update docker.io/helmunittest/helm-unittest docker tag to v3.11.3 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 345bbd8..947ba56 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -105,12 +105,14 @@ podDisruptionBudget: - ingressClass: - enabled: true - isDefaultClass: true -+ # name: my-custom-class - - # Traefik experimental features - experimental: -- v3: -+ #This value is no longer used, set the image.tag to a semver higher than 3.0, e.g. "v3.0.0-beta3" -+ #v3: - # -- Enable traefik version 3 -- enabled: false -+ # enabled: false - plugins: - # -- Enable traefik experimental plugins - enabled: false -@@ -461,6 +463,10 @@ metrics: - ## Tracing - # -- https://doc.traefik.io/traefik/observability/tracing/overview/ - tracing: {} -+ # openTelemetry: # traefik v3+ only -+ # grpc: {} -+ # insecure: true -+ # address: localhost:4317 - # instana: - # localAgentHost: 127.0.0.1 - # localAgentPort: 42699 -@@ -517,7 +523,15 @@ additionalArguments: [] - # - "--log.level=DEBUG" - - # -- Environment variables to be passed to Traefik's binary --env: [] -+env: -+ - name: POD_NAME -+ valueFrom: -+ fieldRef: -+ fieldPath: metadata.name -+ - name: POD_NAMESPACE -+ valueFrom: -+ fieldRef: -+ fieldPath: metadata.namespace - # - name: SOME_VAR - # value: some-var-value - # - name: SOME_VAR_FROM_CONFIG_MAP -@@ -563,7 +577,7 @@ ports: - # NodePort. - # - # -- You SHOULD NOT expose the traefik port on production deployments. -- # If you want to access it from outside of your cluster, -+ # If you want to access it from outside your cluster, - # use `kubectl port-forward` or create a secure ingress - expose: false - # -- The exposed port for this service -@@ -571,7 +585,7 @@ ports: - # -- The port protocol (TCP/UDP) - protocol: TCP - web: -- ## -- Enable this entrypoint as a default entrypoint. When a service doesn't explicity set an entrypoint it will only use this entrypoint. -+ ## -- Enable this entrypoint as a default entrypoint. When a service doesn't explicitly set an entrypoint it will only use this entrypoint. - # asDefault: true - port: 8000 - # hostPort: 8000 -@@ -600,7 +614,7 @@ ports: - # trustedIPs: [] - # insecure: false - websecure: -- ## -- Enable this entrypoint as a default entrypoint. When a service doesn't explicity set an entrypoint it will only use this entrypoint. -+ ## -- Enable this entrypoint as a default entrypoint. When a service doesn't explicitly set an entrypoint it will only use this entrypoint. - # asDefault: true - port: 8443 - # hostPort: 8443 -@@ -666,7 +680,7 @@ ports: - # NodePort. - # - # -- You may not want to expose the metrics port on production deployments. -- # If you want to access it from outside of your cluster, -+ # If you want to access it from outside your cluster, - # use `kubectl port-forward` or create a secure ingress - expose: false - # -- The exposed port for this service -@@ -880,14 +894,15 @@ topologySpreadConstraints: [] - priorityClassName: "" - - # -- Set the container security context --# -- To run the container with ports below 1024 this will need to be adjust to run as root -+# -- To run the container with ports below 1024 this will need to be adjusted to run as root - securityContext: - capabilities: - drop: [ALL] - readOnlyRootFilesystem: true -+ allowPrivilegeEscalation: false - - podSecurityContext: -- # /!\ When setting fsGroup, Kubernetes will recursively changes ownership and -+ # /!\ When setting fsGroup, Kubernetes will recursively change ownership and - # permissions for the contents of each volume to match the fsGroup. This can - # be an issue when storing sensitive content like TLS Certificates /!\ - # fsGroup: 65532 -``` - -## 23.1.0 ![AppVersion: v2.10.1](https://img.shields.io/static/v1?label=AppVersion&message=v2.10.1&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-06-06 - -* release: 🚀 publish v23.1.0 -* fix: 🐛 use k8s version for hpa api version -* fix: 🐛 http3 support on traefik v3 -* fix: use `targetPort` instead of `port` on ServiceMonitor -* feat: ➖ remove Traefik Hub v1 integration -* feat: ✨ add a warning when labelSelector don't match -* feat: common labels for all resources -* feat: allow specifying service loadBalancerClass -* feat: add optional `appProtocol` field on Service ports -* doc: added values README via helm-docs cli - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 71273cc..345bbd8 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1,70 +1,56 @@ - # Default values for Traefik - image: -+ # -- Traefik image host registry - registry: docker.io -+ # -- Traefik image repository - repository: traefik -- # defaults to appVersion -+ # -- defaults to appVersion - tag: "" -+ # -- Traefik image pull policy - pullPolicy: IfNotPresent - --# --# Configure integration with Traefik Hub --# --hub: -- ## Enabling Hub will: -- # * enable Traefik Hub integration on Traefik -- # * add `traefikhub-tunl` endpoint -- # * enable Prometheus metrics with addRoutersLabels -- # * enable allowExternalNameServices on KubernetesIngress provider -- # * enable allowCrossNamespace on KubernetesCRD provider -- # * add an internal (ClusterIP) Service, dedicated for Traefik Hub -- enabled: false -- ## Default port can be changed -- # tunnelPort: 9901 -- ## TLS is optional. Insecure is mutually exclusive with any other options -- # tls: -- # insecure: false -- # ca: "/path/to/ca.pem" -- # cert: "/path/to/cert.pem" -- # key: "/path/to/key.pem" -+# -- Add additional label to all resources -+commonLabels: {} - - # - # Configure the deployment - # - deployment: -+ # -- Enable deployment - enabled: true -- # Can be either Deployment or DaemonSet -+ # -- Deployment or DaemonSet - kind: Deployment -- # Number of pods of the deployment (only applies when kind == Deployment) -+ # -- Number of pods of the deployment (only applies when kind == Deployment) - replicas: 1 -- # Number of old history to retain to allow rollback (If not set, default Kubernetes value is set to 10) -+ # -- Number of old history to retain to allow rollback (If not set, default Kubernetes value is set to 10) - # revisionHistoryLimit: 1 -- # Amount of time (in seconds) before Kubernetes will send the SIGKILL signal if Traefik does not shut down -+ # -- Amount of time (in seconds) before Kubernetes will send the SIGKILL signal if Traefik does not shut down - terminationGracePeriodSeconds: 60 -- # The minimum number of seconds Traefik needs to be up and running before the DaemonSet/Deployment controller considers it available -+ # -- The minimum number of seconds Traefik needs to be up and running before the DaemonSet/Deployment controller considers it available - minReadySeconds: 0 -- # Additional deployment annotations (e.g. for jaeger-operator sidecar injection) -+ # -- Additional deployment annotations (e.g. for jaeger-operator sidecar injection) - annotations: {} -- # Additional deployment labels (e.g. for filtering deployment by custom labels) -+ # -- Additional deployment labels (e.g. for filtering deployment by custom labels) - labels: {} -- # Additional pod annotations (e.g. for mesh injection or prometheus scraping) -+ # -- Additional pod annotations (e.g. for mesh injection or prometheus scraping) - podAnnotations: {} -- # Additional Pod labels (e.g. for filtering Pod by custom labels) -+ # -- Additional Pod labels (e.g. for filtering Pod by custom labels) - podLabels: {} -- # Additional containers (e.g. for metric offloading sidecars) -+ # -- Additional containers (e.g. for metric offloading sidecars) - additionalContainers: [] - # https://docs.datadoghq.com/developers/dogstatsd/unix_socket/?tab=host - # - name: socat-proxy -- # image: alpine/socat:1.0.5 -- # args: ["-s", "-u", "udp-recv:8125", "unix-sendto:/socket/socket"] -- # volumeMounts: -- # - name: dsdsocket -- # mountPath: /socket -- # Additional volumes available for use with initContainers and additionalContainers -+ # image: alpine/socat:1.0.5 -+ # args: ["-s", "-u", "udp-recv:8125", "unix-sendto:/socket/socket"] -+ # volumeMounts: -+ # - name: dsdsocket -+ # mountPath: /socket -+ # -- Additional volumes available for use with initContainers and additionalContainers - additionalVolumes: [] - # - name: dsdsocket - # hostPath: - # path: /var/run/statsd-exporter -- # Additional initContainers (e.g. for setting file permission as shown below) -+ # -- Additional initContainers (e.g. for setting file permission as shown below) - initContainers: [] - # The "volume-permissions" init container is required if you run into permission issues. - # Related issue: https://github.com/traefik/traefik-helm-chart/issues/396 -@@ -78,9 +64,9 @@ deployment: - # volumeMounts: - # - name: data - # mountPath: /data -- # Use process namespace sharing -+ # -- Use process namespace sharing - shareProcessNamespace: false -- # Custom pod DNS policy. Apply if `hostNetwork: true` -+ # -- Custom pod DNS policy. Apply if `hostNetwork: true` - # dnsPolicy: ClusterFirstWithHostNet - dnsConfig: {} - # nameservers: -@@ -92,10 +78,10 @@ deployment: - # - name: ndots - # value: "2" - # - name: edns0 -- # Additional imagePullSecrets -+ # -- Additional imagePullSecrets - imagePullSecrets: [] - # - name: myRegistryKeySecretName -- # Pod lifecycle actions -+ # -- Pod lifecycle actions - lifecycle: {} - # preStop: - # exec: -@@ -107,7 +93,7 @@ deployment: - # host: localhost - # scheme: HTTP - --# Pod disruption budget -+# -- Pod disruption budget - podDisruptionBudget: - enabled: false - # maxUnavailable: 1 -@@ -115,93 +101,112 @@ podDisruptionBudget: - # minAvailable: 0 - # minAvailable: 25% - --# Create a default IngressClass for Traefik -+# -- Create a default IngressClass for Traefik - ingressClass: - enabled: true - isDefaultClass: true - --# Enable experimental features -+# Traefik experimental features - experimental: - v3: -+ # -- Enable traefik version 3 - enabled: false - plugins: -+ # -- Enable traefik experimental plugins - enabled: false - kubernetesGateway: -+ # -- Enable traefik experimental GatewayClass CRD - enabled: false - gateway: -+ # -- Enable traefik regular kubernetes gateway - enabled: true - # certificate: - # group: "core" - # kind: "Secret" - # name: "mysecret" -- # By default, Gateway would be created to the Namespace you are deploying Traefik to. -+ # -- By default, Gateway would be created to the Namespace you are deploying Traefik to. - # You may create that Gateway in another namespace, setting its name below: - # namespace: default - # Additional gateway annotations (e.g. for cert-manager.io/issuer) - # annotations: - # cert-manager.io/issuer: letsencrypt - --# Create an IngressRoute for the dashboard -+## Create an IngressRoute for the dashboard - ingressRoute: - dashboard: -+ # -- Create an IngressRoute for the dashboard - enabled: true -- # Additional ingressRoute annotations (e.g. for kubernetes.io/ingress.class) -+ # -- Additional ingressRoute annotations (e.g. for kubernetes.io/ingress.class) - annotations: {} -- # Additional ingressRoute labels (e.g. for filtering IngressRoute by custom labels) -+ # -- Additional ingressRoute labels (e.g. for filtering IngressRoute by custom labels) - labels: {} -- # The router match rule used for the dashboard ingressRoute -+ # -- The router match rule used for the dashboard ingressRoute - matchRule: PathPrefix(`/dashboard`) || PathPrefix(`/api`) -- # Specify the allowed entrypoints to use for the dashboard ingress route, (e.g. traefik, web, websecure). -+ # -- Specify the allowed entrypoints to use for the dashboard ingress route, (e.g. traefik, web, websecure). - # By default, it's using traefik entrypoint, which is not exposed. - # /!\ Do not expose your dashboard without any protection over the internet /!\ - entryPoints: ["traefik"] -- # Additional ingressRoute middlewares (e.g. for authentication) -+ # -- Additional ingressRoute middlewares (e.g. for authentication) - middlewares: [] -- # TLS options (e.g. secret containing certificate) -+ # -- TLS options (e.g. secret containing certificate) - tls: {} - --# Customize updateStrategy of traefik pods - updateStrategy: -+ # -- Customize updateStrategy: RollingUpdate or OnDelete - type: RollingUpdate - rollingUpdate: - maxUnavailable: 0 - maxSurge: 1 - --# Customize liveness and readiness probe values. - readinessProbe: -+ # -- The number of consecutive failures allowed before considering the probe as failed. - failureThreshold: 1 -+ # -- The number of seconds to wait before starting the first probe. - initialDelaySeconds: 2 -+ # -- The number of seconds to wait between consecutive probes. - periodSeconds: 10 -+ # -- The minimum consecutive successes required to consider the probe successful. - successThreshold: 1 -+ # -- The number of seconds to wait for a probe response before considering it as failed. - timeoutSeconds: 2 -- - livenessProbe: -+ # -- The number of consecutive failures allowed before considering the probe as failed. - failureThreshold: 3 -+ # -- The number of seconds to wait before starting the first probe. - initialDelaySeconds: 2 -+ # -- The number of seconds to wait between consecutive probes. - periodSeconds: 10 -+ # -- The minimum consecutive successes required to consider the probe successful. - successThreshold: 1 -+ # -- The number of seconds to wait for a probe response before considering it as failed. - timeoutSeconds: 2 - --# --# Configure providers --# - providers: - kubernetesCRD: -+ # -- Load Kubernetes IngressRoute provider - enabled: true -+ # -- Allows IngressRoute to reference resources in namespace other than theirs - allowCrossNamespace: false -+ # -- Allows to reference ExternalName services in IngressRoute - allowExternalNameServices: false -+ # -- Allows to return 503 when there is no endpoints available - allowEmptyServices: false - # ingressClass: traefik-internal - # labelSelector: environment=production,method=traefik -+ # -- Array of namespaces to watch. If left empty, Traefik watches all namespaces. - namespaces: [] - # - "default" - - kubernetesIngress: -+ # -- Load Kubernetes IngressRoute provider - enabled: true -+ # -- Allows to reference ExternalName services in Ingress - allowExternalNameServices: false -+ # -- Allows to return 503 when there is no endpoints available - allowEmptyServices: false - # ingressClass: traefik-internal - # labelSelector: environment=production,method=traefik -+ # -- Array of namespaces to watch. If left empty, Traefik watches all namespaces. - namespaces: [] - # - "default" - # IP used for Kubernetes Ingress endpoints -@@ -212,13 +217,13 @@ providers: - # pathOverride: "" - - # --# Add volumes to the traefik pod. The volume name will be passed to tpl. -+# -- Add volumes to the traefik pod. The volume name will be passed to tpl. - # This can be used to mount a cert pair or a configmap that holds a config.toml file. - # After the volume has been mounted, add the configs into traefik by using the `additionalArguments` list below, eg: --# additionalArguments: -+# `additionalArguments: - # - "--providers.file.filename=/config/dynamic.toml" - # - "--ping" --# - "--ping.entrypoint=web" -+# - "--ping.entrypoint=web"` - volumes: [] - # - name: public-cert - # mountPath: "/certs" -@@ -227,25 +232,22 @@ volumes: [] - # mountPath: "/config" - # type: configMap - --# Additional volumeMounts to add to the Traefik container -+# -- Additional volumeMounts to add to the Traefik container - additionalVolumeMounts: [] -- # For instance when using a logshipper for access logs -+ # -- For instance when using a logshipper for access logs - # - name: traefik-logs - # mountPath: /var/log/traefik - --## Logs --## https://docs.traefik.io/observability/logs/ - logs: -- ## Traefik logs concern everything that happens to Traefik itself (startup, configuration, events, shutdown, and so on). - general: -- # By default, the logs use a text format (common), but you can -+ # -- By default, the logs use a text format (common), but you can - # also ask for the json format in the format option - # format: json - # By default, the level is set to ERROR. -- # Alternative logging levels are DEBUG, PANIC, FATAL, ERROR, WARN, and INFO. -+ # -- Alternative logging levels are DEBUG, PANIC, FATAL, ERROR, WARN, and INFO. - level: ERROR - access: -- # To enable access logs -+ # -- To enable access logs - enabled: false - ## By default, logs are written using the Common Log Format (CLF) on stdout. - ## To write logs in JSON, use json in the format option. -@@ -256,21 +258,24 @@ logs: - ## This option represents the number of log lines Traefik will keep in memory before writing - ## them to the selected output. In some cases, this option can greatly help performances. - # bufferingSize: 100 -- ## Filtering https://docs.traefik.io/observability/access-logs/#filtering -+ ## Filtering -+ # -- https://docs.traefik.io/observability/access-logs/#filtering - filters: {} - # statuscodes: "200,300-302" - # retryattempts: true - # minduration: 10ms -- ## Fields -- ## https://docs.traefik.io/observability/access-logs/#limiting-the-fieldsincluding-headers - fields: - general: -+ # -- Available modes: keep, drop, redact. - defaultmode: keep -+ # -- Names of the fields to limit. - names: {} - ## Examples: - # ClientUsername: drop - headers: -+ # -- Available modes: keep, drop, redact. - defaultmode: drop -+ # -- Names of the headers to limit. - names: {} - ## Examples: - # User-Agent: redact -@@ -278,10 +283,10 @@ logs: - # Content-Type: keep - - metrics: -- ## Prometheus is enabled by default. -- ## It can be disabled by setting "prometheus: null" -+ ## -- Prometheus is enabled by default. -+ ## -- It can be disabled by setting "prometheus: null" - prometheus: -- ## Entry point used to expose metrics. -+ # -- Entry point used to expose metrics. - entryPoint: metrics - ## Enable metrics on entry points. Default=true - # addEntryPointsLabels: false -@@ -404,11 +409,9 @@ metrics: - # ## This instructs the reporter to send metrics to the OpenTelemetry Collector using gRPC. - # grpc: true - --## --## enable optional CRDs for Prometheus Operator -+## -- enable optional CRDs for Prometheus Operator - ## - ## Create a dedicated metrics service for use with ServiceMonitor -- ## When hub.enabled is set to true, it's not needed: it will use hub service. - # service: - # enabled: false - # labels: {} -@@ -455,6 +458,8 @@ metrics: - # summary: "Traefik Down" - # description: "{{ $labels.pod }} on {{ $labels.nodename }} is down" - -+## Tracing -+# -- https://doc.traefik.io/traefik/observability/tracing/overview/ - tracing: {} - # instana: - # localAgentHost: 127.0.0.1 -@@ -497,20 +502,21 @@ tracing: {} - # secretToken: "" - # serviceEnvironment: "" - -+# -- Global command arguments to be passed to all traefik's pods - globalArguments: - - "--global.checknewversion" - - "--global.sendanonymoususage" - - # - # Configure Traefik static configuration --# Additional arguments to be passed at Traefik's binary -+# -- Additional arguments to be passed at Traefik's binary - # All available options available on https://docs.traefik.io/reference/static-configuration/cli/ - ## Use curly braces to pass values: `helm install --set="additionalArguments={--providers.kubernetesingress.ingressclass=traefik-internal,--log.level=DEBUG}"` - additionalArguments: [] - # - "--providers.kubernetesingress.ingressclass=traefik-internal" - # - "--log.level=DEBUG" - --# Environment variables to be passed to Traefik's binary -+# -- Environment variables to be passed to Traefik's binary - env: [] - # - name: SOME_VAR - # value: some-var-value -@@ -525,22 +531,20 @@ env: [] - # name: secret-name - # key: secret-key - -+# -- Environment variables to be passed to Traefik's binary from configMaps or secrets - envFrom: [] - # - configMapRef: - # name: config-map-name - # - secretRef: - # name: secret-name - --# Configure ports - ports: -- # The name of this one can't be changed as it is used for the readiness and -- # liveness probes, but you can adjust its config to your liking - traefik: - port: 9000 -- # Use hostPort if set. -+ # -- Use hostPort if set. - # hostPort: 9000 - # -- # Use hostIP if set. If not set, Kubernetes will default to 0.0.0.0, which -+ # -- Use hostIP if set. If not set, Kubernetes will default to 0.0.0.0, which - # means it's listening on all your interfaces and all your IPs. You may want - # to set this value if you need traefik to listen on specific interface - # only. -@@ -558,27 +562,27 @@ ports: - # Defines whether the port is exposed if service.type is LoadBalancer or - # NodePort. - # -- # You SHOULD NOT expose the traefik port on production deployments. -+ # -- You SHOULD NOT expose the traefik port on production deployments. - # If you want to access it from outside of your cluster, - # use `kubectl port-forward` or create a secure ingress - expose: false -- # The exposed port for this service -+ # -- The exposed port for this service - exposedPort: 9000 -- # The port protocol (TCP/UDP) -+ # -- The port protocol (TCP/UDP) - protocol: TCP - web: -- ## Enable this entrypoint as a default entrypoint. When a service doesn't explicity set an entrypoint it will only use this entrypoint. -+ ## -- Enable this entrypoint as a default entrypoint. When a service doesn't explicity set an entrypoint it will only use this entrypoint. - # asDefault: true - port: 8000 - # hostPort: 8000 - # containerPort: 8000 - expose: true - exposedPort: 80 -- ## Different target traefik port on the cluster, useful for IP type LB -+ ## -- Different target traefik port on the cluster, useful for IP type LB - # targetPort: 80 - # The port protocol (TCP/UDP) - protocol: TCP -- # Use nodeport if set. This is useful if you have configured Traefik in a -+ # -- Use nodeport if set. This is useful if you have configured Traefik in a - # LoadBalancer. - # nodePort: 32080 - # Port Redirections -@@ -596,20 +600,22 @@ ports: - # trustedIPs: [] - # insecure: false - websecure: -- ## Enable this entrypoint as a default entrypoint. When a service doesn't explicity set an entrypoint it will only use this entrypoint. -+ ## -- Enable this entrypoint as a default entrypoint. When a service doesn't explicity set an entrypoint it will only use this entrypoint. - # asDefault: true - port: 8443 - # hostPort: 8443 - # containerPort: 8443 - expose: true - exposedPort: 443 -- ## Different target traefik port on the cluster, useful for IP type LB -+ ## -- Different target traefik port on the cluster, useful for IP type LB - # targetPort: 80 -- ## The port protocol (TCP/UDP) -+ ## -- The port protocol (TCP/UDP) - protocol: TCP - # nodePort: 32443 -+ ## -- Specify an application protocol. This may be used as a hint for a Layer 7 load balancer. -+ # appProtocol: https - # -- ## Enable HTTP/3 on the entrypoint -+ ## -- Enable HTTP/3 on the entrypoint - ## Enabling it will also enable http3 experimental feature - ## https://doc.traefik.io/traefik/routing/entrypoints/#http3 - ## There are known limitations when trying to listen on same ports for -@@ -619,12 +625,12 @@ ports: - enabled: false - # advertisedPort: 4443 - # -- ## Trust forwarded headers information (X-Forwarded-*). -+ ## -- Trust forwarded headers information (X-Forwarded-*). - #forwardedHeaders: - # trustedIPs: [] - # insecure: false - # -- ## Enable the Proxy Protocol header parsing for the entry point -+ ## -- Enable the Proxy Protocol header parsing for the entry point - #proxyProtocol: - # trustedIPs: [] - # insecure: false -@@ -642,33 +648,33 @@ ports: - # - foo.example.com - # - bar.example.com - # -- # One can apply Middlewares on an entrypoint -+ # -- One can apply Middlewares on an entrypoint - # https://doc.traefik.io/traefik/middlewares/overview/ - # https://doc.traefik.io/traefik/routing/entrypoints/#middlewares -- # /!\ It introduces here a link between your static configuration and your dynamic configuration /!\ -+ # -- /!\ It introduces here a link between your static configuration and your dynamic configuration /!\ - # It follows the provider naming convention: https://doc.traefik.io/traefik/providers/overview/#provider-namespace - # middlewares: - # - namespace-name1@kubernetescrd - # - namespace-name2@kubernetescrd - middlewares: [] - metrics: -- # When using hostNetwork, use another port to avoid conflict with node exporter: -+ # -- When using hostNetwork, use another port to avoid conflict with node exporter: - # https://github.com/prometheus/prometheus/wiki/Default-port-allocations - port: 9100 - # hostPort: 9100 - # Defines whether the port is exposed if service.type is LoadBalancer or - # NodePort. - # -- # You may not want to expose the metrics port on production deployments. -+ # -- You may not want to expose the metrics port on production deployments. - # If you want to access it from outside of your cluster, - # use `kubectl port-forward` or create a secure ingress - expose: false -- # The exposed port for this service -+ # -- The exposed port for this service - exposedPort: 9100 -- # The port protocol (TCP/UDP) -+ # -- The port protocol (TCP/UDP) - protocol: TCP - --# TLS Options are created as TLSOption CRDs -+# -- TLS Options are created as TLSOption CRDs - # https://doc.traefik.io/traefik/https/tls/#tls-options - # When using `labelSelector`, you'll need to set labels on tlsOption accordingly. - # Example: -@@ -684,7 +690,7 @@ ports: - # - CurveP384 - tlsOptions: {} - --# TLS Store are created as TLSStore CRDs. This is useful if you want to set a default certificate -+# -- TLS Store are created as TLSStore CRDs. This is useful if you want to set a default certificate - # https://doc.traefik.io/traefik/https/tls/#default-certificate - # Example: - # tlsStore: -@@ -693,24 +699,22 @@ tlsOptions: {} - # secretName: tls-cert - tlsStore: {} - --# Options for the main traefik service, where the entrypoints traffic comes --# from. - service: - enabled: true -- ## Single service is using `MixedProtocolLBService` feature gate. -- ## When set to false, it will create two Service, one for TCP and one for UDP. -+ ## -- Single service is using `MixedProtocolLBService` feature gate. -+ ## -- When set to false, it will create two Service, one for TCP and one for UDP. - single: true - type: LoadBalancer -- # Additional annotations applied to both TCP and UDP services (e.g. for cloud provider specific config) -+ # -- Additional annotations applied to both TCP and UDP services (e.g. for cloud provider specific config) - annotations: {} -- # Additional annotations for TCP service only -+ # -- Additional annotations for TCP service only - annotationsTCP: {} -- # Additional annotations for UDP service only -+ # -- Additional annotations for UDP service only - annotationsUDP: {} -- # Additional service labels (e.g. for filtering Service by custom labels) -+ # -- Additional service labels (e.g. for filtering Service by custom labels) - labels: {} -- # Additional entries here will be added to the service spec. -- # Cannot contain type, selector or ports entries. -+ # -- Additional entries here will be added to the service spec. -+ # -- Cannot contain type, selector or ports entries. - spec: {} - # externalTrafficPolicy: Cluster - # loadBalancerIP: "1.2.3.4" -@@ -718,6 +722,8 @@ service: - loadBalancerSourceRanges: [] - # - 192.168.0.1/32 - # - 172.16.0.0/16 -+ ## -- Class of the load balancer implementation -+ # loadBalancerClass: service.k8s.aws/nlb - externalIPs: [] - # - 1.2.3.4 - ## One of SingleStack, PreferDualStack, or RequireDualStack. -@@ -728,7 +734,7 @@ service: - # - IPv4 - # - IPv6 - ## -- ## An additionnal and optional internal Service. -+ ## -- An additionnal and optional internal Service. - ## Same parameters as external Service - # internal: - # type: ClusterIP -@@ -739,9 +745,8 @@ service: - # # externalIPs: [] - # # ipFamilies: [ "IPv4","IPv6" ] - --## Create HorizontalPodAutoscaler object. --## - autoscaling: -+ # -- Create HorizontalPodAutoscaler object. - enabled: false - # minReplicas: 1 - # maxReplicas: 10 -@@ -766,10 +771,10 @@ autoscaling: - # value: 1 - # periodSeconds: 60 - --# Enable persistence using Persistent Volume Claims --# ref: http://kubernetes.io/docs/user-guide/persistent-volumes/ --# It can be used to store TLS certificates, see `storage` in certResolvers - persistence: -+ # -- Enable persistence using Persistent Volume Claims -+ # ref: http://kubernetes.io/docs/user-guide/persistent-volumes/ -+ # It can be used to store TLS certificates, see `storage` in certResolvers - enabled: false - name: data - # existingClaim: "" -@@ -779,8 +784,10 @@ persistence: - # volumeName: "" - path: /data - annotations: {} -- # subPath: "" # only mount a subpath of the Volume into the pod -+ # -- Only mount a subpath of the Volume into the pod -+ # subPath: "" - -+# -- Certificates resolvers configuration - certResolvers: {} - # letsencrypt: - # # for challenge options cf. https://doc.traefik.io/traefik/https/acme/ -@@ -802,13 +809,13 @@ certResolvers: {} - # # It has to match the path with a persistent volume - # storage: /data/acme.json - --# If hostNetwork is true, runs traefik in the host network namespace -+# -- If hostNetwork is true, runs traefik in the host network namespace - # To prevent unschedulabel pods due to port collisions, if hostNetwork=true - # and replicas>1, a pod anti-affinity is recommended and will be set if the - # affinity is left as default. - hostNetwork: false - --# Whether Role Based Access Control objects like roles and rolebindings should be created -+# -- Whether Role Based Access Control objects like roles and rolebindings should be created - rbac: - enabled: true - # If set to false, installs ClusterRole and ClusterRoleBinding so Traefik can be used across namespaces. -@@ -818,19 +825,20 @@ rbac: - # https://kubernetes.io/docs/reference/access-authn-authz/rbac/#user-facing-roles - # aggregateTo: [ "admin" ] - --# Enable to create a PodSecurityPolicy and assign it to the Service Account via RoleBinding or ClusterRoleBinding -+# -- Enable to create a PodSecurityPolicy and assign it to the Service Account via RoleBinding or ClusterRoleBinding - podSecurityPolicy: - enabled: false - --# The service account the pods will use to interact with the Kubernetes API -+# -- The service account the pods will use to interact with the Kubernetes API - serviceAccount: - # If set, an existing service account is used - # If not set, a service account is created automatically using the fullname template - name: "" - --# Additional serviceAccount annotations (e.g. for oidc authentication) -+# -- Additional serviceAccount annotations (e.g. for oidc authentication) - serviceAccountAnnotations: {} - -+# -- The resources parameter defines CPU and memory requirements and limits for Traefik's containers. - resources: {} - # requests: - # cpu: "100m" -@@ -839,8 +847,8 @@ resources: {} - # cpu: "300m" - # memory: "150Mi" - --# This example pod anti-affinity forces the scheduler to put traefik pods --# on nodes where no other traefik pods are scheduled. -+# -- This example pod anti-affinity forces the scheduler to put traefik pods -+# -- on nodes where no other traefik pods are scheduled. - # It should be used when hostNetwork: true to prevent port conflicts - affinity: {} - # podAntiAffinity: -@@ -851,11 +859,15 @@ affinity: {} - # app.kubernetes.io/instance: '{{ .Release.Name }}-{{ .Release.Namespace }}' - # topologyKey: kubernetes.io/hostname - -+# -- nodeSelector is the simplest recommended form of node selection constraint. - nodeSelector: {} -+# -- Tolerations allow the scheduler to schedule pods with matching taints. - tolerations: [] -+# -- You can use topology spread constraints to control -+# how Pods are spread across your cluster among failure-domains. - topologySpreadConstraints: [] --# # This example topologySpreadConstraints forces the scheduler to put traefik pods --# # on nodes where no other traefik pods are scheduled. -+# This example topologySpreadConstraints forces the scheduler to put traefik pods -+# on nodes where no other traefik pods are scheduled. - # - labelSelector: - # matchLabels: - # app: '{{ template "traefik.name" . }}' -@@ -863,29 +875,33 @@ topologySpreadConstraints: [] - # topologyKey: kubernetes.io/hostname - # whenUnsatisfiable: DoNotSchedule - --# Pods can have priority. --# Priority indicates the importance of a Pod relative to other Pods. -+# -- Pods can have priority. -+# -- Priority indicates the importance of a Pod relative to other Pods. - priorityClassName: "" - --# Set the container security context --# To run the container with ports below 1024 this will need to be adjust to run as root -+# -- Set the container security context -+# -- To run the container with ports below 1024 this will need to be adjust to run as root - securityContext: - capabilities: - drop: [ALL] - readOnlyRootFilesystem: true - - podSecurityContext: --# # /!\ When setting fsGroup, Kubernetes will recursively changes ownership and --# # permissions for the contents of each volume to match the fsGroup. This can --# # be an issue when storing sensitive content like TLS Certificates /!\ --# fsGroup: 65532 -+ # /!\ When setting fsGroup, Kubernetes will recursively changes ownership and -+ # permissions for the contents of each volume to match the fsGroup. This can -+ # be an issue when storing sensitive content like TLS Certificates /!\ -+ # fsGroup: 65532 -+ # -- Specifies the policy for changing ownership and permissions of volume contents to match the fsGroup. - fsGroupChangePolicy: "OnRootMismatch" -+ # -- The ID of the group for all containers in the pod to run as. - runAsGroup: 65532 -+ # -- Specifies whether the containers should run as a non-root user. - runAsNonRoot: true -+ # -- The ID of the user for all containers in the pod to run as. - runAsUser: 65532 - - # --# Extra objects to deploy (value evaluated as a template) -+# -- Extra objects to deploy (value evaluated as a template) - # - # In some cases, it can avoid the need for additional, extended or adhoc deployments. - # See #595 for more details and traefik/tests/values/extra.yaml for example. -@@ -895,5 +911,5 @@ extraObjects: [] - # It will not affect optional CRDs such as `ServiceMonitor` and `PrometheusRules` - # namespaceOverride: traefik - # --## This will override the default app.kubernetes.io/instance label for all Objects. -+## -- This will override the default app.kubernetes.io/instance label for all Objects. - # instanceLabelOverride: traefik -``` - -## 23.0.1 ![AppVersion: v2.10.1](https://img.shields.io/static/v1?label=AppVersion&message=v2.10.1&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-04-28 - -* fix: ⬆️ Upgrade traefik Docker tag to v2.10.1 - - -## 23.0.0 ![AppVersion: v2.10.0](https://img.shields.io/static/v1?label=AppVersion&message=v2.10.0&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-04-26 - -* BREAKING CHANGE: Traefik 2.10 comes with CRDs update on API Group - - -## 22.3.0 ![AppVersion: v2.10.0](https://img.shields.io/static/v1?label=AppVersion&message=v2.10.0&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-04-25 - -* ⬆️ Upgrade traefik Docker tag to v2.10.0 -* fix: 🐛 update rbac for both traefik.io and containo.us apigroups (#836) -* breaking: 💥 update CRDs needed for Traefik v2.10 - - -## 22.2.0 ![AppVersion: v2.9.10](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.10&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-04-24 - -* test: 👷 Update unit tests tooling -* fix: 🐛 annotations leaking between aliased subcharts -* fix: indentation on `TLSOption` -* feat: override container port -* feat: allow to set dnsConfig on pod template -* chore: 🔧 new release -* added targetPort support - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 9ece303..71273cc 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -82,6 +82,16 @@ deployment: - shareProcessNamespace: false - # Custom pod DNS policy. Apply if `hostNetwork: true` - # dnsPolicy: ClusterFirstWithHostNet -+ dnsConfig: {} -+ # nameservers: -+ # - 192.0.2.1 # this is an example -+ # searches: -+ # - ns1.svc.cluster-domain.example -+ # - my.dns.search.suffix -+ # options: -+ # - name: ndots -+ # value: "2" -+ # - name: edns0 - # Additional imagePullSecrets - imagePullSecrets: [] - # - name: myRegistryKeySecretName -@@ -561,8 +571,11 @@ ports: - # asDefault: true - port: 8000 - # hostPort: 8000 -+ # containerPort: 8000 - expose: true - exposedPort: 80 -+ ## Different target traefik port on the cluster, useful for IP type LB -+ # targetPort: 80 - # The port protocol (TCP/UDP) - protocol: TCP - # Use nodeport if set. This is useful if you have configured Traefik in a -@@ -587,8 +600,11 @@ ports: - # asDefault: true - port: 8443 - # hostPort: 8443 -+ # containerPort: 8443 - expose: true - exposedPort: 443 -+ ## Different target traefik port on the cluster, useful for IP type LB -+ # targetPort: 80 - ## The port protocol (TCP/UDP) - protocol: TCP - # nodePort: 32443 -``` - -## 22.1.0 ![AppVersion: v2.9.10](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.10&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-04-07 - -* ⬆️ Upgrade traefik Docker tag to v2.9.10 -* feat: add additional labels to tlsoption - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 4762b77..9ece303 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -654,12 +654,15 @@ ports: - - # TLS Options are created as TLSOption CRDs - # https://doc.traefik.io/traefik/https/tls/#tls-options -+# When using `labelSelector`, you'll need to set labels on tlsOption accordingly. - # Example: - # tlsOptions: - # default: -+# labels: {} - # sniStrict: true - # preferServerCipherSuites: true --# foobar: -+# customOptions: -+# labels: {} - # curvePreferences: - # - CurveP521 - # - CurveP384 -``` - -## 22.0.0 ![AppVersion: v2.9.9](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.9&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-03-29 - -* BREAKING CHANGE: `image.repository` introduction may break during the upgrade. See PR #802. - - -## 21.2.1 ![AppVersion: v2.9.9](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.9&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-03-28 - -* 🎨 Introduce `image.registry` and add explicit default (it may impact custom `image.repository`) -* ⬆️ Upgrade traefik Docker tag to v2.9.9 -* :memo: Clarify the need of an initContainer when enabling persistence for TLS Certificates - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index cadc7a6..4762b77 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1,5 +1,6 @@ - # Default values for Traefik - image: -+ registry: docker.io - repository: traefik - # defaults to appVersion - tag: "" -@@ -66,10 +67,14 @@ deployment: - # Additional initContainers (e.g. for setting file permission as shown below) - initContainers: [] - # The "volume-permissions" init container is required if you run into permission issues. -- # Related issue: https://github.com/traefik/traefik/issues/6825 -+ # Related issue: https://github.com/traefik/traefik-helm-chart/issues/396 - # - name: volume-permissions -- # image: busybox:1.35 -- # command: ["sh", "-c", "touch /data/acme.json && chmod -Rv 600 /data/* && chown 65532:65532 /data/acme.json"] -+ # image: busybox:latest -+ # command: ["sh", "-c", "touch /data/acme.json; chmod -v 600 /data/acme.json"] -+ # securityContext: -+ # runAsNonRoot: true -+ # runAsGroup: 65532 -+ # runAsUser: 65532 - # volumeMounts: - # - name: data - # mountPath: /data -@@ -849,13 +854,17 @@ securityContext: - capabilities: - drop: [ALL] - readOnlyRootFilesystem: true -+ -+podSecurityContext: -+# # /!\ When setting fsGroup, Kubernetes will recursively changes ownership and -+# # permissions for the contents of each volume to match the fsGroup. This can -+# # be an issue when storing sensitive content like TLS Certificates /!\ -+# fsGroup: 65532 -+ fsGroupChangePolicy: "OnRootMismatch" - runAsGroup: 65532 - runAsNonRoot: true - runAsUser: 65532 - --podSecurityContext: -- fsGroup: 65532 -- - # - # Extra objects to deploy (value evaluated as a template) - # -``` - -## 21.2.0 ![AppVersion: v2.9.8](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.8&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-03-08 - -* 🚨 Fail when enabling PSP on Kubernetes v1.25+ (#801) -* ⬆️ Upgrade traefik Docker tag to v2.9.8 -* Separate UDP hostPort for HTTP/3 -* :sparkles: release 21.2.0 (#805) - - -## 21.1.0 ![AppVersion: v2.9.7](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.7&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-02-15 - -* ⬆️ Upgrade traefik Docker tag to v2.9.7 -* ✨ release 21.1.0 -* fix: traefik image name for renovate -* feat: Add volumeName to PersistentVolumeClaim (#792) -* Allow setting TLS options on dashboard IngressRoute - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 780b04b..cadc7a6 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -142,6 +142,8 @@ ingressRoute: - entryPoints: ["traefik"] - # Additional ingressRoute middlewares (e.g. for authentication) - middlewares: [] -+ # TLS options (e.g. secret containing certificate) -+ tls: {} - - # Customize updateStrategy of traefik pods - updateStrategy: -@@ -750,6 +752,7 @@ persistence: - accessMode: ReadWriteOnce - size: 128Mi - # storageClass: "" -+ # volumeName: "" - path: /data - annotations: {} - # subPath: "" # only mount a subpath of the Volume into the pod -``` - -## 21.0.0 ![AppVersion: v2.9.6](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.6&color=success&logo=) ![Kubernetes: >=1.16.0-0](https://img.shields.io/static/v1?label=Kubernetes&message=%3E%3D1.16.0-0&color=informational&logo=kubernetes) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2023-02-10 - -* 🙈 Add a setting disable API check on Prometheus Operator (#769) -* 📝 Improve documentation on entrypoint options -* 💥 New release with BREAKING changes (#786) -* ✨ Chart.yaml - add kubeVersion: ">=1.16.0-0" -* fix: allowExternalNameServices for kubernetes ingress when hub enabled (#772) -* fix(service-metrics): invert prometheus svc & fullname length checking -* Configure Renovate (#783) -* :necktie: Improve labels settings behavior on metrics providers (#774) -* :bug: Disabling dashboard ingressroute should delete it (#785) -* :boom: Rename image.name => image.repository (#784) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 42a27f9..780b04b 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1,6 +1,6 @@ - # Default values for Traefik - image: -- name: traefik -+ repository: traefik - # defaults to appVersion - tag: "" - pullPolicy: IfNotPresent -@@ -396,6 +396,8 @@ metrics: - # enabled: false - # labels: {} - # annotations: {} -+ ## When set to true, it won't check if Prometheus Operator CRDs are deployed -+ # disableAPICheck: false - # serviceMonitor: - # metricRelabelings: [] - # - sourceLabels: [__name__] -@@ -580,7 +582,7 @@ ports: - # hostPort: 8443 - expose: true - exposedPort: 443 -- # The port protocol (TCP/UDP) -+ ## The port protocol (TCP/UDP) - protocol: TCP - # nodePort: 32443 - # -@@ -594,6 +596,16 @@ ports: - enabled: false - # advertisedPort: 4443 - # -+ ## Trust forwarded headers information (X-Forwarded-*). -+ #forwardedHeaders: -+ # trustedIPs: [] -+ # insecure: false -+ # -+ ## Enable the Proxy Protocol header parsing for the entry point -+ #proxyProtocol: -+ # trustedIPs: [] -+ # insecure: false -+ # - ## Set TLS at the entrypoint - ## https://doc.traefik.io/traefik/routing/entrypoints/#tls - tls: -@@ -607,16 +619,6 @@ ports: - # - foo.example.com - # - bar.example.com - # -- # Trust forwarded headers information (X-Forwarded-*). -- # forwardedHeaders: -- # trustedIPs: [] -- # insecure: false -- # -- # Enable the Proxy Protocol header parsing for the entry point -- # proxyProtocol: -- # trustedIPs: [] -- # insecure: false -- # - # One can apply Middlewares on an entrypoint - # https://doc.traefik.io/traefik/middlewares/overview/ - # https://doc.traefik.io/traefik/routing/entrypoints/#middlewares -``` - -## 20.8.0 ![AppVersion: v2.9.6](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.6&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-12-09 - -* ✨ update chart to version 20.8.0 -* ✨ add support for default entrypoints -* ✨ add support for OpenTelemetry and Traefik v3 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index b77539d..42a27f9 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -107,6 +107,8 @@ ingressClass: - - # Enable experimental features - experimental: -+ v3: -+ enabled: false - plugins: - enabled: false - kubernetesGateway: -@@ -347,7 +349,43 @@ metrics: - # # addRoutersLabels: true - # ## Enable metrics on services. Default=true - # # addServicesLabels: false -- -+# openTelemetry: -+# ## Address of the OpenTelemetry Collector to send metrics to. -+# address: "localhost:4318" -+# ## Enable metrics on entry points. -+# addEntryPointsLabels: true -+# ## Enable metrics on routers. -+# addRoutersLabels: true -+# ## Enable metrics on services. -+# addServicesLabels: true -+# ## Explicit boundaries for Histogram data points. -+# explicitBoundaries: -+# - "0.1" -+# - "0.3" -+# - "1.2" -+# - "5.0" -+# ## Additional headers sent with metrics by the reporter to the OpenTelemetry Collector. -+# headers: -+# foo: bar -+# test: test -+# ## Allows reporter to send metrics to the OpenTelemetry Collector without using a secured protocol. -+# insecure: true -+# ## Interval at which metrics are sent to the OpenTelemetry Collector. -+# pushInterval: 10s -+# ## Allows to override the default URL path used for sending metrics. This option has no effect when using gRPC transport. -+# path: /foo/v1/traces -+# ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. -+# tls: -+# ## The path to the certificate authority, it defaults to the system bundle. -+# ca: path/to/ca.crt -+# ## The path to the public certificate. When using this option, setting the key option is required. -+# cert: path/to/foo.cert -+# ## The path to the private key. When using this option, setting the cert option is required. -+# key: path/to/key.key -+# ## If set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. -+# insecureSkipVerify: true -+# ## This instructs the reporter to send metrics to the OpenTelemetry Collector using gRPC. -+# grpc: true - - ## - ## enable optional CRDs for Prometheus Operator -@@ -510,6 +548,8 @@ ports: - # The port protocol (TCP/UDP) - protocol: TCP - web: -+ ## Enable this entrypoint as a default entrypoint. When a service doesn't explicity set an entrypoint it will only use this entrypoint. -+ # asDefault: true - port: 8000 - # hostPort: 8000 - expose: true -@@ -534,6 +574,8 @@ ports: - # trustedIPs: [] - # insecure: false - websecure: -+ ## Enable this entrypoint as a default entrypoint. When a service doesn't explicity set an entrypoint it will only use this entrypoint. -+ # asDefault: true - port: 8443 - # hostPort: 8443 - expose: true -``` - -## 20.7.0 ![AppVersion: v2.9.6](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.6&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-12-08 - -* 🐛 Don't fail when prometheus is disabled (#756) -* ⬆️ Update default Traefik release to v2.9.6 (#758) -* ✨ support for Gateway annotations -* add keywords [networking], for artifacthub category quering -* :bug: Fix typo on bufferingSize for access logs (#753) -* :adhesive_bandage: Add quotes for artifacthub changelog parsing (#748) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 4f2fb2a..b77539d 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -120,6 +120,9 @@ experimental: - # By default, Gateway would be created to the Namespace you are deploying Traefik to. - # You may create that Gateway in another namespace, setting its name below: - # namespace: default -+ # Additional gateway annotations (e.g. for cert-manager.io/issuer) -+ # annotations: -+ # cert-manager.io/issuer: letsencrypt - - # Create an IngressRoute for the dashboard - ingressRoute: -@@ -219,7 +222,8 @@ logs: - # By default, the logs use a text format (common), but you can - # also ask for the json format in the format option - # format: json -- # By default, the level is set to ERROR. Alternative logging levels are DEBUG, PANIC, FATAL, ERROR, WARN, and INFO. -+ # By default, the level is set to ERROR. -+ # Alternative logging levels are DEBUG, PANIC, FATAL, ERROR, WARN, and INFO. - level: ERROR - access: - # To enable access logs -``` - -## 20.6.0 ![AppVersion: v2.9.5](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.5&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-30 - -* 🔍️ Add filePath support on access logs (#747) -* :memo: Improve documentation on using PVC with TLS certificates -* :bug: Add missing scheme in help on Traefik Hub integration (#746) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 15f1682..4f2fb2a 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -211,10 +211,10 @@ additionalVolumeMounts: [] - # - name: traefik-logs - # mountPath: /var/log/traefik - --# Logs --# https://docs.traefik.io/observability/logs/ -+## Logs -+## https://docs.traefik.io/observability/logs/ - logs: -- # Traefik logs concern everything that happens to Traefik itself (startup, configuration, events, shutdown, and so on). -+ ## Traefik logs concern everything that happens to Traefik itself (startup, configuration, events, shutdown, and so on). - general: - # By default, the logs use a text format (common), but you can - # also ask for the json format in the format option -@@ -224,31 +224,32 @@ logs: - access: - # To enable access logs - enabled: false -- # By default, logs are written using the Common Log Format (CLF). -- # To write logs in JSON, use json in the format option. -- # If the given format is unsupported, the default (CLF) is used instead. -+ ## By default, logs are written using the Common Log Format (CLF) on stdout. -+ ## To write logs in JSON, use json in the format option. -+ ## If the given format is unsupported, the default (CLF) is used instead. - # format: json -- # To write the logs in an asynchronous fashion, specify a bufferingSize option. -- # This option represents the number of log lines Traefik will keep in memory before writing -- # them to the selected output. In some cases, this option can greatly help performances. -+ # filePath: "/var/log/traefik/access.log -+ ## To write the logs in an asynchronous fashion, specify a bufferingSize option. -+ ## This option represents the number of log lines Traefik will keep in memory before writing -+ ## them to the selected output. In some cases, this option can greatly help performances. - # bufferingSize: 100 -- # Filtering https://docs.traefik.io/observability/access-logs/#filtering -+ ## Filtering https://docs.traefik.io/observability/access-logs/#filtering - filters: {} - # statuscodes: "200,300-302" - # retryattempts: true - # minduration: 10ms -- # Fields -- # https://docs.traefik.io/observability/access-logs/#limiting-the-fieldsincluding-headers -+ ## Fields -+ ## https://docs.traefik.io/observability/access-logs/#limiting-the-fieldsincluding-headers - fields: - general: - defaultmode: keep - names: {} -- # Examples: -+ ## Examples: - # ClientUsername: drop - headers: - defaultmode: drop - names: {} -- # Examples: -+ ## Examples: - # User-Agent: redact - # Authorization: drop - # Content-Type: keep -@@ -693,10 +694,7 @@ autoscaling: - - # Enable persistence using Persistent Volume Claims - # ref: http://kubernetes.io/docs/user-guide/persistent-volumes/ --# After the pvc has been mounted, add the configs into traefik by using the `additionalArguments` list below, eg: --# additionalArguments: --# - "--certificatesresolvers.le.acme.storage=/data/acme.json" --# It will persist TLS certificates. -+# It can be used to store TLS certificates, see `storage` in certResolvers - persistence: - enabled: false - name: data -@@ -726,7 +724,7 @@ certResolvers: {} - # tlsChallenge: true - # httpChallenge: - # entryPoint: "web" --# # match the path to persistence -+# # It has to match the path with a persistent volume - # storage: /data/acme.json - - # If hostNetwork is true, runs traefik in the host network namespace -``` - -## 20.5.3 ![AppVersion: v2.9.5](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.5&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-25 - -* 🐛 Fix template issue with obsolete helm version + add helm version requirement (#743) - - -## 20.5.2 ![AppVersion: v2.9.5](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.5&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-24 - -* ⬆️Update Traefik to v2.9.5 (#740) - - -## 20.5.1 ![AppVersion: v2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-23 - -* 🐛 Fix namespaceSelector on ServiceMonitor (#737) - - -## 20.5.0 ![AppVersion: v2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-23 - -* 🚀 Add complete support on metrics options (#735) -* 🐛 make tests use fixed version - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index e49d02d..15f1682 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -12,7 +12,7 @@ hub: - ## Enabling Hub will: - # * enable Traefik Hub integration on Traefik - # * add `traefikhub-tunl` endpoint -- # * enable addRoutersLabels on prometheus metrics -+ # * enable Prometheus metrics with addRoutersLabels - # * enable allowExternalNameServices on KubernetesIngress provider - # * enable allowCrossNamespace on KubernetesCRD provider - # * add an internal (ClusterIP) Service, dedicated for Traefik Hub -@@ -254,16 +254,96 @@ logs: - # Content-Type: keep - - metrics: -- # datadog: -- # address: 127.0.0.1:8125 -- # influxdb: -- # address: localhost:8089 -- # protocol: udp -+ ## Prometheus is enabled by default. -+ ## It can be disabled by setting "prometheus: null" - prometheus: -+ ## Entry point used to expose metrics. - entryPoint: metrics -- # addRoutersLabels: true -- # statsd: -- # address: localhost:8125 -+ ## Enable metrics on entry points. Default=true -+ # addEntryPointsLabels: false -+ ## Enable metrics on routers. Default=false -+ # addRoutersLabels: true -+ ## Enable metrics on services. Default=true -+ # addServicesLabels: false -+ ## Buckets for latency metrics. Default="0.1,0.3,1.2,5.0" -+ # buckets: "0.5,1.0,2.5" -+ ## When manualRouting is true, it disables the default internal router in -+ ## order to allow creating a custom router for prometheus@internal service. -+ # manualRouting: true -+# datadog: -+# ## Address instructs exporter to send metrics to datadog-agent at this address. -+# address: "127.0.0.1:8125" -+# ## The interval used by the exporter to push metrics to datadog-agent. Default=10s -+# # pushInterval: 30s -+# ## The prefix to use for metrics collection. Default="traefik" -+# # prefix: traefik -+# ## Enable metrics on entry points. Default=true -+# # addEntryPointsLabels: false -+# ## Enable metrics on routers. Default=false -+# # addRoutersLabels: true -+# ## Enable metrics on services. Default=true -+# # addServicesLabels: false -+# influxdb: -+# ## Address instructs exporter to send metrics to influxdb at this address. -+# address: localhost:8089 -+# ## InfluxDB's address protocol (udp or http). Default="udp" -+# protocol: udp -+# ## InfluxDB database used when protocol is http. Default="" -+# # database: "" -+# ## InfluxDB retention policy used when protocol is http. Default="" -+# # retentionPolicy: "" -+# ## InfluxDB username (only with http). Default="" -+# # username: "" -+# ## InfluxDB password (only with http). Default="" -+# # password: "" -+# ## The interval used by the exporter to push metrics to influxdb. Default=10s -+# # pushInterval: 30s -+# ## Additional labels (influxdb tags) on all metrics. -+# # additionalLabels: -+# # env: production -+# # foo: bar -+# ## Enable metrics on entry points. Default=true -+# # addEntryPointsLabels: false -+# ## Enable metrics on routers. Default=false -+# # addRoutersLabels: true -+# ## Enable metrics on services. Default=true -+# # addServicesLabels: false -+# influxdb2: -+# ## Address instructs exporter to send metrics to influxdb v2 at this address. -+# address: localhost:8086 -+# ## Token with which to connect to InfluxDB v2. -+# token: xxx -+# ## Organisation where metrics will be stored. -+# org: "" -+# ## Bucket where metrics will be stored. -+# bucket: "" -+# ## The interval used by the exporter to push metrics to influxdb. Default=10s -+# # pushInterval: 30s -+# ## Additional labels (influxdb tags) on all metrics. -+# # additionalLabels: -+# # env: production -+# # foo: bar -+# ## Enable metrics on entry points. Default=true -+# # addEntryPointsLabels: false -+# ## Enable metrics on routers. Default=false -+# # addRoutersLabels: true -+# ## Enable metrics on services. Default=true -+# # addServicesLabels: false -+# statsd: -+# ## Address instructs exporter to send metrics to statsd at this address. -+# address: localhost:8125 -+# ## The interval used by the exporter to push metrics to influxdb. Default=10s -+# # pushInterval: 30s -+# ## The prefix to use for metrics collection. Default="traefik" -+# # prefix: traefik -+# ## Enable metrics on entry points. Default=true -+# # addEntryPointsLabels: false -+# ## Enable metrics on routers. Default=false -+# # addRoutersLabels: true -+# ## Enable metrics on services. Default=true -+# # addServicesLabels: false -+ -+ - ## - ## enable optional CRDs for Prometheus Operator - ## -``` - -## 20.4.1 ![AppVersion: v2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-21 - -* 🐛 fix namespace references to support namespaceOverride - - -## 20.4.0 ![AppVersion: v2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-21 - -* Add (optional) dedicated metrics service (#727) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index ca15f6a..e49d02d 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -267,6 +267,12 @@ metrics: - ## - ## enable optional CRDs for Prometheus Operator - ## -+ ## Create a dedicated metrics service for use with ServiceMonitor -+ ## When hub.enabled is set to true, it's not needed: it will use hub service. -+ # service: -+ # enabled: false -+ # labels: {} -+ # annotations: {} - # serviceMonitor: - # metricRelabelings: [] - # - sourceLabels: [__name__] -``` - -## 20.3.1 ![AppVersion: v2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-21 - -* 🐛 Fix namespace override which was missing on `ServiceAccount` (#731) - - -## 20.3.0 ![AppVersion: v2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-17 - -* Add overwrite option for instance label value (#725) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index c7f84a7..ca15f6a 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -731,3 +731,6 @@ extraObjects: [] - # This will override the default Release Namespace for Helm. - # It will not affect optional CRDs such as `ServiceMonitor` and `PrometheusRules` - # namespaceOverride: traefik -+# -+## This will override the default app.kubernetes.io/instance label for all Objects. -+# instanceLabelOverride: traefik -``` - -## 20.2.1 ![AppVersion: v2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-17 - -* 🙈 do not namespace ingress class (#723) -* ✨ copy LICENSE and README.md on release - - -## 20.2.0 ![AppVersion: v2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-15 - -* ✨ add support for namespace overrides (#718) -* Document recent changes in the README (#717) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 97a1b71..c7f84a7 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -725,5 +725,9 @@ podSecurityContext: - # Extra objects to deploy (value evaluated as a template) - # - # In some cases, it can avoid the need for additional, extended or adhoc deployments. --# See #595 for more details and traefik/tests/extra.yaml for example. -+# See #595 for more details and traefik/tests/values/extra.yaml for example. - extraObjects: [] -+ -+# This will override the default Release Namespace for Helm. -+# It will not affect optional CRDs such as `ServiceMonitor` and `PrometheusRules` -+# namespaceOverride: traefik -``` - -## 20.1.1 ![AppVersion: v2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=v2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-10 - -* fix: use consistent appVersion with Traefik Proxy - - -## 20.1.0 ![AppVersion: 2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-09 - -* 🔧 Adds more settings for dashboard ingressRoute (#710) -* 🐛 fix chart releases - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 2ec3736..97a1b71 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -129,10 +129,14 @@ ingressRoute: - annotations: {} - # Additional ingressRoute labels (e.g. for filtering IngressRoute by custom labels) - labels: {} -+ # The router match rule used for the dashboard ingressRoute -+ matchRule: PathPrefix(`/dashboard`) || PathPrefix(`/api`) - # Specify the allowed entrypoints to use for the dashboard ingress route, (e.g. traefik, web, websecure). - # By default, it's using traefik entrypoint, which is not exposed. - # /!\ Do not expose your dashboard without any protection over the internet /!\ - entryPoints: ["traefik"] -+ # Additional ingressRoute middlewares (e.g. for authentication) -+ middlewares: [] - - # Customize updateStrategy of traefik pods - updateStrategy: -``` - -## 20.0.0 ![AppVersion: 2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-08 - -* 🐛 remove old deployment workflow -* ✨ migrate to centralised helm repository -* Allow updateStrategy to be configurable - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 413aa88..2ec3736 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -134,9 +134,12 @@ ingressRoute: - # /!\ Do not expose your dashboard without any protection over the internet /!\ - entryPoints: ["traefik"] - --rollingUpdate: -- maxUnavailable: 0 -- maxSurge: 1 -+# Customize updateStrategy of traefik pods -+updateStrategy: -+ type: RollingUpdate -+ rollingUpdate: -+ maxUnavailable: 0 -+ maxSurge: 1 - - # Customize liveness and readiness probe values. - readinessProbe: -``` - -## 19.0.4 ![AppVersion: 2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-08 - -* 🔧 Adds more settings & rename (wrong) scrapeInterval to (valid) interval on ServiceMonitor (#703) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index b24c1cb..413aa88 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -261,10 +261,6 @@ metrics: - ## enable optional CRDs for Prometheus Operator - ## - # serviceMonitor: -- # additionalLabels: -- # foo: bar -- # namespace: "another-namespace" -- # namespaceSelector: {} - # metricRelabelings: [] - # - sourceLabels: [__name__] - # separator: ; -@@ -279,9 +275,17 @@ metrics: - # replacement: $1 - # action: replace - # jobLabel: traefik -- # scrapeInterval: 30s -- # scrapeTimeout: 5s -+ # interval: 30s - # honorLabels: true -+ # # (Optional) -+ # # scrapeTimeout: 5s -+ # # honorTimestamps: true -+ # # enableHttp2: true -+ # # followRedirects: true -+ # # additionalLabels: -+ # # foo: bar -+ # # namespace: "another-namespace" -+ # # namespaceSelector: {} - # prometheusRule: - # additionalLabels: {} - # namespace: "another-namespace" -``` - -## 19.0.3 ![AppVersion: 2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-03 - -* 🎨 Don't require exposed Ports when enabling Hub (#700) - - -## 19.0.2 ![AppVersion: 2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-03 - -* :speech_balloon: Support volume secrets with '.' in name (#695) - - -## 19.0.1 ![AppVersion: 2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-03 - -* 🐛 Fix IngressClass install on EKS (#699) - - -## 19.0.0 ![AppVersion: 2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-11-02 - -* ✨ Provides Default IngressClass for Traefik by default (#693) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 69190f1..b24c1cb 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -100,11 +100,10 @@ podDisruptionBudget: - # minAvailable: 0 - # minAvailable: 25% - --# Use ingressClass. Ignored if Traefik version < 2.3 / kubernetes < 1.18.x -+# Create a default IngressClass for Traefik - ingressClass: -- # true is not unit-testable yet, pending https://github.com/rancher/helm-unittest/pull/12 -- enabled: false -- isDefaultClass: false -+ enabled: true -+ isDefaultClass: true - - # Enable experimental features - experimental: -``` - -## 18.3.0 ![AppVersion: 2.9.4](https://img.shields.io/static/v1?label=AppVersion&message=2.9.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-31 - -* ⬆️ Update Traefik appVersion to 2.9.4 (#696) - - -## 18.2.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-31 - -* 🚩 Add an optional "internal" service (#683) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 8033a87..69190f1 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -416,7 +416,7 @@ ports: - # The port protocol (TCP/UDP) - protocol: TCP - # Use nodeport if set. This is useful if you have configured Traefik in a -- # LoadBalancer -+ # LoadBalancer. - # nodePort: 32080 - # Port Redirections - # Added in 2.2, you can make permanent redirects via entrypoints. -@@ -549,13 +549,24 @@ service: - # - 172.16.0.0/16 - externalIPs: [] - # - 1.2.3.4 -- # One of SingleStack, PreferDualStack, or RequireDualStack. -+ ## One of SingleStack, PreferDualStack, or RequireDualStack. - # ipFamilyPolicy: SingleStack -- # List of IP families (e.g. IPv4 and/or IPv6). -- # ref: https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services -+ ## List of IP families (e.g. IPv4 and/or IPv6). -+ ## ref: https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services - # ipFamilies: - # - IPv4 - # - IPv6 -+ ## -+ ## An additionnal and optional internal Service. -+ ## Same parameters as external Service -+ # internal: -+ # type: ClusterIP -+ # # labels: {} -+ # # annotations: {} -+ # # spec: {} -+ # # loadBalancerSourceRanges: [] -+ # # externalIPs: [] -+ # # ipFamilies: [ "IPv4","IPv6" ] - - ## Create HorizontalPodAutoscaler object. - ## -``` - -## 18.1.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-27 - -* 🚀 Add native support for Traefik Hub (#676) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index acce704..8033a87 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -5,6 +5,27 @@ image: - tag: "" - pullPolicy: IfNotPresent - -+# -+# Configure integration with Traefik Hub -+# -+hub: -+ ## Enabling Hub will: -+ # * enable Traefik Hub integration on Traefik -+ # * add `traefikhub-tunl` endpoint -+ # * enable addRoutersLabels on prometheus metrics -+ # * enable allowExternalNameServices on KubernetesIngress provider -+ # * enable allowCrossNamespace on KubernetesCRD provider -+ # * add an internal (ClusterIP) Service, dedicated for Traefik Hub -+ enabled: false -+ ## Default port can be changed -+ # tunnelPort: 9901 -+ ## TLS is optional. Insecure is mutually exclusive with any other options -+ # tls: -+ # insecure: false -+ # ca: "/path/to/ca.pem" -+ # cert: "/path/to/cert.pem" -+ # key: "/path/to/key.pem" -+ - # - # Configure the deployment - # -@@ -505,6 +526,8 @@ tlsStore: {} - # from. - service: - enabled: true -+ ## Single service is using `MixedProtocolLBService` feature gate. -+ ## When set to false, it will create two Service, one for TCP and one for UDP. - single: true - type: LoadBalancer - # Additional annotations applied to both TCP and UDP services (e.g. for cloud provider specific config) -``` - -## 18.0.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-26 - -* Refactor http3 and merge TCP with UDP ports into a single service (#656) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 807bd09..acce704 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -87,8 +87,6 @@ ingressClass: - - # Enable experimental features - experimental: -- http3: -- enabled: false - plugins: - enabled: false - kubernetesGateway: -@@ -421,12 +419,19 @@ ports: - # The port protocol (TCP/UDP) - protocol: TCP - # nodePort: 32443 -- # Enable HTTP/3. -- # Requires enabling experimental http3 feature and tls. -- # Note that you cannot have a UDP entrypoint with the same port. -- # http3: true -- # Set TLS at the entrypoint -- # https://doc.traefik.io/traefik/routing/entrypoints/#tls -+ # -+ ## Enable HTTP/3 on the entrypoint -+ ## Enabling it will also enable http3 experimental feature -+ ## https://doc.traefik.io/traefik/routing/entrypoints/#http3 -+ ## There are known limitations when trying to listen on same ports for -+ ## TCP & UDP (Http3). There is a workaround in this chart using dual Service. -+ ## https://github.com/kubernetes/kubernetes/issues/47249#issuecomment-587960741 -+ http3: -+ enabled: false -+ # advertisedPort: 4443 -+ # -+ ## Set TLS at the entrypoint -+ ## https://doc.traefik.io/traefik/routing/entrypoints/#tls - tls: - enabled: true - # this is the name of a TLSOption definition -@@ -500,6 +505,7 @@ tlsStore: {} - # from. - service: - enabled: true -+ single: true - type: LoadBalancer - # Additional annotations applied to both TCP and UDP services (e.g. for cloud provider specific config) - annotations: {} -``` - -## 17.0.5 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-21 - -* 📝 Add annotations changelog for artifacthub.io & update Maintainers - - -## 17.0.4 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-21 - -* :art: Add helper function for label selector - - -## 17.0.3 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-20 - -* 🐛 fix changing label selectors - - -## 17.0.2 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-20 - -* fix: setting ports.web.proxyProtocol.insecure=true - - -## 17.0.1 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-20 - -* :bug: Unify all labels selector with traefik chart labels (#681) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 6a90bc6..807bd09 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -639,7 +639,7 @@ affinity: {} - # - labelSelector: - # matchLabels: - # app.kubernetes.io/name: '{{ template "traefik.name" . }}' --# app.kubernetes.io/instance: '{{ .Release.Name }}' -+# app.kubernetes.io/instance: '{{ .Release.Name }}-{{ .Release.Namespace }}' - # topologyKey: kubernetes.io/hostname - - nodeSelector: {} -``` - -## 17.0.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-20 - -* :bug: Fix `ClusterRole`, `ClusterRoleBinding` names and `app.kubernetes.io/instance` label (#662) - - -## 16.2.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-20 - -* Add forwardedHeaders and proxyProtocol config (#673) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 9b5afc4..6a90bc6 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -403,6 +403,16 @@ ports: - # Added in 2.2, you can make permanent redirects via entrypoints. - # https://docs.traefik.io/routing/entrypoints/#redirection - # redirectTo: websecure -+ # -+ # Trust forwarded headers information (X-Forwarded-*). -+ # forwardedHeaders: -+ # trustedIPs: [] -+ # insecure: false -+ # -+ # Enable the Proxy Protocol header parsing for the entry point -+ # proxyProtocol: -+ # trustedIPs: [] -+ # insecure: false - websecure: - port: 8443 - # hostPort: 8443 -@@ -428,6 +438,16 @@ ports: - # - foo.example.com - # - bar.example.com - # -+ # Trust forwarded headers information (X-Forwarded-*). -+ # forwardedHeaders: -+ # trustedIPs: [] -+ # insecure: false -+ # -+ # Enable the Proxy Protocol header parsing for the entry point -+ # proxyProtocol: -+ # trustedIPs: [] -+ # insecure: false -+ # - # One can apply Middlewares on an entrypoint - # https://doc.traefik.io/traefik/middlewares/overview/ - # https://doc.traefik.io/traefik/routing/entrypoints/#middlewares -``` - -## 16.1.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-19 - -* ✨ add optional ServiceMonitor & PrometheusRules CRDs (#425) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 7e335b5..9b5afc4 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -237,8 +237,46 @@ metrics: - prometheus: - entryPoint: metrics - # addRoutersLabels: true -- # statsd: -- # address: localhost:8125 -+ # statsd: -+ # address: localhost:8125 -+## -+## enable optional CRDs for Prometheus Operator -+## -+ # serviceMonitor: -+ # additionalLabels: -+ # foo: bar -+ # namespace: "another-namespace" -+ # namespaceSelector: {} -+ # metricRelabelings: [] -+ # - sourceLabels: [__name__] -+ # separator: ; -+ # regex: ^fluentd_output_status_buffer_(oldest|newest)_.+ -+ # replacement: $1 -+ # action: drop -+ # relabelings: [] -+ # - sourceLabels: [__meta_kubernetes_pod_node_name] -+ # separator: ; -+ # regex: ^(.*)$ -+ # targetLabel: nodename -+ # replacement: $1 -+ # action: replace -+ # jobLabel: traefik -+ # scrapeInterval: 30s -+ # scrapeTimeout: 5s -+ # honorLabels: true -+ # prometheusRule: -+ # additionalLabels: {} -+ # namespace: "another-namespace" -+ # rules: -+ # - alert: TraefikDown -+ # expr: up{job="traefik"} == 0 -+ # for: 5m -+ # labels: -+ # context: traefik -+ # severity: warning -+ # annotations: -+ # summary: "Traefik Down" -+ # description: "{{ $labels.pod }} on {{ $labels.nodename }} is down" - - tracing: {} - # instana: -``` - -## 16.0.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-19 - -* :fire: Remove `Pilot` and `fallbackApiVersion` (#665) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 03fdaed..7e335b5 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -84,15 +84,6 @@ ingressClass: - # true is not unit-testable yet, pending https://github.com/rancher/helm-unittest/pull/12 - enabled: false - isDefaultClass: false -- # Use to force a networking.k8s.io API Version for certain CI/CD applications. E.g. "v1beta1" -- fallbackApiVersion: "" -- --# Activate Pilot integration --pilot: -- enabled: false -- token: "" -- # Toggle Pilot Dashboard -- # dashboard: false - - # Enable experimental features - experimental: -``` - -## 15.3.1 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-18 - -* :art: Improve `IngressRoute` structure (#674) - - -## 15.3.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-18 - -* 📌 Add capacity to enable User-facing role - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 76aac93..03fdaed 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -553,10 +553,12 @@ hostNetwork: false - # Whether Role Based Access Control objects like roles and rolebindings should be created - rbac: - enabled: true -- - # If set to false, installs ClusterRole and ClusterRoleBinding so Traefik can be used across namespaces. - # If set to true, installs Role and RoleBinding. Providers will only watch target namespace. - namespaced: false -+ # Enable user-facing roles -+ # https://kubernetes.io/docs/reference/access-authn-authz/rbac/#user-facing-roles -+ # aggregateTo: [ "admin" ] - - # Enable to create a PodSecurityPolicy and assign it to the Service Account via RoleBinding or ClusterRoleBinding - podSecurityPolicy: -``` - -## 15.2.2 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-17 - -* Fix provider namespace changes - - -## 15.2.1 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-17 - -* 🐛 fix provider namespace changes - - -## 15.2.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-17 - -* :bug: Allow to watch on specific namespaces without using rbac.namespaced (#666) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 781ac15..76aac93 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -555,7 +555,7 @@ rbac: - enabled: true - - # If set to false, installs ClusterRole and ClusterRoleBinding so Traefik can be used across namespaces. -- # If set to true, installs namespace-specific Role and RoleBinding and requires provider configuration be set to that same namespace -+ # If set to true, installs Role and RoleBinding. Providers will only watch target namespace. - namespaced: false - - # Enable to create a PodSecurityPolicy and assign it to the Service Account via RoleBinding or ClusterRoleBinding -``` - -## 15.1.1 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-17 - -* :goal_net: Fail gracefully when http3 is not enabled correctly (#667) - - -## 15.1.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-14 - -* :sparkles: add optional topologySpreadConstraints (#663) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index fc2c371..781ac15 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -593,6 +593,15 @@ affinity: {} - - nodeSelector: {} - tolerations: [] -+topologySpreadConstraints: [] -+# # This example topologySpreadConstraints forces the scheduler to put traefik pods -+# # on nodes where no other traefik pods are scheduled. -+# - labelSelector: -+# matchLabels: -+# app: '{{ template "traefik.name" . }}' -+# maxSkew: 1 -+# topologyKey: kubernetes.io/hostname -+# whenUnsatisfiable: DoNotSchedule - - # Pods can have priority. - # Priority indicates the importance of a Pod relative to other Pods. -``` - -## 15.0.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-13 - -* :rocket: Enable TLS by default on `websecure` port (#657) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 400a29a..fc2c371 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -389,7 +389,7 @@ ports: - # Set TLS at the entrypoint - # https://doc.traefik.io/traefik/routing/entrypoints/#tls - tls: -- enabled: false -+ enabled: true - # this is the name of a TLSOption definition - options: "" - certResolver: "" -``` - -## 14.0.2 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-13 - -* :memo: Add Changelog (#661) - - -## 14.0.1 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-11 - -* :memo: Update workaround for permissions 660 on acme.json - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index a4e4ff2..400a29a 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -45,10 +45,10 @@ deployment: - # Additional initContainers (e.g. for setting file permission as shown below) - initContainers: [] - # The "volume-permissions" init container is required if you run into permission issues. -- # Related issue: https://github.com/traefik/traefik/issues/6972 -+ # Related issue: https://github.com/traefik/traefik/issues/6825 - # - name: volume-permissions -- # image: busybox:1.31.1 -- # command: ["sh", "-c", "chmod -Rv 600 /data/*"] -+ # image: busybox:1.35 -+ # command: ["sh", "-c", "touch /data/acme.json && chmod -Rv 600 /data/* && chown 65532:65532 /data/acme.json"] - # volumeMounts: - # - name: data - # mountPath: /data -``` - -## 14.0.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-11 - -* Limit rbac to only required resources for Ingress and CRD providers - - -## 13.0.1 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-11 - -* Add helper function for common labels - - -## 13.0.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-11 - -* Moved list object to individual objects - - -## 12.0.7 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-10 - -* :lipstick: Affinity templating and example (#557) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 4431c36..a4e4ff2 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -578,19 +578,19 @@ resources: {} - # limits: - # cpu: "300m" - # memory: "150Mi" -+ -+# This example pod anti-affinity forces the scheduler to put traefik pods -+# on nodes where no other traefik pods are scheduled. -+# It should be used when hostNetwork: true to prevent port conflicts - affinity: {} --# # This example pod anti-affinity forces the scheduler to put traefik pods --# # on nodes where no other traefik pods are scheduled. --# # It should be used when hostNetwork: true to prevent port conflicts --# podAntiAffinity: --# requiredDuringSchedulingIgnoredDuringExecution: --# - labelSelector: --# matchExpressions: --# - key: app.kubernetes.io/name --# operator: In --# values: --# - {{ template "traefik.name" . }} --# topologyKey: kubernetes.io/hostname -+# podAntiAffinity: -+# requiredDuringSchedulingIgnoredDuringExecution: -+# - labelSelector: -+# matchLabels: -+# app.kubernetes.io/name: '{{ template "traefik.name" . }}' -+# app.kubernetes.io/instance: '{{ .Release.Name }}' -+# topologyKey: kubernetes.io/hostname -+ - nodeSelector: {} - tolerations: [] - -``` - -## 12.0.6 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-10 - -* :bug: Ignore kustomization file used for CRDs update (#653) - - -## 12.0.5 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-10 - -* :memo: Establish Traefik & CRD update process - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 3526729..4431c36 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -342,6 +342,7 @@ ports: - - # Override the liveness/readiness port. This is useful to integrate traefik - # with an external Load Balancer that performs healthchecks. -+ # Default: ports.traefik.port - # healthchecksPort: 9000 - - # Override the liveness/readiness scheme. Useful for getting ping to -``` - -## 12.0.4 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-10 - -* Allows ingressClass to be used without semver-compatible image tag - - -## 12.0.3 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-10 - -* :bug: Should check hostNetwork when hostPort != containerPort - - -## 12.0.2 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-07 - -* :goal_net: Fail gracefully when hostNetwork is enabled and hostPort != containerPort - - -## 12.0.1 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-07 - -* :bug: Fix a typo on `behavior` for HPA v2 - - -## 12.0.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-06 - -* Update default HPA API Version to `v2` and add support for behavior (#518) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 2bd51f8..3526729 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -488,11 +488,22 @@ autoscaling: - # - type: Resource - # resource: - # name: cpu --# targetAverageUtilization: 60 -+# target: -+# type: Utilization -+# averageUtilization: 60 - # - type: Resource - # resource: - # name: memory --# targetAverageUtilization: 60 -+# target: -+# type: Utilization -+# averageUtilization: 60 -+# behavior: -+# scaleDown: -+# stabilizationWindowSeconds: 300 -+# policies: -+# - type: Pods -+# value: 1 -+# periodSeconds: 60 - - # Enable persistence using Persistent Volume Claims - # ref: http://kubernetes.io/docs/user-guide/persistent-volumes/ -``` - -## 11.1.1 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-05 - -* 🔊 add failure message when using maxUnavailable 0 and hostNetwork - - -## 11.1.0 ![AppVersion: 2.9.1](https://img.shields.io/static/v1?label=AppVersion&message=2.9.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-04 - -* Update Traefik to v2.9.1 - - -## 11.0.0 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-04 - -* tweak default values to avoid downtime when updating - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 844cadc..2bd51f8 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -126,20 +126,20 @@ ingressRoute: - entryPoints: ["traefik"] - - rollingUpdate: -- maxUnavailable: 1 -+ maxUnavailable: 0 - maxSurge: 1 - - # Customize liveness and readiness probe values. - readinessProbe: - failureThreshold: 1 -- initialDelaySeconds: 10 -+ initialDelaySeconds: 2 - periodSeconds: 10 - successThreshold: 1 - timeoutSeconds: 2 - - livenessProbe: - failureThreshold: 3 -- initialDelaySeconds: 10 -+ initialDelaySeconds: 2 - periodSeconds: 10 - successThreshold: 1 - timeoutSeconds: 2 -``` - -## 10.33.0 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-04 - -* :rocket: Add `extraObjects` value that allows creating adhoc resources - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index c926bd9..844cadc 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -598,3 +598,10 @@ securityContext: - - podSecurityContext: - fsGroup: 65532 -+ -+# -+# Extra objects to deploy (value evaluated as a template) -+# -+# In some cases, it can avoid the need for additional, extended or adhoc deployments. -+# See #595 for more details and traefik/tests/extra.yaml for example. -+extraObjects: [] -``` - -## 10.32.0 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-03 - -* Add support setting middleware on entrypoint - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 3957448..c926bd9 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -397,6 +397,16 @@ ports: - # sans: - # - foo.example.com - # - bar.example.com -+ # -+ # One can apply Middlewares on an entrypoint -+ # https://doc.traefik.io/traefik/middlewares/overview/ -+ # https://doc.traefik.io/traefik/routing/entrypoints/#middlewares -+ # /!\ It introduces here a link between your static configuration and your dynamic configuration /!\ -+ # It follows the provider naming convention: https://doc.traefik.io/traefik/providers/overview/#provider-namespace -+ # middlewares: -+ # - namespace-name1@kubernetescrd -+ # - namespace-name2@kubernetescrd -+ middlewares: [] - metrics: - # When using hostNetwork, use another port to avoid conflict with node exporter: - # https://github.com/prometheus/prometheus/wiki/Default-port-allocations -``` - -## 10.31.0 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-03 - -* Support setting dashboard entryPoints for ingressRoute resource - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index c9feb76..3957448 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -120,6 +120,10 @@ ingressRoute: - annotations: {} - # Additional ingressRoute labels (e.g. for filtering IngressRoute by custom labels) - labels: {} -+ # Specify the allowed entrypoints to use for the dashboard ingress route, (e.g. traefik, web, websecure). -+ # By default, it's using traefik entrypoint, which is not exposed. -+ # /!\ Do not expose your dashboard without any protection over the internet /!\ -+ entryPoints: ["traefik"] - - rollingUpdate: - maxUnavailable: 1 -``` - -## 10.30.2 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-10-03 - -* :test_tube: Fail gracefully when asked to provide a service without ports - - -## 10.30.1 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-09-30 - -* :arrow_up: Upgrade helm, ct & unittest (#638) - - -## 10.30.0 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-09-30 - -* Add support HTTPS scheme for healthcheks - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index fed4a8a..c9feb76 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -340,6 +340,10 @@ ports: - # with an external Load Balancer that performs healthchecks. - # healthchecksPort: 9000 - -+ # Override the liveness/readiness scheme. Useful for getting ping to -+ # respond on websecure entryPoint. -+ # healthchecksScheme: HTTPS -+ - # Defines whether the port is exposed if service.type is LoadBalancer or - # NodePort. - # -``` - -## 10.29.0 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-09-29 - -* Add missing tracing options - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index d1708cc..fed4a8a 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -247,12 +247,45 @@ metrics: - - tracing: {} - # instana: -- # enabled: true -+ # localAgentHost: 127.0.0.1 -+ # localAgentPort: 42699 -+ # logLevel: info -+ # enableAutoProfile: true - # datadog: - # localAgentHostPort: 127.0.0.1:8126 - # debug: false - # globalTag: "" - # prioritySampling: false -+ # jaeger: -+ # samplingServerURL: http://localhost:5778/sampling -+ # samplingType: const -+ # samplingParam: 1.0 -+ # localAgentHostPort: 127.0.0.1:6831 -+ # gen128Bit: false -+ # propagation: jaeger -+ # traceContextHeaderName: uber-trace-id -+ # disableAttemptReconnecting: true -+ # collector: -+ # endpoint: "" -+ # user: "" -+ # password: "" -+ # zipkin: -+ # httpEndpoint: http://localhost:9411/api/v2/spans -+ # sameSpan: false -+ # id128Bit: true -+ # sampleRate: 1.0 -+ # haystack: -+ # localAgentHost: 127.0.0.1 -+ # localAgentPort: 35000 -+ # globalTag: "" -+ # traceIDHeaderName: "" -+ # parentIDHeaderName: "" -+ # spanIDHeaderName: "" -+ # baggagePrefixHeaderName: "" -+ # elastic: -+ # serverURL: http://localhost:8200 -+ # secretToken: "" -+ # serviceEnvironment: "" - - globalArguments: - - "--global.checknewversion" -``` - -## 10.28.0 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-09-29 - -* feat: add lifecycle for prestop and poststart - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 19a133c..d1708cc 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -59,6 +59,17 @@ deployment: - # Additional imagePullSecrets - imagePullSecrets: [] - # - name: myRegistryKeySecretName -+ # Pod lifecycle actions -+ lifecycle: {} -+ # preStop: -+ # exec: -+ # command: ["/bin/sh", "-c", "sleep 40"] -+ # postStart: -+ # httpGet: -+ # path: /ping -+ # port: 9000 -+ # host: localhost -+ # scheme: HTTP - - # Pod disruption budget - podDisruptionBudget: -``` - -## 10.27.0 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-09-29 - -* feat: add create gateway option - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index d9c745e..19a133c 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -91,6 +91,8 @@ experimental: - enabled: false - kubernetesGateway: - enabled: false -+ gateway: -+ enabled: true - # certificate: - # group: "core" - # kind: "Secret" -``` - -## 10.26.1 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-09-28 - -* 🐛 fix rbac templating (#636) - - -## 10.26.0 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-09-28 - -* :bug: Fix ingressClass support when rbac.namespaced=true (#499) - - -## 10.25.1 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-09-28 - -* Add ingressclasses to traefik role - - -## 10.25.0 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-09-27 - -* Add TLSStore resource to chart - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index d4011c3..d9c745e 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -373,6 +373,15 @@ ports: - # - CurveP384 - tlsOptions: {} - -+# TLS Store are created as TLSStore CRDs. This is useful if you want to set a default certificate -+# https://doc.traefik.io/traefik/https/tls/#default-certificate -+# Example: -+# tlsStore: -+# default: -+# defaultCertificate: -+# secretName: tls-cert -+tlsStore: {} -+ - # Options for the main traefik service, where the entrypoints traffic comes - # from. - service: -``` - -## 10.24.5 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-09-27 - -* Suggest an alternative port for metrics - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 81f2e85..d4011c3 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -344,6 +344,8 @@ ports: - # - foo.example.com - # - bar.example.com - metrics: -+ # When using hostNetwork, use another port to avoid conflict with node exporter: -+ # https://github.com/prometheus/prometheus/wiki/Default-port-allocations - port: 9100 - # hostPort: 9100 - # Defines whether the port is exposed if service.type is LoadBalancer or -``` - -## 10.24.4 ![AppVersion: 2.8.7](https://img.shields.io/static/v1?label=AppVersion&message=2.8.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-09-26 - -* Update Traefik to v2.8.7 - - -## 10.24.3 ![AppVersion: 2.8.5](https://img.shields.io/static/v1?label=AppVersion&message=2.8.5&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-09-14 - -* Update Traefik version to v2.8.5 - - -## 10.24.2 ![AppVersion: 2.8.4](https://img.shields.io/static/v1?label=AppVersion&message=2.8.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-09-05 - -* Update Traefik version to v2.8.4 - - -## 10.24.1 ![AppVersion: 2.8.0](https://img.shields.io/static/v1?label=AppVersion&message=2.8.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-08-29 - -* Update PodDisruptionBudget apiVersion to policy/v1 - - -## 10.24.0 ![AppVersion: 2.8.0](https://img.shields.io/static/v1?label=AppVersion&message=2.8.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-06-30 - -* Update Traefik version to v2.8.0 - - -## 10.23.0 ![AppVersion: 2.7.1](https://img.shields.io/static/v1?label=AppVersion&message=2.7.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-06-27 - -* Support environment variable usage for Datadog - - -## 10.22.0 ![AppVersion: 2.7.1](https://img.shields.io/static/v1?label=AppVersion&message=2.7.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-06-22 - -* Allow setting revisionHistoryLimit for Deployment and DaemonSet - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index d5785ab..81f2e85 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -14,6 +14,8 @@ deployment: - kind: Deployment - # Number of pods of the deployment (only applies when kind == Deployment) - replicas: 1 -+ # Number of old history to retain to allow rollback (If not set, default Kubernetes value is set to 10) -+ # revisionHistoryLimit: 1 - # Amount of time (in seconds) before Kubernetes will send the SIGKILL signal if Traefik does not shut down - terminationGracePeriodSeconds: 60 - # The minimum number of seconds Traefik needs to be up and running before the DaemonSet/Deployment controller considers it available -``` - -## 10.21.1 ![AppVersion: 2.7.1](https://img.shields.io/static/v1?label=AppVersion&message=2.7.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-06-15 - -* Update Traefik version to 2.7.1 - - -## 10.21.0 ![AppVersion: 2.7.0](https://img.shields.io/static/v1?label=AppVersion&message=2.7.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-06-15 - -* Support allowEmptyServices config for KubernetesCRD - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index e141e29..d5785ab 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -133,6 +133,7 @@ providers: - enabled: true - allowCrossNamespace: false - allowExternalNameServices: false -+ allowEmptyServices: false - # ingressClass: traefik-internal - # labelSelector: environment=production,method=traefik - namespaces: [] -``` - -## 10.20.1 ![AppVersion: 2.7.0](https://img.shields.io/static/v1?label=AppVersion&message=2.7.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-06-01 - -* Add Acme certificate resolver configuration - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index a16b107..e141e29 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -433,6 +433,27 @@ persistence: - annotations: {} - # subPath: "" # only mount a subpath of the Volume into the pod - -+certResolvers: {} -+# letsencrypt: -+# # for challenge options cf. https://doc.traefik.io/traefik/https/acme/ -+# email: email@example.com -+# dnsChallenge: -+# # also add the provider's required configuration under env -+# # or expand then from secrets/configmaps with envfrom -+# # cf. https://doc.traefik.io/traefik/https/acme/#providers -+# provider: digitalocean -+# # add futher options for the dns challenge as needed -+# # cf. https://doc.traefik.io/traefik/https/acme/#dnschallenge -+# delayBeforeCheck: 30 -+# resolvers: -+# - 1.1.1.1 -+# - 8.8.8.8 -+# tlsChallenge: true -+# httpChallenge: -+# entryPoint: "web" -+# # match the path to persistence -+# storage: /data/acme.json -+ - # If hostNetwork is true, runs traefik in the host network namespace - # To prevent unschedulabel pods due to port collisions, if hostNetwork=true - # and replicas>1, a pod anti-affinity is recommended and will be set if the -``` - -## 10.20.0 ![AppVersion: 2.7.0](https://img.shields.io/static/v1?label=AppVersion&message=2.7.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-05-25 - -* Update Traefik Proxy to v2.7.0 - - -## 10.19.5 ![AppVersion: 2.6.6](https://img.shields.io/static/v1?label=AppVersion&message=2.6.6&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-05-04 - -* Upgrade Traefik to 2.6.6 - - -## 10.19.4 ![AppVersion: 2.6.3](https://img.shields.io/static/v1?label=AppVersion&message=2.6.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-03-31 - -* Update Traefik dependency version to 2.6.3 - - -## 10.19.3 ![AppVersion: 2.6.2](https://img.shields.io/static/v1?label=AppVersion&message=2.6.2&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-03-30 - -* Update CRDs to match the ones defined in the reference documentation - - -## 10.19.2 ![AppVersion: 2.6.2](https://img.shields.io/static/v1?label=AppVersion&message=2.6.2&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-03-30 - -* Revert Traefik version to 2.6.2 - - -## 10.19.1 ![AppVersion: 2.6.3](https://img.shields.io/static/v1?label=AppVersion&message=2.6.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-03-30 - -* Update Traefik version to 2.6.3 - - -## 10.19.0 ![AppVersion: 2.6.2](https://img.shields.io/static/v1?label=AppVersion&message=2.6.2&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-03-28 - -* Support ingressClass option for KubernetesIngress provider - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 02ab704..a16b107 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -142,6 +142,7 @@ providers: - enabled: true - allowExternalNameServices: false - allowEmptyServices: false -+ # ingressClass: traefik-internal - # labelSelector: environment=production,method=traefik - namespaces: [] - # - "default" -``` - -## 10.18.0 ![AppVersion: 2.6.2](https://img.shields.io/static/v1?label=AppVersion&message=2.6.2&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-03-28 - -* Support liveness and readyness probes customization - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 15f1103..02ab704 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -110,6 +110,20 @@ rollingUpdate: - maxUnavailable: 1 - maxSurge: 1 - -+# Customize liveness and readiness probe values. -+readinessProbe: -+ failureThreshold: 1 -+ initialDelaySeconds: 10 -+ periodSeconds: 10 -+ successThreshold: 1 -+ timeoutSeconds: 2 -+ -+livenessProbe: -+ failureThreshold: 3 -+ initialDelaySeconds: 10 -+ periodSeconds: 10 -+ successThreshold: 1 -+ timeoutSeconds: 2 - - # - # Configure providers -``` - -## 10.17.0 ![AppVersion: 2.6.2](https://img.shields.io/static/v1?label=AppVersion&message=2.6.2&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-03-28 - -* Support Datadog tracing - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 4dccd1a..15f1103 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -217,6 +217,11 @@ metrics: - tracing: {} - # instana: - # enabled: true -+ # datadog: -+ # localAgentHostPort: 127.0.0.1:8126 -+ # debug: false -+ # globalTag: "" -+ # prioritySampling: false - - globalArguments: - - "--global.checknewversion" -``` - -## 10.16.1 ![AppVersion: 2.6.2](https://img.shields.io/static/v1?label=AppVersion&message=2.6.2&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-03-28 - -* Update Traefik version to 2.6.2 - - -## 10.16.0 ![AppVersion: 2.6.1](https://img.shields.io/static/v1?label=AppVersion&message=2.6.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-03-28 - -* Support allowEmptyServices for KubernetesIngress provider - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 1f9dbbe..4dccd1a 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -127,6 +127,7 @@ providers: - kubernetesIngress: - enabled: true - allowExternalNameServices: false -+ allowEmptyServices: false - # labelSelector: environment=production,method=traefik - namespaces: [] - # - "default" -``` - -## 10.15.0 ![AppVersion: 2.6.1](https://img.shields.io/static/v1?label=AppVersion&message=2.6.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-03-08 - -* Add metrics.prometheus.addRoutersLabels option - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index cd4d49b..1f9dbbe 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -209,6 +209,7 @@ metrics: - # protocol: udp - prometheus: - entryPoint: metrics -+ # addRoutersLabels: true - # statsd: - # address: localhost:8125 - -``` - -## 10.14.2 ![AppVersion: 2.6.1](https://img.shields.io/static/v1?label=AppVersion&message=2.6.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-02-18 - -* Update Traefik to v2.6.1 - - -## 10.14.1 ![AppVersion: 2.6.0](https://img.shields.io/static/v1?label=AppVersion&message=2.6.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-02-09 - -* Add missing inFlightConn TCP middleware CRD - - -## 10.14.0 ![AppVersion: 2.6.0](https://img.shields.io/static/v1?label=AppVersion&message=2.6.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-02-03 - -* Add experimental HTTP/3 support - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index d49122f..cd4d49b 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -83,6 +83,8 @@ pilot: - - # Enable experimental features - experimental: -+ http3: -+ enabled: false - plugins: - enabled: false - kubernetesGateway: -@@ -300,6 +302,10 @@ ports: - # The port protocol (TCP/UDP) - protocol: TCP - # nodePort: 32443 -+ # Enable HTTP/3. -+ # Requires enabling experimental http3 feature and tls. -+ # Note that you cannot have a UDP entrypoint with the same port. -+ # http3: true - # Set TLS at the entrypoint - # https://doc.traefik.io/traefik/routing/entrypoints/#tls - tls: -``` - -## 10.13.0 ![AppVersion: 2.6.0](https://img.shields.io/static/v1?label=AppVersion&message=2.6.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-02-01 - -* Add support for ipFamilies - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 32fce6f..d49122f 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -366,6 +366,11 @@ service: - # - 1.2.3.4 - # One of SingleStack, PreferDualStack, or RequireDualStack. - # ipFamilyPolicy: SingleStack -+ # List of IP families (e.g. IPv4 and/or IPv6). -+ # ref: https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services -+ # ipFamilies: -+ # - IPv4 -+ # - IPv6 - - ## Create HorizontalPodAutoscaler object. - ## -``` - -## 10.12.0 ![AppVersion: 2.6.0](https://img.shields.io/static/v1?label=AppVersion&message=2.6.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-02-01 - -* Add shareProcessNamespace option to podtemplate - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index ab25456..32fce6f 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -50,6 +50,8 @@ deployment: - # volumeMounts: - # - name: data - # mountPath: /data -+ # Use process namespace sharing -+ shareProcessNamespace: false - # Custom pod DNS policy. Apply if `hostNetwork: true` - # dnsPolicy: ClusterFirstWithHostNet - # Additional imagePullSecrets -``` - -## 10.11.1 ![AppVersion: 2.6.0](https://img.shields.io/static/v1?label=AppVersion&message=2.6.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-01-31 - -* Fix anti-affinity example - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 8c72905..ab25456 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -438,13 +438,13 @@ affinity: {} - # # It should be used when hostNetwork: true to prevent port conflicts - # podAntiAffinity: - # requiredDuringSchedulingIgnoredDuringExecution: --# - labelSelector: --# matchExpressions: --# - key: app --# operator: In --# values: --# - {{ template "traefik.name" . }} --# topologyKey: failure-domain.beta.kubernetes.io/zone -+# - labelSelector: -+# matchExpressions: -+# - key: app.kubernetes.io/name -+# operator: In -+# values: -+# - {{ template "traefik.name" . }} -+# topologyKey: kubernetes.io/hostname - nodeSelector: {} - tolerations: [] - -``` - -## 10.11.0 ![AppVersion: 2.6.0](https://img.shields.io/static/v1?label=AppVersion&message=2.6.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-01-31 - -* Add setting to enable Instana tracing - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 7fe4a2c..8c72905 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -208,6 +208,10 @@ metrics: - # statsd: - # address: localhost:8125 - -+tracing: {} -+ # instana: -+ # enabled: true -+ - globalArguments: - - "--global.checknewversion" - - "--global.sendanonymoususage" -``` - -## 10.10.0 ![AppVersion: 2.6.0](https://img.shields.io/static/v1?label=AppVersion&message=2.6.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2022-01-31 - -* Update Traefik to v2.6 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 8ae4bd8..7fe4a2c 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -85,9 +85,8 @@ experimental: - enabled: false - kubernetesGateway: - enabled: false -- appLabelSelector: "traefik" -- certificates: [] -- # - group: "core" -+ # certificate: -+ # group: "core" - # kind: "Secret" - # name: "mysecret" - # By default, Gateway would be created to the Namespace you are deploying Traefik to. -``` - -## 10.9.1 ![AppVersion: 2.5.6](https://img.shields.io/static/v1?label=AppVersion&message=2.5.6&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-12-24 - -* Bump traefik version to 2.5.6 - - -## 10.9.0 ![AppVersion: 2.5.4](https://img.shields.io/static/v1?label=AppVersion&message=2.5.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-12-20 - -* feat: add allowExternalNameServices to KubernetesIngress provider - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 79df205..8ae4bd8 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -123,6 +123,7 @@ providers: - - kubernetesIngress: - enabled: true -+ allowExternalNameServices: false - # labelSelector: environment=production,method=traefik - namespaces: [] - # - "default" -``` - -## 10.8.0 ![AppVersion: 2.5.4](https://img.shields.io/static/v1?label=AppVersion&message=2.5.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-12-20 - -* Add support to specify minReadySeconds on Deployment/DaemonSet - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 7e9186b..79df205 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -16,6 +16,8 @@ deployment: - replicas: 1 - # Amount of time (in seconds) before Kubernetes will send the SIGKILL signal if Traefik does not shut down - terminationGracePeriodSeconds: 60 -+ # The minimum number of seconds Traefik needs to be up and running before the DaemonSet/Deployment controller considers it available -+ minReadySeconds: 0 - # Additional deployment annotations (e.g. for jaeger-operator sidecar injection) - annotations: {} - # Additional deployment labels (e.g. for filtering deployment by custom labels) -``` - -## 10.7.1 ![AppVersion: 2.5.4](https://img.shields.io/static/v1?label=AppVersion&message=2.5.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-12-06 - -* Fix pod disruption when using percentages - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index e0655c8..7e9186b 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -52,13 +52,15 @@ deployment: - # dnsPolicy: ClusterFirstWithHostNet - # Additional imagePullSecrets - imagePullSecrets: [] -- # - name: myRegistryKeySecretName -+ # - name: myRegistryKeySecretName - - # Pod disruption budget - podDisruptionBudget: - enabled: false - # maxUnavailable: 1 -+ # maxUnavailable: 33% - # minAvailable: 0 -+ # minAvailable: 25% - - # Use ingressClass. Ignored if Traefik version < 2.3 / kubernetes < 1.18.x - ingressClass: -``` - -## 10.7.0 ![AppVersion: 2.5.4](https://img.shields.io/static/v1?label=AppVersion&message=2.5.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-12-06 - -* Add support for ipFamilyPolicy - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 3ec7105..e0655c8 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -343,8 +343,8 @@ service: - annotationsUDP: {} - # Additional service labels (e.g. for filtering Service by custom labels) - labels: {} -- # Additional entries here will be added to the service spec. Cannot contains -- # type, selector or ports entries. -+ # Additional entries here will be added to the service spec. -+ # Cannot contain type, selector or ports entries. - spec: {} - # externalTrafficPolicy: Cluster - # loadBalancerIP: "1.2.3.4" -@@ -354,6 +354,8 @@ service: - # - 172.16.0.0/16 - externalIPs: [] - # - 1.2.3.4 -+ # One of SingleStack, PreferDualStack, or RequireDualStack. -+ # ipFamilyPolicy: SingleStack - - ## Create HorizontalPodAutoscaler object. - ## -``` - -## 10.6.2 ![AppVersion: 2.5.4](https://img.shields.io/static/v1?label=AppVersion&message=2.5.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-11-15 - -* Bump Traefik version to 2.5.4 - - -## 10.6.1 ![AppVersion: 2.5.3](https://img.shields.io/static/v1?label=AppVersion&message=2.5.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-11-05 - -* Add missing Gateway API resources to ClusterRole - - -## 10.6.0 ![AppVersion: 2.5.3](https://img.shields.io/static/v1?label=AppVersion&message=2.5.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-10-13 - -* feat: allow termination grace period to be configurable - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index f06ebc6..3ec7105 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -14,6 +14,8 @@ deployment: - kind: Deployment - # Number of pods of the deployment (only applies when kind == Deployment) - replicas: 1 -+ # Amount of time (in seconds) before Kubernetes will send the SIGKILL signal if Traefik does not shut down -+ terminationGracePeriodSeconds: 60 - # Additional deployment annotations (e.g. for jaeger-operator sidecar injection) - annotations: {} - # Additional deployment labels (e.g. for filtering deployment by custom labels) -``` - -## 10.5.0 ![AppVersion: 2.5.3](https://img.shields.io/static/v1?label=AppVersion&message=2.5.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-10-13 - -* feat: add allowExternalNameServices to Kubernetes CRD provider - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 3bcb350..f06ebc6 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -109,6 +109,7 @@ providers: - kubernetesCRD: - enabled: true - allowCrossNamespace: false -+ allowExternalNameServices: false - # ingressClass: traefik-internal - # labelSelector: environment=production,method=traefik - namespaces: [] -``` - -## 10.4.2 ![AppVersion: 2.5.3](https://img.shields.io/static/v1?label=AppVersion&message=2.5.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-10-13 - -* fix(crd): add permissionsPolicy to headers middleware - - -## 10.4.1 ![AppVersion: 2.5.3](https://img.shields.io/static/v1?label=AppVersion&message=2.5.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-10-13 - -* fix(crd): add peerCertURI option to ServersTransport - - -## 10.4.0 ![AppVersion: 2.5.3](https://img.shields.io/static/v1?label=AppVersion&message=2.5.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-10-12 - -* Add Kubernetes CRD labelSelector and ingressClass options - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index f54f5fe..3bcb350 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -109,8 +109,11 @@ providers: - kubernetesCRD: - enabled: true - allowCrossNamespace: false -+ # ingressClass: traefik-internal -+ # labelSelector: environment=production,method=traefik - namespaces: [] - # - "default" -+ - kubernetesIngress: - enabled: true - # labelSelector: environment=production,method=traefik -``` - -## 10.3.6 ![AppVersion: 2.5.3](https://img.shields.io/static/v1?label=AppVersion&message=2.5.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-09-24 - -* Fix missing RequireAnyClientCert value to TLSOption CRD - - -## 10.3.5 ![AppVersion: 2.5.3](https://img.shields.io/static/v1?label=AppVersion&message=2.5.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-09-23 - -* Bump Traefik version to 2.5.3 - - -## 10.3.4 ![AppVersion: 2.5.1](https://img.shields.io/static/v1?label=AppVersion&message=2.5.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-09-17 - -* Add allowCrossNamespace option on kubernetesCRD provider - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 7e3a579..f54f5fe 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -108,6 +108,7 @@ rollingUpdate: - providers: - kubernetesCRD: - enabled: true -+ allowCrossNamespace: false - namespaces: [] - # - "default" - kubernetesIngress: -``` - -## 10.3.3 ![AppVersion: 2.5.1](https://img.shields.io/static/v1?label=AppVersion&message=2.5.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-09-17 - -* fix(crd): missing alpnProtocols in TLSOption - - -## 10.3.2 ![AppVersion: 2.5.1](https://img.shields.io/static/v1?label=AppVersion&message=2.5.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-08-23 - -* Releasing 2.5.1 - - -## 10.3.1 ![AppVersion: 2.5.0](https://img.shields.io/static/v1?label=AppVersion&message=2.5.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-08-20 - -* Fix Ingress RBAC for namespaced scoped deployment - - -## 10.3.0 ![AppVersion: 2.5.0](https://img.shields.io/static/v1?label=AppVersion&message=2.5.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-08-18 - -* Releasing Traefik 2.5.0 - - -## 10.2.0 ![AppVersion: 2.4.13](https://img.shields.io/static/v1?label=AppVersion&message=2.4.13&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-08-18 - -* Allow setting TCP and UDP service annotations separately - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 72a01ea..7e3a579 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -328,8 +328,12 @@ tlsOptions: {} - service: - enabled: true - type: LoadBalancer -- # Additional annotations (e.g. for cloud provider specific config) -+ # Additional annotations applied to both TCP and UDP services (e.g. for cloud provider specific config) - annotations: {} -+ # Additional annotations for TCP service only -+ annotationsTCP: {} -+ # Additional annotations for UDP service only -+ annotationsUDP: {} - # Additional service labels (e.g. for filtering Service by custom labels) - labels: {} - # Additional entries here will be added to the service spec. Cannot contains -``` - -## 10.1.6 ![AppVersion: 2.4.13](https://img.shields.io/static/v1?label=AppVersion&message=2.4.13&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-08-17 - -* fix: missing service labels - - -## 10.1.5 ![AppVersion: 2.4.13](https://img.shields.io/static/v1?label=AppVersion&message=2.4.13&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-08-17 - -* fix(pvc-annotaions): see traefik/traefik-helm-chart#471 - - -## 10.1.4 ![AppVersion: 2.4.13](https://img.shields.io/static/v1?label=AppVersion&message=2.4.13&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-08-17 - -* fix(ingressclass): fallbackApiVersion default shouldn't be `nil` - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 04d336c..72a01ea 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -64,7 +64,7 @@ ingressClass: - enabled: false - isDefaultClass: false - # Use to force a networking.k8s.io API Version for certain CI/CD applications. E.g. "v1beta1" -- fallbackApiVersion: -+ fallbackApiVersion: "" - - # Activate Pilot integration - pilot: -``` - -## 10.1.3 ![AppVersion: 2.4.13](https://img.shields.io/static/v1?label=AppVersion&message=2.4.13&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-08-16 - -* Move Prometheus annotations to Pods - - -## 10.1.2 ![AppVersion: 2.4.13](https://img.shields.io/static/v1?label=AppVersion&message=2.4.13&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-08-10 - -* Version bumped 2.4.13 - - -## 10.1.1 ![AppVersion: 2.4.9](https://img.shields.io/static/v1?label=AppVersion&message=2.4.9&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-07-20 - -* Fixing Prometheus.io/port annotation - - -## 10.1.0 ![AppVersion: 2.4.9](https://img.shields.io/static/v1?label=AppVersion&message=2.4.9&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-07-20 - -* Add metrics framework, and prom annotations - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index f6e370a..04d336c 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -186,6 +186,17 @@ logs: - # Authorization: drop - # Content-Type: keep - -+metrics: -+ # datadog: -+ # address: 127.0.0.1:8125 -+ # influxdb: -+ # address: localhost:8089 -+ # protocol: udp -+ prometheus: -+ entryPoint: metrics -+ # statsd: -+ # address: localhost:8125 -+ - globalArguments: - - "--global.checknewversion" - - "--global.sendanonymoususage" -@@ -284,6 +295,20 @@ ports: - # sans: - # - foo.example.com - # - bar.example.com -+ metrics: -+ port: 9100 -+ # hostPort: 9100 -+ # Defines whether the port is exposed if service.type is LoadBalancer or -+ # NodePort. -+ # -+ # You may not want to expose the metrics port on production deployments. -+ # If you want to access it from outside of your cluster, -+ # use `kubectl port-forward` or create a secure ingress -+ expose: false -+ # The exposed port for this service -+ exposedPort: 9100 -+ # The port protocol (TCP/UDP) -+ protocol: TCP - - # TLS Options are created as TLSOption CRDs - # https://doc.traefik.io/traefik/https/tls/#tls-options -``` - -## 10.0.2 ![AppVersion: 2.4.9](https://img.shields.io/static/v1?label=AppVersion&message=2.4.9&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-07-14 - -* feat(gateway): introduces param / pick Namespace installing Gateway - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 9bf90ea..f6e370a 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -84,6 +84,9 @@ experimental: - # - group: "core" - # kind: "Secret" - # name: "mysecret" -+ # By default, Gateway would be created to the Namespace you are deploying Traefik to. -+ # You may create that Gateway in another namespace, setting its name below: -+ # namespace: default - - # Create an IngressRoute for the dashboard - ingressRoute: -``` - -## 10.0.1 ![AppVersion: 2.4.9](https://img.shields.io/static/v1?label=AppVersion&message=2.4.9&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-07-14 - -* Add RBAC for middlewaretcps - - -## 10.0.0 ![AppVersion: 2.4.9](https://img.shields.io/static/v1?label=AppVersion&message=2.4.9&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-07-07 - -* Update CRD versions - - -## 9.20.1 ![AppVersion: 2.4.8](https://img.shields.io/static/v1?label=AppVersion&message=2.4.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-07-05 - -* Revert CRD templating - - -## 9.20.0 ![AppVersion: 2.4.8](https://img.shields.io/static/v1?label=AppVersion&message=2.4.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-07-05 - -* Add support for apiextensions v1 CRDs - - -## 9.19.2 ![AppVersion: 2.4.8](https://img.shields.io/static/v1?label=AppVersion&message=2.4.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-06-16 - -* Add name-metadata for service "List" object - - -## 9.19.1 ![AppVersion: 2.4.8](https://img.shields.io/static/v1?label=AppVersion&message=2.4.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-05-13 - -* fix simple typo - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index b30afac..9bf90ea 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -363,7 +363,7 @@ rbac: - # If set to true, installs namespace-specific Role and RoleBinding and requires provider configuration be set to that same namespace - namespaced: false - --# Enable to create a PodSecurityPolicy and assign it to the Service Account via RoleBindin or ClusterRoleBinding -+# Enable to create a PodSecurityPolicy and assign it to the Service Account via RoleBinding or ClusterRoleBinding - podSecurityPolicy: - enabled: false - -``` - -## 9.19.0 ![AppVersion: 2.4.8](https://img.shields.io/static/v1?label=AppVersion&message=2.4.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-04-29 - -* Fix IngressClass api version - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 0aa2d6b..b30afac 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -63,6 +63,8 @@ ingressClass: - # true is not unit-testable yet, pending https://github.com/rancher/helm-unittest/pull/12 - enabled: false - isDefaultClass: false -+ # Use to force a networking.k8s.io API Version for certain CI/CD applications. E.g. "v1beta1" -+ fallbackApiVersion: - - # Activate Pilot integration - pilot: -``` - -## 9.18.3 ![AppVersion: 2.4.8](https://img.shields.io/static/v1?label=AppVersion&message=2.4.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-04-26 - -* Fix: ignore provider namespace args on disabled - - -## 9.18.2 ![AppVersion: 2.4.8](https://img.shields.io/static/v1?label=AppVersion&message=2.4.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-04-02 - -* Fix pilot dashboard deactivation - - -## 9.18.1 ![AppVersion: 2.4.8](https://img.shields.io/static/v1?label=AppVersion&message=2.4.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-03-29 - -* Do not disable Traefik Pilot in the dashboard by default - - -## 9.18.0 ![AppVersion: 2.4.8](https://img.shields.io/static/v1?label=AppVersion&message=2.4.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-03-24 - -* Add an option to toggle the pilot dashboard - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 017f771..0aa2d6b 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -68,6 +68,8 @@ ingressClass: - pilot: - enabled: false - token: "" -+ # Toggle Pilot Dashboard -+ # dashboard: false - - # Enable experimental features - experimental: -``` - -## 9.17.6 ![AppVersion: 2.4.8](https://img.shields.io/static/v1?label=AppVersion&message=2.4.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-03-24 - -* Bump Traefik to 2.4.8 - - -## 9.17.5 ![AppVersion: 2.4.7](https://img.shields.io/static/v1?label=AppVersion&message=2.4.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-03-17 - -* feat(labelSelector): option matching Ingresses based on labelSelectors - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 868a985..017f771 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -105,6 +105,7 @@ providers: - # - "default" - kubernetesIngress: - enabled: true -+ # labelSelector: environment=production,method=traefik - namespaces: [] - # - "default" - # IP used for Kubernetes Ingress endpoints -``` - -## 9.17.4 ![AppVersion: 2.4.7](https://img.shields.io/static/v1?label=AppVersion&message=2.4.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-03-17 - -* Add helm resource-policy annotation on PVC - - -## 9.17.3 ![AppVersion: 2.4.7](https://img.shields.io/static/v1?label=AppVersion&message=2.4.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-03-17 - -* Throw error with explicit latest tag - - -## 9.17.2 ![AppVersion: 2.4.7](https://img.shields.io/static/v1?label=AppVersion&message=2.4.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-03-10 - -* fix(keywords): removed by mistake - - -## 9.17.1 ![AppVersion: 2.4.7](https://img.shields.io/static/v1?label=AppVersion&message=2.4.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-03-10 - -* feat(healthchecksPort): Support for overriding the liveness/readiness probes port - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 56abb93..868a985 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -120,6 +120,8 @@ providers: - # After the volume has been mounted, add the configs into traefik by using the `additionalArguments` list below, eg: - # additionalArguments: - # - "--providers.file.filename=/config/dynamic.toml" -+# - "--ping" -+# - "--ping.entrypoint=web" - volumes: [] - # - name: public-cert - # mountPath: "/certs" -@@ -225,6 +227,10 @@ ports: - # only. - # hostIP: 192.168.100.10 - -+ # Override the liveness/readiness port. This is useful to integrate traefik -+ # with an external Load Balancer that performs healthchecks. -+ # healthchecksPort: 9000 -+ - # Defines whether the port is exposed if service.type is LoadBalancer or - # NodePort. - # -``` - -## 9.16.2 ![AppVersion: 2.4.7](https://img.shields.io/static/v1?label=AppVersion&message=2.4.7&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-03-09 - -* Bump Traefik to 2.4.7 - - -## 9.16.1 ![AppVersion: 2.4.6](https://img.shields.io/static/v1?label=AppVersion&message=2.4.6&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-03-09 - -* Adding custom labels to deployment - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index ba24be7..56abb93 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -16,6 +16,8 @@ deployment: - replicas: 1 - # Additional deployment annotations (e.g. for jaeger-operator sidecar injection) - annotations: {} -+ # Additional deployment labels (e.g. for filtering deployment by custom labels) -+ labels: {} - # Additional pod annotations (e.g. for mesh injection or prometheus scraping) - podAnnotations: {} - # Additional Pod labels (e.g. for filtering Pod by custom labels) -``` - -## 9.15.2 ![AppVersion: 2.4.6](https://img.shields.io/static/v1?label=AppVersion&message=2.4.6&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-03-02 - -* Upgrade Traefik to 2.4.6 - - -## 9.15.1 ![AppVersion: 2.4.5](https://img.shields.io/static/v1?label=AppVersion&message=2.4.5&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-03-02 - -* Configurable PVC name - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 1e0e5a9..ba24be7 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -327,6 +327,7 @@ autoscaling: - # It will persist TLS certificates. - persistence: - enabled: false -+ name: data - # existingClaim: "" - accessMode: ReadWriteOnce - size: 128Mi -``` - -## 9.14.4 ![AppVersion: 2.4.5](https://img.shields.io/static/v1?label=AppVersion&message=2.4.5&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-03-02 - -* fix typo - - -## 9.14.3 ![AppVersion: 2.4.5](https://img.shields.io/static/v1?label=AppVersion&message=2.4.5&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-02-19 - -* Bump Traefik to 2.4.5 - - -## 9.14.2 ![AppVersion: 2.4.2](https://img.shields.io/static/v1?label=AppVersion&message=2.4.2&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-02-03 - -* docs: indent nit for dsdsocket example - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 56485ad..1e0e5a9 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -33,7 +33,7 @@ deployment: - additionalVolumes: [] - # - name: dsdsocket - # hostPath: -- # path: /var/run/statsd-exporter -+ # path: /var/run/statsd-exporter - # Additional initContainers (e.g. for setting file permission as shown below) - initContainers: [] - # The "volume-permissions" init container is required if you run into permission issues. -``` - -## 9.14.1 ![AppVersion: 2.4.2](https://img.shields.io/static/v1?label=AppVersion&message=2.4.2&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-02-03 - -* Update Traefik to 2.4.2 - - -## 9.14.0 ![AppVersion: 2.4.0](https://img.shields.io/static/v1?label=AppVersion&message=2.4.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-02-01 - -* Enable Kubernetes Gateway provider with an experimental flag - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 50cab94..56485ad 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -71,6 +71,13 @@ pilot: - experimental: - plugins: - enabled: false -+ kubernetesGateway: -+ enabled: false -+ appLabelSelector: "traefik" -+ certificates: [] -+ # - group: "core" -+ # kind: "Secret" -+ # name: "mysecret" - - # Create an IngressRoute for the dashboard - ingressRoute: -``` - -## 9.13.0 ![AppVersion: 2.4.0](https://img.shields.io/static/v1?label=AppVersion&message=2.4.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2021-01-22 - -* Update Traefik to 2.4 and add resources - - -## 9.12.3 ![AppVersion: 2.3.6](https://img.shields.io/static/v1?label=AppVersion&message=2.3.6&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-12-31 - -* Revert API Upgrade - - -## 9.12.2 ![AppVersion: 2.3.6](https://img.shields.io/static/v1?label=AppVersion&message=2.3.6&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-12-31 - -* Bump Traefik to 2.3.6 - - -## 9.12.1 ![AppVersion: 2.3.3](https://img.shields.io/static/v1?label=AppVersion&message=2.3.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-12-30 - -* Resolve #303, change CRD version from v1beta1 to v1 - - -## 9.12.0 ![AppVersion: 2.3.3](https://img.shields.io/static/v1?label=AppVersion&message=2.3.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-12-30 - -* Implement support for DaemonSet - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 60a721d..50cab94 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -10,7 +10,9 @@ image: - # - deployment: - enabled: true -- # Number of pods of the deployment -+ # Can be either Deployment or DaemonSet -+ kind: Deployment -+ # Number of pods of the deployment (only applies when kind == Deployment) - replicas: 1 - # Additional deployment annotations (e.g. for jaeger-operator sidecar injection) - annotations: {} -``` - -## 9.11.0 ![AppVersion: 2.3.3](https://img.shields.io/static/v1?label=AppVersion&message=2.3.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-11-20 - -* add podLabels - custom labels - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index a187df7..60a721d 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -16,6 +16,8 @@ deployment: - annotations: {} - # Additional pod annotations (e.g. for mesh injection or prometheus scraping) - podAnnotations: {} -+ # Additional Pod labels (e.g. for filtering Pod by custom labels) -+ podLabels: {} - # Additional containers (e.g. for metric offloading sidecars) - additionalContainers: [] - # https://docs.datadoghq.com/developers/dogstatsd/unix_socket/?tab=host -``` - -## 9.10.2 ![AppVersion: 2.3.3](https://img.shields.io/static/v1?label=AppVersion&message=2.3.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-11-20 - -* Bump Traefik to 2.3.3 - - -## 9.10.1 ![AppVersion: 2.3.1](https://img.shields.io/static/v1?label=AppVersion&message=2.3.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-11-04 - -* Specify IngressClass resource when checking for cluster capability - - -## 9.10.0 ![AppVersion: 2.3.1](https://img.shields.io/static/v1?label=AppVersion&message=2.3.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-11-03 - -* Add list of watched provider namespaces - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index e6b85ca..a187df7 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -88,8 +88,12 @@ rollingUpdate: - providers: - kubernetesCRD: - enabled: true -+ namespaces: [] -+ # - "default" - kubernetesIngress: - enabled: true -+ namespaces: [] -+ # - "default" - # IP used for Kubernetes Ingress endpoints - publishedService: - enabled: false -``` - -## 9.9.0 ![AppVersion: 2.3.1](https://img.shields.io/static/v1?label=AppVersion&message=2.3.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-11-03 - -* Add additionalVolumeMounts for traefik container - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 37dd151..e6b85ca 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -111,6 +111,12 @@ volumes: [] - # mountPath: "/config" - # type: configMap - -+# Additional volumeMounts to add to the Traefik container -+additionalVolumeMounts: [] -+ # For instance when using a logshipper for access logs -+ # - name: traefik-logs -+ # mountPath: /var/log/traefik -+ - # Logs - # https://docs.traefik.io/observability/logs/ - logs: -``` - -## 9.8.4 ![AppVersion: 2.3.1](https://img.shields.io/static/v1?label=AppVersion&message=2.3.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-11-03 - -* fix: multiple ImagePullSecrets - - -## 9.8.3 ![AppVersion: 2.3.1](https://img.shields.io/static/v1?label=AppVersion&message=2.3.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-10-30 - -* Add imagePullSecrets - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 87f60c0..37dd151 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -42,6 +42,9 @@ deployment: - # mountPath: /data - # Custom pod DNS policy. Apply if `hostNetwork: true` - # dnsPolicy: ClusterFirstWithHostNet -+ # Additional imagePullSecrets -+ imagePullSecrets: [] -+ # - name: myRegistryKeySecretName - - # Pod disruption budget - podDisruptionBudget: -``` - -## 9.8.2 ![AppVersion: 2.3.1](https://img.shields.io/static/v1?label=AppVersion&message=2.3.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-10-28 - -* Add chart repo to source - - -## 9.8.1 ![AppVersion: 2.3.1](https://img.shields.io/static/v1?label=AppVersion&message=2.3.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-10-23 - -* fix semver compare - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 4ca1f8f..87f60c0 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1,7 +1,8 @@ - # Default values for Traefik - image: - name: traefik -- tag: 2.3.1 -+ # defaults to appVersion -+ tag: "" - pullPolicy: IfNotPresent - - # -``` - -## 9.8.0 ![AppVersion: 2.3.1](https://img.shields.io/static/v1?label=AppVersion&message=2.3.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-10-20 - -* feat: Enable entrypoint tls config + TLSOption - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index eee3622..4ca1f8f 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -231,6 +231,31 @@ ports: - # The port protocol (TCP/UDP) - protocol: TCP - # nodePort: 32443 -+ # Set TLS at the entrypoint -+ # https://doc.traefik.io/traefik/routing/entrypoints/#tls -+ tls: -+ enabled: false -+ # this is the name of a TLSOption definition -+ options: "" -+ certResolver: "" -+ domains: [] -+ # - main: example.com -+ # sans: -+ # - foo.example.com -+ # - bar.example.com -+ -+# TLS Options are created as TLSOption CRDs -+# https://doc.traefik.io/traefik/https/tls/#tls-options -+# Example: -+# tlsOptions: -+# default: -+# sniStrict: true -+# preferServerCipherSuites: true -+# foobar: -+# curvePreferences: -+# - CurveP521 -+# - CurveP384 -+tlsOptions: {} - - # Options for the main traefik service, where the entrypoints traffic comes - # from. -``` - -## 9.7.0 ![AppVersion: 2.3.1](https://img.shields.io/static/v1?label=AppVersion&message=2.3.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-10-15 - -* Add a configuration option for an emptyDir as plugin storage - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index b7153a1..eee3622 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -54,10 +54,16 @@ ingressClass: - enabled: false - isDefaultClass: false - -+# Activate Pilot integration - pilot: - enabled: false - token: "" - -+# Enable experimental features -+experimental: -+ plugins: -+ enabled: false -+ - # Create an IngressRoute for the dashboard - ingressRoute: - dashboard: -``` - -## 9.6.0 ![AppVersion: 2.3.1](https://img.shields.io/static/v1?label=AppVersion&message=2.3.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-10-15 - -* Add additional volumes for init and additional containers - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 9bac45e..b7153a1 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -17,6 +17,18 @@ deployment: - podAnnotations: {} - # Additional containers (e.g. for metric offloading sidecars) - additionalContainers: [] -+ # https://docs.datadoghq.com/developers/dogstatsd/unix_socket/?tab=host -+ # - name: socat-proxy -+ # image: alpine/socat:1.0.5 -+ # args: ["-s", "-u", "udp-recv:8125", "unix-sendto:/socket/socket"] -+ # volumeMounts: -+ # - name: dsdsocket -+ # mountPath: /socket -+ # Additional volumes available for use with initContainers and additionalContainers -+ additionalVolumes: [] -+ # - name: dsdsocket -+ # hostPath: -+ # path: /var/run/statsd-exporter - # Additional initContainers (e.g. for setting file permission as shown below) - initContainers: [] - # The "volume-permissions" init container is required if you run into permission issues. -``` - -## 9.5.2 ![AppVersion: 2.3.1](https://img.shields.io/static/v1?label=AppVersion&message=2.3.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-10-15 - -* Replace extensions with policy because of deprecation - - -## 9.5.1 ![AppVersion: 2.3.1](https://img.shields.io/static/v1?label=AppVersion&message=2.3.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-10-14 - -* Template custom volume name - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 5a8d8ea..9bac45e 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -76,7 +76,7 @@ providers: - # pathOverride: "" - - # --# Add volumes to the traefik pod. -+# Add volumes to the traefik pod. The volume name will be passed to tpl. - # This can be used to mount a cert pair or a configmap that holds a config.toml file. - # After the volume has been mounted, add the configs into traefik by using the `additionalArguments` list below, eg: - # additionalArguments: -@@ -85,7 +85,7 @@ volumes: [] - # - name: public-cert - # mountPath: "/certs" - # type: secret --# - name: configs -+# - name: '{{ printf "%s-configs" .Release.Name }}' - # mountPath: "/config" - # type: configMap - -``` - -## 9.5.0 ![AppVersion: 2.3.1](https://img.shields.io/static/v1?label=AppVersion&message=2.3.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-10-02 - -* Create PodSecurityPolicy and RBAC when needed. - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 8c4d866..5a8d8ea 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -281,6 +281,10 @@ rbac: - # If set to true, installs namespace-specific Role and RoleBinding and requires provider configuration be set to that same namespace - namespaced: false - -+# Enable to create a PodSecurityPolicy and assign it to the Service Account via RoleBindin or ClusterRoleBinding -+podSecurityPolicy: -+ enabled: false -+ - # The service account the pods will use to interact with the Kubernetes API - serviceAccount: - # If set, an existing service account is used -``` - -## 9.4.3 ![AppVersion: 2.3.1](https://img.shields.io/static/v1?label=AppVersion&message=2.3.1&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-10-02 - -* Update traefik to v2.3.1 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 3df75a4..8c4d866 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1,7 +1,7 @@ - # Default values for Traefik - image: - name: traefik -- tag: 2.3.0 -+ tag: 2.3.1 - pullPolicy: IfNotPresent - - # -``` - -## 9.4.2 ![AppVersion: 2.3.0](https://img.shields.io/static/v1?label=AppVersion&message=2.3.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-10-02 - -* Add Artifact Hub repository metadata file - - -## 9.4.1 ![AppVersion: 2.3.0](https://img.shields.io/static/v1?label=AppVersion&message=2.3.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-10-01 - -* Fix broken chart icon url - - -## 9.4.0 ![AppVersion: 2.3.0](https://img.shields.io/static/v1?label=AppVersion&message=2.3.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-10-01 - -* Allow to specify custom labels on Service - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index a6175ff..3df75a4 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -221,6 +221,8 @@ service: - type: LoadBalancer - # Additional annotations (e.g. for cloud provider specific config) - annotations: {} -+ # Additional service labels (e.g. for filtering Service by custom labels) -+ labels: {} - # Additional entries here will be added to the service spec. Cannot contains - # type, selector or ports entries. - spec: {} -``` - -## 9.3.0 ![AppVersion: 2.3.0](https://img.shields.io/static/v1?label=AppVersion&message=2.3.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-09-24 - -* Release Traefik 2.3 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index fba955d..a6175ff 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1,7 +1,7 @@ - # Default values for Traefik - image: - name: traefik -- tag: 2.2.8 -+ tag: 2.3.0 - pullPolicy: IfNotPresent - - # -@@ -36,6 +36,16 @@ podDisruptionBudget: - # maxUnavailable: 1 - # minAvailable: 0 - -+# Use ingressClass. Ignored if Traefik version < 2.3 / kubernetes < 1.18.x -+ingressClass: -+ # true is not unit-testable yet, pending https://github.com/rancher/helm-unittest/pull/12 -+ enabled: false -+ isDefaultClass: false -+ -+pilot: -+ enabled: false -+ token: "" -+ - # Create an IngressRoute for the dashboard - ingressRoute: - dashboard: -``` - -## 9.2.1 ![AppVersion: 2.2.8](https://img.shields.io/static/v1?label=AppVersion&message=2.2.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-09-18 - -* Add new helm url - - -## 9.2.0 ![AppVersion: 2.2.8](https://img.shields.io/static/v1?label=AppVersion&message=2.2.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-09-16 - -* chore: move to new organization. - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 9f52c39..fba955d 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -20,7 +20,7 @@ deployment: - # Additional initContainers (e.g. for setting file permission as shown below) - initContainers: [] - # The "volume-permissions" init container is required if you run into permission issues. -- # Related issue: https://github.com/containous/traefik/issues/6972 -+ # Related issue: https://github.com/traefik/traefik/issues/6972 - # - name: volume-permissions - # image: busybox:1.31.1 - # command: ["sh", "-c", "chmod -Rv 600 /data/*"] -``` - -## 9.1.1 ![AppVersion: 2.2.8](https://img.shields.io/static/v1?label=AppVersion&message=2.2.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-09-04 - -* Update reference to using kubectl proxy to kubectl port-forward - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 7b74a39..9f52c39 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -175,7 +175,7 @@ ports: - # - # You SHOULD NOT expose the traefik port on production deployments. - # If you want to access it from outside of your cluster, -- # use `kubectl proxy` or create a secure ingress -+ # use `kubectl port-forward` or create a secure ingress - expose: false - # The exposed port for this service - exposedPort: 9000 -``` - -## 9.1.0 ![AppVersion: 2.2.8](https://img.shields.io/static/v1?label=AppVersion&message=2.2.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-08-24 - -* PublishedService option - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index e161a14..7b74a39 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -58,6 +58,12 @@ providers: - enabled: true - kubernetesIngress: - enabled: true -+ # IP used for Kubernetes Ingress endpoints -+ publishedService: -+ enabled: false -+ # Published Kubernetes Service to copy status from. Format: namespace/servicename -+ # By default this Traefik service -+ # pathOverride: "" - - # - # Add volumes to the traefik pod. -``` - -## 9.0.0 ![AppVersion: 2.2.8](https://img.shields.io/static/v1?label=AppVersion&message=2.2.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-08-21 - -* feat: Move Chart apiVersion: v2 - - -## 8.13.3 ![AppVersion: 2.2.8](https://img.shields.io/static/v1?label=AppVersion&message=2.2.8&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-08-21 - -* bug: Check for port config - - -## 8.13.2 ![AppVersion: 2.2.8](https://img.shields.io/static/v1?label=AppVersion&message=2.2.8&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-08-19 - -* Fix log level configuration - - -## 8.13.1 ![AppVersion: 2.2.8](https://img.shields.io/static/v1?label=AppVersion&message=2.2.8&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-08-18 - -* Dont redirect to websecure by default - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 67276f7..e161a14 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -188,7 +188,7 @@ ports: - # Port Redirections - # Added in 2.2, you can make permanent redirects via entrypoints. - # https://docs.traefik.io/routing/entrypoints/#redirection -- redirectTo: websecure -+ # redirectTo: websecure - websecure: - port: 8443 - # hostPort: 8443 -``` - -## 8.13.0 ![AppVersion: 2.2.8](https://img.shields.io/static/v1?label=AppVersion&message=2.2.8&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-08-18 - -* Add logging, and http redirect config - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 6f79580..67276f7 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -73,6 +73,48 @@ volumes: [] - # mountPath: "/config" - # type: configMap - -+# Logs -+# https://docs.traefik.io/observability/logs/ -+logs: -+ # Traefik logs concern everything that happens to Traefik itself (startup, configuration, events, shutdown, and so on). -+ general: -+ # By default, the logs use a text format (common), but you can -+ # also ask for the json format in the format option -+ # format: json -+ # By default, the level is set to ERROR. Alternative logging levels are DEBUG, PANIC, FATAL, ERROR, WARN, and INFO. -+ level: ERROR -+ access: -+ # To enable access logs -+ enabled: false -+ # By default, logs are written using the Common Log Format (CLF). -+ # To write logs in JSON, use json in the format option. -+ # If the given format is unsupported, the default (CLF) is used instead. -+ # format: json -+ # To write the logs in an asynchronous fashion, specify a bufferingSize option. -+ # This option represents the number of log lines Traefik will keep in memory before writing -+ # them to the selected output. In some cases, this option can greatly help performances. -+ # bufferingSize: 100 -+ # Filtering https://docs.traefik.io/observability/access-logs/#filtering -+ filters: {} -+ # statuscodes: "200,300-302" -+ # retryattempts: true -+ # minduration: 10ms -+ # Fields -+ # https://docs.traefik.io/observability/access-logs/#limiting-the-fieldsincluding-headers -+ fields: -+ general: -+ defaultmode: keep -+ names: {} -+ # Examples: -+ # ClientUsername: drop -+ headers: -+ defaultmode: drop -+ names: {} -+ # Examples: -+ # User-Agent: redact -+ # Authorization: drop -+ # Content-Type: keep -+ - globalArguments: - - "--global.checknewversion" - - "--global.sendanonymoususage" -@@ -143,6 +185,10 @@ ports: - # Use nodeport if set. This is useful if you have configured Traefik in a - # LoadBalancer - # nodePort: 32080 -+ # Port Redirections -+ # Added in 2.2, you can make permanent redirects via entrypoints. -+ # https://docs.traefik.io/routing/entrypoints/#redirection -+ redirectTo: websecure - websecure: - port: 8443 - # hostPort: 8443 -``` - -## 8.12.0 ![AppVersion: 2.2.8](https://img.shields.io/static/v1?label=AppVersion&message=2.2.8&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-08-14 - -* Add image pull policy - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 10b3949..6f79580 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -2,6 +2,7 @@ - image: - name: traefik - tag: 2.2.8 -+ pullPolicy: IfNotPresent - - # - # Configure the deployment -``` - -## 8.11.0 ![AppVersion: 2.2.8](https://img.shields.io/static/v1?label=AppVersion&message=2.2.8&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-08-12 - -* Add dns policy option - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 80ddaaa..10b3949 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -26,6 +26,8 @@ deployment: - # volumeMounts: - # - name: data - # mountPath: /data -+ # Custom pod DNS policy. Apply if `hostNetwork: true` -+ # dnsPolicy: ClusterFirstWithHostNet - - # Pod disruption budget - podDisruptionBudget: -``` - -## 8.10.0 ![AppVersion: 2.2.8](https://img.shields.io/static/v1?label=AppVersion&message=2.2.8&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-08-11 - -* Add hostIp to port configuration - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 936ab92..80ddaaa 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -112,6 +112,12 @@ ports: - port: 9000 - # Use hostPort if set. - # hostPort: 9000 -+ # -+ # Use hostIP if set. If not set, Kubernetes will default to 0.0.0.0, which -+ # means it's listening on all your interfaces and all your IPs. You may want -+ # to set this value if you need traefik to listen on specific interface -+ # only. -+ # hostIP: 192.168.100.10 - - # Defines whether the port is exposed if service.type is LoadBalancer or - # NodePort. -``` - -## 8.9.2 ![AppVersion: 2.2.8](https://img.shields.io/static/v1?label=AppVersion&message=2.2.8&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-08-10 - -* Bump Traefik to 2.2.8 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 42ee893..936ab92 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1,7 +1,7 @@ - # Default values for Traefik - image: - name: traefik -- tag: 2.2.5 -+ tag: 2.2.8 - - # - # Configure the deployment -``` - -## 8.9.1 ![AppVersion: 2.2.5](https://img.shields.io/static/v1?label=AppVersion&message=2.2.5&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-07-15 - -* Upgrade traefik version - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index a7fb668..42ee893 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1,7 +1,7 @@ - # Default values for Traefik - image: - name: traefik -- tag: 2.2.1 -+ tag: 2.2.5 - - # - # Configure the deployment -``` - -## 8.9.0 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-07-08 - -* run init container to set proper permissions on volume - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 62e3a77..a7fb668 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -16,6 +16,16 @@ deployment: - podAnnotations: {} - # Additional containers (e.g. for metric offloading sidecars) - additionalContainers: [] -+ # Additional initContainers (e.g. for setting file permission as shown below) -+ initContainers: [] -+ # The "volume-permissions" init container is required if you run into permission issues. -+ # Related issue: https://github.com/containous/traefik/issues/6972 -+ # - name: volume-permissions -+ # image: busybox:1.31.1 -+ # command: ["sh", "-c", "chmod -Rv 600 /data/*"] -+ # volumeMounts: -+ # - name: data -+ # mountPath: /data - - # Pod disruption budget - podDisruptionBudget: -``` - -## 8.8.1 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-07-02 - -* Additional container fix - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 85df29c..62e3a77 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -15,7 +15,7 @@ deployment: - # Additional pod annotations (e.g. for mesh injection or prometheus scraping) - podAnnotations: {} - # Additional containers (e.g. for metric offloading sidecars) -- additionalContainers: {} -+ additionalContainers: [] - - # Pod disruption budget - podDisruptionBudget: -``` - -## 8.8.0 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-07-01 - -* added additionalContainers option to chart - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 6a9dfd8..85df29c 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -14,6 +14,8 @@ deployment: - annotations: {} - # Additional pod annotations (e.g. for mesh injection or prometheus scraping) - podAnnotations: {} -+ # Additional containers (e.g. for metric offloading sidecars) -+ additionalContainers: {} - - # Pod disruption budget - podDisruptionBudget: -``` - -## 8.7.2 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-06-30 - -* Update image - - -## 8.7.1 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-06-26 - -* Update values.yaml - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 05f9eab..6a9dfd8 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -196,7 +196,7 @@ rbac: - # If set to true, installs namespace-specific Role and RoleBinding and requires provider configuration be set to that same namespace - namespaced: false - --# The service account the pods will use to interact with the Kubernates API -+# The service account the pods will use to interact with the Kubernetes API - serviceAccount: - # If set, an existing service account is used - # If not set, a service account is created automatically using the fullname template -``` - -## 8.7.0 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-06-23 - -* Add option to disable providers - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 102ae00..05f9eab 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -34,6 +34,16 @@ rollingUpdate: - maxUnavailable: 1 - maxSurge: 1 - -+ -+# -+# Configure providers -+# -+providers: -+ kubernetesCRD: -+ enabled: true -+ kubernetesIngress: -+ enabled: true -+ - # - # Add volumes to the traefik pod. - # This can be used to mount a cert pair or a configmap that holds a config.toml file. -``` - -## 8.6.1 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-06-18 - -* Fix read-only /tmp - - -## 8.6.0 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-06-17 - -* Add existing PVC support(#158) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index b2f4fc3..102ae00 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -164,6 +164,7 @@ autoscaling: - # It will persist TLS certificates. - persistence: - enabled: false -+# existingClaim: "" - accessMode: ReadWriteOnce - size: 128Mi - # storageClass: "" -``` - -## 8.5.0 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-06-16 - -* UDP support - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 9a9b668..b2f4fc3 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -100,11 +100,15 @@ ports: - expose: false - # The exposed port for this service - exposedPort: 9000 -+ # The port protocol (TCP/UDP) -+ protocol: TCP - web: - port: 8000 - # hostPort: 8000 - expose: true - exposedPort: 80 -+ # The port protocol (TCP/UDP) -+ protocol: TCP - # Use nodeport if set. This is useful if you have configured Traefik in a - # LoadBalancer - # nodePort: 32080 -@@ -113,6 +117,8 @@ ports: - # hostPort: 8443 - expose: true - exposedPort: 443 -+ # The port protocol (TCP/UDP) -+ protocol: TCP - # nodePort: 32443 - - # Options for the main traefik service, where the entrypoints traffic comes -``` - -## 8.4.1 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-06-10 - -* Fix PDB with minAvailable set - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index e812b98..9a9b668 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -18,7 +18,7 @@ deployment: - # Pod disruption budget - podDisruptionBudget: - enabled: false -- maxUnavailable: 1 -+ # maxUnavailable: 1 - # minAvailable: 0 - - # Create an IngressRoute for the dashboard -``` - -## 8.4.0 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-06-09 - -* Add pod disruption budget (#192) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 5f44e5c..e812b98 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -15,6 +15,12 @@ deployment: - # Additional pod annotations (e.g. for mesh injection or prometheus scraping) - podAnnotations: {} - -+# Pod disruption budget -+podDisruptionBudget: -+ enabled: false -+ maxUnavailable: 1 -+ # minAvailable: 0 -+ - # Create an IngressRoute for the dashboard - ingressRoute: - dashboard: -``` - -## 8.3.0 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-06-08 - -* Add option to disable RBAC and ServiceAccount - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 96bba18..5f44e5c 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -165,6 +165,20 @@ persistence: - # affinity is left as default. - hostNetwork: false - -+# Whether Role Based Access Control objects like roles and rolebindings should be created -+rbac: -+ enabled: true -+ -+ # If set to false, installs ClusterRole and ClusterRoleBinding so Traefik can be used across namespaces. -+ # If set to true, installs namespace-specific Role and RoleBinding and requires provider configuration be set to that same namespace -+ namespaced: false -+ -+# The service account the pods will use to interact with the Kubernates API -+serviceAccount: -+ # If set, an existing service account is used -+ # If not set, a service account is created automatically using the fullname template -+ name: "" -+ - # Additional serviceAccount annotations (e.g. for oidc authentication) - serviceAccountAnnotations: {} - -``` - -## 8.2.1 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-05-25 - -* Remove suggested providers.kubernetesingress value - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index e35bdf9..96bba18 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -50,9 +50,9 @@ globalArguments: - # Configure Traefik static configuration - # Additional arguments to be passed at Traefik's binary - # All available options available on https://docs.traefik.io/reference/static-configuration/cli/ --## Use curly braces to pass values: `helm install --set="additionalArguments={--providers.kubernetesingress,--log.level=DEBUG}"` -+## Use curly braces to pass values: `helm install --set="additionalArguments={--providers.kubernetesingress.ingressclass=traefik-internal,--log.level=DEBUG}"` - additionalArguments: [] --# - "--providers.kubernetesingress" -+# - "--providers.kubernetesingress.ingressclass=traefik-internal" - # - "--log.level=DEBUG" - - # Environment variables to be passed to Traefik's binary -``` - -## 8.2.0 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-05-18 - -* Add kubernetes ingress by default - - -## 8.1.5 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-05-18 - -* Fix example log params in values.yml - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index abe2334..e35bdf9 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -50,10 +50,10 @@ globalArguments: - # Configure Traefik static configuration - # Additional arguments to be passed at Traefik's binary - # All available options available on https://docs.traefik.io/reference/static-configuration/cli/ --## Use curly braces to pass values: `helm install --set="additionalArguments={--providers.kubernetesingress,--logs.level=DEBUG}"` -+## Use curly braces to pass values: `helm install --set="additionalArguments={--providers.kubernetesingress,--log.level=DEBUG}"` - additionalArguments: [] - # - "--providers.kubernetesingress" --# - "--logs.level=DEBUG" -+# - "--log.level=DEBUG" - - # Environment variables to be passed to Traefik's binary - env: [] -``` - -## 8.1.4 ![AppVersion: 2.2.1](https://img.shields.io/static/v1?label=AppVersion&message=2.2.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-04-30 - -* Update Traefik to v2.2.1 - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 57cc7e1..abe2334 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1,7 +1,7 @@ - # Default values for Traefik - image: - name: traefik -- tag: 2.2.0 -+ tag: 2.2.1 - - # - # Configure the deployment -``` - -## 8.1.3 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-04-29 - -* Clarify additionnal arguments log - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index d639f72..57cc7e1 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -50,9 +50,10 @@ globalArguments: - # Configure Traefik static configuration - # Additional arguments to be passed at Traefik's binary - # All available options available on https://docs.traefik.io/reference/static-configuration/cli/ --## Use curly braces to pass values: `helm install --set="additionalArguments={--providers.kubernetesingress,--global.checknewversion=true}"` -+## Use curly braces to pass values: `helm install --set="additionalArguments={--providers.kubernetesingress,--logs.level=DEBUG}"` - additionalArguments: [] - # - "--providers.kubernetesingress" -+# - "--logs.level=DEBUG" - - # Environment variables to be passed to Traefik's binary - env: [] -``` - -## 8.1.2 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-04-23 - -* Remove invalid flags. (#161) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 0e7aaef..d639f72 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -34,8 +34,6 @@ rollingUpdate: - # After the volume has been mounted, add the configs into traefik by using the `additionalArguments` list below, eg: - # additionalArguments: - # - "--providers.file.filename=/config/dynamic.toml" --# - "--tls.certificates.certFile=/certs/tls.crt" --# - "--tls.certificates.keyFile=/certs/tls.key" - volumes: [] - # - name: public-cert - # mountPath: "/certs" -``` - -## 8.1.1 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-04-23 - -* clarify project philosophy and guidelines - - -## 8.1.0 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-04-22 - -* Add priorityClassName & securityContext - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index d55a40a..0e7aaef 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -191,3 +191,20 @@ affinity: {} - # topologyKey: failure-domain.beta.kubernetes.io/zone - nodeSelector: {} - tolerations: [] -+ -+# Pods can have priority. -+# Priority indicates the importance of a Pod relative to other Pods. -+priorityClassName: "" -+ -+# Set the container security context -+# To run the container with ports below 1024 this will need to be adjust to run as root -+securityContext: -+ capabilities: -+ drop: [ALL] -+ readOnlyRootFilesystem: true -+ runAsGroup: 65532 -+ runAsNonRoot: true -+ runAsUser: 65532 -+ -+podSecurityContext: -+ fsGroup: 65532 -``` - -## 8.0.4 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-04-20 - -* Possibility to bind environment variables via envFrom - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 7f8092e..d55a40a 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -71,6 +71,12 @@ env: [] - # name: secret-name - # key: secret-key - -+envFrom: [] -+# - configMapRef: -+# name: config-map-name -+# - secretRef: -+# name: secret-name -+ - # Configure ports - ports: - # The name of this one can't be changed as it is used for the readiness and -``` - -## 8.0.3 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-04-15 - -* Add support for data volume subPath. (#147) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 152339b..7f8092e 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -152,6 +152,7 @@ persistence: - # storageClass: "" - path: /data - annotations: {} -+ # subPath: "" # only mount a subpath of the Volume into the pod - - # If hostNetwork is true, runs traefik in the host network namespace - # To prevent unschedulabel pods due to port collisions, if hostNetwork=true -``` - -## 8.0.2 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-04-10 - -* Ability to add custom labels to dashboard's IngressRoute - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 5d294b7..152339b 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -21,6 +21,8 @@ ingressRoute: - enabled: true - # Additional ingressRoute annotations (e.g. for kubernetes.io/ingress.class) - annotations: {} -+ # Additional ingressRoute labels (e.g. for filtering IngressRoute by custom labels) -+ labels: {} - - rollingUpdate: - maxUnavailable: 1 -``` - -## 8.0.1 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-04-10 - -* rbac does not need "pods" per documentation - - -## 8.0.0 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-04-07 - -* follow helm best practices - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index e61a9fd..5d294b7 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -10,7 +10,7 @@ deployment: - enabled: true - # Number of pods of the deployment - replicas: 1 -- # Addtional deployment annotations (e.g. for jaeger-operator sidecar injection) -+ # Additional deployment annotations (e.g. for jaeger-operator sidecar injection) - annotations: {} - # Additional pod annotations (e.g. for mesh injection or prometheus scraping) - podAnnotations: {} -@@ -19,7 +19,7 @@ deployment: - ingressRoute: - dashboard: - enabled: true -- # Addtional ingressRoute annotations (e.g. for kubernetes.io/ingress.class) -+ # Additional ingressRoute annotations (e.g. for kubernetes.io/ingress.class) - annotations: {} - - rollingUpdate: -``` - -## 7.2.1 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-04-07 - -* add annotations to ingressRoute - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 15d1c25..e61a9fd 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -19,6 +19,8 @@ deployment: - ingressRoute: - dashboard: - enabled: true -+ # Addtional ingressRoute annotations (e.g. for kubernetes.io/ingress.class) -+ annotations: {} - - rollingUpdate: - maxUnavailable: 1 -``` - -## 7.2.0 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-04-03 - -* Add support for helm 2 - - -## 7.1.0 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-31 - -* Add support for externalIPs - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 6d6d13f..15d1c25 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -116,6 +116,8 @@ service: - loadBalancerSourceRanges: [] - # - 192.168.0.1/32 - # - 172.16.0.0/16 -+ externalIPs: [] -+ # - 1.2.3.4 - - ## Create HorizontalPodAutoscaler object. - ## -``` - -## 7.0.0 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-27 - -* Remove secretsEnv value key - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 1ac720d..6d6d13f 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -52,18 +52,20 @@ globalArguments: - additionalArguments: [] - # - "--providers.kubernetesingress" - --# Secret to be set as environment variables to be passed to Traefik's binary --secretEnv: [] -- # - name: SOME_VAR -- # secretName: my-secret-name -- # secretKey: my-secret-key -- - # Environment variables to be passed to Traefik's binary - env: [] -- # - name: SOME_VAR -- # value: some-var-value -- # - name: SOME_OTHER_VAR -- # value: some-other-var-value -+# - name: SOME_VAR -+# value: some-var-value -+# - name: SOME_VAR_FROM_CONFIG_MAP -+# valueFrom: -+# configMapRef: -+# name: configmap-name -+# key: config-key -+# - name: SOME_SECRET -+# valueFrom: -+# secretKeyRef: -+# name: secret-name -+# key: secret-key - - # Configure ports - ports: -``` - -## 6.4.0 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-27 - -* Add ability to set serviceAccount annotations - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 85abe42..1ac720d 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -151,6 +151,9 @@ persistence: - # affinity is left as default. - hostNetwork: false - -+# Additional serviceAccount annotations (e.g. for oidc authentication) -+serviceAccountAnnotations: {} -+ - resources: {} - # requests: - # cpu: "100m" -``` - -## 6.3.0 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-27 - -* hpa - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 2f5d132..85abe42 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -115,6 +115,22 @@ service: - # - 192.168.0.1/32 - # - 172.16.0.0/16 - -+## Create HorizontalPodAutoscaler object. -+## -+autoscaling: -+ enabled: false -+# minReplicas: 1 -+# maxReplicas: 10 -+# metrics: -+# - type: Resource -+# resource: -+# name: cpu -+# targetAverageUtilization: 60 -+# - type: Resource -+# resource: -+# name: memory -+# targetAverageUtilization: 60 -+ - # Enable persistence using Persistent Volume Claims - # ref: http://kubernetes.io/docs/user-guide/persistent-volumes/ - # After the pvc has been mounted, add the configs into traefik by using the `additionalArguments` list below, eg: -``` - -## 6.2.0 ![AppVersion: 2.2.0](https://img.shields.io/static/v1?label=AppVersion&message=2.2.0&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-26 - -* Update to v2.2 (#96) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index ebd2fde..2f5d132 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1,7 +1,7 @@ - # Default values for Traefik - image: - name: traefik -- tag: 2.1.8 -+ tag: 2.2.0 - - # - # Configure the deployment -``` - -## 6.1.2 ![AppVersion: 2.1.8](https://img.shields.io/static/v1?label=AppVersion&message=2.1.8&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-20 - -* Upgrade traefik version - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 65c7665..ebd2fde 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1,7 +1,7 @@ - # Default values for Traefik - image: - name: traefik -- tag: 2.1.4 -+ tag: 2.1.8 - - # - # Configure the deployment -``` - -## 6.1.1 ![AppVersion: 2.1.4](https://img.shields.io/static/v1?label=AppVersion&message=2.1.4&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-20 - -* Upgrade traefik version - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 89c7ac1..65c7665 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1,7 +1,7 @@ - # Default values for Traefik - image: - name: traefik -- tag: 2.1.3 -+ tag: 2.1.4 - - # - # Configure the deployment -``` - -## 6.1.0 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-20 - -* Add ability to add annotations to deployment - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 8d66111..89c7ac1 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -10,6 +10,8 @@ deployment: - enabled: true - # Number of pods of the deployment - replicas: 1 -+ # Addtional deployment annotations (e.g. for jaeger-operator sidecar injection) -+ annotations: {} - # Additional pod annotations (e.g. for mesh injection or prometheus scraping) - podAnnotations: {} - -``` - -## 6.0.2 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-16 - -* Correct storage class key name - - -## 6.0.1 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-16 - -* Change default values of arrays from objects to actual arrays - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 490b2b6..8d66111 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -51,13 +51,13 @@ additionalArguments: [] - # - "--providers.kubernetesingress" - - # Secret to be set as environment variables to be passed to Traefik's binary --secretEnv: {} -+secretEnv: [] - # - name: SOME_VAR - # secretName: my-secret-name - # secretKey: my-secret-key - - # Environment variables to be passed to Traefik's binary --env: {} -+env: [] - # - name: SOME_VAR - # value: some-var-value - # - name: SOME_OTHER_VAR -@@ -109,7 +109,7 @@ service: - # externalTrafficPolicy: Cluster - # loadBalancerIP: "1.2.3.4" - # clusterIP: "2.3.4.5" -- loadBalancerSourceRanges: {} -+ loadBalancerSourceRanges: [] - # - 192.168.0.1/32 - # - 172.16.0.0/16 - -``` - -## 6.0.0 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-15 - -* Cleanup - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 7aebefe..490b2b6 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -18,15 +18,10 @@ ingressRoute: - dashboard: - enabled: true - --additional: -- checkNewVersion: true -- sendAnonymousUsage: true -- - rollingUpdate: - maxUnavailable: 1 - maxSurge: 1 - -- - # - # Add volumes to the traefik pod. - # This can be used to mount a cert pair or a configmap that holds a config.toml file. -@@ -43,9 +38,14 @@ volumes: [] - # mountPath: "/config" - # type: configMap - -+globalArguments: -+ - "--global.checknewversion" -+ - "--global.sendanonymoususage" -+ - # --# Configure Traefik entry points -+# Configure Traefik static configuration - # Additional arguments to be passed at Traefik's binary -+# All available options available on https://docs.traefik.io/reference/static-configuration/cli/ - ## Use curly braces to pass values: `helm install --set="additionalArguments={--providers.kubernetesingress,--global.checknewversion=true}"` - additionalArguments: [] - # - "--providers.kubernetesingress" -@@ -63,7 +63,7 @@ env: {} - # - name: SOME_OTHER_VAR - # value: some-other-var-value - --# -+# Configure ports - ports: - # The name of this one can't be changed as it is used for the readiness and - # liveness probes, but you can adjust its config to your liking -@@ -94,7 +94,7 @@ ports: - # hostPort: 8443 - expose: true - exposedPort: 443 -- # nodePort: 32443 -+ # nodePort: 32443 - - # Options for the main traefik service, where the entrypoints traffic comes - # from. -@@ -113,9 +113,6 @@ service: - # - 192.168.0.1/32 - # - 172.16.0.0/16 - --logs: -- loglevel: WARN -- - # Enable persistence using Persistent Volume Claims - # ref: http://kubernetes.io/docs/user-guide/persistent-volumes/ - # After the pvc has been mounted, add the configs into traefik by using the `additionalArguments` list below, eg: -``` - -## 5.6.0 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-12 - -* Add field enabled for resources - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 38bb263..7aebefe 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -7,11 +7,17 @@ image: - # Configure the deployment - # - deployment: -+ enabled: true - # Number of pods of the deployment - replicas: 1 - # Additional pod annotations (e.g. for mesh injection or prometheus scraping) - podAnnotations: {} - -+# Create an IngressRoute for the dashboard -+ingressRoute: -+ dashboard: -+ enabled: true -+ - additional: - checkNewVersion: true - sendAnonymousUsage: true -@@ -93,6 +99,7 @@ ports: - # Options for the main traefik service, where the entrypoints traffic comes - # from. - service: -+ enabled: true - type: LoadBalancer - # Additional annotations (e.g. for cloud provider specific config) - annotations: {} -``` - -## 5.5.0 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-12 - -* expose hostnetwork option - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index ecb2833..38bb263 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -123,6 +123,12 @@ persistence: - path: /data - annotations: {} - -+# If hostNetwork is true, runs traefik in the host network namespace -+# To prevent unschedulabel pods due to port collisions, if hostNetwork=true -+# and replicas>1, a pod anti-affinity is recommended and will be set if the -+# affinity is left as default. -+hostNetwork: false -+ - resources: {} - # requests: - # cpu: "100m" -@@ -131,5 +137,17 @@ resources: {} - # cpu: "300m" - # memory: "150Mi" - affinity: {} -+# # This example pod anti-affinity forces the scheduler to put traefik pods -+# # on nodes where no other traefik pods are scheduled. -+# # It should be used when hostNetwork: true to prevent port conflicts -+# podAntiAffinity: -+# requiredDuringSchedulingIgnoredDuringExecution: -+# - labelSelector: -+# matchExpressions: -+# - key: app -+# operator: In -+# values: -+# - {{ template "traefik.name" . }} -+# topologyKey: failure-domain.beta.kubernetes.io/zone - nodeSelector: {} - tolerations: [] -``` - -## 5.4.0 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-12 - -* Add support for hostport - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index ec1d619..ecb2833 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -63,6 +63,9 @@ ports: - # liveness probes, but you can adjust its config to your liking - traefik: - port: 9000 -+ # Use hostPort if set. -+ # hostPort: 9000 -+ - # Defines whether the port is exposed if service.type is LoadBalancer or - # NodePort. - # -@@ -74,6 +77,7 @@ ports: - exposedPort: 9000 - web: - port: 8000 -+ # hostPort: 8000 - expose: true - exposedPort: 80 - # Use nodeport if set. This is useful if you have configured Traefik in a -@@ -81,6 +85,7 @@ ports: - # nodePort: 32080 - websecure: - port: 8443 -+ # hostPort: 8443 - expose: true - exposedPort: 443 - # nodePort: 32443 -``` - -## 5.3.3 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-12 - -* Fix replica check - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 7f31548..ec1d619 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -40,7 +40,7 @@ volumes: [] - # - # Configure Traefik entry points - # Additional arguments to be passed at Traefik's binary --## Use curly braces to pass values: `helm install --set="{--providers.kubernetesingress,--global.checknewversion=true}" ." -+## Use curly braces to pass values: `helm install --set="additionalArguments={--providers.kubernetesingress,--global.checknewversion=true}"` - additionalArguments: [] - # - "--providers.kubernetesingress" - -``` - -## 5.3.2 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-11 - -* Fixed typo in README - - -## 5.3.1 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-11 - -* Production ready - - -## 5.3.0 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-11 - -* Not authorise acme if replica > 1 - - -## 5.2.1 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-11 - -* Fix volume mount - - -## 5.2.0 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-11 - -* Add secret as env var - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index ccea845..7f31548 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -44,12 +44,18 @@ volumes: [] - additionalArguments: [] - # - "--providers.kubernetesingress" - -+# Secret to be set as environment variables to be passed to Traefik's binary -+secretEnv: {} -+ # - name: SOME_VAR -+ # secretName: my-secret-name -+ # secretKey: my-secret-key -+ - # Environment variables to be passed to Traefik's binary - env: {} --# - name: SOME_VAR --# value: some-var-value --# - name: SOME_OTHER_VAR --# value: some-other-var-value -+ # - name: SOME_VAR -+ # value: some-var-value -+ # - name: SOME_OTHER_VAR -+ # value: some-other-var-value - - # - ports: -``` - -## 5.1.0 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-10 - -* Enhance security by add loadBalancerSourceRanges to lockdown ip address. - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 78bbee0..ccea845 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -91,6 +91,9 @@ service: - # externalTrafficPolicy: Cluster - # loadBalancerIP: "1.2.3.4" - # clusterIP: "2.3.4.5" -+ loadBalancerSourceRanges: {} -+ # - 192.168.0.1/32 -+ # - 172.16.0.0/16 - - logs: - loglevel: WARN -``` - -## 5.0.0 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-10 - -* Expose dashboard by default but only on traefik entrypoint - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index a442fca..78bbee0 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -92,15 +92,6 @@ service: - # loadBalancerIP: "1.2.3.4" - # clusterIP: "2.3.4.5" - --dashboard: -- # Enable the dashboard on Traefik -- enable: true -- -- # Expose the dashboard and api through an ingress route at /dashboard -- # and /api This is not secure and SHOULD NOT be enabled on production -- # deployments -- ingressRoute: false -- - logs: - loglevel: WARN - -``` - -## 4.1.3 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-10 - -* Add annotations for PVC (#98) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 8b2f4db..a442fca 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -116,6 +116,7 @@ persistence: - size: 128Mi - # storageClass: "" - path: /data -+ annotations: {} - - resources: {} - # requests: -``` - -## 4.1.2 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-10 - -* Added persistent volume support. (#86) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 2a2554f..8b2f4db 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -103,7 +103,20 @@ dashboard: - - logs: - loglevel: WARN --# -+ -+# Enable persistence using Persistent Volume Claims -+# ref: http://kubernetes.io/docs/user-guide/persistent-volumes/ -+# After the pvc has been mounted, add the configs into traefik by using the `additionalArguments` list below, eg: -+# additionalArguments: -+# - "--certificatesresolvers.le.acme.storage=/data/acme.json" -+# It will persist TLS certificates. -+persistence: -+ enabled: false -+ accessMode: ReadWriteOnce -+ size: 128Mi -+ # storageClass: "" -+ path: /data -+ - resources: {} - # requests: - # cpu: "100m" -``` - -## 4.1.1 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-10 - -* Add values to mount secrets or configmaps as volumes to the traefik pod (#84) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 5401832..2a2554f 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -20,6 +20,23 @@ rollingUpdate: - maxUnavailable: 1 - maxSurge: 1 - -+ -+# -+# Add volumes to the traefik pod. -+# This can be used to mount a cert pair or a configmap that holds a config.toml file. -+# After the volume has been mounted, add the configs into traefik by using the `additionalArguments` list below, eg: -+# additionalArguments: -+# - "--providers.file.filename=/config/dynamic.toml" -+# - "--tls.certificates.certFile=/certs/tls.crt" -+# - "--tls.certificates.keyFile=/certs/tls.key" -+volumes: [] -+# - name: public-cert -+# mountPath: "/certs" -+# type: secret -+# - name: configs -+# mountPath: "/config" -+# type: configMap -+ - # - # Configure Traefik entry points - # Additional arguments to be passed at Traefik's binary -``` - -## 4.1.0 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-10 - -* Add podAnnotations to the deployment (#83) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 5eab74b..5401832 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -9,6 +9,8 @@ image: - deployment: - # Number of pods of the deployment - replicas: 1 -+ # Additional pod annotations (e.g. for mesh injection or prometheus scraping) -+ podAnnotations: {} - - additional: - checkNewVersion: true -``` - -## 4.0.0 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-03-06 - -* Migrate to helm v3 (#94) - - -## 3.5.0 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-02-18 - -* Publish helm chart (#81) - - -## 3.4.0 ![AppVersion: 2.1.3](https://img.shields.io/static/v1?label=AppVersion&message=2.1.3&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-02-13 - -* fix: tests. -* feat: bump traefik to v2.1.3 -* Enable configuration of global checknewversion and sendanonymoususage (#80) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index bcc42f8..5eab74b 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -1,7 +1,7 @@ - # Default values for Traefik - image: - name: traefik -- tag: 2.1.1 -+ tag: 2.1.3 - - # - # Configure the deployment -@@ -10,6 +10,10 @@ deployment: - # Number of pods of the deployment - replicas: 1 - -+additional: -+ checkNewVersion: true -+ sendAnonymousUsage: true -+ - rollingUpdate: - maxUnavailable: 1 - maxSurge: 1 -``` - -## 3.3.3 ![AppVersion: 2.1.1](https://img.shields.io/static/v1?label=AppVersion&message=2.1.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-02-05 - -* fix: deployment environment variables. -* fix: chart version. - - -## 3.3.2 ![AppVersion: 2.1.1](https://img.shields.io/static/v1?label=AppVersion&message=2.1.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-02-03 - -* ix: deployment environment variables. - - -## 3.3.1 ![AppVersion: 2.1.1](https://img.shields.io/static/v1?label=AppVersion&message=2.1.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-01-27 - -* fix: deployment environment variables. - - -## 3.3.0 ![AppVersion: 2.1.1](https://img.shields.io/static/v1?label=AppVersion&message=2.1.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-01-24 - -* Enable configuration of environment variables in traefik deployment (#71) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index 4462359..bcc42f8 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -21,6 +21,13 @@ rollingUpdate: - additionalArguments: [] - # - "--providers.kubernetesingress" - -+# Environment variables to be passed to Traefik's binary -+env: {} -+# - name: SOME_VAR -+# value: some-var-value -+# - name: SOME_OTHER_VAR -+# value: some-other-var-value -+ - # - ports: - # The name of this one can't be changed as it is used for the readiness and -``` - -## 3.2.1 ![AppVersion: 2.1.1](https://img.shields.io/static/v1?label=AppVersion&message=2.1.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-01-22 - -* Add Unit Tests for the chart (#60) - - -## 3.2.0 ![AppVersion: 2.1.1](https://img.shields.io/static/v1?label=AppVersion&message=2.1.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-01-22 - -* Make NodePort configurable (#67) - -### Default value changes - -```diff -diff --git a/traefik/values.yaml b/traefik/values.yaml -index b1fe42a..4462359 100644 ---- a/traefik/values.yaml -+++ b/traefik/values.yaml -@@ -40,10 +40,14 @@ ports: - port: 8000 - expose: true - exposedPort: 80 -+ # Use nodeport if set. This is useful if you have configured Traefik in a -+ # LoadBalancer -+ # nodePort: 32080 - websecure: - port: 8443 - expose: true - exposedPort: 443 -+ # nodePort: 32443 - - # Options for the main traefik service, where the entrypoints traffic comes - # from. -``` - -## 3.1.0 ![AppVersion: 2.1.1](https://img.shields.io/static/v1?label=AppVersion&message=2.1.1&color=success&logo=) ![Helm: v2](https://img.shields.io/static/v1?label=Helm&message=v2&color=inactive&logo=helm) ![Helm: v3](https://img.shields.io/static/v1?label=Helm&message=v3&color=informational&logo=helm) - -**Release date:** 2020-01-20 - -* Switch Chart linting to ct (#59) - -### Default value changes - -```diff -# Default values for Traefik -image: - name: traefik - tag: 2.1.1 - -# -# Configure the deployment -# -deployment: - # Number of pods of the deployment - replicas: 1 - -rollingUpdate: - maxUnavailable: 1 - maxSurge: 1 - -# -# Configure Traefik entry points -# Additional arguments to be passed at Traefik's binary -## Use curly braces to pass values: `helm install --set="{--providers.kubernetesingress,--global.checknewversion=true}" ." -additionalArguments: [] -# - "--providers.kubernetesingress" - -# -ports: - # The name of this one can't be changed as it is used for the readiness and - # liveness probes, but you can adjust its config to your liking - traefik: - port: 9000 - # Defines whether the port is exposed if service.type is LoadBalancer or - # NodePort. - # - # You SHOULD NOT expose the traefik port on production deployments. - # If you want to access it from outside of your cluster, - # use `kubectl proxy` or create a secure ingress - expose: false - # The exposed port for this service - exposedPort: 9000 - web: - port: 8000 - expose: true - exposedPort: 80 - websecure: - port: 8443 - expose: true - exposedPort: 443 - -# Options for the main traefik service, where the entrypoints traffic comes -# from. -service: - type: LoadBalancer - # Additional annotations (e.g. for cloud provider specific config) - annotations: {} - # Additional entries here will be added to the service spec. Cannot contains - # type, selector or ports entries. - spec: {} - # externalTrafficPolicy: Cluster - # loadBalancerIP: "1.2.3.4" - # clusterIP: "2.3.4.5" - -dashboard: - # Enable the dashboard on Traefik - enable: true - - # Expose the dashboard and api through an ingress route at /dashboard - # and /api This is not secure and SHOULD NOT be enabled on production - # deployments - ingressRoute: false - -logs: - loglevel: WARN -# -resources: {} - # requests: - # cpu: "100m" - # memory: "50Mi" - # limits: - # cpu: "300m" - # memory: "150Mi" -affinity: {} -nodeSelector: {} -tolerations: [] -``` - ---- -Autogenerated from Helm Chart and git history using [helm-changelog](https://github.com/mogensen/helm-changelog) diff --git a/charts/traefik/traefik/Chart.yaml b/charts/traefik/traefik/Chart.yaml deleted file mode 100644 index 0d1220a..0000000 --- a/charts/traefik/traefik/Chart.yaml +++ /dev/null @@ -1,39 +0,0 @@ -annotations: - artifacthub.io/changes: "- \"fix: \U0001F41B http3 with internal service\"\n- \"fix: - use correct children indentation for logs.access.filters\"\n- \"fix(schema): :bug: - targetPort can also be a string\"\n- \"fix(certificateResolvers)!: :boom: :bug: - use same syntax in Chart and in Traefik\"\n- \"fix(Traefik)!: :boom: set 8080 - as default port for `traefik` entrypoint\"\n- \"fix(Traefik Hub): RBAC for distributedAcme\"\n- - \"fix(Kubernetes Ingress)!: :boom: :sparkles: enable publishedService by default\"\n- - \"fix(Gateway API): :bug: add missing required RBAC for v3.2 with experimental - Channel\"\n- \"fix(Env Variables)!: allow extending env without overwrite\"\n- - \"feat(deps): update traefik docker tag to v3.2.0\"\n- \"feat(deps): update traefik - docker tag to v3.1.6\"\n- \"feat(Traefik): ✨ support Gateway API statusAddress\"\n- - \"feat(Traefik Proxy): CRDs for v3.2+\"\n- \"feat(Gateway API): :sparkles: standard - install CRD v1.2.0\"\n- \"feat(Gateway API): :sparkles: add infrastructure in - the values\"\n- \"chore: allow TRACE log level\"\n- \"chore(release): \U0001F680 - publish v33.0.0\"\n- \"Update topology spread constraints comments\"\n" -apiVersion: v2 -appVersion: v3.2.0 -description: A Traefik based Kubernetes ingress controller -home: https://traefik.io/ -icon: https://raw.githubusercontent.com/traefik/traefik/v2.3/docs/content/assets/img/traefik.logo.png -keywords: -- traefik -- ingress -- networking -kubeVersion: '>=1.22.0-0' -maintainers: -- email: michel.loiseleur@traefik.io - name: mloiseleur -- email: charlie.haley@traefik.io - name: charlie-haley -- email: remi.buisson@traefik.io - name: darkweaver87 -- name: jnoordsij -name: traefik -sources: -- https://github.com/traefik/traefik -- https://github.com/traefik/traefik-helm-chart -type: application -version: 33.0.0 diff --git a/charts/traefik/traefik/EXAMPLES.md b/charts/traefik/traefik/EXAMPLES.md deleted file mode 100644 index 6c70f5a..0000000 --- a/charts/traefik/traefik/EXAMPLES.md +++ /dev/null @@ -1,1014 +0,0 @@ -# Install as a DaemonSet - -Default install is using a `Deployment` but it's possible to use `DaemonSet` - -```yaml -deployment: - kind: DaemonSet -``` - -# Configure traefik Pod parameters - -## Extending /etc/hosts records - -In some specific cases, you'll need to add extra records to the `/etc/hosts` file for the Traefik containers. -You can configure it using [hostAliases](https://kubernetes.io/docs/tasks/network/customize-hosts-file-for-pods/): - -```yaml -deployment: - hostAliases: - - ip: "127.0.0.1" # this is an example - hostnames: - - "foo.local" - - "bar.local" -``` -## Extending DNS config - -In order to configure additional DNS servers for your traefik pod, you can use `dnsConfig` option: - -```yaml -deployment: - dnsConfig: - nameservers: - - 192.0.2.1 # this is an example - searches: - - ns1.svc.cluster-domain.example - - my.dns.search.suffix - options: - - name: ndots - value: "2" - - name: edns0 -``` - -# Install in a dedicated namespace, with limited RBAC - -Default install is using Cluster-wide RBAC but it can be restricted to target namespace. - -```yaml -rbac: - namespaced: true -``` - -# Install with auto-scaling - -When enabling [HPA](https://kubernetes.io/docs/tasks/run-application/horizontal-pod-autoscale/) -to adjust replicas count according to CPU Usage, you'll need to set resources and nullify replicas. - -```yaml -deployment: - replicas: null -resources: - requests: - cpu: "100m" - memory: "50Mi" - limits: - cpu: "300m" - memory: "150Mi" -autoscaling: - enabled: true - maxReplicas: 2 - metrics: - - type: Resource - resource: - name: cpu - target: - type: Utilization - averageUtilization: 80 -``` - -# Access Traefik dashboard without exposing it - -This Chart does not expose the Traefik local dashboard by default. It's explained in upstream [documentation](https://doc.traefik.io/traefik/operations/api/) why: - -> Enabling the API in production is not recommended, because it will expose all configuration elements, including sensitive data. - -It says also: - -> In production, it should be at least secured by authentication and authorizations. - -Thus, there are multiple ways to expose the dashboard. For instance, after enabling the creation of dashboard `IngressRoute` in the values: - -```yaml -ingressRoute: - dashboard: - enabled: true -``` - -The traefik admin port can be forwarded locally: - -```bash -kubectl port-forward $(kubectl get pods --selector "app.kubernetes.io/name=traefik" --output=name) 8080:8080 -``` - -This command makes the dashboard accessible on the url: http://127.0.0.1:8080/dashboard/ - -# Publish and protect Traefik Dashboard with basic Auth - -To expose the dashboard in a secure way as [recommended](https://doc.traefik.io/traefik/operations/dashboard/#dashboard-router-rule) -in the documentation, it may be useful to override the router rule to specify -a domain to match, or accept requests on the root path (/) in order to redirect -them to /dashboard/. - -```yaml -# Create an IngressRoute for the dashboard -ingressRoute: - dashboard: - enabled: true - # Custom match rule with host domain - matchRule: Host(`traefik-dashboard.example.com`) - entryPoints: ["websecure"] - # Add custom middlewares : authentication and redirection - middlewares: - - name: traefik-dashboard-auth - -# Create the custom middlewares used by the IngressRoute dashboard (can also be created in another way). -# /!\ Yes, you need to replace "changeme" password with a better one. /!\ -extraObjects: - - apiVersion: v1 - kind: Secret - metadata: - name: traefik-dashboard-auth-secret - type: kubernetes.io/basic-auth - stringData: - username: admin - password: changeme - - - apiVersion: traefik.io/v1alpha1 - kind: Middleware - metadata: - name: traefik-dashboard-auth - spec: - basicAuth: - secret: traefik-dashboard-auth-secret -``` - -# Publish and protect Traefik Dashboard with an Ingress - -To expose the dashboard without IngressRoute, it's more complicated and less -secure. You'll need to create an internal Service exposing Traefik API with -special _traefik_ entrypoint. This internal Service can be created from an other tool, with the `extraObjects` section or using [custom services](#add-custom-internal-services). - -You'll need to double check: -1. Service selector with your setup. -2. Middleware annotation on the ingress, _default_ should be replaced with traefik's namespace - -```yaml -ingressRoute: - dashboard: - enabled: false -additionalArguments: -- "--api.insecure=true" -# Create the service, middleware and Ingress used to expose the dashboard (can also be created in another way). -# /!\ Yes, you need to replace "changeme" password with a better one. /!\ -extraObjects: - - apiVersion: v1 - kind: Service - metadata: - name: traefik-api - spec: - type: ClusterIP - selector: - app.kubernetes.io/name: traefik - app.kubernetes.io/instance: traefik-default - ports: - - port: 8080 - name: traefik - targetPort: 8080 - protocol: TCP - - - apiVersion: v1 - kind: Secret - metadata: - name: traefik-dashboard-auth-secret - type: kubernetes.io/basic-auth - stringData: - username: admin - password: changeme - - - apiVersion: traefik.io/v1alpha1 - kind: Middleware - metadata: - name: traefik-dashboard-auth - spec: - basicAuth: - secret: traefik-dashboard-auth-secret - - - apiVersion: networking.k8s.io/v1 - kind: Ingress - metadata: - name: traefik-dashboard - annotations: - traefik.ingress.kubernetes.io/router.entrypoints: websecure - traefik.ingress.kubernetes.io/router.middlewares: default-traefik-dashboard-auth@kubernetescrd - spec: - rules: - - host: traefik-dashboard.example.com - http: - paths: - - path: / - pathType: Prefix - backend: - service: - name: traefik-api - port: - name: traefik -``` - - -# Install on AWS - -It can use [native AWS support](https://kubernetes.io/docs/concepts/services-networking/service/#aws-nlb-support) on Kubernetes - -```yaml -service: - annotations: - service.beta.kubernetes.io/aws-load-balancer-type: nlb -``` - -Or if [AWS LB controller](https://kubernetes-sigs.github.io/aws-load-balancer-controller/v2.2/guide/service/annotations/#legacy-cloud-provider) is installed : -```yaml -service: - annotations: - service.beta.kubernetes.io/aws-load-balancer-type: nlb-ip -``` - -# Install on GCP - -A [regional IP with a Service](https://cloud.google.com/kubernetes-engine/docs/tutorials/configuring-domain-name-static-ip#use_a_service) can be used -```yaml -service: - spec: - loadBalancerIP: "1.2.3.4" -``` - -Or a [global IP on Ingress](https://cloud.google.com/kubernetes-engine/docs/tutorials/configuring-domain-name-static-ip#use_an_ingress) -```yaml -service: - type: NodePort -extraObjects: - - apiVersion: networking.k8s.io/v1 - kind: Ingress - metadata: - name: traefik - annotations: - kubernetes.io/ingress.global-static-ip-name: "myGlobalIpName" - spec: - defaultBackend: - service: - name: traefik - port: - number: 80 -``` - -Or a [global IP on a Gateway](https://cloud.google.com/kubernetes-engine/docs/how-to/deploying-gateways) with continuous HTTPS encryption. - -```yaml -ports: - websecure: - appProtocol: HTTPS # Hint for Google L7 load balancer -service: - type: ClusterIP -extraObjects: -- apiVersion: gateway.networking.k8s.io/v1beta1 - kind: Gateway - metadata: - name: traefik - annotations: - networking.gke.io/certmap: "myCertificateMap" - spec: - gatewayClassName: gke-l7-global-external-managed - addresses: - - type: NamedAddress - value: "myGlobalIPName" - listeners: - - name: https - protocol: HTTPS - port: 443 -- apiVersion: gateway.networking.k8s.io/v1beta1 - kind: HTTPRoute - metadata: - name: traefik - spec: - parentRefs: - - kind: Gateway - name: traefik - rules: - - backendRefs: - - name: traefik - port: 443 -- apiVersion: networking.gke.io/v1 - kind: HealthCheckPolicy - metadata: - name: traefik - spec: - default: - config: - type: HTTP - httpHealthCheck: - port: 8080 - requestPath: /ping - targetRef: - group: "" - kind: Service - name: traefik -``` - -# Install on Azure - -A [static IP on a resource group](https://learn.microsoft.com/en-us/azure/aks/static-ip) can be used: - -```yaml -service: - spec: - loadBalancerIP: "1.2.3.4" - annotations: - service.beta.kubernetes.io/azure-load-balancer-resource-group: myResourceGroup -``` - -Here is a more complete example, using also native Let's encrypt feature of Traefik Proxy with Azure DNS: - -```yaml -persistence: - enabled: true - size: 128Mi -certificatesResolvers: - letsencrypt: - acme: - email: "{{ letsencrypt_email }}" - #caServer: https://acme-v02.api.letsencrypt.org/directory # Production server - caServer: https://acme-staging-v02.api.letsencrypt.org/directory # Staging server - dnsChallenge: - provider: azuredns - storage: /data/acme.json -env: - - name: AZURE_CLIENT_ID - value: "{{ azure_dns_challenge_application_id }}" - - name: AZURE_CLIENT_SECRET - valueFrom: - secretKeyRef: - name: azuredns-secret - key: client-secret - - name: AZURE_SUBSCRIPTION_ID - value: "{{ azure_subscription_id }}" - - name: AZURE_TENANT_ID - value: "{{ azure_tenant_id }}" - - name: AZURE_RESOURCE_GROUP - value: "{{ azure_resource_group }}" -deployment: - initContainers: - - name: volume-permissions - image: busybox:latest - command: ["sh", "-c", "ls -la /; touch /data/acme.json; chmod -v 600 /data/acme.json"] - volumeMounts: - - mountPath: /data - name: data -podSecurityContext: - fsGroup: 65532 - fsGroupChangePolicy: "OnRootMismatch" -service: - spec: - type: LoadBalancer - annotations: - service.beta.kubernetes.io/azure-load-balancer-resource-group: "{{ azure_node_resource_group }}" - service.beta.kubernetes.io/azure-pip-name: "{{ azure_resource_group }}" - service.beta.kubernetes.io/azure-dns-label-name: "{{ azure_resource_group }}" - service.beta.kubernetes.io/azure-allowed-ip-ranges: "{{ ip_range | join(',') }}" -extraObjects: - - apiVersion: v1 - kind: Secret - metadata: - name: azuredns-secret - namespace: traefik - type: Opaque - stringData: - client-secret: "{{ azure_dns_challenge_application_secret }}" -``` - -# Use an IngressClass - -Default install comes with an `IngressClass` resource that can be enabled on providers. - -Here's how one can enable it on CRD & Ingress Kubernetes provider: - -```yaml -ingressClass: - name: traefik -providers: - kubernetesCRD: - ingressClass: traefik - kubernetesIngress: - ingressClass: traefik -``` - -# Use HTTP3 - -By default, it will use a Load balancers with mixed protocols on `websecure` -entrypoint. They are available since v1.20 and in beta as of Kubernetes v1.24. -Availability may depend on your Kubernetes provider. - -When using TCP and UDP with a single service, you may encounter [this issue](https://github.com/kubernetes/kubernetes/issues/47249#issuecomment-587960741) from Kubernetes. -If you want to avoid this issue, you can set `ports.websecure.http3.advertisedPort` -to an other value than 443 - -```yaml -ports: - websecure: - http3: - enabled: true -``` - -You can also create two `Service`, one for TCP and one for UDP: - -```yaml -ports: - websecure: - http3: - enabled: true -service: - single: false -``` - -# Use PROXY protocol on Digital Ocean - -PROXY protocol is a protocol for sending client connection information, such as origin IP addresses and port numbers, to the final backend server, rather than discarding it at the load balancer. - -```yaml -.DOTrustedIPs: &DOTrustedIPs - - 127.0.0.1/32 - - 10.120.0.0/16 - -service: - enabled: true - type: LoadBalancer - annotations: - # This will tell DigitalOcean to enable the proxy protocol. - service.beta.kubernetes.io/do-loadbalancer-enable-proxy-protocol: "true" - spec: - # This is the default and should stay as cluster to keep the DO health checks working. - externalTrafficPolicy: Cluster - -ports: - web: - forwardedHeaders: - trustedIPs: *DOTrustedIPs - proxyProtocol: - trustedIPs: *DOTrustedIPs - websecure: - forwardedHeaders: - trustedIPs: *DOTrustedIPs - proxyProtocol: - trustedIPs: *DOTrustedIPs -``` - -# Enable plugin storage - -This chart follows common security practices: it runs as non root with a readonly root filesystem. -When enabling a plugin which needs storage, you have to add it to the deployment. - -Here is a simple example with crowdsec. You may want to replace with your plugin or see complete exemple on crowdsec [here](https://github.com/maxlerebourg/crowdsec-bouncer-traefik-plugin/blob/main/examples/kubernetes/README.md). - -```yaml -deployment: - additionalVolumes: - - name: plugins -additionalVolumeMounts: -- name: plugins - mountPath: /plugins-storage -additionalArguments: -- "--experimental.plugins.bouncer.moduleName=github.com/maxlerebourg/crowdsec-bouncer-traefik-plugin" -- "--experimental.plugins.bouncer.version=v1.1.9" -``` - -# Use Traefik native Let's Encrypt integration, without cert-manager - -In Traefik Proxy, ACME certificates are stored in a JSON file. - -This file needs to have 0600 permissions, meaning, only the owner of the file has full read and write access to it. -By default, Kubernetes recursively changes ownership and permissions for the content of each volume. - -=> An initContainer can be used to avoid an issue on this sensitive file. -See [#396](https://github.com/traefik/traefik-helm-chart/issues/396) for more details. - -Once the provider is ready, it can be used in an `IngressRoute`: - -```yaml -apiVersion: traefik.io/v1alpha1 -kind: IngressRoute -metadata: - name: [...] -spec: - entryPoints: [...] - routes: [...] - tls: - certResolver: letsencrypt -``` - -:information_source: Change `apiVersion` to `traefik.containo.us/v1alpha1` for charts prior to v28.0.0 - -See [the list of supported providers](https://doc.traefik.io/traefik/https/acme/#providers) for others. - -## Example with CloudFlare - -This example needs a CloudFlare token in a Kubernetes `Secret` and a working `StorageClass`. - -**Step 1**: Create `Secret` with CloudFlare token: - -```yaml ---- -apiVersion: v1 -kind: Secret -metadata: - name: cloudflare -type: Opaque -stringData: - token: {{ SET_A_VALID_TOKEN_HERE }} -``` - -**Step 2**: - -```yaml -persistence: - enabled: true - storageClass: xxx -certificatesResolvers: - letsencrypt: - acme: - dnsChallenge: - provider: cloudflare - storage: /data/acme.json -env: - - name: CF_DNS_API_TOKEN - valueFrom: - secretKeyRef: - name: cloudflare - key: token -deployment: - initContainers: - - name: volume-permissions - image: busybox:latest - command: ["sh", "-c", "touch /data/acme.json; chmod -v 600 /data/acme.json"] - volumeMounts: - - mountPath: /data - name: data -podSecurityContext: - fsGroup: 65532 - fsGroupChangePolicy: "OnRootMismatch" -``` - ->[!NOTE] -> With [Traefik Hub](https://traefik.io/traefik-hub/), certificates can be stored as a `Secret` on Kubernetes with `distributedAcme` resolver. - -# Provide default certificate with cert-manager and CloudFlare DNS - -Setup: - -* cert-manager installed in `cert-manager` namespace -* A cloudflare account on a DNS Zone - -**Step 1**: Create `Secret` and `Issuer` needed by `cert-manager` with your API Token. -See [cert-manager documentation](https://cert-manager.io/docs/configuration/acme/dns01/cloudflare/) -for creating this token with needed rights: - -```yaml ---- -apiVersion: v1 -kind: Secret -metadata: - name: cloudflare - namespace: traefik -type: Opaque -stringData: - api-token: XXX ---- -apiVersion: cert-manager.io/v1 -kind: Issuer -metadata: - name: cloudflare - namespace: traefik -spec: - acme: - server: https://acme-v02.api.letsencrypt.org/directory - email: email@example.com - privateKeySecretRef: - name: cloudflare-key - solvers: - - dns01: - cloudflare: - apiTokenSecretRef: - name: cloudflare - key: api-token -``` - -**Step 2**: Create `Certificate` in traefik namespace - -```yaml -apiVersion: cert-manager.io/v1 -kind: Certificate -metadata: - name: wildcard-example-com - namespace: traefik -spec: - secretName: wildcard-example-com-tls - dnsNames: - - "example.com" - - "*.example.com" - issuerRef: - name: cloudflare - kind: Issuer -``` - -**Step 3**: Check that it's ready - -```bash -kubectl get certificate -n traefik -``` - -If needed, logs of cert-manager pod can give you more information - -**Step 4**: Use it on the TLS Store in **values.yaml** file for this Helm Chart - -```yaml -tlsStore: - default: - defaultCertificate: - secretName: wildcard-example-com-tls -``` - -**Step 5**: Enjoy. All your `IngressRoute` use this certificate by default now. - -They should use websecure entrypoint like this: - -```yaml -apiVersion: traefik.io/v1alpha1 -kind: IngressRoute -metadata: - name: example-com-tls -spec: - entryPoints: - - websecure - routes: - - match: Host(`test.example.com`) - kind: Rule - services: - - name: XXXX - port: 80 -``` - -# Add custom (internal) services - -In some cases you might want to have more than one Traefik service within your cluster, -e.g. a default (external) one and a service that is only exposed internally to pods within your cluster. - -The `service.additionalServices` allows you to add an arbitrary amount of services, -provided as a name to service details mapping; for example you can use the following values: - -```yaml -service: - additionalServices: - internal: - type: ClusterIP - labels: - traefik-service-label: internal -``` - -Ports can then be exposed on this service by using the port name to boolean mapping `expose` on the respective port; -e.g. to expose the `traefik` API port on your internal service so pods within your cluster can use it, you can do: - -```yaml -ports: - traefik: - expose: - # Sensitive data should not be exposed on the internet - # => Keep this disabled ! - default: false - internal: true -``` - -This will then provide an additional Service manifest, looking like this: - -```yaml ---- -# Source: traefik/templates/service.yaml -apiVersion: v1 -kind: Service -metadata: - name: traefik-internal - namespace: traefik -[...] -spec: - type: ClusterIP - selector: - app.kubernetes.io/name: traefik - app.kubernetes.io/instance: traefik-traefik - ports: - - port: 8080 - name: "traefik" - targetPort: traefik - protocol: TCP -``` - -# Use this Chart as a dependency of your own chart - - -First, let's create a default Helm Chart, with Traefik as a dependency. -```bash -helm create foo -cd foo -echo " -dependencies: - - name: traefik - version: "24.0.0" - repository: "https://traefik.github.io/charts" -" >> Chart.yaml -``` - -Second, let's tune some values like enabling HPA: - -```bash -cat <<-EOF >> values.yaml -traefik: - autoscaling: - enabled: true - maxReplicas: 3 -EOF -``` - -Third, one can see if it works as expected: -```bash -helm dependency update -helm dependency build -helm template . | grep -A 14 -B 3 Horizontal -``` - -It should produce this output: - -```yaml ---- -# Source: foo/charts/traefik/templates/hpa.yaml -apiVersion: autoscaling/v2 -kind: HorizontalPodAutoscaler -metadata: - name: release-name-traefik - namespace: flux-system - labels: - app.kubernetes.io/name: traefik - app.kubernetes.io/instance: release-name-flux-system - helm.sh/chart: traefik-24.0.0 - app.kubernetes.io/managed-by: Helm -spec: - scaleTargetRef: - apiVersion: apps/v1 - kind: Deployment - name: release-name-traefik - maxReplicas: 3 -``` - -# Configure TLS - -The [TLS options](https://doc.traefik.io/traefik/https/tls/#tls-options) allow one to configure some parameters of the TLS connection. - -```yaml -tlsOptions: - default: - labels: {} - sniStrict: true - custom-options: - labels: {} - curvePreferences: - - CurveP521 - - CurveP384 -``` - -# Use latest build of Traefik v3 from master - -An experimental build of Traefik Proxy is available on a specific repository. - -It can be used with those _values_: - -```yaml -image: - repository: traefik/traefik - tag: experimental-v3.0 -``` - -# Use Prometheus Operator - -An optional support of this operator is included in this Chart. See documentation of this operator for more details. - -It can be used with those _values_: - -```yaml -metrics: - prometheus: - service: - enabled: true - disableAPICheck: false - serviceMonitor: - enabled: true - metricRelabelings: - - sourceLabels: [__name__] - separator: ; - regex: ^fluentd_output_status_buffer_(oldest|newest)_.+ - replacement: $1 - action: drop - relabelings: - - sourceLabels: [__meta_kubernetes_pod_node_name] - separator: ; - regex: ^(.*)$ - targetLabel: nodename - replacement: $1 - action: replace - jobLabel: traefik - interval: 30s - honorLabels: true - prometheusRule: - enabled: true - rules: - - alert: TraefikDown - expr: up{job="traefik"} == 0 - for: 5m - labels: - context: traefik - severity: warning - annotations: - summary: "Traefik Down" - description: "{{ $labels.pod }} on {{ $labels.nodename }} is down" -``` - -# Use kubernetes Gateway API - -One can use the new stable kubernetes gateway API provider setting the following _values_: - -```yaml -providers: - kubernetesGateway: - enabled: true -``` - -
- -With those values, a whoami service can be exposed with a HTTPRoute - -```yaml ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: whoami -spec: - replicas: 2 - selector: - matchLabels: - app: whoami - template: - metadata: - labels: - app: whoami - spec: - containers: - - name: whoami - image: traefik/whoami - ---- -apiVersion: v1 -kind: Service -metadata: - name: whoami -spec: - selector: - app: whoami - ports: - - protocol: TCP - port: 80 - ---- -apiVersion: gateway.networking.k8s.io/v1 -kind: HTTPRoute -metadata: - name: whoami -spec: - parentRefs: - - name: traefik-gateway - hostnames: - - whoami.docker.localhost - rules: - - matches: - - path: - type: Exact - value: / - - backendRefs: - - name: whoami - port: 80 - weight: 1 -``` - -Once it's applied, whoami should be accessible on http://whoami.docker.localhost/ - -
- -:information_source: In this example, `Deployment` and `HTTPRoute` should be deployed in the same namespace as the Traefik Gateway: Chart namespace. - -# Use Kubernetes Gateway API with cert-manager - -One can use the new stable kubernetes gateway API provider with automatic TLS certificates delivery (with cert-manager) setting the following _values_: - -```yaml -providers: - kubernetesGateway: - enabled: true -gateway: - enabled: true - annotations: - cert-manager.io/issuer: selfsigned-issuer - listeners: - websecure: - hostname: whoami.docker.localhost - port: 8443 - protocol: HTTPS - certificateRefs: - - name: whoami-tls -``` - -Install cert-manager: - -```bash -helm repo add jetstack https://charts.jetstack.io --force-update -helm upgrade --install \ -cert-manager jetstack/cert-manager \ ---namespace cert-manager \ ---create-namespace \ ---version v1.15.1 \ ---set crds.enabled=true \ ---set "extraArgs={--enable-gateway-api}" -``` - -
- -With those values, a whoami service can be exposed with HTTPRoute on both HTTP and HTTPS - -```yaml ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: whoami -spec: - replicas: 2 - selector: - matchLabels: - app: whoami - template: - metadata: - labels: - app: whoami - spec: - containers: - - name: whoami - image: traefik/whoami - ---- -apiVersion: v1 -kind: Service -metadata: - name: whoami -spec: - selector: - app: whoami - ports: - - protocol: TCP - port: 80 - ---- -apiVersion: gateway.networking.k8s.io/v1 -kind: HTTPRoute -metadata: - name: whoami -spec: - parentRefs: - - name: traefik-gateway - hostnames: - - whoami.docker.localhost - rules: - - matches: - - path: - type: Exact - value: / - - backendRefs: - - name: whoami - port: 80 - weight: 1 - ---- -apiVersion: cert-manager.io/v1 -kind: Issuer -metadata: - name: selfsigned-issuer -spec: - selfSigned: {} -``` - -Once it's applied, whoami should be accessible on https://whoami.docker.localhost/ - -
diff --git a/charts/traefik/traefik/Guidelines.md b/charts/traefik/traefik/Guidelines.md deleted file mode 100644 index 3b72a40..0000000 --- a/charts/traefik/traefik/Guidelines.md +++ /dev/null @@ -1,34 +0,0 @@ -# Traefik Helm Chart Guidelines - -This document outlines the guidelines for developing, managing and extending the Traefik helm chart. - -This Helm Chart is documented using field description from comments with [helm-docs](https://github.com/norwoodj/helm-docs). - -It comes with a JSON schema generated from values with [helm schema](https://github.com/losisin/helm-values-schema-json) plugin. - -## Feature Example - -```yaml -logs: - general: - # -- Set [logs format](https://doc.traefik.io/traefik/observability/logs/#format) - format: # @schema enum:["common", "json", null]; type:[string, null]; default: "common" -``` - -Documention is on the first comment, starting with `# --` -Specific instructions for schema, when needed, are done with the inline comment starting with `# @schema`. - -## Whitespace - -Extra whitespace is to be avoided in templating. Conditionals should chomp whitespace: - -```yaml -{{- if .Values }} -{{- end }} -``` - -There should be an empty commented line between each primary key in the values.yaml file to separate features from each other. - -## Values YAML Design - -The values.yaml file is designed to be user-friendly. It does not have to resemble the templated configuration if it is not conducive. Similarly, value names to not have to correspond to fields in the template if it is not conducive. diff --git a/charts/traefik/traefik/LICENSE b/charts/traefik/traefik/LICENSE deleted file mode 100644 index 907ff83..0000000 --- a/charts/traefik/traefik/LICENSE +++ /dev/null @@ -1,202 +0,0 @@ - Apache License - Version 2.0, January 2004 - http://www.apache.org/licenses/ - - TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION - - 1. Definitions. - - "License" shall mean the terms and conditions for use, reproduction, - and distribution as defined by Sections 1 through 9 of this document. - - "Licensor" shall mean the copyright owner or entity authorized by - the copyright owner that is granting the License. - - "Legal Entity" shall mean the union of the acting entity and all - other entities that control, are controlled by, or are under common - control with that entity. For the purposes of this definition, - "control" means (i) the power, direct or indirect, to cause the - direction or management of such entity, whether by contract or - otherwise, or (ii) ownership of fifty percent (50%) or more of the - outstanding shares, or (iii) beneficial ownership of such entity. - - "You" (or "Your") shall mean an individual or Legal Entity - exercising permissions granted by this License. - - "Source" form shall mean the preferred form for making modifications, - including but not limited to software source code, documentation - source, and configuration files. - - "Object" form shall mean any form resulting from mechanical - transformation or translation of a Source form, including but - not limited to compiled object code, generated documentation, - and conversions to other media types. - - "Work" shall mean the work of authorship, whether in Source or - Object form, made available under the License, as indicated by a - copyright notice that is included in or attached to the work - (an example is provided in the Appendix below). - - "Derivative Works" shall mean any work, whether in Source or Object - form, that is based on (or derived from) the Work and for which the - editorial revisions, annotations, elaborations, or other modifications - represent, as a whole, an original work of authorship. For the purposes - of this License, Derivative Works shall not include works that remain - separable from, or merely link (or bind by name) to the interfaces of, - the Work and Derivative Works thereof. - - "Contribution" shall mean any work of authorship, including - the original version of the Work and any modifications or additions - to that Work or Derivative Works thereof, that is intentionally - submitted to Licensor for inclusion in the Work by the copyright owner - or by an individual or Legal Entity authorized to submit on behalf of - the copyright owner. For the purposes of this definition, "submitted" - means any form of electronic, verbal, or written communication sent - to the Licensor or its representatives, including but not limited to - communication on electronic mailing lists, source code control systems, - and issue tracking systems that are managed by, or on behalf of, the - Licensor for the purpose of discussing and improving the Work, but - excluding communication that is conspicuously marked or otherwise - designated in writing by the copyright owner as "Not a Contribution." - - "Contributor" shall mean Licensor and any individual or Legal Entity - on behalf of whom a Contribution has been received by Licensor and - subsequently incorporated within the Work. - - 2. Grant of Copyright License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - copyright license to reproduce, prepare Derivative Works of, - publicly display, publicly perform, sublicense, and distribute the - Work and such Derivative Works in Source or Object form. - - 3. Grant of Patent License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - (except as stated in this section) patent license to make, have made, - use, offer to sell, sell, import, and otherwise transfer the Work, - where such license applies only to those patent claims licensable - by such Contributor that are necessarily infringed by their - Contribution(s) alone or by combination of their Contribution(s) - with the Work to which such Contribution(s) was submitted. If You - institute patent litigation against any entity (including a - cross-claim or counterclaim in a lawsuit) alleging that the Work - or a Contribution incorporated within the Work constitutes direct - or contributory patent infringement, then any patent licenses - granted to You under this License for that Work shall terminate - as of the date such litigation is filed. - - 4. Redistribution. You may reproduce and distribute copies of the - Work or Derivative Works thereof in any medium, with or without - modifications, and in Source or Object form, provided that You - meet the following conditions: - - (a) You must give any other recipients of the Work or - Derivative Works a copy of this License; and - - (b) You must cause any modified files to carry prominent notices - stating that You changed the files; and - - (c) You must retain, in the Source form of any Derivative Works - that You distribute, all copyright, patent, trademark, and - attribution notices from the Source form of the Work, - excluding those notices that do not pertain to any part of - the Derivative Works; and - - (d) If the Work includes a "NOTICE" text file as part of its - distribution, then any Derivative Works that You distribute must - include a readable copy of the attribution notices contained - within such NOTICE file, excluding those notices that do not - pertain to any part of the Derivative Works, in at least one - of the following places: within a NOTICE text file distributed - as part of the Derivative Works; within the Source form or - documentation, if provided along with the Derivative Works; or, - within a display generated by the Derivative Works, if and - wherever such third-party notices normally appear. The contents - of the NOTICE file are for informational purposes only and - do not modify the License. You may add Your own attribution - notices within Derivative Works that You distribute, alongside - or as an addendum to the NOTICE text from the Work, provided - that such additional attribution notices cannot be construed - as modifying the License. - - You may add Your own copyright statement to Your modifications and - may provide additional or different license terms and conditions - for use, reproduction, or distribution of Your modifications, or - for any such Derivative Works as a whole, provided Your use, - reproduction, and distribution of the Work otherwise complies with - the conditions stated in this License. - - 5. Submission of Contributions. Unless You explicitly state otherwise, - any Contribution intentionally submitted for inclusion in the Work - by You to the Licensor shall be under the terms and conditions of - this License, 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 - with Licensor regarding such Contributions. - - 6. Trademarks. This License does not grant permission to use the trade - names, trademarks, service marks, or product names of the Licensor, - except as required for reasonable and customary use in describing the - origin of the Work and reproducing the content of the NOTICE file. - - 7. Disclaimer of Warranty. Unless required by applicable law or - agreed to in writing, Licensor provides the Work (and each - Contributor provides its Contributions) on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or - implied, including, without limitation, any warranties or conditions - of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A - PARTICULAR PURPOSE. You are solely responsible for determining the - appropriateness of using or redistributing the Work and assume any - risks associated with Your exercise of permissions under this License. - - 8. Limitation of Liability. In no event and under no legal theory, - whether in tort (including negligence), contract, or otherwise, - unless required by applicable law (such as deliberate and grossly - negligent acts) or agreed to in writing, shall any Contributor be - liable to You for damages, including any direct, indirect, special, - incidental, or consequential damages of any character arising as a - result of this License or out of the use or inability to use the - Work (including but not limited to damages for loss of goodwill, - work stoppage, computer failure or malfunction, or any and all - other commercial damages or losses), even if such Contributor - has been advised of the possibility of such damages. - - 9. Accepting Warranty or Additional Liability. While redistributing - the Work or Derivative Works thereof, You may choose to offer, - and charge a fee for, acceptance of support, warranty, indemnity, - or other liability obligations and/or rights consistent with this - License. However, in accepting such obligations, You may act only - on Your own behalf and on Your sole responsibility, not on behalf - of any other Contributor, and only if You agree to indemnify, - defend, and hold each Contributor harmless for any liability - incurred by, or claims asserted against, such Contributor by reason - of your accepting any such warranty or additional liability. - - END OF TERMS AND CONDITIONS - - APPENDIX: How to apply the Apache License to your work. - - To apply the Apache License to your work, attach the following - boilerplate notice, with the fields enclosed by brackets "[]" - replaced with your own identifying information. (Don't include - the brackets!) The text should be enclosed in the appropriate - comment syntax for the file format. We also recommend that a - file or class name and description of purpose be included on the - same "printed page" as the copyright notice for easier - identification within third-party archives. - - Copyright 2020 Containous - Copyright 2020 Traefik Labs - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. diff --git a/charts/traefik/traefik/README.md b/charts/traefik/traefik/README.md deleted file mode 100644 index cd963c1..0000000 --- a/charts/traefik/traefik/README.md +++ /dev/null @@ -1,158 +0,0 @@ -# Traefik - -[Traefik](https://traefik.io/) is a modern HTTP reverse proxy and load balancer made to deploy -microservices with ease. - -## Introduction - -Starting with v28.x, this chart now bootstraps Traefik Proxy version 3 as a Kubernetes ingress controller, -using Custom Resources `IngressRoute`: . - -It's possible to use this chart with Traefik Proxy v2 using v27.x -This chart support policy is aligned with [upstream support policy](https://doc.traefik.io/traefik/deprecation/releases/) of Traefik Proxy. - -See [Migration guide from v2 to v3](https://doc.traefik.io/traefik/v3.0/migration/v2-to-v3/) and upgrading section of this chart on CRDs. - -### Philosophy - -The Traefik HelmChart is focused on Traefik deployment configuration. - -To keep this HelmChart as generic as possible we tend -to avoid integrating any third party solutions nor any specific use cases. - -Accordingly, the encouraged approach to fulfill your needs: - -1. Override the default Traefik configuration values ([yaml file or cli](https://helm.sh/docs/chart_template_guide/values_files/)) -2. Append your own configurations (`kubectl apply -f myconf.yaml`) - -[Examples](https://github.com/traefik/traefik-helm-chart/blob/master/EXAMPLES.md) of common usage are provided. - -If needed, one may use [extraObjects](./traefik/tests/values/extra.yaml) or extend this HelmChart [as a Subchart](https://helm.sh/docs/chart_template_guide/subcharts_and_globals/). - -## Installing - -### Prerequisites - -1. [x] Helm **v3 > 3.9.0** [installed](https://helm.sh/docs/using_helm/#installing-helm): `helm version` -2. [x] Traefik's chart repository: `helm repo add traefik https://traefik.github.io/charts` - -### Kubernetes Version Support - -Due to changes in CRD version support, the following versions of the chart are usable and supported on the following Kubernetes versions: - -| | Kubernetes v1.15 and below | Kubernetes v1.16-v1.21 | Kubernetes v1.22 and above | -|-------------------------|-----------------------------|------------------------|----------------------------| -| Chart v9.20.2 and below | [x] | [x] | | -| Chart v10.0.0 and above | | [x] | [x] | -| Chart v22.0.0 and above | | | [x] | - -### CRDs Support of Traefik Proxy - -Due to changes in API Group of Traefik CRDs from `containo.us` to `traefik.io`, this Chart install CRDs needed by default Traefik Proxy version, following this table: - -| | `containo.us` | `traefik.io` | -|-------------------------|-----------------------------|------------------------| -| Chart v22.0.0 and below | [x] | | -| Chart v23.0.0 and above | [x] | [x] | -| Chart v28.0.0 and above | | [x] | - -### Deploying Traefik - -```bash -helm install traefik traefik/traefik -``` - -or: - -```bash -helm install traefik oci://ghcr.io/traefik/helm/traefik -``` - -You can customize the install with a `values` file. There are some [EXAMPLES](./EXAMPLES.md) provided. -Complete documentation on all available parameters is in the [default file](./traefik/values.yaml). - -```bash -helm install -f myvalues.yaml traefik traefik/traefik -``` - -🛂 **Warning**: Helm v2 support was removed in the chart version 10.0.0. - -## Upgrading - -One can check what has changed in the [Changelog](./traefik/Changelog.md). - -:information_source: With Helm v3, CRDs created by this chart can not be updated, cf. the [Helm Documentation on CRDs](https://helm.sh/docs/chart_best_practices/custom_resource_definitions). - -:warning: Please read carefully release notes of this chart before upgrading CRDs. - -```bash -# Update repository -helm repo update -# See current Chart & Traefik version -helm search repo traefik/traefik -# Update CRDs (Traefik Proxy v3 CRDs) -kubectl apply --server-side --force-conflicts -k https://github.com/traefik/traefik-helm-chart/traefik/crds/ -# Upgrade Traefik -helm upgrade traefik traefik/traefik -``` - -New major version indicates that there is an incompatible breaking change. - -#### Upgrade up to 27.X - -When upgrading on Traefik Proxy v2 version, one need to stay at Traefik Helm Chart v27.x. The command to upgrade to the latest Traefik Proxy v2 CRD is: - -```bash -kubectl apply --server-side --force-conflicts -k https://github.com/traefik/traefik-helm-chart/traefik/crds/?ref=v27 -``` - -### Upgrading after 18.X+ - -It's detailed in [release notes](https://github.com/traefik/traefik-helm-chart/releases). - -### Upgrading from 17.x to 18.x - -Since v18.x, this chart by default merges TCP and UDP ports into a single (LoadBalancer) `Service`. -Load balancers with mixed protocols are available since v1.20 and in -[beta as of Kubernetes v1.24](https://kubernetes.io/docs/concepts/services-networking/service/#load-balancers-with-mixed-protocol-types). -Availability may depend on your Kubernetes provider. - -To retain the old default behavior, set `service.single` to `false` in your values. - -When using TCP and UDP with a single service, you may encounter -[this issue](https://github.com/kubernetes/kubernetes/issues/47249#issuecomment-587960741) -from Kubernetes. - -On HTTP/3, if you want to avoid this issue, you can set -`ports.websecure.http3.advertisedPort` to an other value than `443` - -If you were previously using HTTP/3, you should update your values as follows: - - Replace the old value (`true`) of `ports.websecure.http3` with a key `enabled: true` - - Remove `experimental.http3.enabled=true` entry - -### Upgrading from 16.x to 17.x - -Since v17.x, this chart provides unified labels following -[Kubernetes recommendation](https://kubernetes.io/docs/concepts/overview/working-with-objects/common-labels/). - -This version needs to change an immutable field, which is not supported by -Kubernetes and Helm, see [this issue](https://github.com/helm/helm/issues/7350) -for more details. -So you will have to delete your `Service`, `Deployment` or `DaemonSet` in -order to be able to upgrade. - -You may also upgrade by deploying another Traefik to a different namespace and -removing after your first Traefik. - -Alternatively, since version 20.3.0 of this chart, you may set `instanceLabelOverride` to the previous value of that label. -This will override the new `Release.Name-Release.Namespace` pattern to avoid any (longer) downtime. - -## Contributing - -If you want to contribute to this chart, please read the [Contributing Guide](./CONTRIBUTING.md). - -Thanks to all the people who have already contributed! - - - - diff --git a/charts/traefik/traefik/VALUES.md b/charts/traefik/traefik/VALUES.md deleted file mode 100644 index aa3d014..0000000 --- a/charts/traefik/traefik/VALUES.md +++ /dev/null @@ -1,321 +0,0 @@ -# traefik - -![Version: 33.0.0](https://img.shields.io/badge/Version-33.0.0-informational?style=flat-square) ![Type: application](https://img.shields.io/badge/Type-application-informational?style=flat-square) ![AppVersion: v3.2.0](https://img.shields.io/badge/AppVersion-v3.2.0-informational?style=flat-square) - -A Traefik based Kubernetes ingress controller - -**Homepage:** - -## Maintainers - -| Name | Email | Url | -| ---- | ------ | --- | -| mloiseleur | | | -| charlie-haley | | | -| darkweaver87 | | | -| jnoordsij | | | - -## Source Code - -* -* - -## Requirements - -Kubernetes: `>=1.22.0-0` - -## Values - -| Key | Type | Default | Description | -|-----|------|---------|-------------| -| additionalArguments | list | `[]` | Additional arguments to be passed at Traefik's binary See [CLI Reference](https://docs.traefik.io/reference/static-configuration/cli/) Use curly braces to pass values: `helm install --set="additionalArguments={--providers.kubernetesingress.ingressclass=traefik-internal,--log.level=DEBUG}"` | -| additionalVolumeMounts | list | `[]` | Additional volumeMounts to add to the Traefik container | -| affinity | object | `{}` | on nodes where no other traefik pods are scheduled. It should be used when hostNetwork: true to prevent port conflicts | -| autoscaling.enabled | bool | `false` | Create HorizontalPodAutoscaler object. See EXAMPLES.md for more details. | -| certificatesResolvers | object | `{}` | Certificates resolvers configuration. Ref: https://doc.traefik.io/traefik/https/acme/#certificate-resolvers See EXAMPLES.md for more details. | -| commonLabels | object | `{}` | Add additional label to all resources | -| core.defaultRuleSyntax | string | `""` | Can be used to use globally v2 router syntax See https://doc.traefik.io/traefik/v3.0/migration/v2-to-v3/#new-v3-syntax-notable-changes | -| deployment.additionalContainers | list | `[]` | Additional containers (e.g. for metric offloading sidecars) | -| deployment.additionalVolumes | list | `[]` | Additional volumes available for use with initContainers and additionalContainers | -| deployment.annotations | object | `{}` | Additional deployment annotations (e.g. for jaeger-operator sidecar injection) | -| deployment.dnsConfig | object | `{}` | Custom pod [DNS config](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.30/#poddnsconfig-v1-core) | -| deployment.dnsPolicy | string | `""` | Custom pod DNS policy. Apply if `hostNetwork: true` | -| deployment.enabled | bool | `true` | Enable deployment | -| deployment.healthchecksHost | string | `""` | | -| deployment.healthchecksPort | string | `nil` | | -| deployment.healthchecksScheme | string | `nil` | | -| deployment.hostAliases | list | `[]` | Custom [host aliases](https://kubernetes.io/docs/tasks/network/customize-hosts-file-for-pods/) | -| deployment.imagePullSecrets | list | `[]` | Pull secret for fetching traefik container image | -| deployment.initContainers | list | `[]` | Additional initContainers (e.g. for setting file permission as shown below) | -| deployment.kind | string | `"Deployment"` | Deployment or DaemonSet | -| deployment.labels | object | `{}` | Additional deployment labels (e.g. for filtering deployment by custom labels) | -| deployment.lifecycle | object | `{}` | Pod lifecycle actions | -| deployment.livenessPath | string | `""` | Override the liveness path. Default: /ping | -| deployment.minReadySeconds | int | `0` | The minimum number of seconds Traefik needs to be up and running before the DaemonSet/Deployment controller considers it available | -| deployment.podAnnotations | object | `{}` | Additional pod annotations (e.g. for mesh injection or prometheus scraping) It supports templating. One can set it with values like traefik/name: '{{ template "traefik.name" . }}' | -| deployment.podLabels | object | `{}` | Additional Pod labels (e.g. for filtering Pod by custom labels) | -| deployment.readinessPath | string | `""` | | -| deployment.replicas | int | `1` | Number of pods of the deployment (only applies when kind == Deployment) | -| deployment.revisionHistoryLimit | string | `nil` | Number of old history to retain to allow rollback (If not set, default Kubernetes value is set to 10) | -| deployment.runtimeClassName | string | `""` | Set a runtimeClassName on pod | -| deployment.shareProcessNamespace | bool | `false` | Use process namespace sharing | -| deployment.terminationGracePeriodSeconds | int | `60` | Amount of time (in seconds) before Kubernetes will send the SIGKILL signal if Traefik does not shut down | -| env | list | See _values.yaml_ | Additional Environment variables to be passed to Traefik's binary | -| envFrom | list | `[]` | Environment variables to be passed to Traefik's binary from configMaps or secrets | -| experimental.kubernetesGateway.enabled | bool | `false` | Enable traefik experimental GatewayClass CRD | -| experimental.plugins | object | `{}` | Enable traefik experimental plugins | -| extraObjects | list | `[]` | Extra objects to deploy (value evaluated as a template) In some cases, it can avoid the need for additional, extended or adhoc deployments. See #595 for more details and traefik/tests/values/extra.yaml for example. | -| gateway.annotations | object | `{}` | Additional gateway annotations (e.g. for cert-manager.io/issuer) | -| gateway.enabled | bool | `true` | When providers.kubernetesGateway.enabled, deploy a default gateway | -| gateway.infrastructure | object | `{}` | [Infrastructure](https://kubernetes.io/blog/2023/11/28/gateway-api-ga/#gateway-infrastructure-labels) | -| gateway.listeners | object | `{"web":{"hostname":"","namespacePolicy":null,"port":8000,"protocol":"HTTP"}}` | Define listeners | -| gateway.listeners.web.hostname | string | `""` | Optional hostname. See [Hostname](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.Hostname) | -| gateway.listeners.web.namespacePolicy | string | `nil` | Routes are restricted to namespace of the gateway [by default](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.FromNamespaces | -| gateway.listeners.web.port | int | `8000` | Port is the network port. Multiple listeners may use the same port, subject to the Listener compatibility rules. The port must match a port declared in ports section. | -| gateway.name | string | `""` | Set a custom name to gateway | -| gateway.namespace | string | `""` | By default, Gateway is created in the same `Namespace` than Traefik. | -| gatewayClass.enabled | bool | `true` | When providers.kubernetesGateway.enabled and gateway.enabled, deploy a default gatewayClass | -| gatewayClass.labels | object | `{}` | Additional gatewayClass labels (e.g. for filtering gateway objects by custom labels) | -| gatewayClass.name | string | `""` | Set a custom name to GatewayClass | -| globalArguments | list | `["--global.checknewversion","--global.sendanonymoususage"]` | Global command arguments to be passed to all traefik's pods | -| hostNetwork | bool | `false` | If hostNetwork is true, runs traefik in the host network namespace To prevent unschedulabel pods due to port collisions, if hostNetwork=true and replicas>1, a pod anti-affinity is recommended and will be set if the affinity is left as default. | -| hub.apimanagement.admission.listenAddr | string | `""` | WebHook admission server listen address. Default: "0.0.0.0:9943". | -| hub.apimanagement.admission.secretName | string | `""` | Certificate of the WebHook admission server. Default: "hub-agent-cert". | -| hub.apimanagement.enabled | bool | `false` | Set to true in order to enable API Management. Requires a valid license token. | -| hub.redis.cluster | string | `nil` | Enable Redis Cluster. Default: true. | -| hub.redis.database | string | `nil` | Database used to store information. Default: "0". | -| hub.redis.endpoints | string | `""` | Endpoints of the Redis instances to connect to. Default: "". | -| hub.redis.password | string | `""` | The password to use when connecting to Redis endpoints. Default: "". | -| hub.redis.sentinel.masterset | string | `""` | Name of the set of main nodes to use for main selection. Required when using Sentinel. Default: "". | -| hub.redis.sentinel.password | string | `""` | Password to use for sentinel authentication (can be different from endpoint password). Default: "". | -| hub.redis.sentinel.username | string | `""` | Username to use for sentinel authentication (can be different from endpoint username). Default: "". | -| hub.redis.timeout | string | `""` | Timeout applied on connection with redis. Default: "0s". | -| hub.redis.tls.ca | string | `""` | Path to the certificate authority used for the secured connection. | -| hub.redis.tls.cert | string | `""` | Path to the public certificate used for the secure connection. | -| hub.redis.tls.insecureSkipVerify | bool | `false` | When insecureSkipVerify is set to true, the TLS connection accepts any certificate presented by the server. Default: false. | -| hub.redis.tls.key | string | `""` | Path to the private key used for the secure connection. | -| hub.redis.username | string | `""` | The username to use when connecting to Redis endpoints. Default: "". | -| hub.sendlogs | string | `nil` | | -| hub.token | string | `""` | Name of `Secret` with key 'token' set to a valid license token. It enables API Gateway. | -| image.pullPolicy | string | `"IfNotPresent"` | Traefik image pull policy | -| image.registry | string | `"docker.io"` | Traefik image host registry | -| image.repository | string | `"traefik"` | Traefik image repository | -| image.tag | string | `nil` | defaults to appVersion | -| ingressClass | object | `{"enabled":true,"isDefaultClass":true,"name":""}` | Create a default IngressClass for Traefik | -| ingressRoute.dashboard.annotations | object | `{}` | Additional ingressRoute annotations (e.g. for kubernetes.io/ingress.class) | -| ingressRoute.dashboard.enabled | bool | `false` | Create an IngressRoute for the dashboard | -| ingressRoute.dashboard.entryPoints | list | `["traefik"]` | Specify the allowed entrypoints to use for the dashboard ingress route, (e.g. traefik, web, websecure). By default, it's using traefik entrypoint, which is not exposed. /!\ Do not expose your dashboard without any protection over the internet /!\ | -| ingressRoute.dashboard.labels | object | `{}` | Additional ingressRoute labels (e.g. for filtering IngressRoute by custom labels) | -| ingressRoute.dashboard.matchRule | string | `"PathPrefix(`/dashboard`) || PathPrefix(`/api`)"` | The router match rule used for the dashboard ingressRoute | -| ingressRoute.dashboard.middlewares | list | `[]` | Additional ingressRoute middlewares (e.g. for authentication) | -| ingressRoute.dashboard.services | list | `[{"kind":"TraefikService","name":"api@internal"}]` | The internal service used for the dashboard ingressRoute | -| ingressRoute.dashboard.tls | object | `{}` | TLS options (e.g. secret containing certificate) | -| ingressRoute.healthcheck.annotations | object | `{}` | Additional ingressRoute annotations (e.g. for kubernetes.io/ingress.class) | -| ingressRoute.healthcheck.enabled | bool | `false` | Create an IngressRoute for the healthcheck probe | -| ingressRoute.healthcheck.entryPoints | list | `["traefik"]` | Specify the allowed entrypoints to use for the healthcheck ingress route, (e.g. traefik, web, websecure). By default, it's using traefik entrypoint, which is not exposed. | -| ingressRoute.healthcheck.labels | object | `{}` | Additional ingressRoute labels (e.g. for filtering IngressRoute by custom labels) | -| ingressRoute.healthcheck.matchRule | string | `"PathPrefix(`/ping`)"` | The router match rule used for the healthcheck ingressRoute | -| ingressRoute.healthcheck.middlewares | list | `[]` | Additional ingressRoute middlewares (e.g. for authentication) | -| ingressRoute.healthcheck.services | list | `[{"kind":"TraefikService","name":"ping@internal"}]` | The internal service used for the healthcheck ingressRoute | -| ingressRoute.healthcheck.tls | object | `{}` | TLS options (e.g. secret containing certificate) | -| instanceLabelOverride | string | `""` | | -| livenessProbe.failureThreshold | int | `3` | The number of consecutive failures allowed before considering the probe as failed. | -| livenessProbe.initialDelaySeconds | int | `2` | The number of seconds to wait before starting the first probe. | -| livenessProbe.periodSeconds | int | `10` | The number of seconds to wait between consecutive probes. | -| livenessProbe.successThreshold | int | `1` | The minimum consecutive successes required to consider the probe successful. | -| livenessProbe.timeoutSeconds | int | `2` | The number of seconds to wait for a probe response before considering it as failed. | -| logs.access.addInternals | bool | `false` | Enables accessLogs for internal resources. Default: false. | -| logs.access.bufferingSize | string | `nil` | Set [bufferingSize](https://doc.traefik.io/traefik/observability/access-logs/#bufferingsize) | -| logs.access.enabled | bool | `false` | To enable access logs | -| logs.access.fields.general.defaultmode | string | `"keep"` | Set default mode for fields.names | -| logs.access.fields.general.names | object | `{}` | Names of the fields to limit. | -| logs.access.fields.headers | object | `{"defaultmode":"drop","names":{}}` | [Limit logged fields or headers](https://doc.traefik.io/traefik/observability/access-logs/#limiting-the-fieldsincluding-headers) | -| logs.access.fields.headers.defaultmode | string | `"drop"` | Set default mode for fields.headers | -| logs.access.filters | object | `{"minduration":"","retryattempts":false,"statuscodes":""}` | Set [filtering](https://docs.traefik.io/observability/access-logs/#filtering) | -| logs.access.filters.minduration | string | `""` | Set minDuration, to keep access logs when requests take longer than the specified duration | -| logs.access.filters.retryattempts | bool | `false` | Set retryAttempts, to keep the access logs when at least one retry has happened | -| logs.access.filters.statuscodes | string | `""` | Set statusCodes, to limit the access logs to requests with a status codes in the specified range | -| logs.access.format | string | `nil` | Set [access log format](https://doc.traefik.io/traefik/observability/access-logs/#format) | -| logs.general.filePath | string | `""` | To write the logs into a log file, use the filePath option. | -| logs.general.format | string | `nil` | Set [logs format](https://doc.traefik.io/traefik/observability/logs/#format) | -| logs.general.level | string | `"INFO"` | Alternative logging levels are TRACE, DEBUG, INFO, WARN, ERROR, FATAL, and PANIC. | -| logs.general.noColor | bool | `false` | When set to true and format is common, it disables the colorized output. | -| metrics.addInternals | bool | `false` | | -| metrics.otlp.addEntryPointsLabels | string | `nil` | Enable metrics on entry points. Default: true | -| metrics.otlp.addRoutersLabels | string | `nil` | Enable metrics on routers. Default: false | -| metrics.otlp.addServicesLabels | string | `nil` | Enable metrics on services. Default: true | -| metrics.otlp.enabled | bool | `false` | Set to true in order to enable the OpenTelemetry metrics | -| metrics.otlp.explicitBoundaries | list | `[]` | Explicit boundaries for Histogram data points. Default: [.005, .01, .025, .05, .1, .25, .5, 1, 2.5, 5, 10] | -| metrics.otlp.grpc.enabled | bool | `false` | Set to true in order to send metrics to the OpenTelemetry Collector using gRPC | -| metrics.otlp.grpc.endpoint | string | `""` | Format: ://:. Default: http://localhost:4318/v1/metrics | -| metrics.otlp.grpc.insecure | bool | `false` | Allows reporter to send metrics to the OpenTelemetry Collector without using a secured protocol. | -| metrics.otlp.grpc.tls.ca | string | `""` | The path to the certificate authority, it defaults to the system bundle. | -| metrics.otlp.grpc.tls.cert | string | `""` | The path to the public certificate. When using this option, setting the key option is required. | -| metrics.otlp.grpc.tls.insecureSkipVerify | bool | `false` | When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. | -| metrics.otlp.grpc.tls.key | string | `""` | The path to the private key. When using this option, setting the cert option is required. | -| metrics.otlp.http.enabled | bool | `false` | Set to true in order to send metrics to the OpenTelemetry Collector using HTTP. | -| metrics.otlp.http.endpoint | string | `""` | Format: ://:. Default: http://localhost:4318/v1/metrics | -| metrics.otlp.http.headers | object | `{}` | Additional headers sent with metrics by the reporter to the OpenTelemetry Collector. | -| metrics.otlp.http.tls.ca | string | `""` | The path to the certificate authority, it defaults to the system bundle. | -| metrics.otlp.http.tls.cert | string | `""` | The path to the public certificate. When using this option, setting the key option is required. | -| metrics.otlp.http.tls.insecureSkipVerify | string | `nil` | When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. | -| metrics.otlp.http.tls.key | string | `""` | The path to the private key. When using this option, setting the cert option is required. | -| metrics.otlp.pushInterval | string | `""` | Interval at which metrics are sent to the OpenTelemetry Collector. Default: 10s | -| metrics.prometheus.addEntryPointsLabels | string | `nil` | | -| metrics.prometheus.addRoutersLabels | string | `nil` | | -| metrics.prometheus.addServicesLabels | string | `nil` | | -| metrics.prometheus.buckets | string | `""` | | -| metrics.prometheus.disableAPICheck | string | `nil` | When set to true, it won't check if Prometheus Operator CRDs are deployed | -| metrics.prometheus.entryPoint | string | `"metrics"` | Entry point used to expose metrics. | -| metrics.prometheus.manualRouting | bool | `false` | | -| metrics.prometheus.prometheusRule.additionalLabels | object | `{}` | | -| metrics.prometheus.prometheusRule.enabled | bool | `false` | Enable optional CR for Prometheus Operator. See EXAMPLES.md for more details. | -| metrics.prometheus.prometheusRule.namespace | string | `""` | | -| metrics.prometheus.service.annotations | object | `{}` | | -| metrics.prometheus.service.enabled | bool | `false` | Create a dedicated metrics service to use with ServiceMonitor | -| metrics.prometheus.service.labels | object | `{}` | | -| metrics.prometheus.serviceMonitor.additionalLabels | object | `{}` | | -| metrics.prometheus.serviceMonitor.enableHttp2 | bool | `false` | | -| metrics.prometheus.serviceMonitor.enabled | bool | `false` | Enable optional CR for Prometheus Operator. See EXAMPLES.md for more details. | -| metrics.prometheus.serviceMonitor.followRedirects | bool | `false` | | -| metrics.prometheus.serviceMonitor.honorLabels | bool | `false` | | -| metrics.prometheus.serviceMonitor.honorTimestamps | bool | `false` | | -| metrics.prometheus.serviceMonitor.interval | string | `""` | | -| metrics.prometheus.serviceMonitor.jobLabel | string | `""` | | -| metrics.prometheus.serviceMonitor.metricRelabelings | list | `[]` | | -| metrics.prometheus.serviceMonitor.namespace | string | `""` | | -| metrics.prometheus.serviceMonitor.namespaceSelector | object | `{}` | | -| metrics.prometheus.serviceMonitor.relabelings | list | `[]` | | -| metrics.prometheus.serviceMonitor.scrapeTimeout | string | `""` | | -| namespaceOverride | string | `""` | This field override the default Release Namespace for Helm. It will not affect optional CRDs such as `ServiceMonitor` and `PrometheusRules` | -| nodeSelector | object | `{}` | nodeSelector is the simplest recommended form of node selection constraint. | -| persistence.accessMode | string | `"ReadWriteOnce"` | | -| persistence.annotations | object | `{}` | | -| persistence.enabled | bool | `false` | Enable persistence using Persistent Volume Claims ref: http://kubernetes.io/docs/user-guide/persistent-volumes/. It can be used to store TLS certificates along with `certificatesResolvers..acme.storage` option | -| persistence.existingClaim | string | `""` | | -| persistence.name | string | `"data"` | | -| persistence.path | string | `"/data"` | | -| persistence.size | string | `"128Mi"` | | -| persistence.storageClass | string | `""` | | -| persistence.subPath | string | `""` | Only mount a subpath of the Volume into the pod | -| persistence.volumeName | string | `""` | | -| podDisruptionBudget | object | `{"enabled":false,"maxUnavailable":null,"minAvailable":null}` | [Pod Disruption Budget](https://kubernetes.io/docs/reference/kubernetes-api/policy-resources/pod-disruption-budget-v1/) | -| podSecurityContext | object | See _values.yaml_ | [Pod Security Context](https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/pod-v1/#security-context) | -| podSecurityPolicy | object | `{"enabled":false}` | Enable to create a PodSecurityPolicy and assign it to the Service Account via RoleBinding or ClusterRoleBinding | -| ports.metrics.expose | object | `{"default":false}` | You may not want to expose the metrics port on production deployments. If you want to access it from outside your cluster, use `kubectl port-forward` or create a secure ingress | -| ports.metrics.exposedPort | int | `9100` | The exposed port for this service | -| ports.metrics.port | int | `9100` | When using hostNetwork, use another port to avoid conflict with node exporter: https://github.com/prometheus/prometheus/wiki/Default-port-allocations | -| ports.metrics.protocol | string | `"TCP"` | The port protocol (TCP/UDP) | -| ports.traefik.expose | object | `{"default":false}` | You SHOULD NOT expose the traefik port on production deployments. If you want to access it from outside your cluster, use `kubectl port-forward` or create a secure ingress | -| ports.traefik.exposedPort | int | `8080` | The exposed port for this service | -| ports.traefik.hostIP | string | `nil` | Use hostIP if set. If not set, Kubernetes will default to 0.0.0.0, which means it's listening on all your interfaces and all your IPs. You may want to set this value if you need traefik to listen on specific interface only. | -| ports.traefik.hostPort | string | `nil` | Use hostPort if set. | -| ports.traefik.port | int | `8080` | | -| ports.traefik.protocol | string | `"TCP"` | The port protocol (TCP/UDP) | -| ports.web.expose.default | bool | `true` | | -| ports.web.exposedPort | int | `80` | | -| ports.web.forwardedHeaders.insecure | bool | `false` | | -| ports.web.forwardedHeaders.trustedIPs | list | `[]` | Trust forwarded headers information (X-Forwarded-*). | -| ports.web.nodePort | string | `nil` | See [upstream documentation](https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport) | -| ports.web.port | int | `8000` | | -| ports.web.protocol | string | `"TCP"` | | -| ports.web.proxyProtocol.insecure | bool | `false` | | -| ports.web.proxyProtocol.trustedIPs | list | `[]` | Enable the Proxy Protocol header parsing for the entry point | -| ports.web.redirectTo | object | `{}` | | -| ports.web.targetPort | string | `nil` | | -| ports.web.transport | object | `{"keepAliveMaxRequests":null,"keepAliveMaxTime":null,"lifeCycle":{"graceTimeOut":null,"requestAcceptGraceTimeout":null},"respondingTimeouts":{"idleTimeout":null,"readTimeout":null,"writeTimeout":null}}` | Set transport settings for the entrypoint; see also https://doc.traefik.io/traefik/routing/entrypoints/#transport | -| ports.websecure.allowACMEByPass | bool | `false` | See [upstream documentation](https://doc.traefik.io/traefik/routing/entrypoints/#allowacmebypass) | -| ports.websecure.appProtocol | string | `nil` | See [upstream documentation](https://kubernetes.io/docs/concepts/services-networking/service/#application-protocol) | -| ports.websecure.containerPort | string | `nil` | | -| ports.websecure.expose.default | bool | `true` | | -| ports.websecure.exposedPort | int | `443` | | -| ports.websecure.forwardedHeaders.insecure | bool | `false` | | -| ports.websecure.forwardedHeaders.trustedIPs | list | `[]` | Trust forwarded headers information (X-Forwarded-*). | -| ports.websecure.hostPort | string | `nil` | | -| ports.websecure.http3.advertisedPort | string | `nil` | | -| ports.websecure.http3.enabled | bool | `false` | | -| ports.websecure.middlewares | list | `[]` | /!\ It introduces here a link between your static configuration and your dynamic configuration /!\ It follows the provider naming convention: https://doc.traefik.io/traefik/providers/overview/#provider-namespace - namespace-name1@kubernetescrd - namespace-name2@kubernetescrd | -| ports.websecure.nodePort | string | `nil` | See [upstream documentation](https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport) | -| ports.websecure.port | int | `8443` | | -| ports.websecure.protocol | string | `"TCP"` | | -| ports.websecure.proxyProtocol.insecure | bool | `false` | | -| ports.websecure.proxyProtocol.trustedIPs | list | `[]` | Enable the Proxy Protocol header parsing for the entry point | -| ports.websecure.targetPort | string | `nil` | | -| ports.websecure.tls | object | `{"certResolver":"","domains":[],"enabled":true,"options":""}` | See [upstream documentation](https://doc.traefik.io/traefik/routing/entrypoints/#tls) | -| ports.websecure.transport | object | `{"keepAliveMaxRequests":null,"keepAliveMaxTime":null,"lifeCycle":{"graceTimeOut":null,"requestAcceptGraceTimeout":null},"respondingTimeouts":{"idleTimeout":null,"readTimeout":null,"writeTimeout":null}}` | See [upstream documentation](https://doc.traefik.io/traefik/routing/entrypoints/#transport) | -| priorityClassName | string | `""` | [Pod Priority and Preemption](https://kubernetes.io/docs/concepts/scheduling-eviction/pod-priority-preemption/) | -| providers.file.content | string | `""` | File content (YAML format, go template supported) (see https://doc.traefik.io/traefik/providers/file/) | -| providers.file.enabled | bool | `false` | Create a file provider | -| providers.file.watch | bool | `true` | Allows Traefik to automatically watch for file changes | -| providers.kubernetesCRD.allowCrossNamespace | bool | `false` | Allows IngressRoute to reference resources in namespace other than theirs | -| providers.kubernetesCRD.allowEmptyServices | bool | `true` | Allows to return 503 when there is no endpoints available | -| providers.kubernetesCRD.allowExternalNameServices | bool | `false` | Allows to reference ExternalName services in IngressRoute | -| providers.kubernetesCRD.enabled | bool | `true` | Load Kubernetes IngressRoute provider | -| providers.kubernetesCRD.ingressClass | string | `""` | When the parameter is set, only resources containing an annotation with the same value are processed. Otherwise, resources missing the annotation, having an empty value, or the value traefik are processed. It will also set required annotation on Dashboard and Healthcheck IngressRoute when enabled. | -| providers.kubernetesCRD.namespaces | list | `[]` | Array of namespaces to watch. If left empty, Traefik watches all namespaces. | -| providers.kubernetesCRD.nativeLBByDefault | bool | `false` | Defines whether to use Native Kubernetes load-balancing mode by default. | -| providers.kubernetesGateway.enabled | bool | `false` | Enable Traefik Gateway provider for Gateway API | -| providers.kubernetesGateway.experimentalChannel | bool | `false` | Toggles support for the Experimental Channel resources (Gateway API release channels documentation). This option currently enables support for TCPRoute and TLSRoute. | -| providers.kubernetesGateway.labelselector | string | `""` | A label selector can be defined to filter on specific GatewayClass objects only. | -| providers.kubernetesGateway.namespaces | list | `[]` | Array of namespaces to watch. If left empty, Traefik watches all namespaces. | -| providers.kubernetesGateway.statusAddress.hostname | string | `""` | This Hostname will get copied to the Gateway status.addresses. | -| providers.kubernetesGateway.statusAddress.ip | string | `""` | This IP will get copied to the Gateway status.addresses, and currently only supports one IP value (IPv4 or IPv6). | -| providers.kubernetesGateway.statusAddress.service | object | `{"name":"{{ (include \"traefik.fullname\" .) }}","namespace":"{{ .Release.Namespace }}"}` | The Kubernetes service to copy status addresses from. When using third parties tools like External-DNS, this option can be used to copy the service loadbalancer.status (containing the service's endpoints IPs) to the gateways. Default to Service of this Chart. | -| providers.kubernetesIngress.allowEmptyServices | bool | `true` | Allows to return 503 when there is no endpoints available | -| providers.kubernetesIngress.allowExternalNameServices | bool | `false` | Allows to reference ExternalName services in Ingress | -| providers.kubernetesIngress.enabled | bool | `true` | Load Kubernetes Ingress provider | -| providers.kubernetesIngress.ingressClass | string | `nil` | When ingressClass is set, only Ingresses containing an annotation with the same value are processed. Otherwise, Ingresses missing the annotation, having an empty value, or the value traefik are processed. | -| providers.kubernetesIngress.namespaces | list | `[]` | Array of namespaces to watch. If left empty, Traefik watches all namespaces. | -| providers.kubernetesIngress.nativeLBByDefault | bool | `false` | Defines whether to use Native Kubernetes load-balancing mode by default. | -| providers.kubernetesIngress.publishedService.enabled | bool | `true` | Enable [publishedService](https://doc.traefik.io/traefik/providers/kubernetes-ingress/#publishedservice) | -| providers.kubernetesIngress.publishedService.pathOverride | string | `""` | Override path of Kubernetes Service used to copy status from. Format: namespace/servicename. Default to Service deployed with this Chart. | -| rbac | object | `{"aggregateTo":[],"enabled":true,"namespaced":false,"secretResourceNames":[]}` | Whether Role Based Access Control objects like roles and rolebindings should be created | -| readinessProbe.failureThreshold | int | `1` | The number of consecutive failures allowed before considering the probe as failed. | -| readinessProbe.initialDelaySeconds | int | `2` | The number of seconds to wait before starting the first probe. | -| readinessProbe.periodSeconds | int | `10` | The number of seconds to wait between consecutive probes. | -| readinessProbe.successThreshold | int | `1` | The minimum consecutive successes required to consider the probe successful. | -| readinessProbe.timeoutSeconds | int | `2` | The number of seconds to wait for a probe response before considering it as failed. | -| resources | object | `{}` | [Resources](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/) for `traefik` container. | -| securityContext | object | See _values.yaml_ | [SecurityContext](https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/pod-v1/#security-context-1) | -| service.additionalServices | object | `{}` | | -| service.annotations | object | `{}` | Additional annotations applied to both TCP and UDP services (e.g. for cloud provider specific config) | -| service.annotationsTCP | object | `{}` | Additional annotations for TCP service only | -| service.annotationsUDP | object | `{}` | Additional annotations for UDP service only | -| service.enabled | bool | `true` | | -| service.externalIPs | list | `[]` | | -| service.labels | object | `{}` | Additional service labels (e.g. for filtering Service by custom labels) | -| service.loadBalancerSourceRanges | list | `[]` | | -| service.single | bool | `true` | | -| service.spec | object | `{}` | Cannot contain type, selector or ports entries. | -| service.type | string | `"LoadBalancer"` | | -| serviceAccount | object | `{"name":""}` | The service account the pods will use to interact with the Kubernetes API | -| serviceAccountAnnotations | object | `{}` | Additional serviceAccount annotations (e.g. for oidc authentication) | -| startupProbe | object | `{}` | Define [Startup Probe](https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/#define-startup-probes) | -| tlsOptions | object | `{}` | TLS Options are created as [TLSOption CRDs](https://doc.traefik.io/traefik/https/tls/#tls-options) When using `labelSelector`, you'll need to set labels on tlsOption accordingly. See EXAMPLE.md for details. | -| tlsStore | object | `{}` | TLS Store are created as [TLSStore CRDs](https://doc.traefik.io/traefik/https/tls/#default-certificate). This is useful if you want to set a default certificate. See EXAMPLE.md for details. | -| tolerations | list | `[]` | Tolerations allow the scheduler to schedule pods with matching taints. | -| topologySpreadConstraints | list | `[]` | You can use topology spread constraints to control how Pods are spread across your cluster among failure-domains. | -| tracing | object | `{"addInternals":false,"otlp":{"enabled":false,"grpc":{"enabled":false,"endpoint":"","insecure":false,"tls":{"ca":"","cert":"","insecureSkipVerify":false,"key":""}},"http":{"enabled":false,"endpoint":"","headers":{},"tls":{"ca":"","cert":"","insecureSkipVerify":false,"key":""}}}}` | https://doc.traefik.io/traefik/observability/tracing/overview/ | -| tracing.addInternals | bool | `false` | Enables tracing for internal resources. Default: false. | -| tracing.otlp.enabled | bool | `false` | See https://doc.traefik.io/traefik/v3.0/observability/tracing/opentelemetry/ | -| tracing.otlp.grpc.enabled | bool | `false` | Set to true in order to send metrics to the OpenTelemetry Collector using gRPC | -| tracing.otlp.grpc.endpoint | string | `""` | Format: ://:. Default: http://localhost:4318/v1/metrics | -| tracing.otlp.grpc.insecure | bool | `false` | Allows reporter to send metrics to the OpenTelemetry Collector without using a secured protocol. | -| tracing.otlp.grpc.tls.ca | string | `""` | The path to the certificate authority, it defaults to the system bundle. | -| tracing.otlp.grpc.tls.cert | string | `""` | The path to the public certificate. When using this option, setting the key option is required. | -| tracing.otlp.grpc.tls.insecureSkipVerify | bool | `false` | When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. | -| tracing.otlp.grpc.tls.key | string | `""` | The path to the private key. When using this option, setting the cert option is required. | -| tracing.otlp.http.enabled | bool | `false` | Set to true in order to send metrics to the OpenTelemetry Collector using HTTP. | -| tracing.otlp.http.endpoint | string | `""` | Format: ://:. Default: http://localhost:4318/v1/metrics | -| tracing.otlp.http.headers | object | `{}` | Additional headers sent with metrics by the reporter to the OpenTelemetry Collector. | -| tracing.otlp.http.tls.ca | string | `""` | The path to the certificate authority, it defaults to the system bundle. | -| tracing.otlp.http.tls.cert | string | `""` | The path to the public certificate. When using this option, setting the key option is required. | -| tracing.otlp.http.tls.insecureSkipVerify | bool | `false` | When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. | -| tracing.otlp.http.tls.key | string | `""` | The path to the private key. When using this option, setting the cert option is required. | -| updateStrategy.rollingUpdate.maxSurge | int | `1` | | -| updateStrategy.rollingUpdate.maxUnavailable | int | `0` | | -| updateStrategy.type | string | `"RollingUpdate"` | Customize updateStrategy of Deployment or DaemonSet | -| volumes | list | `[]` | Add volumes to the traefik pod. The volume name will be passed to tpl. This can be used to mount a cert pair or a configmap that holds a config.toml file. After the volume has been mounted, add the configs into traefik by using the `additionalArguments` list below, eg: `additionalArguments: - "--providers.file.filename=/config/dynamic.toml" - "--ping" - "--ping.entrypoint=web"` | - ----------------------------------------------- -Autogenerated from chart metadata using [helm-docs v1.14.2](https://github.com/norwoodj/helm-docs/releases/v1.14.2) diff --git a/charts/traefik/traefik/crds/gateway-standard-install-v1.2.0.yaml b/charts/traefik/traefik/crds/gateway-standard-install-v1.2.0.yaml deleted file mode 100644 index 0efb1ac..0000000 --- a/charts/traefik/traefik/crds/gateway-standard-install-v1.2.0.yaml +++ /dev/null @@ -1,10345 +0,0 @@ -# Copyright 2024 The Kubernetes Authors. -# -# Licensed under the Apache License, Version 2.0 (the "License"); -# you may not use this file except in compliance with the License. -# You may obtain a copy of the License at -# -# http://www.apache.org/licenses/LICENSE-2.0 -# -# Unless required by applicable law or agreed to in writing, software -# distributed under the License is distributed on an "AS IS" BASIS, -# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -# See the License for the specific language governing permissions and -# limitations under the License. - -# -# Gateway API Standard channel install -# ---- -# -# config/crd/standard/gateway.networking.k8s.io_gatewayclasses.yaml -# -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - api-approved.kubernetes.io: https://github.com/kubernetes-sigs/gateway-api/pull/3328 - gateway.networking.k8s.io/bundle-version: v1.2.0 - gateway.networking.k8s.io/channel: standard - creationTimestamp: null - name: gatewayclasses.gateway.networking.k8s.io -spec: - group: gateway.networking.k8s.io - names: - categories: - - gateway-api - kind: GatewayClass - listKind: GatewayClassList - plural: gatewayclasses - shortNames: - - gc - singular: gatewayclass - scope: Cluster - versions: - - additionalPrinterColumns: - - jsonPath: .spec.controllerName - name: Controller - type: string - - jsonPath: .status.conditions[?(@.type=="Accepted")].status - name: Accepted - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .spec.description - name: Description - priority: 1 - type: string - name: v1 - schema: - openAPIV3Schema: - description: |- - GatewayClass describes a class of Gateways available to the user for creating - Gateway resources. - - It is recommended that this resource be used as a template for Gateways. This - means that a Gateway is based on the state of the GatewayClass at the time it - was created and changes to the GatewayClass or associated parameters are not - propagated down to existing Gateways. This recommendation is intended to - limit the blast radius of changes to GatewayClass or associated parameters. - If implementations choose to propagate GatewayClass changes to existing - Gateways, that MUST be clearly documented by the implementation. - - Whenever one or more Gateways are using a GatewayClass, implementations SHOULD - add the `gateway-exists-finalizer.gateway.networking.k8s.io` finalizer on the - associated GatewayClass. This ensures that a GatewayClass associated with a - Gateway is not deleted while in use. - - GatewayClass is a Cluster level resource. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: Spec defines the desired state of GatewayClass. - properties: - controllerName: - description: |- - ControllerName is the name of the controller that is managing Gateways of - this class. The value of this field MUST be a domain prefixed path. - - Example: "example.net/gateway-controller". - - This field is not mutable and cannot be empty. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ - type: string - x-kubernetes-validations: - - message: Value is immutable - rule: self == oldSelf - description: - description: Description helps describe a GatewayClass with more details. - maxLength: 64 - type: string - parametersRef: - description: |- - ParametersRef is a reference to a resource that contains the configuration - parameters corresponding to the GatewayClass. This is optional if the - controller does not require any additional configuration. - - ParametersRef can reference a standard Kubernetes resource, i.e. ConfigMap, - or an implementation-specific custom resource. The resource can be - cluster-scoped or namespace-scoped. - - If the referent cannot be found, refers to an unsupported kind, or when - the data within that resource is malformed, the GatewayClass SHOULD be - rejected with the "Accepted" status condition set to "False" and an - "InvalidParameters" reason. - - A Gateway for this GatewayClass may provide its own `parametersRef`. When both are specified, - the merging behavior is implementation specific. - It is generally recommended that GatewayClass provides defaults that can be overridden by a Gateway. - - Support: Implementation-specific - properties: - group: - description: Group is the group of the referent. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: Kind is kind of the referent. - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the referent. - This field is required when referring to a Namespace-scoped resource and - MUST be unset when referring to a Cluster-scoped resource. - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - required: - - group - - kind - - name - type: object - required: - - controllerName - type: object - status: - default: - conditions: - - lastTransitionTime: "1970-01-01T00:00:00Z" - message: Waiting for controller - reason: Pending - status: Unknown - type: Accepted - description: |- - Status defines the current state of GatewayClass. - - Implementations MUST populate status on all GatewayClass resources which - specify their controller name. - properties: - conditions: - default: - - lastTransitionTime: "1970-01-01T00:00:00Z" - message: Waiting for controller - reason: Pending - status: Unknown - type: Accepted - description: |- - Conditions is the current status from the controller for - this GatewayClass. - - Controllers should prefer to publish conditions using values - of GatewayClassConditionType for the type of each Condition. - items: - description: Condition contains details for one aspect of the current - state of this API Resource. - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: type of condition in CamelCase or in foo.example.com/CamelCase. - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - maxItems: 8 - type: array - x-kubernetes-list-map-keys: - - type - x-kubernetes-list-type: map - type: object - required: - - spec - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.controllerName - name: Controller - type: string - - jsonPath: .status.conditions[?(@.type=="Accepted")].status - name: Accepted - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .spec.description - name: Description - priority: 1 - type: string - name: v1beta1 - schema: - openAPIV3Schema: - description: |- - GatewayClass describes a class of Gateways available to the user for creating - Gateway resources. - - It is recommended that this resource be used as a template for Gateways. This - means that a Gateway is based on the state of the GatewayClass at the time it - was created and changes to the GatewayClass or associated parameters are not - propagated down to existing Gateways. This recommendation is intended to - limit the blast radius of changes to GatewayClass or associated parameters. - If implementations choose to propagate GatewayClass changes to existing - Gateways, that MUST be clearly documented by the implementation. - - Whenever one or more Gateways are using a GatewayClass, implementations SHOULD - add the `gateway-exists-finalizer.gateway.networking.k8s.io` finalizer on the - associated GatewayClass. This ensures that a GatewayClass associated with a - Gateway is not deleted while in use. - - GatewayClass is a Cluster level resource. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: Spec defines the desired state of GatewayClass. - properties: - controllerName: - description: |- - ControllerName is the name of the controller that is managing Gateways of - this class. The value of this field MUST be a domain prefixed path. - - Example: "example.net/gateway-controller". - - This field is not mutable and cannot be empty. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ - type: string - x-kubernetes-validations: - - message: Value is immutable - rule: self == oldSelf - description: - description: Description helps describe a GatewayClass with more details. - maxLength: 64 - type: string - parametersRef: - description: |- - ParametersRef is a reference to a resource that contains the configuration - parameters corresponding to the GatewayClass. This is optional if the - controller does not require any additional configuration. - - ParametersRef can reference a standard Kubernetes resource, i.e. ConfigMap, - or an implementation-specific custom resource. The resource can be - cluster-scoped or namespace-scoped. - - If the referent cannot be found, refers to an unsupported kind, or when - the data within that resource is malformed, the GatewayClass SHOULD be - rejected with the "Accepted" status condition set to "False" and an - "InvalidParameters" reason. - - A Gateway for this GatewayClass may provide its own `parametersRef`. When both are specified, - the merging behavior is implementation specific. - It is generally recommended that GatewayClass provides defaults that can be overridden by a Gateway. - - Support: Implementation-specific - properties: - group: - description: Group is the group of the referent. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: Kind is kind of the referent. - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the referent. - This field is required when referring to a Namespace-scoped resource and - MUST be unset when referring to a Cluster-scoped resource. - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - required: - - group - - kind - - name - type: object - required: - - controllerName - type: object - status: - default: - conditions: - - lastTransitionTime: "1970-01-01T00:00:00Z" - message: Waiting for controller - reason: Pending - status: Unknown - type: Accepted - description: |- - Status defines the current state of GatewayClass. - - Implementations MUST populate status on all GatewayClass resources which - specify their controller name. - properties: - conditions: - default: - - lastTransitionTime: "1970-01-01T00:00:00Z" - message: Waiting for controller - reason: Pending - status: Unknown - type: Accepted - description: |- - Conditions is the current status from the controller for - this GatewayClass. - - Controllers should prefer to publish conditions using values - of GatewayClassConditionType for the type of each Condition. - items: - description: Condition contains details for one aspect of the current - state of this API Resource. - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: type of condition in CamelCase or in foo.example.com/CamelCase. - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - maxItems: 8 - type: array - x-kubernetes-list-map-keys: - - type - x-kubernetes-list-type: map - type: object - required: - - spec - type: object - served: true - storage: false - subresources: - status: {} -status: - acceptedNames: - kind: "" - plural: "" - conditions: null - storedVersions: null ---- -# -# config/crd/standard/gateway.networking.k8s.io_gateways.yaml -# -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - api-approved.kubernetes.io: https://github.com/kubernetes-sigs/gateway-api/pull/3328 - gateway.networking.k8s.io/bundle-version: v1.2.0 - gateway.networking.k8s.io/channel: standard - creationTimestamp: null - name: gateways.gateway.networking.k8s.io -spec: - group: gateway.networking.k8s.io - names: - categories: - - gateway-api - kind: Gateway - listKind: GatewayList - plural: gateways - shortNames: - - gtw - singular: gateway - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.gatewayClassName - name: Class - type: string - - jsonPath: .status.addresses[*].value - name: Address - type: string - - jsonPath: .status.conditions[?(@.type=="Programmed")].status - name: Programmed - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1 - schema: - openAPIV3Schema: - description: |- - Gateway represents an instance of a service-traffic handling infrastructure - by binding Listeners to a set of IP addresses. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: Spec defines the desired state of Gateway. - properties: - addresses: - description: |+ - Addresses requested for this Gateway. This is optional and behavior can - depend on the implementation. If a value is set in the spec and the - requested address is invalid or unavailable, the implementation MUST - indicate this in the associated entry in GatewayStatus.Addresses. - - The Addresses field represents a request for the address(es) on the - "outside of the Gateway", that traffic bound for this Gateway will use. - This could be the IP address or hostname of an external load balancer or - other networking infrastructure, or some other address that traffic will - be sent to. - - If no Addresses are specified, the implementation MAY schedule the - Gateway in an implementation-specific manner, assigning an appropriate - set of Addresses. - - The implementation MUST bind all Listeners to every GatewayAddress that - it assigns to the Gateway and add a corresponding entry in - GatewayStatus.Addresses. - - Support: Extended - - items: - description: GatewayAddress describes an address that can be bound - to a Gateway. - oneOf: - - properties: - type: - enum: - - IPAddress - value: - anyOf: - - format: ipv4 - - format: ipv6 - - properties: - type: - not: - enum: - - IPAddress - properties: - type: - default: IPAddress - description: Type of the address. - maxLength: 253 - minLength: 1 - pattern: ^Hostname|IPAddress|NamedAddress|[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ - type: string - value: - description: |- - Value of the address. The validity of the values will depend - on the type and support by the controller. - - Examples: `1.2.3.4`, `128::1`, `my-ip-address`. - maxLength: 253 - minLength: 1 - type: string - required: - - value - type: object - x-kubernetes-validations: - - message: Hostname value must only contain valid characters (matching - ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$) - rule: 'self.type == ''Hostname'' ? self.value.matches(r"""^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$"""): - true' - maxItems: 16 - type: array - x-kubernetes-validations: - - message: IPAddress values must be unique - rule: 'self.all(a1, a1.type == ''IPAddress'' ? self.exists_one(a2, - a2.type == a1.type && a2.value == a1.value) : true )' - - message: Hostname values must be unique - rule: 'self.all(a1, a1.type == ''Hostname'' ? self.exists_one(a2, - a2.type == a1.type && a2.value == a1.value) : true )' - gatewayClassName: - description: |- - GatewayClassName used for this Gateway. This is the name of a - GatewayClass resource. - maxLength: 253 - minLength: 1 - type: string - infrastructure: - description: |- - Infrastructure defines infrastructure level attributes about this Gateway instance. - - Support: Extended - properties: - annotations: - additionalProperties: - description: |- - AnnotationValue is the value of an annotation in Gateway API. This is used - for validation of maps such as TLS options. This roughly matches Kubernetes - annotation validation, although the length validation in that case is based - on the entire size of the annotations struct. - maxLength: 4096 - minLength: 0 - type: string - description: |- - Annotations that SHOULD be applied to any resources created in response to this Gateway. - - For implementations creating other Kubernetes objects, this should be the `metadata.annotations` field on resources. - For other implementations, this refers to any relevant (implementation specific) "annotations" concepts. - - An implementation may chose to add additional implementation-specific annotations as they see fit. - - Support: Extended - maxProperties: 8 - type: object - x-kubernetes-validations: - - message: Annotation keys must be in the form of an optional - DNS subdomain prefix followed by a required name segment of - up to 63 characters. - rule: self.all(key, key.matches(r"""^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?([A-Za-z0-9][-A-Za-z0-9_.]{0,61})?[A-Za-z0-9]$""")) - - message: If specified, the annotation key's prefix must be a - DNS subdomain not longer than 253 characters in total. - rule: self.all(key, key.split("/")[0].size() < 253) - labels: - additionalProperties: - description: |- - LabelValue is the value of a label in the Gateway API. This is used for validation - of maps such as Gateway infrastructure labels. This matches the Kubernetes - label validation rules: - * must be 63 characters or less (can be empty), - * unless empty, must begin and end with an alphanumeric character ([a-z0-9A-Z]), - * could contain dashes (-), underscores (_), dots (.), and alphanumerics between. - - Valid values include: - - * MyValue - * my.name - * 123-my-value - maxLength: 63 - minLength: 0 - pattern: ^(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])?$ - type: string - description: |- - Labels that SHOULD be applied to any resources created in response to this Gateway. - - For implementations creating other Kubernetes objects, this should be the `metadata.labels` field on resources. - For other implementations, this refers to any relevant (implementation specific) "labels" concepts. - - An implementation may chose to add additional implementation-specific labels as they see fit. - - If an implementation maps these labels to Pods, or any other resource that would need to be recreated when labels - change, it SHOULD clearly warn about this behavior in documentation. - - Support: Extended - maxProperties: 8 - type: object - x-kubernetes-validations: - - message: Label keys must be in the form of an optional DNS subdomain - prefix followed by a required name segment of up to 63 characters. - rule: self.all(key, key.matches(r"""^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?([A-Za-z0-9][-A-Za-z0-9_.]{0,61})?[A-Za-z0-9]$""")) - - message: If specified, the label key's prefix must be a DNS - subdomain not longer than 253 characters in total. - rule: self.all(key, key.split("/")[0].size() < 253) - parametersRef: - description: |- - ParametersRef is a reference to a resource that contains the configuration - parameters corresponding to the Gateway. This is optional if the - controller does not require any additional configuration. - - This follows the same semantics as GatewayClass's `parametersRef`, but on a per-Gateway basis - - The Gateway's GatewayClass may provide its own `parametersRef`. When both are specified, - the merging behavior is implementation specific. - It is generally recommended that GatewayClass provides defaults that can be overridden by a Gateway. - - Support: Implementation-specific - properties: - group: - description: Group is the group of the referent. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: Kind is kind of the referent. - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - required: - - group - - kind - - name - type: object - type: object - listeners: - description: |- - Listeners associated with this Gateway. Listeners define - logical endpoints that are bound on this Gateway's addresses. - At least one Listener MUST be specified. - - Each Listener in a set of Listeners (for example, in a single Gateway) - MUST be _distinct_, in that a traffic flow MUST be able to be assigned to - exactly one listener. (This section uses "set of Listeners" rather than - "Listeners in a single Gateway" because implementations MAY merge configuration - from multiple Gateways onto a single data plane, and these rules _also_ - apply in that case). - - Practically, this means that each listener in a set MUST have a unique - combination of Port, Protocol, and, if supported by the protocol, Hostname. - - Some combinations of port, protocol, and TLS settings are considered - Core support and MUST be supported by implementations based on their - targeted conformance profile: - - HTTP Profile - - 1. HTTPRoute, Port: 80, Protocol: HTTP - 2. HTTPRoute, Port: 443, Protocol: HTTPS, TLS Mode: Terminate, TLS keypair provided - - TLS Profile - - 1. TLSRoute, Port: 443, Protocol: TLS, TLS Mode: Passthrough - - "Distinct" Listeners have the following property: - - The implementation can match inbound requests to a single distinct - Listener. When multiple Listeners share values for fields (for - example, two Listeners with the same Port value), the implementation - can match requests to only one of the Listeners using other - Listener fields. - - For example, the following Listener scenarios are distinct: - - 1. Multiple Listeners with the same Port that all use the "HTTP" - Protocol that all have unique Hostname values. - 2. Multiple Listeners with the same Port that use either the "HTTPS" or - "TLS" Protocol that all have unique Hostname values. - 3. A mixture of "TCP" and "UDP" Protocol Listeners, where no Listener - with the same Protocol has the same Port value. - - Some fields in the Listener struct have possible values that affect - whether the Listener is distinct. Hostname is particularly relevant - for HTTP or HTTPS protocols. - - When using the Hostname value to select between same-Port, same-Protocol - Listeners, the Hostname value must be different on each Listener for the - Listener to be distinct. - - When the Listeners are distinct based on Hostname, inbound request - hostnames MUST match from the most specific to least specific Hostname - values to choose the correct Listener and its associated set of Routes. - - Exact matches must be processed before wildcard matches, and wildcard - matches must be processed before fallback (empty Hostname value) - matches. For example, `"foo.example.com"` takes precedence over - `"*.example.com"`, and `"*.example.com"` takes precedence over `""`. - - Additionally, if there are multiple wildcard entries, more specific - wildcard entries must be processed before less specific wildcard entries. - For example, `"*.foo.example.com"` takes precedence over `"*.example.com"`. - The precise definition here is that the higher the number of dots in the - hostname to the right of the wildcard character, the higher the precedence. - - The wildcard character will match any number of characters _and dots_ to - the left, however, so `"*.example.com"` will match both - `"foo.bar.example.com"` _and_ `"bar.example.com"`. - - If a set of Listeners contains Listeners that are not distinct, then those - Listeners are Conflicted, and the implementation MUST set the "Conflicted" - condition in the Listener Status to "True". - - Implementations MAY choose to accept a Gateway with some Conflicted - Listeners only if they only accept the partial Listener set that contains - no Conflicted Listeners. To put this another way, implementations may - accept a partial Listener set only if they throw out *all* the conflicting - Listeners. No picking one of the conflicting listeners as the winner. - This also means that the Gateway must have at least one non-conflicting - Listener in this case, otherwise it violates the requirement that at - least one Listener must be present. - - The implementation MUST set a "ListenersNotValid" condition on the - Gateway Status when the Gateway contains Conflicted Listeners whether or - not they accept the Gateway. That Condition SHOULD clearly - indicate in the Message which Listeners are conflicted, and which are - Accepted. Additionally, the Listener status for those listeners SHOULD - indicate which Listeners are conflicted and not Accepted. - - A Gateway's Listeners are considered "compatible" if: - - 1. They are distinct. - 2. The implementation can serve them in compliance with the Addresses - requirement that all Listeners are available on all assigned - addresses. - - Compatible combinations in Extended support are expected to vary across - implementations. A combination that is compatible for one implementation - may not be compatible for another. - - For example, an implementation that cannot serve both TCP and UDP listeners - on the same address, or cannot mix HTTPS and generic TLS listens on the same port - would not consider those cases compatible, even though they are distinct. - - Note that requests SHOULD match at most one Listener. For example, if - Listeners are defined for "foo.example.com" and "*.example.com", a - request to "foo.example.com" SHOULD only be routed using routes attached - to the "foo.example.com" Listener (and not the "*.example.com" Listener). - This concept is known as "Listener Isolation". Implementations that do - not support Listener Isolation MUST clearly document this. - - Implementations MAY merge separate Gateways onto a single set of - Addresses if all Listeners across all Gateways are compatible. - - Support: Core - items: - description: |- - Listener embodies the concept of a logical endpoint where a Gateway accepts - network connections. - properties: - allowedRoutes: - default: - namespaces: - from: Same - description: |- - AllowedRoutes defines the types of routes that MAY be attached to a - Listener and the trusted namespaces where those Route resources MAY be - present. - - Although a client request may match multiple route rules, only one rule - may ultimately receive the request. Matching precedence MUST be - determined in order of the following criteria: - - * The most specific match as defined by the Route type. - * The oldest Route based on creation timestamp. For example, a Route with - a creation timestamp of "2020-09-08 01:02:03" is given precedence over - a Route with a creation timestamp of "2020-09-08 01:02:04". - * If everything else is equivalent, the Route appearing first in - alphabetical order (namespace/name) should be given precedence. For - example, foo/bar is given precedence over foo/baz. - - All valid rules within a Route attached to this Listener should be - implemented. Invalid Route rules can be ignored (sometimes that will mean - the full Route). If a Route rule transitions from valid to invalid, - support for that Route rule should be dropped to ensure consistency. For - example, even if a filter specified by a Route rule is invalid, the rest - of the rules within that Route should still be supported. - - Support: Core - properties: - kinds: - description: |- - Kinds specifies the groups and kinds of Routes that are allowed to bind - to this Gateway Listener. When unspecified or empty, the kinds of Routes - selected are determined using the Listener protocol. - - A RouteGroupKind MUST correspond to kinds of Routes that are compatible - with the application protocol specified in the Listener's Protocol field. - If an implementation does not support or recognize this resource type, it - MUST set the "ResolvedRefs" condition to False for this Listener with the - "InvalidRouteKinds" reason. - - Support: Core - items: - description: RouteGroupKind indicates the group and kind - of a Route resource. - properties: - group: - default: gateway.networking.k8s.io - description: Group is the group of the Route. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: Kind is the kind of the Route. - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - required: - - kind - type: object - maxItems: 8 - type: array - namespaces: - default: - from: Same - description: |- - Namespaces indicates namespaces from which Routes may be attached to this - Listener. This is restricted to the namespace of this Gateway by default. - - Support: Core - properties: - from: - default: Same - description: |- - From indicates where Routes will be selected for this Gateway. Possible - values are: - - * All: Routes in all namespaces may be used by this Gateway. - * Selector: Routes in namespaces selected by the selector may be used by - this Gateway. - * Same: Only Routes in the same namespace may be used by this Gateway. - - Support: Core - enum: - - All - - Selector - - Same - type: string - selector: - description: |- - Selector must be specified when From is set to "Selector". In that case, - only Routes in Namespaces matching this Selector will be selected by this - Gateway. This field is ignored for other values of "From". - - Support: Core - properties: - matchExpressions: - description: matchExpressions is a list of label - selector requirements. The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the - selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - x-kubernetes-list-type: atomic - required: - - key - - operator - type: object - type: array - x-kubernetes-list-type: atomic - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - type: object - type: object - hostname: - description: |- - Hostname specifies the virtual hostname to match for protocol types that - define this concept. When unspecified, all hostnames are matched. This - field is ignored for protocols that don't require hostname based - matching. - - Implementations MUST apply Hostname matching appropriately for each of - the following protocols: - - * TLS: The Listener Hostname MUST match the SNI. - * HTTP: The Listener Hostname MUST match the Host header of the request. - * HTTPS: The Listener Hostname SHOULD match at both the TLS and HTTP - protocol layers as described above. If an implementation does not - ensure that both the SNI and Host header match the Listener hostname, - it MUST clearly document that. - - For HTTPRoute and TLSRoute resources, there is an interaction with the - `spec.hostnames` array. When both listener and route specify hostnames, - there MUST be an intersection between the values for a Route to be - accepted. For more information, refer to the Route specific Hostnames - documentation. - - Hostnames that are prefixed with a wildcard label (`*.`) are interpreted - as a suffix match. That means that a match for `*.example.com` would match - both `test.example.com`, and `foo.test.example.com`, but not `example.com`. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - name: - description: |- - Name is the name of the Listener. This name MUST be unique within a - Gateway. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - port: - description: |- - Port is the network port. Multiple listeners may use the - same port, subject to the Listener compatibility rules. - - Support: Core - format: int32 - maximum: 65535 - minimum: 1 - type: integer - protocol: - description: |- - Protocol specifies the network protocol this listener expects to receive. - - Support: Core - maxLength: 255 - minLength: 1 - pattern: ^[a-zA-Z0-9]([-a-zA-Z0-9]*[a-zA-Z0-9])?$|[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9]+$ - type: string - tls: - description: |- - TLS is the TLS configuration for the Listener. This field is required if - the Protocol field is "HTTPS" or "TLS". It is invalid to set this field - if the Protocol field is "HTTP", "TCP", or "UDP". - - The association of SNIs to Certificate defined in GatewayTLSConfig is - defined based on the Hostname field for this listener. - - The GatewayClass MUST use the longest matching SNI out of all - available certificates for any TLS handshake. - - Support: Core - properties: - certificateRefs: - description: |- - CertificateRefs contains a series of references to Kubernetes objects that - contains TLS certificates and private keys. These certificates are used to - establish a TLS handshake for requests that match the hostname of the - associated listener. - - A single CertificateRef to a Kubernetes Secret has "Core" support. - Implementations MAY choose to support attaching multiple certificates to - a Listener, but this behavior is implementation-specific. - - References to a resource in different namespace are invalid UNLESS there - is a ReferenceGrant in the target namespace that allows the certificate - to be attached. If a ReferenceGrant does not allow this reference, the - "ResolvedRefs" condition MUST be set to False for this listener with the - "RefNotPermitted" reason. - - This field is required to have at least one element when the mode is set - to "Terminate" (default) and is optional otherwise. - - CertificateRefs can reference to standard Kubernetes resources, i.e. - Secret, or implementation-specific custom resources. - - Support: Core - A single reference to a Kubernetes Secret of type kubernetes.io/tls - - Support: Implementation-specific (More than one reference or other resource types) - items: - description: |- - SecretObjectReference identifies an API object including its namespace, - defaulting to Secret. - - The API object must be valid in the cluster; the Group and Kind must - be registered in the cluster for this reference to be valid. - - References to objects with invalid Group and Kind are not valid, and must - be rejected by the implementation, with appropriate Conditions set - on the containing object. - properties: - group: - default: "" - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Secret - description: Kind is kind of the referent. For example - "Secret". - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the referenced object. When unspecified, the local - namespace is inferred. - - Note that when a namespace different than the local namespace is specified, - a ReferenceGrant object is required in the referent namespace to allow that - namespace's owner to accept the reference. See the ReferenceGrant - documentation for details. - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - required: - - name - type: object - maxItems: 64 - type: array - mode: - default: Terminate - description: |- - Mode defines the TLS behavior for the TLS session initiated by the client. - There are two possible modes: - - - Terminate: The TLS session between the downstream client and the - Gateway is terminated at the Gateway. This mode requires certificates - to be specified in some way, such as populating the certificateRefs - field. - - Passthrough: The TLS session is NOT terminated by the Gateway. This - implies that the Gateway can't decipher the TLS stream except for - the ClientHello message of the TLS protocol. The certificateRefs field - is ignored in this mode. - - Support: Core - enum: - - Terminate - - Passthrough - type: string - options: - additionalProperties: - description: |- - AnnotationValue is the value of an annotation in Gateway API. This is used - for validation of maps such as TLS options. This roughly matches Kubernetes - annotation validation, although the length validation in that case is based - on the entire size of the annotations struct. - maxLength: 4096 - minLength: 0 - type: string - description: |- - Options are a list of key/value pairs to enable extended TLS - configuration for each implementation. For example, configuring the - minimum TLS version or supported cipher suites. - - A set of common keys MAY be defined by the API in the future. To avoid - any ambiguity, implementation-specific definitions MUST use - domain-prefixed names, such as `example.com/my-custom-option`. - Un-prefixed names are reserved for key names defined by Gateway API. - - Support: Implementation-specific - maxProperties: 16 - type: object - type: object - x-kubernetes-validations: - - message: certificateRefs or options must be specified when - mode is Terminate - rule: 'self.mode == ''Terminate'' ? size(self.certificateRefs) - > 0 || size(self.options) > 0 : true' - required: - - name - - port - - protocol - type: object - maxItems: 64 - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - x-kubernetes-validations: - - message: tls must not be specified for protocols ['HTTP', 'TCP', - 'UDP'] - rule: 'self.all(l, l.protocol in [''HTTP'', ''TCP'', ''UDP''] ? - !has(l.tls) : true)' - - message: tls mode must be Terminate for protocol HTTPS - rule: 'self.all(l, (l.protocol == ''HTTPS'' && has(l.tls)) ? (l.tls.mode - == '''' || l.tls.mode == ''Terminate'') : true)' - - message: hostname must not be specified for protocols ['TCP', 'UDP'] - rule: 'self.all(l, l.protocol in [''TCP'', ''UDP''] ? (!has(l.hostname) - || l.hostname == '''') : true)' - - message: Listener name must be unique within the Gateway - rule: self.all(l1, self.exists_one(l2, l1.name == l2.name)) - - message: Combination of port, protocol and hostname must be unique - for each listener - rule: 'self.all(l1, self.exists_one(l2, l1.port == l2.port && l1.protocol - == l2.protocol && (has(l1.hostname) && has(l2.hostname) ? l1.hostname - == l2.hostname : !has(l1.hostname) && !has(l2.hostname))))' - required: - - gatewayClassName - - listeners - type: object - status: - default: - conditions: - - lastTransitionTime: "1970-01-01T00:00:00Z" - message: Waiting for controller - reason: Pending - status: Unknown - type: Accepted - - lastTransitionTime: "1970-01-01T00:00:00Z" - message: Waiting for controller - reason: Pending - status: Unknown - type: Programmed - description: Status defines the current state of Gateway. - properties: - addresses: - description: |+ - Addresses lists the network addresses that have been bound to the - Gateway. - - This list may differ from the addresses provided in the spec under some - conditions: - - * no addresses are specified, all addresses are dynamically assigned - * a combination of specified and dynamic addresses are assigned - * a specified address was unusable (e.g. already in use) - - items: - description: GatewayStatusAddress describes a network address that - is bound to a Gateway. - oneOf: - - properties: - type: - enum: - - IPAddress - value: - anyOf: - - format: ipv4 - - format: ipv6 - - properties: - type: - not: - enum: - - IPAddress - properties: - type: - default: IPAddress - description: Type of the address. - maxLength: 253 - minLength: 1 - pattern: ^Hostname|IPAddress|NamedAddress|[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ - type: string - value: - description: |- - Value of the address. The validity of the values will depend - on the type and support by the controller. - - Examples: `1.2.3.4`, `128::1`, `my-ip-address`. - maxLength: 253 - minLength: 1 - type: string - required: - - value - type: object - x-kubernetes-validations: - - message: Hostname value must only contain valid characters (matching - ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$) - rule: 'self.type == ''Hostname'' ? self.value.matches(r"""^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$"""): - true' - maxItems: 16 - type: array - conditions: - default: - - lastTransitionTime: "1970-01-01T00:00:00Z" - message: Waiting for controller - reason: Pending - status: Unknown - type: Accepted - - lastTransitionTime: "1970-01-01T00:00:00Z" - message: Waiting for controller - reason: Pending - status: Unknown - type: Programmed - description: |- - Conditions describe the current conditions of the Gateway. - - Implementations should prefer to express Gateway conditions - using the `GatewayConditionType` and `GatewayConditionReason` - constants so that operators and tools can converge on a common - vocabulary to describe Gateway state. - - Known condition types are: - - * "Accepted" - * "Programmed" - * "Ready" - items: - description: Condition contains details for one aspect of the current - state of this API Resource. - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: type of condition in CamelCase or in foo.example.com/CamelCase. - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - maxItems: 8 - type: array - x-kubernetes-list-map-keys: - - type - x-kubernetes-list-type: map - listeners: - description: Listeners provide status for each unique listener port - defined in the Spec. - items: - description: ListenerStatus is the status associated with a Listener. - properties: - attachedRoutes: - description: |- - AttachedRoutes represents the total number of Routes that have been - successfully attached to this Listener. - - Successful attachment of a Route to a Listener is based solely on the - combination of the AllowedRoutes field on the corresponding Listener - and the Route's ParentRefs field. A Route is successfully attached to - a Listener when it is selected by the Listener's AllowedRoutes field - AND the Route has a valid ParentRef selecting the whole Gateway - resource or a specific Listener as a parent resource (more detail on - attachment semantics can be found in the documentation on the various - Route kinds ParentRefs fields). Listener or Route status does not impact - successful attachment, i.e. the AttachedRoutes field count MUST be set - for Listeners with condition Accepted: false and MUST count successfully - attached Routes that may themselves have Accepted: false conditions. - - Uses for this field include troubleshooting Route attachment and - measuring blast radius/impact of changes to a Listener. - format: int32 - type: integer - conditions: - description: Conditions describe the current condition of this - listener. - items: - description: Condition contains details for one aspect of - the current state of this API Resource. - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, - Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: type of condition in CamelCase or in foo.example.com/CamelCase. - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - maxItems: 8 - type: array - x-kubernetes-list-map-keys: - - type - x-kubernetes-list-type: map - name: - description: Name is the name of the Listener that this status - corresponds to. - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - supportedKinds: - description: |- - SupportedKinds is the list indicating the Kinds supported by this - listener. This MUST represent the kinds an implementation supports for - that Listener configuration. - - If kinds are specified in Spec that are not supported, they MUST NOT - appear in this list and an implementation MUST set the "ResolvedRefs" - condition to "False" with the "InvalidRouteKinds" reason. If both valid - and invalid Route kinds are specified, the implementation MUST - reference the valid Route kinds that have been specified. - items: - description: RouteGroupKind indicates the group and kind of - a Route resource. - properties: - group: - default: gateway.networking.k8s.io - description: Group is the group of the Route. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: Kind is the kind of the Route. - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - required: - - kind - type: object - maxItems: 8 - type: array - required: - - attachedRoutes - - conditions - - name - - supportedKinds - type: object - maxItems: 64 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - required: - - spec - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.gatewayClassName - name: Class - type: string - - jsonPath: .status.addresses[*].value - name: Address - type: string - - jsonPath: .status.conditions[?(@.type=="Programmed")].status - name: Programmed - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1beta1 - schema: - openAPIV3Schema: - description: |- - Gateway represents an instance of a service-traffic handling infrastructure - by binding Listeners to a set of IP addresses. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: Spec defines the desired state of Gateway. - properties: - addresses: - description: |+ - Addresses requested for this Gateway. This is optional and behavior can - depend on the implementation. If a value is set in the spec and the - requested address is invalid or unavailable, the implementation MUST - indicate this in the associated entry in GatewayStatus.Addresses. - - The Addresses field represents a request for the address(es) on the - "outside of the Gateway", that traffic bound for this Gateway will use. - This could be the IP address or hostname of an external load balancer or - other networking infrastructure, or some other address that traffic will - be sent to. - - If no Addresses are specified, the implementation MAY schedule the - Gateway in an implementation-specific manner, assigning an appropriate - set of Addresses. - - The implementation MUST bind all Listeners to every GatewayAddress that - it assigns to the Gateway and add a corresponding entry in - GatewayStatus.Addresses. - - Support: Extended - - items: - description: GatewayAddress describes an address that can be bound - to a Gateway. - oneOf: - - properties: - type: - enum: - - IPAddress - value: - anyOf: - - format: ipv4 - - format: ipv6 - - properties: - type: - not: - enum: - - IPAddress - properties: - type: - default: IPAddress - description: Type of the address. - maxLength: 253 - minLength: 1 - pattern: ^Hostname|IPAddress|NamedAddress|[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ - type: string - value: - description: |- - Value of the address. The validity of the values will depend - on the type and support by the controller. - - Examples: `1.2.3.4`, `128::1`, `my-ip-address`. - maxLength: 253 - minLength: 1 - type: string - required: - - value - type: object - x-kubernetes-validations: - - message: Hostname value must only contain valid characters (matching - ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$) - rule: 'self.type == ''Hostname'' ? self.value.matches(r"""^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$"""): - true' - maxItems: 16 - type: array - x-kubernetes-validations: - - message: IPAddress values must be unique - rule: 'self.all(a1, a1.type == ''IPAddress'' ? self.exists_one(a2, - a2.type == a1.type && a2.value == a1.value) : true )' - - message: Hostname values must be unique - rule: 'self.all(a1, a1.type == ''Hostname'' ? self.exists_one(a2, - a2.type == a1.type && a2.value == a1.value) : true )' - gatewayClassName: - description: |- - GatewayClassName used for this Gateway. This is the name of a - GatewayClass resource. - maxLength: 253 - minLength: 1 - type: string - infrastructure: - description: |- - Infrastructure defines infrastructure level attributes about this Gateway instance. - - Support: Extended - properties: - annotations: - additionalProperties: - description: |- - AnnotationValue is the value of an annotation in Gateway API. This is used - for validation of maps such as TLS options. This roughly matches Kubernetes - annotation validation, although the length validation in that case is based - on the entire size of the annotations struct. - maxLength: 4096 - minLength: 0 - type: string - description: |- - Annotations that SHOULD be applied to any resources created in response to this Gateway. - - For implementations creating other Kubernetes objects, this should be the `metadata.annotations` field on resources. - For other implementations, this refers to any relevant (implementation specific) "annotations" concepts. - - An implementation may chose to add additional implementation-specific annotations as they see fit. - - Support: Extended - maxProperties: 8 - type: object - x-kubernetes-validations: - - message: Annotation keys must be in the form of an optional - DNS subdomain prefix followed by a required name segment of - up to 63 characters. - rule: self.all(key, key.matches(r"""^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?([A-Za-z0-9][-A-Za-z0-9_.]{0,61})?[A-Za-z0-9]$""")) - - message: If specified, the annotation key's prefix must be a - DNS subdomain not longer than 253 characters in total. - rule: self.all(key, key.split("/")[0].size() < 253) - labels: - additionalProperties: - description: |- - LabelValue is the value of a label in the Gateway API. This is used for validation - of maps such as Gateway infrastructure labels. This matches the Kubernetes - label validation rules: - * must be 63 characters or less (can be empty), - * unless empty, must begin and end with an alphanumeric character ([a-z0-9A-Z]), - * could contain dashes (-), underscores (_), dots (.), and alphanumerics between. - - Valid values include: - - * MyValue - * my.name - * 123-my-value - maxLength: 63 - minLength: 0 - pattern: ^(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])?$ - type: string - description: |- - Labels that SHOULD be applied to any resources created in response to this Gateway. - - For implementations creating other Kubernetes objects, this should be the `metadata.labels` field on resources. - For other implementations, this refers to any relevant (implementation specific) "labels" concepts. - - An implementation may chose to add additional implementation-specific labels as they see fit. - - If an implementation maps these labels to Pods, or any other resource that would need to be recreated when labels - change, it SHOULD clearly warn about this behavior in documentation. - - Support: Extended - maxProperties: 8 - type: object - x-kubernetes-validations: - - message: Label keys must be in the form of an optional DNS subdomain - prefix followed by a required name segment of up to 63 characters. - rule: self.all(key, key.matches(r"""^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?([A-Za-z0-9][-A-Za-z0-9_.]{0,61})?[A-Za-z0-9]$""")) - - message: If specified, the label key's prefix must be a DNS - subdomain not longer than 253 characters in total. - rule: self.all(key, key.split("/")[0].size() < 253) - parametersRef: - description: |- - ParametersRef is a reference to a resource that contains the configuration - parameters corresponding to the Gateway. This is optional if the - controller does not require any additional configuration. - - This follows the same semantics as GatewayClass's `parametersRef`, but on a per-Gateway basis - - The Gateway's GatewayClass may provide its own `parametersRef`. When both are specified, - the merging behavior is implementation specific. - It is generally recommended that GatewayClass provides defaults that can be overridden by a Gateway. - - Support: Implementation-specific - properties: - group: - description: Group is the group of the referent. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: Kind is kind of the referent. - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - required: - - group - - kind - - name - type: object - type: object - listeners: - description: |- - Listeners associated with this Gateway. Listeners define - logical endpoints that are bound on this Gateway's addresses. - At least one Listener MUST be specified. - - Each Listener in a set of Listeners (for example, in a single Gateway) - MUST be _distinct_, in that a traffic flow MUST be able to be assigned to - exactly one listener. (This section uses "set of Listeners" rather than - "Listeners in a single Gateway" because implementations MAY merge configuration - from multiple Gateways onto a single data plane, and these rules _also_ - apply in that case). - - Practically, this means that each listener in a set MUST have a unique - combination of Port, Protocol, and, if supported by the protocol, Hostname. - - Some combinations of port, protocol, and TLS settings are considered - Core support and MUST be supported by implementations based on their - targeted conformance profile: - - HTTP Profile - - 1. HTTPRoute, Port: 80, Protocol: HTTP - 2. HTTPRoute, Port: 443, Protocol: HTTPS, TLS Mode: Terminate, TLS keypair provided - - TLS Profile - - 1. TLSRoute, Port: 443, Protocol: TLS, TLS Mode: Passthrough - - "Distinct" Listeners have the following property: - - The implementation can match inbound requests to a single distinct - Listener. When multiple Listeners share values for fields (for - example, two Listeners with the same Port value), the implementation - can match requests to only one of the Listeners using other - Listener fields. - - For example, the following Listener scenarios are distinct: - - 1. Multiple Listeners with the same Port that all use the "HTTP" - Protocol that all have unique Hostname values. - 2. Multiple Listeners with the same Port that use either the "HTTPS" or - "TLS" Protocol that all have unique Hostname values. - 3. A mixture of "TCP" and "UDP" Protocol Listeners, where no Listener - with the same Protocol has the same Port value. - - Some fields in the Listener struct have possible values that affect - whether the Listener is distinct. Hostname is particularly relevant - for HTTP or HTTPS protocols. - - When using the Hostname value to select between same-Port, same-Protocol - Listeners, the Hostname value must be different on each Listener for the - Listener to be distinct. - - When the Listeners are distinct based on Hostname, inbound request - hostnames MUST match from the most specific to least specific Hostname - values to choose the correct Listener and its associated set of Routes. - - Exact matches must be processed before wildcard matches, and wildcard - matches must be processed before fallback (empty Hostname value) - matches. For example, `"foo.example.com"` takes precedence over - `"*.example.com"`, and `"*.example.com"` takes precedence over `""`. - - Additionally, if there are multiple wildcard entries, more specific - wildcard entries must be processed before less specific wildcard entries. - For example, `"*.foo.example.com"` takes precedence over `"*.example.com"`. - The precise definition here is that the higher the number of dots in the - hostname to the right of the wildcard character, the higher the precedence. - - The wildcard character will match any number of characters _and dots_ to - the left, however, so `"*.example.com"` will match both - `"foo.bar.example.com"` _and_ `"bar.example.com"`. - - If a set of Listeners contains Listeners that are not distinct, then those - Listeners are Conflicted, and the implementation MUST set the "Conflicted" - condition in the Listener Status to "True". - - Implementations MAY choose to accept a Gateway with some Conflicted - Listeners only if they only accept the partial Listener set that contains - no Conflicted Listeners. To put this another way, implementations may - accept a partial Listener set only if they throw out *all* the conflicting - Listeners. No picking one of the conflicting listeners as the winner. - This also means that the Gateway must have at least one non-conflicting - Listener in this case, otherwise it violates the requirement that at - least one Listener must be present. - - The implementation MUST set a "ListenersNotValid" condition on the - Gateway Status when the Gateway contains Conflicted Listeners whether or - not they accept the Gateway. That Condition SHOULD clearly - indicate in the Message which Listeners are conflicted, and which are - Accepted. Additionally, the Listener status for those listeners SHOULD - indicate which Listeners are conflicted and not Accepted. - - A Gateway's Listeners are considered "compatible" if: - - 1. They are distinct. - 2. The implementation can serve them in compliance with the Addresses - requirement that all Listeners are available on all assigned - addresses. - - Compatible combinations in Extended support are expected to vary across - implementations. A combination that is compatible for one implementation - may not be compatible for another. - - For example, an implementation that cannot serve both TCP and UDP listeners - on the same address, or cannot mix HTTPS and generic TLS listens on the same port - would not consider those cases compatible, even though they are distinct. - - Note that requests SHOULD match at most one Listener. For example, if - Listeners are defined for "foo.example.com" and "*.example.com", a - request to "foo.example.com" SHOULD only be routed using routes attached - to the "foo.example.com" Listener (and not the "*.example.com" Listener). - This concept is known as "Listener Isolation". Implementations that do - not support Listener Isolation MUST clearly document this. - - Implementations MAY merge separate Gateways onto a single set of - Addresses if all Listeners across all Gateways are compatible. - - Support: Core - items: - description: |- - Listener embodies the concept of a logical endpoint where a Gateway accepts - network connections. - properties: - allowedRoutes: - default: - namespaces: - from: Same - description: |- - AllowedRoutes defines the types of routes that MAY be attached to a - Listener and the trusted namespaces where those Route resources MAY be - present. - - Although a client request may match multiple route rules, only one rule - may ultimately receive the request. Matching precedence MUST be - determined in order of the following criteria: - - * The most specific match as defined by the Route type. - * The oldest Route based on creation timestamp. For example, a Route with - a creation timestamp of "2020-09-08 01:02:03" is given precedence over - a Route with a creation timestamp of "2020-09-08 01:02:04". - * If everything else is equivalent, the Route appearing first in - alphabetical order (namespace/name) should be given precedence. For - example, foo/bar is given precedence over foo/baz. - - All valid rules within a Route attached to this Listener should be - implemented. Invalid Route rules can be ignored (sometimes that will mean - the full Route). If a Route rule transitions from valid to invalid, - support for that Route rule should be dropped to ensure consistency. For - example, even if a filter specified by a Route rule is invalid, the rest - of the rules within that Route should still be supported. - - Support: Core - properties: - kinds: - description: |- - Kinds specifies the groups and kinds of Routes that are allowed to bind - to this Gateway Listener. When unspecified or empty, the kinds of Routes - selected are determined using the Listener protocol. - - A RouteGroupKind MUST correspond to kinds of Routes that are compatible - with the application protocol specified in the Listener's Protocol field. - If an implementation does not support or recognize this resource type, it - MUST set the "ResolvedRefs" condition to False for this Listener with the - "InvalidRouteKinds" reason. - - Support: Core - items: - description: RouteGroupKind indicates the group and kind - of a Route resource. - properties: - group: - default: gateway.networking.k8s.io - description: Group is the group of the Route. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: Kind is the kind of the Route. - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - required: - - kind - type: object - maxItems: 8 - type: array - namespaces: - default: - from: Same - description: |- - Namespaces indicates namespaces from which Routes may be attached to this - Listener. This is restricted to the namespace of this Gateway by default. - - Support: Core - properties: - from: - default: Same - description: |- - From indicates where Routes will be selected for this Gateway. Possible - values are: - - * All: Routes in all namespaces may be used by this Gateway. - * Selector: Routes in namespaces selected by the selector may be used by - this Gateway. - * Same: Only Routes in the same namespace may be used by this Gateway. - - Support: Core - enum: - - All - - Selector - - Same - type: string - selector: - description: |- - Selector must be specified when From is set to "Selector". In that case, - only Routes in Namespaces matching this Selector will be selected by this - Gateway. This field is ignored for other values of "From". - - Support: Core - properties: - matchExpressions: - description: matchExpressions is a list of label - selector requirements. The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the - selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - x-kubernetes-list-type: atomic - required: - - key - - operator - type: object - type: array - x-kubernetes-list-type: atomic - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - type: object - type: object - hostname: - description: |- - Hostname specifies the virtual hostname to match for protocol types that - define this concept. When unspecified, all hostnames are matched. This - field is ignored for protocols that don't require hostname based - matching. - - Implementations MUST apply Hostname matching appropriately for each of - the following protocols: - - * TLS: The Listener Hostname MUST match the SNI. - * HTTP: The Listener Hostname MUST match the Host header of the request. - * HTTPS: The Listener Hostname SHOULD match at both the TLS and HTTP - protocol layers as described above. If an implementation does not - ensure that both the SNI and Host header match the Listener hostname, - it MUST clearly document that. - - For HTTPRoute and TLSRoute resources, there is an interaction with the - `spec.hostnames` array. When both listener and route specify hostnames, - there MUST be an intersection between the values for a Route to be - accepted. For more information, refer to the Route specific Hostnames - documentation. - - Hostnames that are prefixed with a wildcard label (`*.`) are interpreted - as a suffix match. That means that a match for `*.example.com` would match - both `test.example.com`, and `foo.test.example.com`, but not `example.com`. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - name: - description: |- - Name is the name of the Listener. This name MUST be unique within a - Gateway. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - port: - description: |- - Port is the network port. Multiple listeners may use the - same port, subject to the Listener compatibility rules. - - Support: Core - format: int32 - maximum: 65535 - minimum: 1 - type: integer - protocol: - description: |- - Protocol specifies the network protocol this listener expects to receive. - - Support: Core - maxLength: 255 - minLength: 1 - pattern: ^[a-zA-Z0-9]([-a-zA-Z0-9]*[a-zA-Z0-9])?$|[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9]+$ - type: string - tls: - description: |- - TLS is the TLS configuration for the Listener. This field is required if - the Protocol field is "HTTPS" or "TLS". It is invalid to set this field - if the Protocol field is "HTTP", "TCP", or "UDP". - - The association of SNIs to Certificate defined in GatewayTLSConfig is - defined based on the Hostname field for this listener. - - The GatewayClass MUST use the longest matching SNI out of all - available certificates for any TLS handshake. - - Support: Core - properties: - certificateRefs: - description: |- - CertificateRefs contains a series of references to Kubernetes objects that - contains TLS certificates and private keys. These certificates are used to - establish a TLS handshake for requests that match the hostname of the - associated listener. - - A single CertificateRef to a Kubernetes Secret has "Core" support. - Implementations MAY choose to support attaching multiple certificates to - a Listener, but this behavior is implementation-specific. - - References to a resource in different namespace are invalid UNLESS there - is a ReferenceGrant in the target namespace that allows the certificate - to be attached. If a ReferenceGrant does not allow this reference, the - "ResolvedRefs" condition MUST be set to False for this listener with the - "RefNotPermitted" reason. - - This field is required to have at least one element when the mode is set - to "Terminate" (default) and is optional otherwise. - - CertificateRefs can reference to standard Kubernetes resources, i.e. - Secret, or implementation-specific custom resources. - - Support: Core - A single reference to a Kubernetes Secret of type kubernetes.io/tls - - Support: Implementation-specific (More than one reference or other resource types) - items: - description: |- - SecretObjectReference identifies an API object including its namespace, - defaulting to Secret. - - The API object must be valid in the cluster; the Group and Kind must - be registered in the cluster for this reference to be valid. - - References to objects with invalid Group and Kind are not valid, and must - be rejected by the implementation, with appropriate Conditions set - on the containing object. - properties: - group: - default: "" - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Secret - description: Kind is kind of the referent. For example - "Secret". - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the referenced object. When unspecified, the local - namespace is inferred. - - Note that when a namespace different than the local namespace is specified, - a ReferenceGrant object is required in the referent namespace to allow that - namespace's owner to accept the reference. See the ReferenceGrant - documentation for details. - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - required: - - name - type: object - maxItems: 64 - type: array - mode: - default: Terminate - description: |- - Mode defines the TLS behavior for the TLS session initiated by the client. - There are two possible modes: - - - Terminate: The TLS session between the downstream client and the - Gateway is terminated at the Gateway. This mode requires certificates - to be specified in some way, such as populating the certificateRefs - field. - - Passthrough: The TLS session is NOT terminated by the Gateway. This - implies that the Gateway can't decipher the TLS stream except for - the ClientHello message of the TLS protocol. The certificateRefs field - is ignored in this mode. - - Support: Core - enum: - - Terminate - - Passthrough - type: string - options: - additionalProperties: - description: |- - AnnotationValue is the value of an annotation in Gateway API. This is used - for validation of maps such as TLS options. This roughly matches Kubernetes - annotation validation, although the length validation in that case is based - on the entire size of the annotations struct. - maxLength: 4096 - minLength: 0 - type: string - description: |- - Options are a list of key/value pairs to enable extended TLS - configuration for each implementation. For example, configuring the - minimum TLS version or supported cipher suites. - - A set of common keys MAY be defined by the API in the future. To avoid - any ambiguity, implementation-specific definitions MUST use - domain-prefixed names, such as `example.com/my-custom-option`. - Un-prefixed names are reserved for key names defined by Gateway API. - - Support: Implementation-specific - maxProperties: 16 - type: object - type: object - x-kubernetes-validations: - - message: certificateRefs or options must be specified when - mode is Terminate - rule: 'self.mode == ''Terminate'' ? size(self.certificateRefs) - > 0 || size(self.options) > 0 : true' - required: - - name - - port - - protocol - type: object - maxItems: 64 - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - x-kubernetes-validations: - - message: tls must not be specified for protocols ['HTTP', 'TCP', - 'UDP'] - rule: 'self.all(l, l.protocol in [''HTTP'', ''TCP'', ''UDP''] ? - !has(l.tls) : true)' - - message: tls mode must be Terminate for protocol HTTPS - rule: 'self.all(l, (l.protocol == ''HTTPS'' && has(l.tls)) ? (l.tls.mode - == '''' || l.tls.mode == ''Terminate'') : true)' - - message: hostname must not be specified for protocols ['TCP', 'UDP'] - rule: 'self.all(l, l.protocol in [''TCP'', ''UDP''] ? (!has(l.hostname) - || l.hostname == '''') : true)' - - message: Listener name must be unique within the Gateway - rule: self.all(l1, self.exists_one(l2, l1.name == l2.name)) - - message: Combination of port, protocol and hostname must be unique - for each listener - rule: 'self.all(l1, self.exists_one(l2, l1.port == l2.port && l1.protocol - == l2.protocol && (has(l1.hostname) && has(l2.hostname) ? l1.hostname - == l2.hostname : !has(l1.hostname) && !has(l2.hostname))))' - required: - - gatewayClassName - - listeners - type: object - status: - default: - conditions: - - lastTransitionTime: "1970-01-01T00:00:00Z" - message: Waiting for controller - reason: Pending - status: Unknown - type: Accepted - - lastTransitionTime: "1970-01-01T00:00:00Z" - message: Waiting for controller - reason: Pending - status: Unknown - type: Programmed - description: Status defines the current state of Gateway. - properties: - addresses: - description: |+ - Addresses lists the network addresses that have been bound to the - Gateway. - - This list may differ from the addresses provided in the spec under some - conditions: - - * no addresses are specified, all addresses are dynamically assigned - * a combination of specified and dynamic addresses are assigned - * a specified address was unusable (e.g. already in use) - - items: - description: GatewayStatusAddress describes a network address that - is bound to a Gateway. - oneOf: - - properties: - type: - enum: - - IPAddress - value: - anyOf: - - format: ipv4 - - format: ipv6 - - properties: - type: - not: - enum: - - IPAddress - properties: - type: - default: IPAddress - description: Type of the address. - maxLength: 253 - minLength: 1 - pattern: ^Hostname|IPAddress|NamedAddress|[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ - type: string - value: - description: |- - Value of the address. The validity of the values will depend - on the type and support by the controller. - - Examples: `1.2.3.4`, `128::1`, `my-ip-address`. - maxLength: 253 - minLength: 1 - type: string - required: - - value - type: object - x-kubernetes-validations: - - message: Hostname value must only contain valid characters (matching - ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$) - rule: 'self.type == ''Hostname'' ? self.value.matches(r"""^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$"""): - true' - maxItems: 16 - type: array - conditions: - default: - - lastTransitionTime: "1970-01-01T00:00:00Z" - message: Waiting for controller - reason: Pending - status: Unknown - type: Accepted - - lastTransitionTime: "1970-01-01T00:00:00Z" - message: Waiting for controller - reason: Pending - status: Unknown - type: Programmed - description: |- - Conditions describe the current conditions of the Gateway. - - Implementations should prefer to express Gateway conditions - using the `GatewayConditionType` and `GatewayConditionReason` - constants so that operators and tools can converge on a common - vocabulary to describe Gateway state. - - Known condition types are: - - * "Accepted" - * "Programmed" - * "Ready" - items: - description: Condition contains details for one aspect of the current - state of this API Resource. - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: type of condition in CamelCase or in foo.example.com/CamelCase. - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - maxItems: 8 - type: array - x-kubernetes-list-map-keys: - - type - x-kubernetes-list-type: map - listeners: - description: Listeners provide status for each unique listener port - defined in the Spec. - items: - description: ListenerStatus is the status associated with a Listener. - properties: - attachedRoutes: - description: |- - AttachedRoutes represents the total number of Routes that have been - successfully attached to this Listener. - - Successful attachment of a Route to a Listener is based solely on the - combination of the AllowedRoutes field on the corresponding Listener - and the Route's ParentRefs field. A Route is successfully attached to - a Listener when it is selected by the Listener's AllowedRoutes field - AND the Route has a valid ParentRef selecting the whole Gateway - resource or a specific Listener as a parent resource (more detail on - attachment semantics can be found in the documentation on the various - Route kinds ParentRefs fields). Listener or Route status does not impact - successful attachment, i.e. the AttachedRoutes field count MUST be set - for Listeners with condition Accepted: false and MUST count successfully - attached Routes that may themselves have Accepted: false conditions. - - Uses for this field include troubleshooting Route attachment and - measuring blast radius/impact of changes to a Listener. - format: int32 - type: integer - conditions: - description: Conditions describe the current condition of this - listener. - items: - description: Condition contains details for one aspect of - the current state of this API Resource. - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, - Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: type of condition in CamelCase or in foo.example.com/CamelCase. - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - maxItems: 8 - type: array - x-kubernetes-list-map-keys: - - type - x-kubernetes-list-type: map - name: - description: Name is the name of the Listener that this status - corresponds to. - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - supportedKinds: - description: |- - SupportedKinds is the list indicating the Kinds supported by this - listener. This MUST represent the kinds an implementation supports for - that Listener configuration. - - If kinds are specified in Spec that are not supported, they MUST NOT - appear in this list and an implementation MUST set the "ResolvedRefs" - condition to "False" with the "InvalidRouteKinds" reason. If both valid - and invalid Route kinds are specified, the implementation MUST - reference the valid Route kinds that have been specified. - items: - description: RouteGroupKind indicates the group and kind of - a Route resource. - properties: - group: - default: gateway.networking.k8s.io - description: Group is the group of the Route. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: Kind is the kind of the Route. - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - required: - - kind - type: object - maxItems: 8 - type: array - required: - - attachedRoutes - - conditions - - name - - supportedKinds - type: object - maxItems: 64 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - required: - - spec - type: object - served: true - storage: false - subresources: - status: {} -status: - acceptedNames: - kind: "" - plural: "" - conditions: null - storedVersions: null ---- -# -# config/crd/standard/gateway.networking.k8s.io_grpcroutes.yaml -# -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - api-approved.kubernetes.io: https://github.com/kubernetes-sigs/gateway-api/pull/3328 - gateway.networking.k8s.io/bundle-version: v1.2.0 - gateway.networking.k8s.io/channel: standard - creationTimestamp: null - name: grpcroutes.gateway.networking.k8s.io -spec: - group: gateway.networking.k8s.io - names: - categories: - - gateway-api - kind: GRPCRoute - listKind: GRPCRouteList - plural: grpcroutes - singular: grpcroute - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.hostnames - name: Hostnames - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1 - schema: - openAPIV3Schema: - description: |- - GRPCRoute provides a way to route gRPC requests. This includes the capability - to match requests by hostname, gRPC service, gRPC method, or HTTP/2 header. - Filters can be used to specify additional processing steps. Backends specify - where matching requests will be routed. - - GRPCRoute falls under extended support within the Gateway API. Within the - following specification, the word "MUST" indicates that an implementation - supporting GRPCRoute must conform to the indicated requirement, but an - implementation not supporting this route type need not follow the requirement - unless explicitly indicated. - - Implementations supporting `GRPCRoute` with the `HTTPS` `ProtocolType` MUST - accept HTTP/2 connections without an initial upgrade from HTTP/1.1, i.e. via - ALPN. If the implementation does not support this, then it MUST set the - "Accepted" condition to "False" for the affected listener with a reason of - "UnsupportedProtocol". Implementations MAY also accept HTTP/2 connections - with an upgrade from HTTP/1. - - Implementations supporting `GRPCRoute` with the `HTTP` `ProtocolType` MUST - support HTTP/2 over cleartext TCP (h2c, - https://www.rfc-editor.org/rfc/rfc7540#section-3.1) without an initial - upgrade from HTTP/1.1, i.e. with prior knowledge - (https://www.rfc-editor.org/rfc/rfc7540#section-3.4). If the implementation - does not support this, then it MUST set the "Accepted" condition to "False" - for the affected listener with a reason of "UnsupportedProtocol". - Implementations MAY also accept HTTP/2 connections with an upgrade from - HTTP/1, i.e. without prior knowledge. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: Spec defines the desired state of GRPCRoute. - properties: - hostnames: - description: |- - Hostnames defines a set of hostnames to match against the GRPC - Host header to select a GRPCRoute to process the request. This matches - the RFC 1123 definition of a hostname with 2 notable exceptions: - - 1. IPs are not allowed. - 2. A hostname may be prefixed with a wildcard label (`*.`). The wildcard - label MUST appear by itself as the first label. - - If a hostname is specified by both the Listener and GRPCRoute, there - MUST be at least one intersecting hostname for the GRPCRoute to be - attached to the Listener. For example: - - * A Listener with `test.example.com` as the hostname matches GRPCRoutes - that have either not specified any hostnames, or have specified at - least one of `test.example.com` or `*.example.com`. - * A Listener with `*.example.com` as the hostname matches GRPCRoutes - that have either not specified any hostnames or have specified at least - one hostname that matches the Listener hostname. For example, - `test.example.com` and `*.example.com` would both match. On the other - hand, `example.com` and `test.example.net` would not match. - - Hostnames that are prefixed with a wildcard label (`*.`) are interpreted - as a suffix match. That means that a match for `*.example.com` would match - both `test.example.com`, and `foo.test.example.com`, but not `example.com`. - - If both the Listener and GRPCRoute have specified hostnames, any - GRPCRoute hostnames that do not match the Listener hostname MUST be - ignored. For example, if a Listener specified `*.example.com`, and the - GRPCRoute specified `test.example.com` and `test.example.net`, - `test.example.net` MUST NOT be considered for a match. - - If both the Listener and GRPCRoute have specified hostnames, and none - match with the criteria above, then the GRPCRoute MUST NOT be accepted by - the implementation. The implementation MUST raise an 'Accepted' Condition - with a status of `False` in the corresponding RouteParentStatus. - - If a Route (A) of type HTTPRoute or GRPCRoute is attached to a - Listener and that listener already has another Route (B) of the other - type attached and the intersection of the hostnames of A and B is - non-empty, then the implementation MUST accept exactly one of these two - routes, determined by the following criteria, in order: - - * The oldest Route based on creation timestamp. - * The Route appearing first in alphabetical order by - "{namespace}/{name}". - - The rejected Route MUST raise an 'Accepted' condition with a status of - 'False' in the corresponding RouteParentStatus. - - Support: Core - items: - description: |- - Hostname is the fully qualified domain name of a network host. This matches - the RFC 1123 definition of a hostname with 2 notable exceptions: - - 1. IPs are not allowed. - 2. A hostname may be prefixed with a wildcard label (`*.`). The wildcard - label must appear by itself as the first label. - - Hostname can be "precise" which is a domain name without the terminating - dot of a network host (e.g. "foo.example.com") or "wildcard", which is a - domain name prefixed with a single wildcard label (e.g. `*.example.com`). - - Note that as per RFC1035 and RFC1123, a *label* must consist of lower case - alphanumeric characters or '-', and must start and end with an alphanumeric - character. No other punctuation is allowed. - maxLength: 253 - minLength: 1 - pattern: ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - maxItems: 16 - type: array - parentRefs: - description: |+ - ParentRefs references the resources (usually Gateways) that a Route wants - to be attached to. Note that the referenced parent resource needs to - allow this for the attachment to be complete. For Gateways, that means - the Gateway needs to allow attachment from Routes of this kind and - namespace. For Services, that means the Service must either be in the same - namespace for a "producer" route, or the mesh implementation must support - and allow "consumer" routes for the referenced Service. ReferenceGrant is - not applicable for governing ParentRefs to Services - it is not possible to - create a "producer" route for a Service in a different namespace from the - Route. - - There are two kinds of parent resources with "Core" support: - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - This API may be extended in the future to support additional kinds of parent - resources. - - ParentRefs must be _distinct_. This means either that: - - * They select different objects. If this is the case, then parentRef - entries are distinct. In terms of fields, this means that the - multi-part key defined by `group`, `kind`, `namespace`, and `name` must - be unique across all parentRef entries in the Route. - * They do not select different objects, but for each optional field used, - each ParentRef that selects the same object must set the same set of - optional fields to different values. If one ParentRef sets a - combination of optional fields, all must set the same combination. - - Some examples: - - * If one ParentRef sets `sectionName`, all ParentRefs referencing the - same object must also set `sectionName`. - * If one ParentRef sets `port`, all ParentRefs referencing the same - object must also set `port`. - * If one ParentRef sets `sectionName` and `port`, all ParentRefs - referencing the same object must also set `sectionName` and `port`. - - It is possible to separately reference multiple distinct objects that may - be collapsed by an implementation. For example, some implementations may - choose to merge compatible Gateway Listeners together. If that is the - case, the list of routes attached to those resources should also be - merged. - - Note that for ParentRefs that cross namespace boundaries, there are specific - rules. Cross-namespace references are only valid if they are explicitly - allowed by something in the namespace they are referring to. For example, - Gateway has the AllowedRoutes field, and ReferenceGrant provides a - generic way to enable other kinds of cross-namespace reference. - - - - - - - items: - description: |- - ParentReference identifies an API object (usually a Gateway) that can be considered - a parent of this resource (usually a route). There are two kinds of parent resources - with "Core" support: - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - This API may be extended in the future to support additional kinds of parent - resources. - - The API object must be valid in the cluster; the Group and Kind must - be registered in the cluster for this reference to be valid. - properties: - group: - default: gateway.networking.k8s.io - description: |- - Group is the group of the referent. - When unspecified, "gateway.networking.k8s.io" is inferred. - To set the core API group (such as for a "Service" kind referent), - Group must be explicitly set to "" (empty string). - - Support: Core - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Gateway - description: |- - Kind is kind of the referent. - - There are two kinds of parent resources with "Core" support: - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - Support for other resources is Implementation-Specific. - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: |- - Name is the name of the referent. - - Support: Core - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the referent. When unspecified, this refers - to the local namespace of the Route. - - Note that there are specific rules for ParentRefs which cross namespace - boundaries. Cross-namespace references are only valid if they are explicitly - allowed by something in the namespace they are referring to. For example: - Gateway has the AllowedRoutes field, and ReferenceGrant provides a - generic way to enable any other kind of cross-namespace reference. - - - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port is the network port this Route targets. It can be interpreted - differently based on the type of parent resource. - - When the parent resource is a Gateway, this targets all listeners - listening on the specified port that also support this kind of Route(and - select this Route). It's not recommended to set `Port` unless the - networking behaviors specified in a Route must apply to a specific port - as opposed to a listener(s) whose port(s) may be changed. When both Port - and SectionName are specified, the name and port of the selected listener - must match both specified values. - - - - Implementations MAY choose to support other parent resources. - Implementations supporting other types of parent resources MUST clearly - document how/if Port is interpreted. - - For the purpose of status, an attachment is considered successful as - long as the parent resource accepts it partially. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment - from the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, - the Route MUST be considered detached from the Gateway. - - Support: Extended - format: int32 - maximum: 65535 - minimum: 1 - type: integer - sectionName: - description: |- - SectionName is the name of a section within the target resource. In the - following resources, SectionName is interpreted as the following: - - * Gateway: Listener name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - * Service: Port name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - - Implementations MAY choose to support attaching Routes to other resources. - If that is the case, they MUST clearly document how SectionName is - interpreted. - - When unspecified (empty string), this will reference the entire resource. - For the purpose of status, an attachment is considered successful if at - least one section in the parent resource accepts it. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from - the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, the - Route MUST be considered detached from the Gateway. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - required: - - name - type: object - maxItems: 32 - type: array - x-kubernetes-validations: - - message: sectionName must be specified when parentRefs includes - 2 or more references to the same parent - rule: 'self.all(p1, self.all(p2, p1.group == p2.group && p1.kind - == p2.kind && p1.name == p2.name && (((!has(p1.__namespace__) - || p1.__namespace__ == '''') && (!has(p2.__namespace__) || p2.__namespace__ - == '''')) || (has(p1.__namespace__) && has(p2.__namespace__) && - p1.__namespace__ == p2.__namespace__ )) ? ((!has(p1.sectionName) - || p1.sectionName == '''') == (!has(p2.sectionName) || p2.sectionName - == '''')) : true))' - - message: sectionName must be unique when parentRefs includes 2 or - more references to the same parent - rule: self.all(p1, self.exists_one(p2, p1.group == p2.group && p1.kind - == p2.kind && p1.name == p2.name && (((!has(p1.__namespace__) - || p1.__namespace__ == '') && (!has(p2.__namespace__) || p2.__namespace__ - == '')) || (has(p1.__namespace__) && has(p2.__namespace__) && - p1.__namespace__ == p2.__namespace__ )) && (((!has(p1.sectionName) - || p1.sectionName == '') && (!has(p2.sectionName) || p2.sectionName - == '')) || (has(p1.sectionName) && has(p2.sectionName) && p1.sectionName - == p2.sectionName)))) - rules: - description: |+ - Rules are a list of GRPC matchers, filters and actions. - - items: - description: |- - GRPCRouteRule defines the semantics for matching a gRPC request based on - conditions (matches), processing it (filters), and forwarding the request to - an API object (backendRefs). - properties: - backendRefs: - description: |- - BackendRefs defines the backend(s) where matching requests should be - sent. - - Failure behavior here depends on how many BackendRefs are specified and - how many are invalid. - - If *all* entries in BackendRefs are invalid, and there are also no filters - specified in this route rule, *all* traffic which matches this rule MUST - receive an `UNAVAILABLE` status. - - See the GRPCBackendRef definition for the rules about what makes a single - GRPCBackendRef invalid. - - When a GRPCBackendRef is invalid, `UNAVAILABLE` statuses MUST be returned for - requests that would have otherwise been routed to an invalid backend. If - multiple backends are specified, and some are invalid, the proportion of - requests that would otherwise have been routed to an invalid backend - MUST receive an `UNAVAILABLE` status. - - For example, if two backends are specified with equal weights, and one is - invalid, 50 percent of traffic MUST receive an `UNAVAILABLE` status. - Implementations may choose how that 50 percent is determined. - - Support: Core for Kubernetes Service - - Support: Implementation-specific for any other resource - - Support for weight: Core - items: - description: |- - GRPCBackendRef defines how a GRPCRoute forwards a gRPC request. - - Note that when a namespace different than the local namespace is specified, a - ReferenceGrant object is required in the referent namespace to allow that - namespace's owner to accept the reference. See the ReferenceGrant - documentation for details. - - - - When the BackendRef points to a Kubernetes Service, implementations SHOULD - honor the appProtocol field if it is set for the target Service Port. - - Implementations supporting appProtocol SHOULD recognize the Kubernetes - Standard Application Protocols defined in KEP-3726. - - If a Service appProtocol isn't specified, an implementation MAY infer the - backend protocol through its own means. Implementations MAY infer the - protocol from the Route type referring to the backend Service. - - If a Route is not able to send traffic to the backend using the specified - protocol then the backend is considered invalid. Implementations MUST set the - "ResolvedRefs" condition to "False" with the "UnsupportedProtocol" reason. - - - properties: - filters: - description: |- - Filters defined at this level MUST be executed if and only if the - request is being forwarded to the backend defined here. - - Support: Implementation-specific (For broader support of filters, use the - Filters field in GRPCRouteRule.) - items: - description: |- - GRPCRouteFilter defines processing steps that must be completed during the - request or response lifecycle. GRPCRouteFilters are meant as an extension - point to express processing that may be done in Gateway implementations. Some - examples include request or response modification, implementing - authentication strategies, rate-limiting, and traffic shaping. API - guarantee/conformance is defined based on the type of the filter. - properties: - extensionRef: - description: |- - ExtensionRef is an optional, implementation-specific extension to the - "filter" behavior. For example, resource "myroutefilter" in group - "networking.example.net"). ExtensionRef MUST NOT be used for core and - extended filters. - - Support: Implementation-specific - - This filter can be used multiple times within the same rule. - properties: - group: - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: Kind is kind of the referent. For - example "HTTPRoute" or "Service". - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - required: - - group - - kind - - name - type: object - requestHeaderModifier: - description: |- - RequestHeaderModifier defines a schema for a filter that modifies request - headers. - - Support: Core - properties: - add: - description: |- - Add adds the given header(s) (name, value) to the request - before the action. It appends to any existing values associated - with the header name. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - add: - - name: "my-header" - value: "bar,baz" - - Output: - GET /foo HTTP/1.1 - my-header: foo,bar,baz - items: - description: HTTPHeader represents an HTTP - Header name and value as defined by RFC - 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP - Header to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - remove: - description: |- - Remove the given header(s) from the HTTP request before the action. The - value of Remove is a list of HTTP header names. Note that the header - names are case-insensitive (see - https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). - - Input: - GET /foo HTTP/1.1 - my-header1: foo - my-header2: bar - my-header3: baz - - Config: - remove: ["my-header1", "my-header3"] - - Output: - GET /foo HTTP/1.1 - my-header2: bar - items: - type: string - maxItems: 16 - type: array - x-kubernetes-list-type: set - set: - description: |- - Set overwrites the request with the given header (name, value) - before the action. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - set: - - name: "my-header" - value: "bar" - - Output: - GET /foo HTTP/1.1 - my-header: bar - items: - description: HTTPHeader represents an HTTP - Header name and value as defined by RFC - 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP - Header to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - requestMirror: - description: |+ - RequestMirror defines a schema for a filter that mirrors requests. - Requests are sent to the specified destination, but responses from - that destination are ignored. - - This filter can be used multiple times within the same rule. Note that - not all implementations will be able to support mirroring to multiple - backends. - - Support: Extended - - properties: - backendRef: - description: |- - BackendRef references a resource where mirrored requests are sent. - - Mirrored requests must be sent only to a single destination endpoint - within this BackendRef, irrespective of how many endpoints are present - within this BackendRef. - - If the referent cannot be found, this BackendRef is invalid and must be - dropped from the Gateway. The controller must ensure the "ResolvedRefs" - condition on the Route status is set to `status: False` and not configure - this backend in the underlying implementation. - - If there is a cross-namespace reference to an *existing* object - that is not allowed by a ReferenceGrant, the controller must ensure the - "ResolvedRefs" condition on the Route is set to `status: False`, - with the "RefNotPermitted" reason and not configure this backend in the - underlying implementation. - - In either error case, the Message of the `ResolvedRefs` Condition - should be used to provide more detail about the problem. - - Support: Extended for Kubernetes Service - - Support: Implementation-specific for any other resource - properties: - group: - default: "" - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Service - description: |- - Kind is the Kubernetes resource kind of the referent. For example - "Service". - - Defaults to "Service" when not specified. - - ExternalName services can refer to CNAME DNS records that may live - outside of the cluster and as such are difficult to reason about in - terms of conformance. They also may not be safe to forward to (see - CVE-2021-25740 for more information). Implementations SHOULD NOT - support ExternalName Services. - - Support: Core (Services with a type other than ExternalName) - - Support: Implementation-specific (Services with type ExternalName) - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the backend. When unspecified, the local - namespace is inferred. - - Note that when a namespace different than the local namespace is specified, - a ReferenceGrant object is required in the referent namespace to allow that - namespace's owner to accept the reference. See the ReferenceGrant - documentation for details. - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port specifies the destination port number to use for this resource. - Port is required when the referent is a Kubernetes Service. In this - case, the port number is the service port number, not the target port. - For other resources, destination port might be derived from the referent - resource or this field. - format: int32 - maximum: 65535 - minimum: 1 - type: integer - required: - - name - type: object - x-kubernetes-validations: - - message: Must have port for Service reference - rule: '(size(self.group) == 0 && self.kind - == ''Service'') ? has(self.port) : true' - required: - - backendRef - type: object - responseHeaderModifier: - description: |- - ResponseHeaderModifier defines a schema for a filter that modifies response - headers. - - Support: Extended - properties: - add: - description: |- - Add adds the given header(s) (name, value) to the request - before the action. It appends to any existing values associated - with the header name. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - add: - - name: "my-header" - value: "bar,baz" - - Output: - GET /foo HTTP/1.1 - my-header: foo,bar,baz - items: - description: HTTPHeader represents an HTTP - Header name and value as defined by RFC - 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP - Header to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - remove: - description: |- - Remove the given header(s) from the HTTP request before the action. The - value of Remove is a list of HTTP header names. Note that the header - names are case-insensitive (see - https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). - - Input: - GET /foo HTTP/1.1 - my-header1: foo - my-header2: bar - my-header3: baz - - Config: - remove: ["my-header1", "my-header3"] - - Output: - GET /foo HTTP/1.1 - my-header2: bar - items: - type: string - maxItems: 16 - type: array - x-kubernetes-list-type: set - set: - description: |- - Set overwrites the request with the given header (name, value) - before the action. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - set: - - name: "my-header" - value: "bar" - - Output: - GET /foo HTTP/1.1 - my-header: bar - items: - description: HTTPHeader represents an HTTP - Header name and value as defined by RFC - 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP - Header to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: - description: |+ - Type identifies the type of filter to apply. As with other API fields, - types are classified into three conformance levels: - - - Core: Filter types and their corresponding configuration defined by - "Support: Core" in this package, e.g. "RequestHeaderModifier". All - implementations supporting GRPCRoute MUST support core filters. - - - Extended: Filter types and their corresponding configuration defined by - "Support: Extended" in this package, e.g. "RequestMirror". Implementers - are encouraged to support extended filters. - - - Implementation-specific: Filters that are defined and supported by specific vendors. - In the future, filters showing convergence in behavior across multiple - implementations will be considered for inclusion in extended or core - conformance levels. Filter-specific configuration for such filters - is specified using the ExtensionRef field. `Type` MUST be set to - "ExtensionRef" for custom filters. - - Implementers are encouraged to define custom implementation types to - extend the core API with implementation-specific behavior. - - If a reference to a custom filter type cannot be resolved, the filter - MUST NOT be skipped. Instead, requests that would have been processed by - that filter MUST receive a HTTP error response. - - enum: - - ResponseHeaderModifier - - RequestHeaderModifier - - RequestMirror - - ExtensionRef - type: string - required: - - type - type: object - x-kubernetes-validations: - - message: filter.requestHeaderModifier must be nil - if the filter.type is not RequestHeaderModifier - rule: '!(has(self.requestHeaderModifier) && self.type - != ''RequestHeaderModifier'')' - - message: filter.requestHeaderModifier must be specified - for RequestHeaderModifier filter.type - rule: '!(!has(self.requestHeaderModifier) && self.type - == ''RequestHeaderModifier'')' - - message: filter.responseHeaderModifier must be nil - if the filter.type is not ResponseHeaderModifier - rule: '!(has(self.responseHeaderModifier) && self.type - != ''ResponseHeaderModifier'')' - - message: filter.responseHeaderModifier must be specified - for ResponseHeaderModifier filter.type - rule: '!(!has(self.responseHeaderModifier) && self.type - == ''ResponseHeaderModifier'')' - - message: filter.requestMirror must be nil if the filter.type - is not RequestMirror - rule: '!(has(self.requestMirror) && self.type != ''RequestMirror'')' - - message: filter.requestMirror must be specified for - RequestMirror filter.type - rule: '!(!has(self.requestMirror) && self.type == - ''RequestMirror'')' - - message: filter.extensionRef must be nil if the filter.type - is not ExtensionRef - rule: '!(has(self.extensionRef) && self.type != ''ExtensionRef'')' - - message: filter.extensionRef must be specified for - ExtensionRef filter.type - rule: '!(!has(self.extensionRef) && self.type == ''ExtensionRef'')' - maxItems: 16 - type: array - x-kubernetes-validations: - - message: RequestHeaderModifier filter cannot be repeated - rule: self.filter(f, f.type == 'RequestHeaderModifier').size() - <= 1 - - message: ResponseHeaderModifier filter cannot be repeated - rule: self.filter(f, f.type == 'ResponseHeaderModifier').size() - <= 1 - group: - default: "" - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Service - description: |- - Kind is the Kubernetes resource kind of the referent. For example - "Service". - - Defaults to "Service" when not specified. - - ExternalName services can refer to CNAME DNS records that may live - outside of the cluster and as such are difficult to reason about in - terms of conformance. They also may not be safe to forward to (see - CVE-2021-25740 for more information). Implementations SHOULD NOT - support ExternalName Services. - - Support: Core (Services with a type other than ExternalName) - - Support: Implementation-specific (Services with type ExternalName) - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the backend. When unspecified, the local - namespace is inferred. - - Note that when a namespace different than the local namespace is specified, - a ReferenceGrant object is required in the referent namespace to allow that - namespace's owner to accept the reference. See the ReferenceGrant - documentation for details. - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port specifies the destination port number to use for this resource. - Port is required when the referent is a Kubernetes Service. In this - case, the port number is the service port number, not the target port. - For other resources, destination port might be derived from the referent - resource or this field. - format: int32 - maximum: 65535 - minimum: 1 - type: integer - weight: - default: 1 - description: |- - Weight specifies the proportion of requests forwarded to the referenced - backend. This is computed as weight/(sum of all weights in this - BackendRefs list). For non-zero values, there may be some epsilon from - the exact proportion defined here depending on the precision an - implementation supports. Weight is not a percentage and the sum of - weights does not need to equal 100. - - If only one backend is specified and it has a weight greater than 0, 100% - of the traffic is forwarded to that backend. If weight is set to 0, no - traffic should be forwarded for this entry. If unspecified, weight - defaults to 1. - - Support for this field varies based on the context where used. - format: int32 - maximum: 1000000 - minimum: 0 - type: integer - required: - - name - type: object - x-kubernetes-validations: - - message: Must have port for Service reference - rule: '(size(self.group) == 0 && self.kind == ''Service'') - ? has(self.port) : true' - maxItems: 16 - type: array - filters: - description: |- - Filters define the filters that are applied to requests that match - this rule. - - The effects of ordering of multiple behaviors are currently unspecified. - This can change in the future based on feedback during the alpha stage. - - Conformance-levels at this level are defined based on the type of filter: - - - ALL core filters MUST be supported by all implementations that support - GRPCRoute. - - Implementers are encouraged to support extended filters. - - Implementation-specific custom filters have no API guarantees across - implementations. - - Specifying the same filter multiple times is not supported unless explicitly - indicated in the filter. - - If an implementation can not support a combination of filters, it must clearly - document that limitation. In cases where incompatible or unsupported - filters are specified and cause the `Accepted` condition to be set to status - `False`, implementations may use the `IncompatibleFilters` reason to specify - this configuration error. - - Support: Core - items: - description: |- - GRPCRouteFilter defines processing steps that must be completed during the - request or response lifecycle. GRPCRouteFilters are meant as an extension - point to express processing that may be done in Gateway implementations. Some - examples include request or response modification, implementing - authentication strategies, rate-limiting, and traffic shaping. API - guarantee/conformance is defined based on the type of the filter. - properties: - extensionRef: - description: |- - ExtensionRef is an optional, implementation-specific extension to the - "filter" behavior. For example, resource "myroutefilter" in group - "networking.example.net"). ExtensionRef MUST NOT be used for core and - extended filters. - - Support: Implementation-specific - - This filter can be used multiple times within the same rule. - properties: - group: - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: Kind is kind of the referent. For example - "HTTPRoute" or "Service". - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - required: - - group - - kind - - name - type: object - requestHeaderModifier: - description: |- - RequestHeaderModifier defines a schema for a filter that modifies request - headers. - - Support: Core - properties: - add: - description: |- - Add adds the given header(s) (name, value) to the request - before the action. It appends to any existing values associated - with the header name. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - add: - - name: "my-header" - value: "bar,baz" - - Output: - GET /foo HTTP/1.1 - my-header: foo,bar,baz - items: - description: HTTPHeader represents an HTTP Header - name and value as defined by RFC 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP Header - to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - remove: - description: |- - Remove the given header(s) from the HTTP request before the action. The - value of Remove is a list of HTTP header names. Note that the header - names are case-insensitive (see - https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). - - Input: - GET /foo HTTP/1.1 - my-header1: foo - my-header2: bar - my-header3: baz - - Config: - remove: ["my-header1", "my-header3"] - - Output: - GET /foo HTTP/1.1 - my-header2: bar - items: - type: string - maxItems: 16 - type: array - x-kubernetes-list-type: set - set: - description: |- - Set overwrites the request with the given header (name, value) - before the action. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - set: - - name: "my-header" - value: "bar" - - Output: - GET /foo HTTP/1.1 - my-header: bar - items: - description: HTTPHeader represents an HTTP Header - name and value as defined by RFC 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP Header - to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - requestMirror: - description: |+ - RequestMirror defines a schema for a filter that mirrors requests. - Requests are sent to the specified destination, but responses from - that destination are ignored. - - This filter can be used multiple times within the same rule. Note that - not all implementations will be able to support mirroring to multiple - backends. - - Support: Extended - - properties: - backendRef: - description: |- - BackendRef references a resource where mirrored requests are sent. - - Mirrored requests must be sent only to a single destination endpoint - within this BackendRef, irrespective of how many endpoints are present - within this BackendRef. - - If the referent cannot be found, this BackendRef is invalid and must be - dropped from the Gateway. The controller must ensure the "ResolvedRefs" - condition on the Route status is set to `status: False` and not configure - this backend in the underlying implementation. - - If there is a cross-namespace reference to an *existing* object - that is not allowed by a ReferenceGrant, the controller must ensure the - "ResolvedRefs" condition on the Route is set to `status: False`, - with the "RefNotPermitted" reason and not configure this backend in the - underlying implementation. - - In either error case, the Message of the `ResolvedRefs` Condition - should be used to provide more detail about the problem. - - Support: Extended for Kubernetes Service - - Support: Implementation-specific for any other resource - properties: - group: - default: "" - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Service - description: |- - Kind is the Kubernetes resource kind of the referent. For example - "Service". - - Defaults to "Service" when not specified. - - ExternalName services can refer to CNAME DNS records that may live - outside of the cluster and as such are difficult to reason about in - terms of conformance. They also may not be safe to forward to (see - CVE-2021-25740 for more information). Implementations SHOULD NOT - support ExternalName Services. - - Support: Core (Services with a type other than ExternalName) - - Support: Implementation-specific (Services with type ExternalName) - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the backend. When unspecified, the local - namespace is inferred. - - Note that when a namespace different than the local namespace is specified, - a ReferenceGrant object is required in the referent namespace to allow that - namespace's owner to accept the reference. See the ReferenceGrant - documentation for details. - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port specifies the destination port number to use for this resource. - Port is required when the referent is a Kubernetes Service. In this - case, the port number is the service port number, not the target port. - For other resources, destination port might be derived from the referent - resource or this field. - format: int32 - maximum: 65535 - minimum: 1 - type: integer - required: - - name - type: object - x-kubernetes-validations: - - message: Must have port for Service reference - rule: '(size(self.group) == 0 && self.kind == ''Service'') - ? has(self.port) : true' - required: - - backendRef - type: object - responseHeaderModifier: - description: |- - ResponseHeaderModifier defines a schema for a filter that modifies response - headers. - - Support: Extended - properties: - add: - description: |- - Add adds the given header(s) (name, value) to the request - before the action. It appends to any existing values associated - with the header name. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - add: - - name: "my-header" - value: "bar,baz" - - Output: - GET /foo HTTP/1.1 - my-header: foo,bar,baz - items: - description: HTTPHeader represents an HTTP Header - name and value as defined by RFC 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP Header - to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - remove: - description: |- - Remove the given header(s) from the HTTP request before the action. The - value of Remove is a list of HTTP header names. Note that the header - names are case-insensitive (see - https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). - - Input: - GET /foo HTTP/1.1 - my-header1: foo - my-header2: bar - my-header3: baz - - Config: - remove: ["my-header1", "my-header3"] - - Output: - GET /foo HTTP/1.1 - my-header2: bar - items: - type: string - maxItems: 16 - type: array - x-kubernetes-list-type: set - set: - description: |- - Set overwrites the request with the given header (name, value) - before the action. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - set: - - name: "my-header" - value: "bar" - - Output: - GET /foo HTTP/1.1 - my-header: bar - items: - description: HTTPHeader represents an HTTP Header - name and value as defined by RFC 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP Header - to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: - description: |+ - Type identifies the type of filter to apply. As with other API fields, - types are classified into three conformance levels: - - - Core: Filter types and their corresponding configuration defined by - "Support: Core" in this package, e.g. "RequestHeaderModifier". All - implementations supporting GRPCRoute MUST support core filters. - - - Extended: Filter types and their corresponding configuration defined by - "Support: Extended" in this package, e.g. "RequestMirror". Implementers - are encouraged to support extended filters. - - - Implementation-specific: Filters that are defined and supported by specific vendors. - In the future, filters showing convergence in behavior across multiple - implementations will be considered for inclusion in extended or core - conformance levels. Filter-specific configuration for such filters - is specified using the ExtensionRef field. `Type` MUST be set to - "ExtensionRef" for custom filters. - - Implementers are encouraged to define custom implementation types to - extend the core API with implementation-specific behavior. - - If a reference to a custom filter type cannot be resolved, the filter - MUST NOT be skipped. Instead, requests that would have been processed by - that filter MUST receive a HTTP error response. - - enum: - - ResponseHeaderModifier - - RequestHeaderModifier - - RequestMirror - - ExtensionRef - type: string - required: - - type - type: object - x-kubernetes-validations: - - message: filter.requestHeaderModifier must be nil if the - filter.type is not RequestHeaderModifier - rule: '!(has(self.requestHeaderModifier) && self.type != - ''RequestHeaderModifier'')' - - message: filter.requestHeaderModifier must be specified - for RequestHeaderModifier filter.type - rule: '!(!has(self.requestHeaderModifier) && self.type == - ''RequestHeaderModifier'')' - - message: filter.responseHeaderModifier must be nil if the - filter.type is not ResponseHeaderModifier - rule: '!(has(self.responseHeaderModifier) && self.type != - ''ResponseHeaderModifier'')' - - message: filter.responseHeaderModifier must be specified - for ResponseHeaderModifier filter.type - rule: '!(!has(self.responseHeaderModifier) && self.type - == ''ResponseHeaderModifier'')' - - message: filter.requestMirror must be nil if the filter.type - is not RequestMirror - rule: '!(has(self.requestMirror) && self.type != ''RequestMirror'')' - - message: filter.requestMirror must be specified for RequestMirror - filter.type - rule: '!(!has(self.requestMirror) && self.type == ''RequestMirror'')' - - message: filter.extensionRef must be nil if the filter.type - is not ExtensionRef - rule: '!(has(self.extensionRef) && self.type != ''ExtensionRef'')' - - message: filter.extensionRef must be specified for ExtensionRef - filter.type - rule: '!(!has(self.extensionRef) && self.type == ''ExtensionRef'')' - maxItems: 16 - type: array - x-kubernetes-validations: - - message: RequestHeaderModifier filter cannot be repeated - rule: self.filter(f, f.type == 'RequestHeaderModifier').size() - <= 1 - - message: ResponseHeaderModifier filter cannot be repeated - rule: self.filter(f, f.type == 'ResponseHeaderModifier').size() - <= 1 - matches: - description: |- - Matches define conditions used for matching the rule against incoming - gRPC requests. Each match is independent, i.e. this rule will be matched - if **any** one of the matches is satisfied. - - For example, take the following matches configuration: - - ``` - matches: - - method: - service: foo.bar - headers: - values: - version: 2 - - method: - service: foo.bar.v2 - ``` - - For a request to match against this rule, it MUST satisfy - EITHER of the two conditions: - - - service of foo.bar AND contains the header `version: 2` - - service of foo.bar.v2 - - See the documentation for GRPCRouteMatch on how to specify multiple - match conditions to be ANDed together. - - If no matches are specified, the implementation MUST match every gRPC request. - - Proxy or Load Balancer routing configuration generated from GRPCRoutes - MUST prioritize rules based on the following criteria, continuing on - ties. Merging MUST not be done between GRPCRoutes and HTTPRoutes. - Precedence MUST be given to the rule with the largest number of: - - * Characters in a matching non-wildcard hostname. - * Characters in a matching hostname. - * Characters in a matching service. - * Characters in a matching method. - * Header matches. - - If ties still exist across multiple Routes, matching precedence MUST be - determined in order of the following criteria, continuing on ties: - - * The oldest Route based on creation timestamp. - * The Route appearing first in alphabetical order by - "{namespace}/{name}". - - If ties still exist within the Route that has been given precedence, - matching precedence MUST be granted to the first matching rule meeting - the above criteria. - items: - description: |- - GRPCRouteMatch defines the predicate used to match requests to a given - action. Multiple match types are ANDed together, i.e. the match will - evaluate to true only if all conditions are satisfied. - - For example, the match below will match a gRPC request only if its service - is `foo` AND it contains the `version: v1` header: - - ``` - matches: - - method: - type: Exact - service: "foo" - headers: - - name: "version" - value "v1" - - ``` - properties: - headers: - description: |- - Headers specifies gRPC request header matchers. Multiple match values are - ANDed together, meaning, a request MUST match all the specified headers - to select the route. - items: - description: |- - GRPCHeaderMatch describes how to select a gRPC route by matching gRPC request - headers. - properties: - name: - description: |- - Name is the name of the gRPC Header to be matched. - - If multiple entries specify equivalent header names, only the first - entry with an equivalent name MUST be considered for a match. Subsequent - entries with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - type: - default: Exact - description: Type specifies how to match against - the value of the header. - enum: - - Exact - - RegularExpression - type: string - value: - description: Value is the value of the gRPC Header - to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - method: - description: |- - Method specifies a gRPC request service/method matcher. If this field is - not specified, all services and methods will match. - properties: - method: - description: |- - Value of the method to match against. If left empty or omitted, will - match all services. - - At least one of Service and Method MUST be a non-empty string. - maxLength: 1024 - type: string - service: - description: |- - Value of the service to match against. If left empty or omitted, will - match any service. - - At least one of Service and Method MUST be a non-empty string. - maxLength: 1024 - type: string - type: - default: Exact - description: |- - Type specifies how to match against the service and/or method. - Support: Core (Exact with service and method specified) - - Support: Implementation-specific (Exact with method specified but no service specified) - - Support: Implementation-specific (RegularExpression) - enum: - - Exact - - RegularExpression - type: string - type: object - x-kubernetes-validations: - - message: One or both of 'service' or 'method' must be - specified - rule: 'has(self.type) ? has(self.service) || has(self.method) - : true' - - message: service must only contain valid characters - (matching ^(?i)\.?[a-z_][a-z_0-9]*(\.[a-z_][a-z_0-9]*)*$) - rule: '(!has(self.type) || self.type == ''Exact'') && - has(self.service) ? self.service.matches(r"""^(?i)\.?[a-z_][a-z_0-9]*(\.[a-z_][a-z_0-9]*)*$"""): - true' - - message: method must only contain valid characters (matching - ^[A-Za-z_][A-Za-z_0-9]*$) - rule: '(!has(self.type) || self.type == ''Exact'') && - has(self.method) ? self.method.matches(r"""^[A-Za-z_][A-Za-z_0-9]*$"""): - true' - type: object - maxItems: 8 - type: array - type: object - maxItems: 16 - type: array - x-kubernetes-validations: - - message: While 16 rules and 64 matches per rule are allowed, the - total number of matches across all rules in a route must be less - than 128 - rule: '(self.size() > 0 ? (has(self[0].matches) ? self[0].matches.size() - : 0) : 0) + (self.size() > 1 ? (has(self[1].matches) ? self[1].matches.size() - : 0) : 0) + (self.size() > 2 ? (has(self[2].matches) ? self[2].matches.size() - : 0) : 0) + (self.size() > 3 ? (has(self[3].matches) ? self[3].matches.size() - : 0) : 0) + (self.size() > 4 ? (has(self[4].matches) ? self[4].matches.size() - : 0) : 0) + (self.size() > 5 ? (has(self[5].matches) ? self[5].matches.size() - : 0) : 0) + (self.size() > 6 ? (has(self[6].matches) ? self[6].matches.size() - : 0) : 0) + (self.size() > 7 ? (has(self[7].matches) ? self[7].matches.size() - : 0) : 0) + (self.size() > 8 ? (has(self[8].matches) ? self[8].matches.size() - : 0) : 0) + (self.size() > 9 ? (has(self[9].matches) ? self[9].matches.size() - : 0) : 0) + (self.size() > 10 ? (has(self[10].matches) ? self[10].matches.size() - : 0) : 0) + (self.size() > 11 ? (has(self[11].matches) ? self[11].matches.size() - : 0) : 0) + (self.size() > 12 ? (has(self[12].matches) ? self[12].matches.size() - : 0) : 0) + (self.size() > 13 ? (has(self[13].matches) ? self[13].matches.size() - : 0) : 0) + (self.size() > 14 ? (has(self[14].matches) ? self[14].matches.size() - : 0) : 0) + (self.size() > 15 ? (has(self[15].matches) ? self[15].matches.size() - : 0) : 0) <= 128' - type: object - status: - description: Status defines the current state of GRPCRoute. - properties: - parents: - description: |- - Parents is a list of parent resources (usually Gateways) that are - associated with the route, and the status of the route with respect to - each parent. When this route attaches to a parent, the controller that - manages the parent must add an entry to this list when the controller - first sees the route and should update the entry as appropriate when the - route or gateway is modified. - - Note that parent references that cannot be resolved by an implementation - of this API will not be added to this list. Implementations of this API - can only populate Route status for the Gateways/parent resources they are - responsible for. - - A maximum of 32 Gateways will be represented in this list. An empty list - means the route has not been attached to any Gateway. - items: - description: |- - RouteParentStatus describes the status of a route with respect to an - associated Parent. - properties: - conditions: - description: |- - Conditions describes the status of the route with respect to the Gateway. - Note that the route's availability is also subject to the Gateway's own - status conditions and listener status. - - If the Route's ParentRef specifies an existing Gateway that supports - Routes of this kind AND that Gateway's controller has sufficient access, - then that Gateway's controller MUST set the "Accepted" condition on the - Route, to indicate whether the route has been accepted or rejected by the - Gateway, and why. - - A Route MUST be considered "Accepted" if at least one of the Route's - rules is implemented by the Gateway. - - There are a number of cases where the "Accepted" condition may not be set - due to lack of controller visibility, that includes when: - - * The Route refers to a non-existent parent. - * The Route is of a type that the controller does not support. - * The Route is in a namespace the controller does not have access to. - items: - description: Condition contains details for one aspect of - the current state of this API Resource. - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, - Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: type of condition in CamelCase or in foo.example.com/CamelCase. - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - maxItems: 8 - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - type - x-kubernetes-list-type: map - controllerName: - description: |- - ControllerName is a domain/path string that indicates the name of the - controller that wrote this status. This corresponds with the - controllerName field on GatewayClass. - - Example: "example.net/gateway-controller". - - The format of this field is DOMAIN "/" PATH, where DOMAIN and PATH are - valid Kubernetes names - (https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names). - - Controllers MUST populate this field when writing status. Controllers should ensure that - entries to status populated with their ControllerName are cleaned up when they are no - longer necessary. - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ - type: string - parentRef: - description: |- - ParentRef corresponds with a ParentRef in the spec that this - RouteParentStatus struct describes the status of. - properties: - group: - default: gateway.networking.k8s.io - description: |- - Group is the group of the referent. - When unspecified, "gateway.networking.k8s.io" is inferred. - To set the core API group (such as for a "Service" kind referent), - Group must be explicitly set to "" (empty string). - - Support: Core - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Gateway - description: |- - Kind is kind of the referent. - - There are two kinds of parent resources with "Core" support: - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - Support for other resources is Implementation-Specific. - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: |- - Name is the name of the referent. - - Support: Core - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the referent. When unspecified, this refers - to the local namespace of the Route. - - Note that there are specific rules for ParentRefs which cross namespace - boundaries. Cross-namespace references are only valid if they are explicitly - allowed by something in the namespace they are referring to. For example: - Gateway has the AllowedRoutes field, and ReferenceGrant provides a - generic way to enable any other kind of cross-namespace reference. - - - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port is the network port this Route targets. It can be interpreted - differently based on the type of parent resource. - - When the parent resource is a Gateway, this targets all listeners - listening on the specified port that also support this kind of Route(and - select this Route). It's not recommended to set `Port` unless the - networking behaviors specified in a Route must apply to a specific port - as opposed to a listener(s) whose port(s) may be changed. When both Port - and SectionName are specified, the name and port of the selected listener - must match both specified values. - - - - Implementations MAY choose to support other parent resources. - Implementations supporting other types of parent resources MUST clearly - document how/if Port is interpreted. - - For the purpose of status, an attachment is considered successful as - long as the parent resource accepts it partially. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment - from the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, - the Route MUST be considered detached from the Gateway. - - Support: Extended - format: int32 - maximum: 65535 - minimum: 1 - type: integer - sectionName: - description: |- - SectionName is the name of a section within the target resource. In the - following resources, SectionName is interpreted as the following: - - * Gateway: Listener name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - * Service: Port name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - - Implementations MAY choose to support attaching Routes to other resources. - If that is the case, they MUST clearly document how SectionName is - interpreted. - - When unspecified (empty string), this will reference the entire resource. - For the purpose of status, an attachment is considered successful if at - least one section in the parent resource accepts it. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from - the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, the - Route MUST be considered detached from the Gateway. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - required: - - name - type: object - required: - - controllerName - - parentRef - type: object - maxItems: 32 - type: array - required: - - parents - type: object - type: object - served: true - storage: true - subresources: - status: {} -status: - acceptedNames: - kind: "" - plural: "" - conditions: null - storedVersions: null ---- -# -# config/crd/standard/gateway.networking.k8s.io_httproutes.yaml -# -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - api-approved.kubernetes.io: https://github.com/kubernetes-sigs/gateway-api/pull/3328 - gateway.networking.k8s.io/bundle-version: v1.2.0 - gateway.networking.k8s.io/channel: standard - creationTimestamp: null - name: httproutes.gateway.networking.k8s.io -spec: - group: gateway.networking.k8s.io - names: - categories: - - gateway-api - kind: HTTPRoute - listKind: HTTPRouteList - plural: httproutes - singular: httproute - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.hostnames - name: Hostnames - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1 - schema: - openAPIV3Schema: - description: |- - HTTPRoute provides a way to route HTTP requests. This includes the capability - to match requests by hostname, path, header, or query param. Filters can be - used to specify additional processing steps. Backends specify where matching - requests should be routed. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: Spec defines the desired state of HTTPRoute. - properties: - hostnames: - description: |- - Hostnames defines a set of hostnames that should match against the HTTP Host - header to select a HTTPRoute used to process the request. Implementations - MUST ignore any port value specified in the HTTP Host header while - performing a match and (absent of any applicable header modification - configuration) MUST forward this header unmodified to the backend. - - Valid values for Hostnames are determined by RFC 1123 definition of a - hostname with 2 notable exceptions: - - 1. IPs are not allowed. - 2. A hostname may be prefixed with a wildcard label (`*.`). The wildcard - label must appear by itself as the first label. - - If a hostname is specified by both the Listener and HTTPRoute, there - must be at least one intersecting hostname for the HTTPRoute to be - attached to the Listener. For example: - - * A Listener with `test.example.com` as the hostname matches HTTPRoutes - that have either not specified any hostnames, or have specified at - least one of `test.example.com` or `*.example.com`. - * A Listener with `*.example.com` as the hostname matches HTTPRoutes - that have either not specified any hostnames or have specified at least - one hostname that matches the Listener hostname. For example, - `*.example.com`, `test.example.com`, and `foo.test.example.com` would - all match. On the other hand, `example.com` and `test.example.net` would - not match. - - Hostnames that are prefixed with a wildcard label (`*.`) are interpreted - as a suffix match. That means that a match for `*.example.com` would match - both `test.example.com`, and `foo.test.example.com`, but not `example.com`. - - If both the Listener and HTTPRoute have specified hostnames, any - HTTPRoute hostnames that do not match the Listener hostname MUST be - ignored. For example, if a Listener specified `*.example.com`, and the - HTTPRoute specified `test.example.com` and `test.example.net`, - `test.example.net` must not be considered for a match. - - If both the Listener and HTTPRoute have specified hostnames, and none - match with the criteria above, then the HTTPRoute is not accepted. The - implementation must raise an 'Accepted' Condition with a status of - `False` in the corresponding RouteParentStatus. - - In the event that multiple HTTPRoutes specify intersecting hostnames (e.g. - overlapping wildcard matching and exact matching hostnames), precedence must - be given to rules from the HTTPRoute with the largest number of: - - * Characters in a matching non-wildcard hostname. - * Characters in a matching hostname. - - If ties exist across multiple Routes, the matching precedence rules for - HTTPRouteMatches takes over. - - Support: Core - items: - description: |- - Hostname is the fully qualified domain name of a network host. This matches - the RFC 1123 definition of a hostname with 2 notable exceptions: - - 1. IPs are not allowed. - 2. A hostname may be prefixed with a wildcard label (`*.`). The wildcard - label must appear by itself as the first label. - - Hostname can be "precise" which is a domain name without the terminating - dot of a network host (e.g. "foo.example.com") or "wildcard", which is a - domain name prefixed with a single wildcard label (e.g. `*.example.com`). - - Note that as per RFC1035 and RFC1123, a *label* must consist of lower case - alphanumeric characters or '-', and must start and end with an alphanumeric - character. No other punctuation is allowed. - maxLength: 253 - minLength: 1 - pattern: ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - maxItems: 16 - type: array - parentRefs: - description: |+ - ParentRefs references the resources (usually Gateways) that a Route wants - to be attached to. Note that the referenced parent resource needs to - allow this for the attachment to be complete. For Gateways, that means - the Gateway needs to allow attachment from Routes of this kind and - namespace. For Services, that means the Service must either be in the same - namespace for a "producer" route, or the mesh implementation must support - and allow "consumer" routes for the referenced Service. ReferenceGrant is - not applicable for governing ParentRefs to Services - it is not possible to - create a "producer" route for a Service in a different namespace from the - Route. - - There are two kinds of parent resources with "Core" support: - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - This API may be extended in the future to support additional kinds of parent - resources. - - ParentRefs must be _distinct_. This means either that: - - * They select different objects. If this is the case, then parentRef - entries are distinct. In terms of fields, this means that the - multi-part key defined by `group`, `kind`, `namespace`, and `name` must - be unique across all parentRef entries in the Route. - * They do not select different objects, but for each optional field used, - each ParentRef that selects the same object must set the same set of - optional fields to different values. If one ParentRef sets a - combination of optional fields, all must set the same combination. - - Some examples: - - * If one ParentRef sets `sectionName`, all ParentRefs referencing the - same object must also set `sectionName`. - * If one ParentRef sets `port`, all ParentRefs referencing the same - object must also set `port`. - * If one ParentRef sets `sectionName` and `port`, all ParentRefs - referencing the same object must also set `sectionName` and `port`. - - It is possible to separately reference multiple distinct objects that may - be collapsed by an implementation. For example, some implementations may - choose to merge compatible Gateway Listeners together. If that is the - case, the list of routes attached to those resources should also be - merged. - - Note that for ParentRefs that cross namespace boundaries, there are specific - rules. Cross-namespace references are only valid if they are explicitly - allowed by something in the namespace they are referring to. For example, - Gateway has the AllowedRoutes field, and ReferenceGrant provides a - generic way to enable other kinds of cross-namespace reference. - - - - - - - items: - description: |- - ParentReference identifies an API object (usually a Gateway) that can be considered - a parent of this resource (usually a route). There are two kinds of parent resources - with "Core" support: - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - This API may be extended in the future to support additional kinds of parent - resources. - - The API object must be valid in the cluster; the Group and Kind must - be registered in the cluster for this reference to be valid. - properties: - group: - default: gateway.networking.k8s.io - description: |- - Group is the group of the referent. - When unspecified, "gateway.networking.k8s.io" is inferred. - To set the core API group (such as for a "Service" kind referent), - Group must be explicitly set to "" (empty string). - - Support: Core - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Gateway - description: |- - Kind is kind of the referent. - - There are two kinds of parent resources with "Core" support: - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - Support for other resources is Implementation-Specific. - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: |- - Name is the name of the referent. - - Support: Core - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the referent. When unspecified, this refers - to the local namespace of the Route. - - Note that there are specific rules for ParentRefs which cross namespace - boundaries. Cross-namespace references are only valid if they are explicitly - allowed by something in the namespace they are referring to. For example: - Gateway has the AllowedRoutes field, and ReferenceGrant provides a - generic way to enable any other kind of cross-namespace reference. - - - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port is the network port this Route targets. It can be interpreted - differently based on the type of parent resource. - - When the parent resource is a Gateway, this targets all listeners - listening on the specified port that also support this kind of Route(and - select this Route). It's not recommended to set `Port` unless the - networking behaviors specified in a Route must apply to a specific port - as opposed to a listener(s) whose port(s) may be changed. When both Port - and SectionName are specified, the name and port of the selected listener - must match both specified values. - - - - Implementations MAY choose to support other parent resources. - Implementations supporting other types of parent resources MUST clearly - document how/if Port is interpreted. - - For the purpose of status, an attachment is considered successful as - long as the parent resource accepts it partially. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment - from the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, - the Route MUST be considered detached from the Gateway. - - Support: Extended - format: int32 - maximum: 65535 - minimum: 1 - type: integer - sectionName: - description: |- - SectionName is the name of a section within the target resource. In the - following resources, SectionName is interpreted as the following: - - * Gateway: Listener name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - * Service: Port name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - - Implementations MAY choose to support attaching Routes to other resources. - If that is the case, they MUST clearly document how SectionName is - interpreted. - - When unspecified (empty string), this will reference the entire resource. - For the purpose of status, an attachment is considered successful if at - least one section in the parent resource accepts it. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from - the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, the - Route MUST be considered detached from the Gateway. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - required: - - name - type: object - maxItems: 32 - type: array - x-kubernetes-validations: - - message: sectionName must be specified when parentRefs includes - 2 or more references to the same parent - rule: 'self.all(p1, self.all(p2, p1.group == p2.group && p1.kind - == p2.kind && p1.name == p2.name && (((!has(p1.__namespace__) - || p1.__namespace__ == '''') && (!has(p2.__namespace__) || p2.__namespace__ - == '''')) || (has(p1.__namespace__) && has(p2.__namespace__) && - p1.__namespace__ == p2.__namespace__ )) ? ((!has(p1.sectionName) - || p1.sectionName == '''') == (!has(p2.sectionName) || p2.sectionName - == '''')) : true))' - - message: sectionName must be unique when parentRefs includes 2 or - more references to the same parent - rule: self.all(p1, self.exists_one(p2, p1.group == p2.group && p1.kind - == p2.kind && p1.name == p2.name && (((!has(p1.__namespace__) - || p1.__namespace__ == '') && (!has(p2.__namespace__) || p2.__namespace__ - == '')) || (has(p1.__namespace__) && has(p2.__namespace__) && - p1.__namespace__ == p2.__namespace__ )) && (((!has(p1.sectionName) - || p1.sectionName == '') && (!has(p2.sectionName) || p2.sectionName - == '')) || (has(p1.sectionName) && has(p2.sectionName) && p1.sectionName - == p2.sectionName)))) - rules: - default: - - matches: - - path: - type: PathPrefix - value: / - description: |+ - Rules are a list of HTTP matchers, filters and actions. - - items: - description: |- - HTTPRouteRule defines semantics for matching an HTTP request based on - conditions (matches), processing it (filters), and forwarding the request to - an API object (backendRefs). - properties: - backendRefs: - description: |- - BackendRefs defines the backend(s) where matching requests should be - sent. - - Failure behavior here depends on how many BackendRefs are specified and - how many are invalid. - - If *all* entries in BackendRefs are invalid, and there are also no filters - specified in this route rule, *all* traffic which matches this rule MUST - receive a 500 status code. - - See the HTTPBackendRef definition for the rules about what makes a single - HTTPBackendRef invalid. - - When a HTTPBackendRef is invalid, 500 status codes MUST be returned for - requests that would have otherwise been routed to an invalid backend. If - multiple backends are specified, and some are invalid, the proportion of - requests that would otherwise have been routed to an invalid backend - MUST receive a 500 status code. - - For example, if two backends are specified with equal weights, and one is - invalid, 50 percent of traffic must receive a 500. Implementations may - choose how that 50 percent is determined. - - When a HTTPBackendRef refers to a Service that has no ready endpoints, - implementations SHOULD return a 503 for requests to that backend instead. - If an implementation chooses to do this, all of the above rules for 500 responses - MUST also apply for responses that return a 503. - - Support: Core for Kubernetes Service - - Support: Extended for Kubernetes ServiceImport - - Support: Implementation-specific for any other resource - - Support for weight: Core - items: - description: |- - HTTPBackendRef defines how a HTTPRoute forwards a HTTP request. - - Note that when a namespace different than the local namespace is specified, a - ReferenceGrant object is required in the referent namespace to allow that - namespace's owner to accept the reference. See the ReferenceGrant - documentation for details. - - - - When the BackendRef points to a Kubernetes Service, implementations SHOULD - honor the appProtocol field if it is set for the target Service Port. - - Implementations supporting appProtocol SHOULD recognize the Kubernetes - Standard Application Protocols defined in KEP-3726. - - If a Service appProtocol isn't specified, an implementation MAY infer the - backend protocol through its own means. Implementations MAY infer the - protocol from the Route type referring to the backend Service. - - If a Route is not able to send traffic to the backend using the specified - protocol then the backend is considered invalid. Implementations MUST set the - "ResolvedRefs" condition to "False" with the "UnsupportedProtocol" reason. - - - properties: - filters: - description: |- - Filters defined at this level should be executed if and only if the - request is being forwarded to the backend defined here. - - Support: Implementation-specific (For broader support of filters, use the - Filters field in HTTPRouteRule.) - items: - description: |- - HTTPRouteFilter defines processing steps that must be completed during the - request or response lifecycle. HTTPRouteFilters are meant as an extension - point to express processing that may be done in Gateway implementations. Some - examples include request or response modification, implementing - authentication strategies, rate-limiting, and traffic shaping. API - guarantee/conformance is defined based on the type of the filter. - properties: - extensionRef: - description: |- - ExtensionRef is an optional, implementation-specific extension to the - "filter" behavior. For example, resource "myroutefilter" in group - "networking.example.net"). ExtensionRef MUST NOT be used for core and - extended filters. - - This filter can be used multiple times within the same rule. - - Support: Implementation-specific - properties: - group: - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: Kind is kind of the referent. For - example "HTTPRoute" or "Service". - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - required: - - group - - kind - - name - type: object - requestHeaderModifier: - description: |- - RequestHeaderModifier defines a schema for a filter that modifies request - headers. - - Support: Core - properties: - add: - description: |- - Add adds the given header(s) (name, value) to the request - before the action. It appends to any existing values associated - with the header name. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - add: - - name: "my-header" - value: "bar,baz" - - Output: - GET /foo HTTP/1.1 - my-header: foo,bar,baz - items: - description: HTTPHeader represents an HTTP - Header name and value as defined by RFC - 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP - Header to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - remove: - description: |- - Remove the given header(s) from the HTTP request before the action. The - value of Remove is a list of HTTP header names. Note that the header - names are case-insensitive (see - https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). - - Input: - GET /foo HTTP/1.1 - my-header1: foo - my-header2: bar - my-header3: baz - - Config: - remove: ["my-header1", "my-header3"] - - Output: - GET /foo HTTP/1.1 - my-header2: bar - items: - type: string - maxItems: 16 - type: array - x-kubernetes-list-type: set - set: - description: |- - Set overwrites the request with the given header (name, value) - before the action. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - set: - - name: "my-header" - value: "bar" - - Output: - GET /foo HTTP/1.1 - my-header: bar - items: - description: HTTPHeader represents an HTTP - Header name and value as defined by RFC - 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP - Header to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - requestMirror: - description: |+ - RequestMirror defines a schema for a filter that mirrors requests. - Requests are sent to the specified destination, but responses from - that destination are ignored. - - This filter can be used multiple times within the same rule. Note that - not all implementations will be able to support mirroring to multiple - backends. - - Support: Extended - - properties: - backendRef: - description: |- - BackendRef references a resource where mirrored requests are sent. - - Mirrored requests must be sent only to a single destination endpoint - within this BackendRef, irrespective of how many endpoints are present - within this BackendRef. - - If the referent cannot be found, this BackendRef is invalid and must be - dropped from the Gateway. The controller must ensure the "ResolvedRefs" - condition on the Route status is set to `status: False` and not configure - this backend in the underlying implementation. - - If there is a cross-namespace reference to an *existing* object - that is not allowed by a ReferenceGrant, the controller must ensure the - "ResolvedRefs" condition on the Route is set to `status: False`, - with the "RefNotPermitted" reason and not configure this backend in the - underlying implementation. - - In either error case, the Message of the `ResolvedRefs` Condition - should be used to provide more detail about the problem. - - Support: Extended for Kubernetes Service - - Support: Implementation-specific for any other resource - properties: - group: - default: "" - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Service - description: |- - Kind is the Kubernetes resource kind of the referent. For example - "Service". - - Defaults to "Service" when not specified. - - ExternalName services can refer to CNAME DNS records that may live - outside of the cluster and as such are difficult to reason about in - terms of conformance. They also may not be safe to forward to (see - CVE-2021-25740 for more information). Implementations SHOULD NOT - support ExternalName Services. - - Support: Core (Services with a type other than ExternalName) - - Support: Implementation-specific (Services with type ExternalName) - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the backend. When unspecified, the local - namespace is inferred. - - Note that when a namespace different than the local namespace is specified, - a ReferenceGrant object is required in the referent namespace to allow that - namespace's owner to accept the reference. See the ReferenceGrant - documentation for details. - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port specifies the destination port number to use for this resource. - Port is required when the referent is a Kubernetes Service. In this - case, the port number is the service port number, not the target port. - For other resources, destination port might be derived from the referent - resource or this field. - format: int32 - maximum: 65535 - minimum: 1 - type: integer - required: - - name - type: object - x-kubernetes-validations: - - message: Must have port for Service reference - rule: '(size(self.group) == 0 && self.kind - == ''Service'') ? has(self.port) : true' - required: - - backendRef - type: object - requestRedirect: - description: |- - RequestRedirect defines a schema for a filter that responds to the - request with an HTTP redirection. - - Support: Core - properties: - hostname: - description: |- - Hostname is the hostname to be used in the value of the `Location` - header in the response. - When empty, the hostname in the `Host` header of the request is used. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - path: - description: |- - Path defines parameters used to modify the path of the incoming request. - The modified path is then used to construct the `Location` header. When - empty, the request path is used as-is. - - Support: Extended - properties: - replaceFullPath: - description: |- - ReplaceFullPath specifies the value with which to replace the full path - of a request during a rewrite or redirect. - maxLength: 1024 - type: string - replacePrefixMatch: - description: |- - ReplacePrefixMatch specifies the value with which to replace the prefix - match of a request during a rewrite or redirect. For example, a request - to "/foo/bar" with a prefix match of "/foo" and a ReplacePrefixMatch - of "/xyz" would be modified to "/xyz/bar". - - Note that this matches the behavior of the PathPrefix match type. This - matches full path elements. A path element refers to the list of labels - in the path split by the `/` separator. When specified, a trailing `/` is - ignored. For example, the paths `/abc`, `/abc/`, and `/abc/def` would all - match the prefix `/abc`, but the path `/abcd` would not. - - ReplacePrefixMatch is only compatible with a `PathPrefix` HTTPRouteMatch. - Using any other HTTPRouteMatch type on the same HTTPRouteRule will result in - the implementation setting the Accepted Condition for the Route to `status: False`. - - Request Path | Prefix Match | Replace Prefix | Modified Path - maxLength: 1024 - type: string - type: - description: |- - Type defines the type of path modifier. Additional types may be - added in a future release of the API. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - enum: - - ReplaceFullPath - - ReplacePrefixMatch - type: string - required: - - type - type: object - x-kubernetes-validations: - - message: replaceFullPath must be specified - when type is set to 'ReplaceFullPath' - rule: 'self.type == ''ReplaceFullPath'' ? - has(self.replaceFullPath) : true' - - message: type must be 'ReplaceFullPath' when - replaceFullPath is set - rule: 'has(self.replaceFullPath) ? self.type - == ''ReplaceFullPath'' : true' - - message: replacePrefixMatch must be specified - when type is set to 'ReplacePrefixMatch' - rule: 'self.type == ''ReplacePrefixMatch'' - ? has(self.replacePrefixMatch) : true' - - message: type must be 'ReplacePrefixMatch' - when replacePrefixMatch is set - rule: 'has(self.replacePrefixMatch) ? self.type - == ''ReplacePrefixMatch'' : true' - port: - description: |- - Port is the port to be used in the value of the `Location` - header in the response. - - If no port is specified, the redirect port MUST be derived using the - following rules: - - * If redirect scheme is not-empty, the redirect port MUST be the well-known - port associated with the redirect scheme. Specifically "http" to port 80 - and "https" to port 443. If the redirect scheme does not have a - well-known port, the listener port of the Gateway SHOULD be used. - * If redirect scheme is empty, the redirect port MUST be the Gateway - Listener port. - - Implementations SHOULD NOT add the port number in the 'Location' - header in the following cases: - - * A Location header that will use HTTP (whether that is determined via - the Listener protocol or the Scheme field) _and_ use port 80. - * A Location header that will use HTTPS (whether that is determined via - the Listener protocol or the Scheme field) _and_ use port 443. - - Support: Extended - format: int32 - maximum: 65535 - minimum: 1 - type: integer - scheme: - description: |- - Scheme is the scheme to be used in the value of the `Location` header in - the response. When empty, the scheme of the request is used. - - Scheme redirects can affect the port of the redirect, for more information, - refer to the documentation for the port field of this filter. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - - Support: Extended - enum: - - http - - https - type: string - statusCode: - default: 302 - description: |- - StatusCode is the HTTP status code to be used in response. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - - Support: Core - enum: - - 301 - - 302 - type: integer - type: object - responseHeaderModifier: - description: |- - ResponseHeaderModifier defines a schema for a filter that modifies response - headers. - - Support: Extended - properties: - add: - description: |- - Add adds the given header(s) (name, value) to the request - before the action. It appends to any existing values associated - with the header name. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - add: - - name: "my-header" - value: "bar,baz" - - Output: - GET /foo HTTP/1.1 - my-header: foo,bar,baz - items: - description: HTTPHeader represents an HTTP - Header name and value as defined by RFC - 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP - Header to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - remove: - description: |- - Remove the given header(s) from the HTTP request before the action. The - value of Remove is a list of HTTP header names. Note that the header - names are case-insensitive (see - https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). - - Input: - GET /foo HTTP/1.1 - my-header1: foo - my-header2: bar - my-header3: baz - - Config: - remove: ["my-header1", "my-header3"] - - Output: - GET /foo HTTP/1.1 - my-header2: bar - items: - type: string - maxItems: 16 - type: array - x-kubernetes-list-type: set - set: - description: |- - Set overwrites the request with the given header (name, value) - before the action. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - set: - - name: "my-header" - value: "bar" - - Output: - GET /foo HTTP/1.1 - my-header: bar - items: - description: HTTPHeader represents an HTTP - Header name and value as defined by RFC - 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP - Header to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: - description: |- - Type identifies the type of filter to apply. As with other API fields, - types are classified into three conformance levels: - - - Core: Filter types and their corresponding configuration defined by - "Support: Core" in this package, e.g. "RequestHeaderModifier". All - implementations must support core filters. - - - Extended: Filter types and their corresponding configuration defined by - "Support: Extended" in this package, e.g. "RequestMirror". Implementers - are encouraged to support extended filters. - - - Implementation-specific: Filters that are defined and supported by - specific vendors. - In the future, filters showing convergence in behavior across multiple - implementations will be considered for inclusion in extended or core - conformance levels. Filter-specific configuration for such filters - is specified using the ExtensionRef field. `Type` should be set to - "ExtensionRef" for custom filters. - - Implementers are encouraged to define custom implementation types to - extend the core API with implementation-specific behavior. - - If a reference to a custom filter type cannot be resolved, the filter - MUST NOT be skipped. Instead, requests that would have been processed by - that filter MUST receive a HTTP error response. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - enum: - - RequestHeaderModifier - - ResponseHeaderModifier - - RequestMirror - - RequestRedirect - - URLRewrite - - ExtensionRef - type: string - urlRewrite: - description: |- - URLRewrite defines a schema for a filter that modifies a request during forwarding. - - Support: Extended - properties: - hostname: - description: |- - Hostname is the value to be used to replace the Host header value during - forwarding. - - Support: Extended - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - path: - description: |- - Path defines a path rewrite. - - Support: Extended - properties: - replaceFullPath: - description: |- - ReplaceFullPath specifies the value with which to replace the full path - of a request during a rewrite or redirect. - maxLength: 1024 - type: string - replacePrefixMatch: - description: |- - ReplacePrefixMatch specifies the value with which to replace the prefix - match of a request during a rewrite or redirect. For example, a request - to "/foo/bar" with a prefix match of "/foo" and a ReplacePrefixMatch - of "/xyz" would be modified to "/xyz/bar". - - Note that this matches the behavior of the PathPrefix match type. This - matches full path elements. A path element refers to the list of labels - in the path split by the `/` separator. When specified, a trailing `/` is - ignored. For example, the paths `/abc`, `/abc/`, and `/abc/def` would all - match the prefix `/abc`, but the path `/abcd` would not. - - ReplacePrefixMatch is only compatible with a `PathPrefix` HTTPRouteMatch. - Using any other HTTPRouteMatch type on the same HTTPRouteRule will result in - the implementation setting the Accepted Condition for the Route to `status: False`. - - Request Path | Prefix Match | Replace Prefix | Modified Path - maxLength: 1024 - type: string - type: - description: |- - Type defines the type of path modifier. Additional types may be - added in a future release of the API. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - enum: - - ReplaceFullPath - - ReplacePrefixMatch - type: string - required: - - type - type: object - x-kubernetes-validations: - - message: replaceFullPath must be specified - when type is set to 'ReplaceFullPath' - rule: 'self.type == ''ReplaceFullPath'' ? - has(self.replaceFullPath) : true' - - message: type must be 'ReplaceFullPath' when - replaceFullPath is set - rule: 'has(self.replaceFullPath) ? self.type - == ''ReplaceFullPath'' : true' - - message: replacePrefixMatch must be specified - when type is set to 'ReplacePrefixMatch' - rule: 'self.type == ''ReplacePrefixMatch'' - ? has(self.replacePrefixMatch) : true' - - message: type must be 'ReplacePrefixMatch' - when replacePrefixMatch is set - rule: 'has(self.replacePrefixMatch) ? self.type - == ''ReplacePrefixMatch'' : true' - type: object - required: - - type - type: object - x-kubernetes-validations: - - message: filter.requestHeaderModifier must be nil - if the filter.type is not RequestHeaderModifier - rule: '!(has(self.requestHeaderModifier) && self.type - != ''RequestHeaderModifier'')' - - message: filter.requestHeaderModifier must be specified - for RequestHeaderModifier filter.type - rule: '!(!has(self.requestHeaderModifier) && self.type - == ''RequestHeaderModifier'')' - - message: filter.responseHeaderModifier must be nil - if the filter.type is not ResponseHeaderModifier - rule: '!(has(self.responseHeaderModifier) && self.type - != ''ResponseHeaderModifier'')' - - message: filter.responseHeaderModifier must be specified - for ResponseHeaderModifier filter.type - rule: '!(!has(self.responseHeaderModifier) && self.type - == ''ResponseHeaderModifier'')' - - message: filter.requestMirror must be nil if the filter.type - is not RequestMirror - rule: '!(has(self.requestMirror) && self.type != ''RequestMirror'')' - - message: filter.requestMirror must be specified for - RequestMirror filter.type - rule: '!(!has(self.requestMirror) && self.type == - ''RequestMirror'')' - - message: filter.requestRedirect must be nil if the - filter.type is not RequestRedirect - rule: '!(has(self.requestRedirect) && self.type != - ''RequestRedirect'')' - - message: filter.requestRedirect must be specified - for RequestRedirect filter.type - rule: '!(!has(self.requestRedirect) && self.type == - ''RequestRedirect'')' - - message: filter.urlRewrite must be nil if the filter.type - is not URLRewrite - rule: '!(has(self.urlRewrite) && self.type != ''URLRewrite'')' - - message: filter.urlRewrite must be specified for URLRewrite - filter.type - rule: '!(!has(self.urlRewrite) && self.type == ''URLRewrite'')' - - message: filter.extensionRef must be nil if the filter.type - is not ExtensionRef - rule: '!(has(self.extensionRef) && self.type != ''ExtensionRef'')' - - message: filter.extensionRef must be specified for - ExtensionRef filter.type - rule: '!(!has(self.extensionRef) && self.type == ''ExtensionRef'')' - maxItems: 16 - type: array - x-kubernetes-validations: - - message: May specify either httpRouteFilterRequestRedirect - or httpRouteFilterRequestRewrite, but not both - rule: '!(self.exists(f, f.type == ''RequestRedirect'') - && self.exists(f, f.type == ''URLRewrite''))' - - message: May specify either httpRouteFilterRequestRedirect - or httpRouteFilterRequestRewrite, but not both - rule: '!(self.exists(f, f.type == ''RequestRedirect'') - && self.exists(f, f.type == ''URLRewrite''))' - - message: RequestHeaderModifier filter cannot be repeated - rule: self.filter(f, f.type == 'RequestHeaderModifier').size() - <= 1 - - message: ResponseHeaderModifier filter cannot be repeated - rule: self.filter(f, f.type == 'ResponseHeaderModifier').size() - <= 1 - - message: RequestRedirect filter cannot be repeated - rule: self.filter(f, f.type == 'RequestRedirect').size() - <= 1 - - message: URLRewrite filter cannot be repeated - rule: self.filter(f, f.type == 'URLRewrite').size() - <= 1 - group: - default: "" - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Service - description: |- - Kind is the Kubernetes resource kind of the referent. For example - "Service". - - Defaults to "Service" when not specified. - - ExternalName services can refer to CNAME DNS records that may live - outside of the cluster and as such are difficult to reason about in - terms of conformance. They also may not be safe to forward to (see - CVE-2021-25740 for more information). Implementations SHOULD NOT - support ExternalName Services. - - Support: Core (Services with a type other than ExternalName) - - Support: Implementation-specific (Services with type ExternalName) - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the backend. When unspecified, the local - namespace is inferred. - - Note that when a namespace different than the local namespace is specified, - a ReferenceGrant object is required in the referent namespace to allow that - namespace's owner to accept the reference. See the ReferenceGrant - documentation for details. - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port specifies the destination port number to use for this resource. - Port is required when the referent is a Kubernetes Service. In this - case, the port number is the service port number, not the target port. - For other resources, destination port might be derived from the referent - resource or this field. - format: int32 - maximum: 65535 - minimum: 1 - type: integer - weight: - default: 1 - description: |- - Weight specifies the proportion of requests forwarded to the referenced - backend. This is computed as weight/(sum of all weights in this - BackendRefs list). For non-zero values, there may be some epsilon from - the exact proportion defined here depending on the precision an - implementation supports. Weight is not a percentage and the sum of - weights does not need to equal 100. - - If only one backend is specified and it has a weight greater than 0, 100% - of the traffic is forwarded to that backend. If weight is set to 0, no - traffic should be forwarded for this entry. If unspecified, weight - defaults to 1. - - Support for this field varies based on the context where used. - format: int32 - maximum: 1000000 - minimum: 0 - type: integer - required: - - name - type: object - x-kubernetes-validations: - - message: Must have port for Service reference - rule: '(size(self.group) == 0 && self.kind == ''Service'') - ? has(self.port) : true' - maxItems: 16 - type: array - filters: - description: |- - Filters define the filters that are applied to requests that match - this rule. - - Wherever possible, implementations SHOULD implement filters in the order - they are specified. - - Implementations MAY choose to implement this ordering strictly, rejecting - any combination or order of filters that can not be supported. If implementations - choose a strict interpretation of filter ordering, they MUST clearly document - that behavior. - - To reject an invalid combination or order of filters, implementations SHOULD - consider the Route Rules with this configuration invalid. If all Route Rules - in a Route are invalid, the entire Route would be considered invalid. If only - a portion of Route Rules are invalid, implementations MUST set the - "PartiallyInvalid" condition for the Route. - - Conformance-levels at this level are defined based on the type of filter: - - - ALL core filters MUST be supported by all implementations. - - Implementers are encouraged to support extended filters. - - Implementation-specific custom filters have no API guarantees across - implementations. - - Specifying the same filter multiple times is not supported unless explicitly - indicated in the filter. - - All filters are expected to be compatible with each other except for the - URLRewrite and RequestRedirect filters, which may not be combined. If an - implementation can not support other combinations of filters, they must clearly - document that limitation. In cases where incompatible or unsupported - filters are specified and cause the `Accepted` condition to be set to status - `False`, implementations may use the `IncompatibleFilters` reason to specify - this configuration error. - - Support: Core - items: - description: |- - HTTPRouteFilter defines processing steps that must be completed during the - request or response lifecycle. HTTPRouteFilters are meant as an extension - point to express processing that may be done in Gateway implementations. Some - examples include request or response modification, implementing - authentication strategies, rate-limiting, and traffic shaping. API - guarantee/conformance is defined based on the type of the filter. - properties: - extensionRef: - description: |- - ExtensionRef is an optional, implementation-specific extension to the - "filter" behavior. For example, resource "myroutefilter" in group - "networking.example.net"). ExtensionRef MUST NOT be used for core and - extended filters. - - This filter can be used multiple times within the same rule. - - Support: Implementation-specific - properties: - group: - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: Kind is kind of the referent. For example - "HTTPRoute" or "Service". - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - required: - - group - - kind - - name - type: object - requestHeaderModifier: - description: |- - RequestHeaderModifier defines a schema for a filter that modifies request - headers. - - Support: Core - properties: - add: - description: |- - Add adds the given header(s) (name, value) to the request - before the action. It appends to any existing values associated - with the header name. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - add: - - name: "my-header" - value: "bar,baz" - - Output: - GET /foo HTTP/1.1 - my-header: foo,bar,baz - items: - description: HTTPHeader represents an HTTP Header - name and value as defined by RFC 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP Header - to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - remove: - description: |- - Remove the given header(s) from the HTTP request before the action. The - value of Remove is a list of HTTP header names. Note that the header - names are case-insensitive (see - https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). - - Input: - GET /foo HTTP/1.1 - my-header1: foo - my-header2: bar - my-header3: baz - - Config: - remove: ["my-header1", "my-header3"] - - Output: - GET /foo HTTP/1.1 - my-header2: bar - items: - type: string - maxItems: 16 - type: array - x-kubernetes-list-type: set - set: - description: |- - Set overwrites the request with the given header (name, value) - before the action. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - set: - - name: "my-header" - value: "bar" - - Output: - GET /foo HTTP/1.1 - my-header: bar - items: - description: HTTPHeader represents an HTTP Header - name and value as defined by RFC 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP Header - to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - requestMirror: - description: |+ - RequestMirror defines a schema for a filter that mirrors requests. - Requests are sent to the specified destination, but responses from - that destination are ignored. - - This filter can be used multiple times within the same rule. Note that - not all implementations will be able to support mirroring to multiple - backends. - - Support: Extended - - properties: - backendRef: - description: |- - BackendRef references a resource where mirrored requests are sent. - - Mirrored requests must be sent only to a single destination endpoint - within this BackendRef, irrespective of how many endpoints are present - within this BackendRef. - - If the referent cannot be found, this BackendRef is invalid and must be - dropped from the Gateway. The controller must ensure the "ResolvedRefs" - condition on the Route status is set to `status: False` and not configure - this backend in the underlying implementation. - - If there is a cross-namespace reference to an *existing* object - that is not allowed by a ReferenceGrant, the controller must ensure the - "ResolvedRefs" condition on the Route is set to `status: False`, - with the "RefNotPermitted" reason and not configure this backend in the - underlying implementation. - - In either error case, the Message of the `ResolvedRefs` Condition - should be used to provide more detail about the problem. - - Support: Extended for Kubernetes Service - - Support: Implementation-specific for any other resource - properties: - group: - default: "" - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Service - description: |- - Kind is the Kubernetes resource kind of the referent. For example - "Service". - - Defaults to "Service" when not specified. - - ExternalName services can refer to CNAME DNS records that may live - outside of the cluster and as such are difficult to reason about in - terms of conformance. They also may not be safe to forward to (see - CVE-2021-25740 for more information). Implementations SHOULD NOT - support ExternalName Services. - - Support: Core (Services with a type other than ExternalName) - - Support: Implementation-specific (Services with type ExternalName) - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the backend. When unspecified, the local - namespace is inferred. - - Note that when a namespace different than the local namespace is specified, - a ReferenceGrant object is required in the referent namespace to allow that - namespace's owner to accept the reference. See the ReferenceGrant - documentation for details. - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port specifies the destination port number to use for this resource. - Port is required when the referent is a Kubernetes Service. In this - case, the port number is the service port number, not the target port. - For other resources, destination port might be derived from the referent - resource or this field. - format: int32 - maximum: 65535 - minimum: 1 - type: integer - required: - - name - type: object - x-kubernetes-validations: - - message: Must have port for Service reference - rule: '(size(self.group) == 0 && self.kind == ''Service'') - ? has(self.port) : true' - required: - - backendRef - type: object - requestRedirect: - description: |- - RequestRedirect defines a schema for a filter that responds to the - request with an HTTP redirection. - - Support: Core - properties: - hostname: - description: |- - Hostname is the hostname to be used in the value of the `Location` - header in the response. - When empty, the hostname in the `Host` header of the request is used. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - path: - description: |- - Path defines parameters used to modify the path of the incoming request. - The modified path is then used to construct the `Location` header. When - empty, the request path is used as-is. - - Support: Extended - properties: - replaceFullPath: - description: |- - ReplaceFullPath specifies the value with which to replace the full path - of a request during a rewrite or redirect. - maxLength: 1024 - type: string - replacePrefixMatch: - description: |- - ReplacePrefixMatch specifies the value with which to replace the prefix - match of a request during a rewrite or redirect. For example, a request - to "/foo/bar" with a prefix match of "/foo" and a ReplacePrefixMatch - of "/xyz" would be modified to "/xyz/bar". - - Note that this matches the behavior of the PathPrefix match type. This - matches full path elements. A path element refers to the list of labels - in the path split by the `/` separator. When specified, a trailing `/` is - ignored. For example, the paths `/abc`, `/abc/`, and `/abc/def` would all - match the prefix `/abc`, but the path `/abcd` would not. - - ReplacePrefixMatch is only compatible with a `PathPrefix` HTTPRouteMatch. - Using any other HTTPRouteMatch type on the same HTTPRouteRule will result in - the implementation setting the Accepted Condition for the Route to `status: False`. - - Request Path | Prefix Match | Replace Prefix | Modified Path - maxLength: 1024 - type: string - type: - description: |- - Type defines the type of path modifier. Additional types may be - added in a future release of the API. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - enum: - - ReplaceFullPath - - ReplacePrefixMatch - type: string - required: - - type - type: object - x-kubernetes-validations: - - message: replaceFullPath must be specified when - type is set to 'ReplaceFullPath' - rule: 'self.type == ''ReplaceFullPath'' ? has(self.replaceFullPath) - : true' - - message: type must be 'ReplaceFullPath' when replaceFullPath - is set - rule: 'has(self.replaceFullPath) ? self.type == - ''ReplaceFullPath'' : true' - - message: replacePrefixMatch must be specified when - type is set to 'ReplacePrefixMatch' - rule: 'self.type == ''ReplacePrefixMatch'' ? has(self.replacePrefixMatch) - : true' - - message: type must be 'ReplacePrefixMatch' when - replacePrefixMatch is set - rule: 'has(self.replacePrefixMatch) ? self.type - == ''ReplacePrefixMatch'' : true' - port: - description: |- - Port is the port to be used in the value of the `Location` - header in the response. - - If no port is specified, the redirect port MUST be derived using the - following rules: - - * If redirect scheme is not-empty, the redirect port MUST be the well-known - port associated with the redirect scheme. Specifically "http" to port 80 - and "https" to port 443. If the redirect scheme does not have a - well-known port, the listener port of the Gateway SHOULD be used. - * If redirect scheme is empty, the redirect port MUST be the Gateway - Listener port. - - Implementations SHOULD NOT add the port number in the 'Location' - header in the following cases: - - * A Location header that will use HTTP (whether that is determined via - the Listener protocol or the Scheme field) _and_ use port 80. - * A Location header that will use HTTPS (whether that is determined via - the Listener protocol or the Scheme field) _and_ use port 443. - - Support: Extended - format: int32 - maximum: 65535 - minimum: 1 - type: integer - scheme: - description: |- - Scheme is the scheme to be used in the value of the `Location` header in - the response. When empty, the scheme of the request is used. - - Scheme redirects can affect the port of the redirect, for more information, - refer to the documentation for the port field of this filter. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - - Support: Extended - enum: - - http - - https - type: string - statusCode: - default: 302 - description: |- - StatusCode is the HTTP status code to be used in response. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - - Support: Core - enum: - - 301 - - 302 - type: integer - type: object - responseHeaderModifier: - description: |- - ResponseHeaderModifier defines a schema for a filter that modifies response - headers. - - Support: Extended - properties: - add: - description: |- - Add adds the given header(s) (name, value) to the request - before the action. It appends to any existing values associated - with the header name. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - add: - - name: "my-header" - value: "bar,baz" - - Output: - GET /foo HTTP/1.1 - my-header: foo,bar,baz - items: - description: HTTPHeader represents an HTTP Header - name and value as defined by RFC 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP Header - to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - remove: - description: |- - Remove the given header(s) from the HTTP request before the action. The - value of Remove is a list of HTTP header names. Note that the header - names are case-insensitive (see - https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). - - Input: - GET /foo HTTP/1.1 - my-header1: foo - my-header2: bar - my-header3: baz - - Config: - remove: ["my-header1", "my-header3"] - - Output: - GET /foo HTTP/1.1 - my-header2: bar - items: - type: string - maxItems: 16 - type: array - x-kubernetes-list-type: set - set: - description: |- - Set overwrites the request with the given header (name, value) - before the action. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - set: - - name: "my-header" - value: "bar" - - Output: - GET /foo HTTP/1.1 - my-header: bar - items: - description: HTTPHeader represents an HTTP Header - name and value as defined by RFC 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP Header - to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: - description: |- - Type identifies the type of filter to apply. As with other API fields, - types are classified into three conformance levels: - - - Core: Filter types and their corresponding configuration defined by - "Support: Core" in this package, e.g. "RequestHeaderModifier". All - implementations must support core filters. - - - Extended: Filter types and their corresponding configuration defined by - "Support: Extended" in this package, e.g. "RequestMirror". Implementers - are encouraged to support extended filters. - - - Implementation-specific: Filters that are defined and supported by - specific vendors. - In the future, filters showing convergence in behavior across multiple - implementations will be considered for inclusion in extended or core - conformance levels. Filter-specific configuration for such filters - is specified using the ExtensionRef field. `Type` should be set to - "ExtensionRef" for custom filters. - - Implementers are encouraged to define custom implementation types to - extend the core API with implementation-specific behavior. - - If a reference to a custom filter type cannot be resolved, the filter - MUST NOT be skipped. Instead, requests that would have been processed by - that filter MUST receive a HTTP error response. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - enum: - - RequestHeaderModifier - - ResponseHeaderModifier - - RequestMirror - - RequestRedirect - - URLRewrite - - ExtensionRef - type: string - urlRewrite: - description: |- - URLRewrite defines a schema for a filter that modifies a request during forwarding. - - Support: Extended - properties: - hostname: - description: |- - Hostname is the value to be used to replace the Host header value during - forwarding. - - Support: Extended - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - path: - description: |- - Path defines a path rewrite. - - Support: Extended - properties: - replaceFullPath: - description: |- - ReplaceFullPath specifies the value with which to replace the full path - of a request during a rewrite or redirect. - maxLength: 1024 - type: string - replacePrefixMatch: - description: |- - ReplacePrefixMatch specifies the value with which to replace the prefix - match of a request during a rewrite or redirect. For example, a request - to "/foo/bar" with a prefix match of "/foo" and a ReplacePrefixMatch - of "/xyz" would be modified to "/xyz/bar". - - Note that this matches the behavior of the PathPrefix match type. This - matches full path elements. A path element refers to the list of labels - in the path split by the `/` separator. When specified, a trailing `/` is - ignored. For example, the paths `/abc`, `/abc/`, and `/abc/def` would all - match the prefix `/abc`, but the path `/abcd` would not. - - ReplacePrefixMatch is only compatible with a `PathPrefix` HTTPRouteMatch. - Using any other HTTPRouteMatch type on the same HTTPRouteRule will result in - the implementation setting the Accepted Condition for the Route to `status: False`. - - Request Path | Prefix Match | Replace Prefix | Modified Path - maxLength: 1024 - type: string - type: - description: |- - Type defines the type of path modifier. Additional types may be - added in a future release of the API. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - enum: - - ReplaceFullPath - - ReplacePrefixMatch - type: string - required: - - type - type: object - x-kubernetes-validations: - - message: replaceFullPath must be specified when - type is set to 'ReplaceFullPath' - rule: 'self.type == ''ReplaceFullPath'' ? has(self.replaceFullPath) - : true' - - message: type must be 'ReplaceFullPath' when replaceFullPath - is set - rule: 'has(self.replaceFullPath) ? self.type == - ''ReplaceFullPath'' : true' - - message: replacePrefixMatch must be specified when - type is set to 'ReplacePrefixMatch' - rule: 'self.type == ''ReplacePrefixMatch'' ? has(self.replacePrefixMatch) - : true' - - message: type must be 'ReplacePrefixMatch' when - replacePrefixMatch is set - rule: 'has(self.replacePrefixMatch) ? self.type - == ''ReplacePrefixMatch'' : true' - type: object - required: - - type - type: object - x-kubernetes-validations: - - message: filter.requestHeaderModifier must be nil if the - filter.type is not RequestHeaderModifier - rule: '!(has(self.requestHeaderModifier) && self.type != - ''RequestHeaderModifier'')' - - message: filter.requestHeaderModifier must be specified - for RequestHeaderModifier filter.type - rule: '!(!has(self.requestHeaderModifier) && self.type == - ''RequestHeaderModifier'')' - - message: filter.responseHeaderModifier must be nil if the - filter.type is not ResponseHeaderModifier - rule: '!(has(self.responseHeaderModifier) && self.type != - ''ResponseHeaderModifier'')' - - message: filter.responseHeaderModifier must be specified - for ResponseHeaderModifier filter.type - rule: '!(!has(self.responseHeaderModifier) && self.type - == ''ResponseHeaderModifier'')' - - message: filter.requestMirror must be nil if the filter.type - is not RequestMirror - rule: '!(has(self.requestMirror) && self.type != ''RequestMirror'')' - - message: filter.requestMirror must be specified for RequestMirror - filter.type - rule: '!(!has(self.requestMirror) && self.type == ''RequestMirror'')' - - message: filter.requestRedirect must be nil if the filter.type - is not RequestRedirect - rule: '!(has(self.requestRedirect) && self.type != ''RequestRedirect'')' - - message: filter.requestRedirect must be specified for RequestRedirect - filter.type - rule: '!(!has(self.requestRedirect) && self.type == ''RequestRedirect'')' - - message: filter.urlRewrite must be nil if the filter.type - is not URLRewrite - rule: '!(has(self.urlRewrite) && self.type != ''URLRewrite'')' - - message: filter.urlRewrite must be specified for URLRewrite - filter.type - rule: '!(!has(self.urlRewrite) && self.type == ''URLRewrite'')' - - message: filter.extensionRef must be nil if the filter.type - is not ExtensionRef - rule: '!(has(self.extensionRef) && self.type != ''ExtensionRef'')' - - message: filter.extensionRef must be specified for ExtensionRef - filter.type - rule: '!(!has(self.extensionRef) && self.type == ''ExtensionRef'')' - maxItems: 16 - type: array - x-kubernetes-validations: - - message: May specify either httpRouteFilterRequestRedirect - or httpRouteFilterRequestRewrite, but not both - rule: '!(self.exists(f, f.type == ''RequestRedirect'') && - self.exists(f, f.type == ''URLRewrite''))' - - message: RequestHeaderModifier filter cannot be repeated - rule: self.filter(f, f.type == 'RequestHeaderModifier').size() - <= 1 - - message: ResponseHeaderModifier filter cannot be repeated - rule: self.filter(f, f.type == 'ResponseHeaderModifier').size() - <= 1 - - message: RequestRedirect filter cannot be repeated - rule: self.filter(f, f.type == 'RequestRedirect').size() <= - 1 - - message: URLRewrite filter cannot be repeated - rule: self.filter(f, f.type == 'URLRewrite').size() <= 1 - matches: - default: - - path: - type: PathPrefix - value: / - description: |- - Matches define conditions used for matching the rule against incoming - HTTP requests. Each match is independent, i.e. this rule will be matched - if **any** one of the matches is satisfied. - - For example, take the following matches configuration: - - ``` - matches: - - path: - value: "/foo" - headers: - - name: "version" - value: "v2" - - path: - value: "/v2/foo" - ``` - - For a request to match against this rule, a request must satisfy - EITHER of the two conditions: - - - path prefixed with `/foo` AND contains the header `version: v2` - - path prefix of `/v2/foo` - - See the documentation for HTTPRouteMatch on how to specify multiple - match conditions that should be ANDed together. - - If no matches are specified, the default is a prefix - path match on "/", which has the effect of matching every - HTTP request. - - Proxy or Load Balancer routing configuration generated from HTTPRoutes - MUST prioritize matches based on the following criteria, continuing on - ties. Across all rules specified on applicable Routes, precedence must be - given to the match having: - - * "Exact" path match. - * "Prefix" path match with largest number of characters. - * Method match. - * Largest number of header matches. - * Largest number of query param matches. - - Note: The precedence of RegularExpression path matches are implementation-specific. - - If ties still exist across multiple Routes, matching precedence MUST be - determined in order of the following criteria, continuing on ties: - - * The oldest Route based on creation timestamp. - * The Route appearing first in alphabetical order by - "{namespace}/{name}". - - If ties still exist within an HTTPRoute, matching precedence MUST be granted - to the FIRST matching rule (in list order) with a match meeting the above - criteria. - - When no rules matching a request have been successfully attached to the - parent a request is coming from, a HTTP 404 status code MUST be returned. - items: - description: "HTTPRouteMatch defines the predicate used to - match requests to a given\naction. Multiple match types - are ANDed together, i.e. the match will\nevaluate to true - only if all conditions are satisfied.\n\nFor example, the - match below will match a HTTP request only if its path\nstarts - with `/foo` AND it contains the `version: v1` header:\n\n```\nmatch:\n\n\tpath:\n\t - \ value: \"/foo\"\n\theaders:\n\t- name: \"version\"\n\t - \ value \"v1\"\n\n```" - properties: - headers: - description: |- - Headers specifies HTTP request header matchers. Multiple match values are - ANDed together, meaning, a request must match all the specified headers - to select the route. - items: - description: |- - HTTPHeaderMatch describes how to select a HTTP route by matching HTTP request - headers. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, only the first - entry with an equivalent name MUST be considered for a match. Subsequent - entries with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - - When a header is repeated in an HTTP request, it is - implementation-specific behavior as to how this is represented. - Generally, proxies should follow the guidance from the RFC: - https://www.rfc-editor.org/rfc/rfc7230.html#section-3.2.2 regarding - processing a repeated header, with special handling for "Set-Cookie". - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - type: - default: Exact - description: |- - Type specifies how to match against the value of the header. - - Support: Core (Exact) - - Support: Implementation-specific (RegularExpression) - - Since RegularExpression HeaderMatchType has implementation-specific - conformance, implementations can support POSIX, PCRE or any other dialects - of regular expressions. Please read the implementation's documentation to - determine the supported dialect. - enum: - - Exact - - RegularExpression - type: string - value: - description: Value is the value of HTTP Header to - be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - method: - description: |- - Method specifies HTTP method matcher. - When specified, this route will be matched only if the request has the - specified method. - - Support: Extended - enum: - - GET - - HEAD - - POST - - PUT - - DELETE - - CONNECT - - OPTIONS - - TRACE - - PATCH - type: string - path: - default: - type: PathPrefix - value: / - description: |- - Path specifies a HTTP request path matcher. If this field is not - specified, a default prefix match on the "/" path is provided. - properties: - type: - default: PathPrefix - description: |- - Type specifies how to match against the path Value. - - Support: Core (Exact, PathPrefix) - - Support: Implementation-specific (RegularExpression) - enum: - - Exact - - PathPrefix - - RegularExpression - type: string - value: - default: / - description: Value of the HTTP path to match against. - maxLength: 1024 - type: string - type: object - x-kubernetes-validations: - - message: value must be an absolute path and start with - '/' when type one of ['Exact', 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? self.value.startsWith(''/'') - : true' - - message: must not contain '//' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.contains(''//'') - : true' - - message: must not contain '/./' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.contains(''/./'') - : true' - - message: must not contain '/../' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.contains(''/../'') - : true' - - message: must not contain '%2f' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.contains(''%2f'') - : true' - - message: must not contain '%2F' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.contains(''%2F'') - : true' - - message: must not contain '#' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.contains(''#'') - : true' - - message: must not end with '/..' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.endsWith(''/..'') - : true' - - message: must not end with '/.' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.endsWith(''/.'') - : true' - - message: type must be one of ['Exact', 'PathPrefix', - 'RegularExpression'] - rule: self.type in ['Exact','PathPrefix'] || self.type - == 'RegularExpression' - - message: must only contain valid characters (matching - ^(?:[-A-Za-z0-9/._~!$&'()*+,;=:@]|[%][0-9a-fA-F]{2})+$) - for types ['Exact', 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? self.value.matches(r"""^(?:[-A-Za-z0-9/._~!$&''()*+,;=:@]|[%][0-9a-fA-F]{2})+$""") - : true' - queryParams: - description: |- - QueryParams specifies HTTP query parameter matchers. Multiple match - values are ANDed together, meaning, a request must match all the - specified query parameters to select the route. - - Support: Extended - items: - description: |- - HTTPQueryParamMatch describes how to select a HTTP route by matching HTTP - query parameters. - properties: - name: - description: |- - Name is the name of the HTTP query param to be matched. This must be an - exact string match. (See - https://tools.ietf.org/html/rfc7230#section-2.7.3). - - If multiple entries specify equivalent query param names, only the first - entry with an equivalent name MUST be considered for a match. Subsequent - entries with an equivalent query param name MUST be ignored. - - If a query param is repeated in an HTTP request, the behavior is - purposely left undefined, since different data planes have different - capabilities. However, it is *recommended* that implementations should - match against the first value of the param if the data plane supports it, - as this behavior is expected in other load balancing contexts outside of - the Gateway API. - - Users SHOULD NOT route traffic based on repeated query params to guard - themselves against potential differences in the implementations. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - type: - default: Exact - description: |- - Type specifies how to match against the value of the query parameter. - - Support: Extended (Exact) - - Support: Implementation-specific (RegularExpression) - - Since RegularExpression QueryParamMatchType has Implementation-specific - conformance, implementations can support POSIX, PCRE or any other - dialects of regular expressions. Please read the implementation's - documentation to determine the supported dialect. - enum: - - Exact - - RegularExpression - type: string - value: - description: Value is the value of HTTP query param - to be matched. - maxLength: 1024 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - maxItems: 64 - type: array - timeouts: - description: |- - Timeouts defines the timeouts that can be configured for an HTTP request. - - Support: Extended - properties: - backendRequest: - description: |- - BackendRequest specifies a timeout for an individual request from the gateway - to a backend. This covers the time from when the request first starts being - sent from the gateway to when the full response has been received from the backend. - - Setting a timeout to the zero duration (e.g. "0s") SHOULD disable the timeout - completely. Implementations that cannot completely disable the timeout MUST - instead interpret the zero duration as the longest possible value to which - the timeout can be set. - - An entire client HTTP transaction with a gateway, covered by the Request timeout, - may result in more than one call from the gateway to the destination backend, - for example, if automatic retries are supported. - - The value of BackendRequest must be a Gateway API Duration string as defined by - GEP-2257. When this field is unspecified, its behavior is implementation-specific; - when specified, the value of BackendRequest must be no more than the value of the - Request timeout (since the Request timeout encompasses the BackendRequest timeout). - - Support: Extended - pattern: ^([0-9]{1,5}(h|m|s|ms)){1,4}$ - type: string - request: - description: |- - Request specifies the maximum duration for a gateway to respond to an HTTP request. - If the gateway has not been able to respond before this deadline is met, the gateway - MUST return a timeout error. - - For example, setting the `rules.timeouts.request` field to the value `10s` in an - `HTTPRoute` will cause a timeout if a client request is taking longer than 10 seconds - to complete. - - Setting a timeout to the zero duration (e.g. "0s") SHOULD disable the timeout - completely. Implementations that cannot completely disable the timeout MUST - instead interpret the zero duration as the longest possible value to which - the timeout can be set. - - This timeout is intended to cover as close to the whole request-response transaction - as possible although an implementation MAY choose to start the timeout after the entire - request stream has been received instead of immediately after the transaction is - initiated by the client. - - The value of Request is a Gateway API Duration string as defined by GEP-2257. When this - field is unspecified, request timeout behavior is implementation-specific. - - Support: Extended - pattern: ^([0-9]{1,5}(h|m|s|ms)){1,4}$ - type: string - type: object - x-kubernetes-validations: - - message: backendRequest timeout cannot be longer than request - timeout - rule: '!(has(self.request) && has(self.backendRequest) && - duration(self.request) != duration(''0s'') && duration(self.backendRequest) - > duration(self.request))' - type: object - x-kubernetes-validations: - - message: RequestRedirect filter must not be used together with - backendRefs - rule: '(has(self.backendRefs) && size(self.backendRefs) > 0) ? - (!has(self.filters) || self.filters.all(f, !has(f.requestRedirect))): - true' - - message: When using RequestRedirect filter with path.replacePrefixMatch, - exactly one PathPrefix match must be specified - rule: '(has(self.filters) && self.filters.exists_one(f, has(f.requestRedirect) - && has(f.requestRedirect.path) && f.requestRedirect.path.type - == ''ReplacePrefixMatch'' && has(f.requestRedirect.path.replacePrefixMatch))) - ? ((size(self.matches) != 1 || !has(self.matches[0].path) || - self.matches[0].path.type != ''PathPrefix'') ? false : true) - : true' - - message: When using URLRewrite filter with path.replacePrefixMatch, - exactly one PathPrefix match must be specified - rule: '(has(self.filters) && self.filters.exists_one(f, has(f.urlRewrite) - && has(f.urlRewrite.path) && f.urlRewrite.path.type == ''ReplacePrefixMatch'' - && has(f.urlRewrite.path.replacePrefixMatch))) ? ((size(self.matches) - != 1 || !has(self.matches[0].path) || self.matches[0].path.type - != ''PathPrefix'') ? false : true) : true' - - message: Within backendRefs, when using RequestRedirect filter - with path.replacePrefixMatch, exactly one PathPrefix match must - be specified - rule: '(has(self.backendRefs) && self.backendRefs.exists_one(b, - (has(b.filters) && b.filters.exists_one(f, has(f.requestRedirect) - && has(f.requestRedirect.path) && f.requestRedirect.path.type - == ''ReplacePrefixMatch'' && has(f.requestRedirect.path.replacePrefixMatch))) - )) ? ((size(self.matches) != 1 || !has(self.matches[0].path) - || self.matches[0].path.type != ''PathPrefix'') ? false : true) - : true' - - message: Within backendRefs, When using URLRewrite filter with - path.replacePrefixMatch, exactly one PathPrefix match must be - specified - rule: '(has(self.backendRefs) && self.backendRefs.exists_one(b, - (has(b.filters) && b.filters.exists_one(f, has(f.urlRewrite) - && has(f.urlRewrite.path) && f.urlRewrite.path.type == ''ReplacePrefixMatch'' - && has(f.urlRewrite.path.replacePrefixMatch))) )) ? ((size(self.matches) - != 1 || !has(self.matches[0].path) || self.matches[0].path.type - != ''PathPrefix'') ? false : true) : true' - maxItems: 16 - type: array - x-kubernetes-validations: - - message: While 16 rules and 64 matches per rule are allowed, the - total number of matches across all rules in a route must be less - than 128 - rule: '(self.size() > 0 ? self[0].matches.size() : 0) + (self.size() - > 1 ? self[1].matches.size() : 0) + (self.size() > 2 ? self[2].matches.size() - : 0) + (self.size() > 3 ? self[3].matches.size() : 0) + (self.size() - > 4 ? self[4].matches.size() : 0) + (self.size() > 5 ? self[5].matches.size() - : 0) + (self.size() > 6 ? self[6].matches.size() : 0) + (self.size() - > 7 ? self[7].matches.size() : 0) + (self.size() > 8 ? self[8].matches.size() - : 0) + (self.size() > 9 ? self[9].matches.size() : 0) + (self.size() - > 10 ? self[10].matches.size() : 0) + (self.size() > 11 ? self[11].matches.size() - : 0) + (self.size() > 12 ? self[12].matches.size() : 0) + (self.size() - > 13 ? self[13].matches.size() : 0) + (self.size() > 14 ? self[14].matches.size() - : 0) + (self.size() > 15 ? self[15].matches.size() : 0) <= 128' - type: object - status: - description: Status defines the current state of HTTPRoute. - properties: - parents: - description: |- - Parents is a list of parent resources (usually Gateways) that are - associated with the route, and the status of the route with respect to - each parent. When this route attaches to a parent, the controller that - manages the parent must add an entry to this list when the controller - first sees the route and should update the entry as appropriate when the - route or gateway is modified. - - Note that parent references that cannot be resolved by an implementation - of this API will not be added to this list. Implementations of this API - can only populate Route status for the Gateways/parent resources they are - responsible for. - - A maximum of 32 Gateways will be represented in this list. An empty list - means the route has not been attached to any Gateway. - items: - description: |- - RouteParentStatus describes the status of a route with respect to an - associated Parent. - properties: - conditions: - description: |- - Conditions describes the status of the route with respect to the Gateway. - Note that the route's availability is also subject to the Gateway's own - status conditions and listener status. - - If the Route's ParentRef specifies an existing Gateway that supports - Routes of this kind AND that Gateway's controller has sufficient access, - then that Gateway's controller MUST set the "Accepted" condition on the - Route, to indicate whether the route has been accepted or rejected by the - Gateway, and why. - - A Route MUST be considered "Accepted" if at least one of the Route's - rules is implemented by the Gateway. - - There are a number of cases where the "Accepted" condition may not be set - due to lack of controller visibility, that includes when: - - * The Route refers to a non-existent parent. - * The Route is of a type that the controller does not support. - * The Route is in a namespace the controller does not have access to. - items: - description: Condition contains details for one aspect of - the current state of this API Resource. - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, - Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: type of condition in CamelCase or in foo.example.com/CamelCase. - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - maxItems: 8 - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - type - x-kubernetes-list-type: map - controllerName: - description: |- - ControllerName is a domain/path string that indicates the name of the - controller that wrote this status. This corresponds with the - controllerName field on GatewayClass. - - Example: "example.net/gateway-controller". - - The format of this field is DOMAIN "/" PATH, where DOMAIN and PATH are - valid Kubernetes names - (https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names). - - Controllers MUST populate this field when writing status. Controllers should ensure that - entries to status populated with their ControllerName are cleaned up when they are no - longer necessary. - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ - type: string - parentRef: - description: |- - ParentRef corresponds with a ParentRef in the spec that this - RouteParentStatus struct describes the status of. - properties: - group: - default: gateway.networking.k8s.io - description: |- - Group is the group of the referent. - When unspecified, "gateway.networking.k8s.io" is inferred. - To set the core API group (such as for a "Service" kind referent), - Group must be explicitly set to "" (empty string). - - Support: Core - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Gateway - description: |- - Kind is kind of the referent. - - There are two kinds of parent resources with "Core" support: - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - Support for other resources is Implementation-Specific. - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: |- - Name is the name of the referent. - - Support: Core - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the referent. When unspecified, this refers - to the local namespace of the Route. - - Note that there are specific rules for ParentRefs which cross namespace - boundaries. Cross-namespace references are only valid if they are explicitly - allowed by something in the namespace they are referring to. For example: - Gateway has the AllowedRoutes field, and ReferenceGrant provides a - generic way to enable any other kind of cross-namespace reference. - - - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port is the network port this Route targets. It can be interpreted - differently based on the type of parent resource. - - When the parent resource is a Gateway, this targets all listeners - listening on the specified port that also support this kind of Route(and - select this Route). It's not recommended to set `Port` unless the - networking behaviors specified in a Route must apply to a specific port - as opposed to a listener(s) whose port(s) may be changed. When both Port - and SectionName are specified, the name and port of the selected listener - must match both specified values. - - - - Implementations MAY choose to support other parent resources. - Implementations supporting other types of parent resources MUST clearly - document how/if Port is interpreted. - - For the purpose of status, an attachment is considered successful as - long as the parent resource accepts it partially. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment - from the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, - the Route MUST be considered detached from the Gateway. - - Support: Extended - format: int32 - maximum: 65535 - minimum: 1 - type: integer - sectionName: - description: |- - SectionName is the name of a section within the target resource. In the - following resources, SectionName is interpreted as the following: - - * Gateway: Listener name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - * Service: Port name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - - Implementations MAY choose to support attaching Routes to other resources. - If that is the case, they MUST clearly document how SectionName is - interpreted. - - When unspecified (empty string), this will reference the entire resource. - For the purpose of status, an attachment is considered successful if at - least one section in the parent resource accepts it. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from - the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, the - Route MUST be considered detached from the Gateway. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - required: - - name - type: object - required: - - controllerName - - parentRef - type: object - maxItems: 32 - type: array - required: - - parents - type: object - required: - - spec - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.hostnames - name: Hostnames - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1beta1 - schema: - openAPIV3Schema: - description: |- - HTTPRoute provides a way to route HTTP requests. This includes the capability - to match requests by hostname, path, header, or query param. Filters can be - used to specify additional processing steps. Backends specify where matching - requests should be routed. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: Spec defines the desired state of HTTPRoute. - properties: - hostnames: - description: |- - Hostnames defines a set of hostnames that should match against the HTTP Host - header to select a HTTPRoute used to process the request. Implementations - MUST ignore any port value specified in the HTTP Host header while - performing a match and (absent of any applicable header modification - configuration) MUST forward this header unmodified to the backend. - - Valid values for Hostnames are determined by RFC 1123 definition of a - hostname with 2 notable exceptions: - - 1. IPs are not allowed. - 2. A hostname may be prefixed with a wildcard label (`*.`). The wildcard - label must appear by itself as the first label. - - If a hostname is specified by both the Listener and HTTPRoute, there - must be at least one intersecting hostname for the HTTPRoute to be - attached to the Listener. For example: - - * A Listener with `test.example.com` as the hostname matches HTTPRoutes - that have either not specified any hostnames, or have specified at - least one of `test.example.com` or `*.example.com`. - * A Listener with `*.example.com` as the hostname matches HTTPRoutes - that have either not specified any hostnames or have specified at least - one hostname that matches the Listener hostname. For example, - `*.example.com`, `test.example.com`, and `foo.test.example.com` would - all match. On the other hand, `example.com` and `test.example.net` would - not match. - - Hostnames that are prefixed with a wildcard label (`*.`) are interpreted - as a suffix match. That means that a match for `*.example.com` would match - both `test.example.com`, and `foo.test.example.com`, but not `example.com`. - - If both the Listener and HTTPRoute have specified hostnames, any - HTTPRoute hostnames that do not match the Listener hostname MUST be - ignored. For example, if a Listener specified `*.example.com`, and the - HTTPRoute specified `test.example.com` and `test.example.net`, - `test.example.net` must not be considered for a match. - - If both the Listener and HTTPRoute have specified hostnames, and none - match with the criteria above, then the HTTPRoute is not accepted. The - implementation must raise an 'Accepted' Condition with a status of - `False` in the corresponding RouteParentStatus. - - In the event that multiple HTTPRoutes specify intersecting hostnames (e.g. - overlapping wildcard matching and exact matching hostnames), precedence must - be given to rules from the HTTPRoute with the largest number of: - - * Characters in a matching non-wildcard hostname. - * Characters in a matching hostname. - - If ties exist across multiple Routes, the matching precedence rules for - HTTPRouteMatches takes over. - - Support: Core - items: - description: |- - Hostname is the fully qualified domain name of a network host. This matches - the RFC 1123 definition of a hostname with 2 notable exceptions: - - 1. IPs are not allowed. - 2. A hostname may be prefixed with a wildcard label (`*.`). The wildcard - label must appear by itself as the first label. - - Hostname can be "precise" which is a domain name without the terminating - dot of a network host (e.g. "foo.example.com") or "wildcard", which is a - domain name prefixed with a single wildcard label (e.g. `*.example.com`). - - Note that as per RFC1035 and RFC1123, a *label* must consist of lower case - alphanumeric characters or '-', and must start and end with an alphanumeric - character. No other punctuation is allowed. - maxLength: 253 - minLength: 1 - pattern: ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - maxItems: 16 - type: array - parentRefs: - description: |+ - ParentRefs references the resources (usually Gateways) that a Route wants - to be attached to. Note that the referenced parent resource needs to - allow this for the attachment to be complete. For Gateways, that means - the Gateway needs to allow attachment from Routes of this kind and - namespace. For Services, that means the Service must either be in the same - namespace for a "producer" route, or the mesh implementation must support - and allow "consumer" routes for the referenced Service. ReferenceGrant is - not applicable for governing ParentRefs to Services - it is not possible to - create a "producer" route for a Service in a different namespace from the - Route. - - There are two kinds of parent resources with "Core" support: - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - This API may be extended in the future to support additional kinds of parent - resources. - - ParentRefs must be _distinct_. This means either that: - - * They select different objects. If this is the case, then parentRef - entries are distinct. In terms of fields, this means that the - multi-part key defined by `group`, `kind`, `namespace`, and `name` must - be unique across all parentRef entries in the Route. - * They do not select different objects, but for each optional field used, - each ParentRef that selects the same object must set the same set of - optional fields to different values. If one ParentRef sets a - combination of optional fields, all must set the same combination. - - Some examples: - - * If one ParentRef sets `sectionName`, all ParentRefs referencing the - same object must also set `sectionName`. - * If one ParentRef sets `port`, all ParentRefs referencing the same - object must also set `port`. - * If one ParentRef sets `sectionName` and `port`, all ParentRefs - referencing the same object must also set `sectionName` and `port`. - - It is possible to separately reference multiple distinct objects that may - be collapsed by an implementation. For example, some implementations may - choose to merge compatible Gateway Listeners together. If that is the - case, the list of routes attached to those resources should also be - merged. - - Note that for ParentRefs that cross namespace boundaries, there are specific - rules. Cross-namespace references are only valid if they are explicitly - allowed by something in the namespace they are referring to. For example, - Gateway has the AllowedRoutes field, and ReferenceGrant provides a - generic way to enable other kinds of cross-namespace reference. - - - - - - - items: - description: |- - ParentReference identifies an API object (usually a Gateway) that can be considered - a parent of this resource (usually a route). There are two kinds of parent resources - with "Core" support: - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - This API may be extended in the future to support additional kinds of parent - resources. - - The API object must be valid in the cluster; the Group and Kind must - be registered in the cluster for this reference to be valid. - properties: - group: - default: gateway.networking.k8s.io - description: |- - Group is the group of the referent. - When unspecified, "gateway.networking.k8s.io" is inferred. - To set the core API group (such as for a "Service" kind referent), - Group must be explicitly set to "" (empty string). - - Support: Core - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Gateway - description: |- - Kind is kind of the referent. - - There are two kinds of parent resources with "Core" support: - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - Support for other resources is Implementation-Specific. - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: |- - Name is the name of the referent. - - Support: Core - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the referent. When unspecified, this refers - to the local namespace of the Route. - - Note that there are specific rules for ParentRefs which cross namespace - boundaries. Cross-namespace references are only valid if they are explicitly - allowed by something in the namespace they are referring to. For example: - Gateway has the AllowedRoutes field, and ReferenceGrant provides a - generic way to enable any other kind of cross-namespace reference. - - - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port is the network port this Route targets. It can be interpreted - differently based on the type of parent resource. - - When the parent resource is a Gateway, this targets all listeners - listening on the specified port that also support this kind of Route(and - select this Route). It's not recommended to set `Port` unless the - networking behaviors specified in a Route must apply to a specific port - as opposed to a listener(s) whose port(s) may be changed. When both Port - and SectionName are specified, the name and port of the selected listener - must match both specified values. - - - - Implementations MAY choose to support other parent resources. - Implementations supporting other types of parent resources MUST clearly - document how/if Port is interpreted. - - For the purpose of status, an attachment is considered successful as - long as the parent resource accepts it partially. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment - from the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, - the Route MUST be considered detached from the Gateway. - - Support: Extended - format: int32 - maximum: 65535 - minimum: 1 - type: integer - sectionName: - description: |- - SectionName is the name of a section within the target resource. In the - following resources, SectionName is interpreted as the following: - - * Gateway: Listener name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - * Service: Port name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - - Implementations MAY choose to support attaching Routes to other resources. - If that is the case, they MUST clearly document how SectionName is - interpreted. - - When unspecified (empty string), this will reference the entire resource. - For the purpose of status, an attachment is considered successful if at - least one section in the parent resource accepts it. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from - the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, the - Route MUST be considered detached from the Gateway. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - required: - - name - type: object - maxItems: 32 - type: array - x-kubernetes-validations: - - message: sectionName must be specified when parentRefs includes - 2 or more references to the same parent - rule: 'self.all(p1, self.all(p2, p1.group == p2.group && p1.kind - == p2.kind && p1.name == p2.name && (((!has(p1.__namespace__) - || p1.__namespace__ == '''') && (!has(p2.__namespace__) || p2.__namespace__ - == '''')) || (has(p1.__namespace__) && has(p2.__namespace__) && - p1.__namespace__ == p2.__namespace__ )) ? ((!has(p1.sectionName) - || p1.sectionName == '''') == (!has(p2.sectionName) || p2.sectionName - == '''')) : true))' - - message: sectionName must be unique when parentRefs includes 2 or - more references to the same parent - rule: self.all(p1, self.exists_one(p2, p1.group == p2.group && p1.kind - == p2.kind && p1.name == p2.name && (((!has(p1.__namespace__) - || p1.__namespace__ == '') && (!has(p2.__namespace__) || p2.__namespace__ - == '')) || (has(p1.__namespace__) && has(p2.__namespace__) && - p1.__namespace__ == p2.__namespace__ )) && (((!has(p1.sectionName) - || p1.sectionName == '') && (!has(p2.sectionName) || p2.sectionName - == '')) || (has(p1.sectionName) && has(p2.sectionName) && p1.sectionName - == p2.sectionName)))) - rules: - default: - - matches: - - path: - type: PathPrefix - value: / - description: |+ - Rules are a list of HTTP matchers, filters and actions. - - items: - description: |- - HTTPRouteRule defines semantics for matching an HTTP request based on - conditions (matches), processing it (filters), and forwarding the request to - an API object (backendRefs). - properties: - backendRefs: - description: |- - BackendRefs defines the backend(s) where matching requests should be - sent. - - Failure behavior here depends on how many BackendRefs are specified and - how many are invalid. - - If *all* entries in BackendRefs are invalid, and there are also no filters - specified in this route rule, *all* traffic which matches this rule MUST - receive a 500 status code. - - See the HTTPBackendRef definition for the rules about what makes a single - HTTPBackendRef invalid. - - When a HTTPBackendRef is invalid, 500 status codes MUST be returned for - requests that would have otherwise been routed to an invalid backend. If - multiple backends are specified, and some are invalid, the proportion of - requests that would otherwise have been routed to an invalid backend - MUST receive a 500 status code. - - For example, if two backends are specified with equal weights, and one is - invalid, 50 percent of traffic must receive a 500. Implementations may - choose how that 50 percent is determined. - - When a HTTPBackendRef refers to a Service that has no ready endpoints, - implementations SHOULD return a 503 for requests to that backend instead. - If an implementation chooses to do this, all of the above rules for 500 responses - MUST also apply for responses that return a 503. - - Support: Core for Kubernetes Service - - Support: Extended for Kubernetes ServiceImport - - Support: Implementation-specific for any other resource - - Support for weight: Core - items: - description: |- - HTTPBackendRef defines how a HTTPRoute forwards a HTTP request. - - Note that when a namespace different than the local namespace is specified, a - ReferenceGrant object is required in the referent namespace to allow that - namespace's owner to accept the reference. See the ReferenceGrant - documentation for details. - - - - When the BackendRef points to a Kubernetes Service, implementations SHOULD - honor the appProtocol field if it is set for the target Service Port. - - Implementations supporting appProtocol SHOULD recognize the Kubernetes - Standard Application Protocols defined in KEP-3726. - - If a Service appProtocol isn't specified, an implementation MAY infer the - backend protocol through its own means. Implementations MAY infer the - protocol from the Route type referring to the backend Service. - - If a Route is not able to send traffic to the backend using the specified - protocol then the backend is considered invalid. Implementations MUST set the - "ResolvedRefs" condition to "False" with the "UnsupportedProtocol" reason. - - - properties: - filters: - description: |- - Filters defined at this level should be executed if and only if the - request is being forwarded to the backend defined here. - - Support: Implementation-specific (For broader support of filters, use the - Filters field in HTTPRouteRule.) - items: - description: |- - HTTPRouteFilter defines processing steps that must be completed during the - request or response lifecycle. HTTPRouteFilters are meant as an extension - point to express processing that may be done in Gateway implementations. Some - examples include request or response modification, implementing - authentication strategies, rate-limiting, and traffic shaping. API - guarantee/conformance is defined based on the type of the filter. - properties: - extensionRef: - description: |- - ExtensionRef is an optional, implementation-specific extension to the - "filter" behavior. For example, resource "myroutefilter" in group - "networking.example.net"). ExtensionRef MUST NOT be used for core and - extended filters. - - This filter can be used multiple times within the same rule. - - Support: Implementation-specific - properties: - group: - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: Kind is kind of the referent. For - example "HTTPRoute" or "Service". - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - required: - - group - - kind - - name - type: object - requestHeaderModifier: - description: |- - RequestHeaderModifier defines a schema for a filter that modifies request - headers. - - Support: Core - properties: - add: - description: |- - Add adds the given header(s) (name, value) to the request - before the action. It appends to any existing values associated - with the header name. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - add: - - name: "my-header" - value: "bar,baz" - - Output: - GET /foo HTTP/1.1 - my-header: foo,bar,baz - items: - description: HTTPHeader represents an HTTP - Header name and value as defined by RFC - 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP - Header to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - remove: - description: |- - Remove the given header(s) from the HTTP request before the action. The - value of Remove is a list of HTTP header names. Note that the header - names are case-insensitive (see - https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). - - Input: - GET /foo HTTP/1.1 - my-header1: foo - my-header2: bar - my-header3: baz - - Config: - remove: ["my-header1", "my-header3"] - - Output: - GET /foo HTTP/1.1 - my-header2: bar - items: - type: string - maxItems: 16 - type: array - x-kubernetes-list-type: set - set: - description: |- - Set overwrites the request with the given header (name, value) - before the action. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - set: - - name: "my-header" - value: "bar" - - Output: - GET /foo HTTP/1.1 - my-header: bar - items: - description: HTTPHeader represents an HTTP - Header name and value as defined by RFC - 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP - Header to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - requestMirror: - description: |+ - RequestMirror defines a schema for a filter that mirrors requests. - Requests are sent to the specified destination, but responses from - that destination are ignored. - - This filter can be used multiple times within the same rule. Note that - not all implementations will be able to support mirroring to multiple - backends. - - Support: Extended - - properties: - backendRef: - description: |- - BackendRef references a resource where mirrored requests are sent. - - Mirrored requests must be sent only to a single destination endpoint - within this BackendRef, irrespective of how many endpoints are present - within this BackendRef. - - If the referent cannot be found, this BackendRef is invalid and must be - dropped from the Gateway. The controller must ensure the "ResolvedRefs" - condition on the Route status is set to `status: False` and not configure - this backend in the underlying implementation. - - If there is a cross-namespace reference to an *existing* object - that is not allowed by a ReferenceGrant, the controller must ensure the - "ResolvedRefs" condition on the Route is set to `status: False`, - with the "RefNotPermitted" reason and not configure this backend in the - underlying implementation. - - In either error case, the Message of the `ResolvedRefs` Condition - should be used to provide more detail about the problem. - - Support: Extended for Kubernetes Service - - Support: Implementation-specific for any other resource - properties: - group: - default: "" - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Service - description: |- - Kind is the Kubernetes resource kind of the referent. For example - "Service". - - Defaults to "Service" when not specified. - - ExternalName services can refer to CNAME DNS records that may live - outside of the cluster and as such are difficult to reason about in - terms of conformance. They also may not be safe to forward to (see - CVE-2021-25740 for more information). Implementations SHOULD NOT - support ExternalName Services. - - Support: Core (Services with a type other than ExternalName) - - Support: Implementation-specific (Services with type ExternalName) - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the backend. When unspecified, the local - namespace is inferred. - - Note that when a namespace different than the local namespace is specified, - a ReferenceGrant object is required in the referent namespace to allow that - namespace's owner to accept the reference. See the ReferenceGrant - documentation for details. - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port specifies the destination port number to use for this resource. - Port is required when the referent is a Kubernetes Service. In this - case, the port number is the service port number, not the target port. - For other resources, destination port might be derived from the referent - resource or this field. - format: int32 - maximum: 65535 - minimum: 1 - type: integer - required: - - name - type: object - x-kubernetes-validations: - - message: Must have port for Service reference - rule: '(size(self.group) == 0 && self.kind - == ''Service'') ? has(self.port) : true' - required: - - backendRef - type: object - requestRedirect: - description: |- - RequestRedirect defines a schema for a filter that responds to the - request with an HTTP redirection. - - Support: Core - properties: - hostname: - description: |- - Hostname is the hostname to be used in the value of the `Location` - header in the response. - When empty, the hostname in the `Host` header of the request is used. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - path: - description: |- - Path defines parameters used to modify the path of the incoming request. - The modified path is then used to construct the `Location` header. When - empty, the request path is used as-is. - - Support: Extended - properties: - replaceFullPath: - description: |- - ReplaceFullPath specifies the value with which to replace the full path - of a request during a rewrite or redirect. - maxLength: 1024 - type: string - replacePrefixMatch: - description: |- - ReplacePrefixMatch specifies the value with which to replace the prefix - match of a request during a rewrite or redirect. For example, a request - to "/foo/bar" with a prefix match of "/foo" and a ReplacePrefixMatch - of "/xyz" would be modified to "/xyz/bar". - - Note that this matches the behavior of the PathPrefix match type. This - matches full path elements. A path element refers to the list of labels - in the path split by the `/` separator. When specified, a trailing `/` is - ignored. For example, the paths `/abc`, `/abc/`, and `/abc/def` would all - match the prefix `/abc`, but the path `/abcd` would not. - - ReplacePrefixMatch is only compatible with a `PathPrefix` HTTPRouteMatch. - Using any other HTTPRouteMatch type on the same HTTPRouteRule will result in - the implementation setting the Accepted Condition for the Route to `status: False`. - - Request Path | Prefix Match | Replace Prefix | Modified Path - maxLength: 1024 - type: string - type: - description: |- - Type defines the type of path modifier. Additional types may be - added in a future release of the API. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - enum: - - ReplaceFullPath - - ReplacePrefixMatch - type: string - required: - - type - type: object - x-kubernetes-validations: - - message: replaceFullPath must be specified - when type is set to 'ReplaceFullPath' - rule: 'self.type == ''ReplaceFullPath'' ? - has(self.replaceFullPath) : true' - - message: type must be 'ReplaceFullPath' when - replaceFullPath is set - rule: 'has(self.replaceFullPath) ? self.type - == ''ReplaceFullPath'' : true' - - message: replacePrefixMatch must be specified - when type is set to 'ReplacePrefixMatch' - rule: 'self.type == ''ReplacePrefixMatch'' - ? has(self.replacePrefixMatch) : true' - - message: type must be 'ReplacePrefixMatch' - when replacePrefixMatch is set - rule: 'has(self.replacePrefixMatch) ? self.type - == ''ReplacePrefixMatch'' : true' - port: - description: |- - Port is the port to be used in the value of the `Location` - header in the response. - - If no port is specified, the redirect port MUST be derived using the - following rules: - - * If redirect scheme is not-empty, the redirect port MUST be the well-known - port associated with the redirect scheme. Specifically "http" to port 80 - and "https" to port 443. If the redirect scheme does not have a - well-known port, the listener port of the Gateway SHOULD be used. - * If redirect scheme is empty, the redirect port MUST be the Gateway - Listener port. - - Implementations SHOULD NOT add the port number in the 'Location' - header in the following cases: - - * A Location header that will use HTTP (whether that is determined via - the Listener protocol or the Scheme field) _and_ use port 80. - * A Location header that will use HTTPS (whether that is determined via - the Listener protocol or the Scheme field) _and_ use port 443. - - Support: Extended - format: int32 - maximum: 65535 - minimum: 1 - type: integer - scheme: - description: |- - Scheme is the scheme to be used in the value of the `Location` header in - the response. When empty, the scheme of the request is used. - - Scheme redirects can affect the port of the redirect, for more information, - refer to the documentation for the port field of this filter. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - - Support: Extended - enum: - - http - - https - type: string - statusCode: - default: 302 - description: |- - StatusCode is the HTTP status code to be used in response. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - - Support: Core - enum: - - 301 - - 302 - type: integer - type: object - responseHeaderModifier: - description: |- - ResponseHeaderModifier defines a schema for a filter that modifies response - headers. - - Support: Extended - properties: - add: - description: |- - Add adds the given header(s) (name, value) to the request - before the action. It appends to any existing values associated - with the header name. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - add: - - name: "my-header" - value: "bar,baz" - - Output: - GET /foo HTTP/1.1 - my-header: foo,bar,baz - items: - description: HTTPHeader represents an HTTP - Header name and value as defined by RFC - 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP - Header to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - remove: - description: |- - Remove the given header(s) from the HTTP request before the action. The - value of Remove is a list of HTTP header names. Note that the header - names are case-insensitive (see - https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). - - Input: - GET /foo HTTP/1.1 - my-header1: foo - my-header2: bar - my-header3: baz - - Config: - remove: ["my-header1", "my-header3"] - - Output: - GET /foo HTTP/1.1 - my-header2: bar - items: - type: string - maxItems: 16 - type: array - x-kubernetes-list-type: set - set: - description: |- - Set overwrites the request with the given header (name, value) - before the action. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - set: - - name: "my-header" - value: "bar" - - Output: - GET /foo HTTP/1.1 - my-header: bar - items: - description: HTTPHeader represents an HTTP - Header name and value as defined by RFC - 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP - Header to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: - description: |- - Type identifies the type of filter to apply. As with other API fields, - types are classified into three conformance levels: - - - Core: Filter types and their corresponding configuration defined by - "Support: Core" in this package, e.g. "RequestHeaderModifier". All - implementations must support core filters. - - - Extended: Filter types and their corresponding configuration defined by - "Support: Extended" in this package, e.g. "RequestMirror". Implementers - are encouraged to support extended filters. - - - Implementation-specific: Filters that are defined and supported by - specific vendors. - In the future, filters showing convergence in behavior across multiple - implementations will be considered for inclusion in extended or core - conformance levels. Filter-specific configuration for such filters - is specified using the ExtensionRef field. `Type` should be set to - "ExtensionRef" for custom filters. - - Implementers are encouraged to define custom implementation types to - extend the core API with implementation-specific behavior. - - If a reference to a custom filter type cannot be resolved, the filter - MUST NOT be skipped. Instead, requests that would have been processed by - that filter MUST receive a HTTP error response. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - enum: - - RequestHeaderModifier - - ResponseHeaderModifier - - RequestMirror - - RequestRedirect - - URLRewrite - - ExtensionRef - type: string - urlRewrite: - description: |- - URLRewrite defines a schema for a filter that modifies a request during forwarding. - - Support: Extended - properties: - hostname: - description: |- - Hostname is the value to be used to replace the Host header value during - forwarding. - - Support: Extended - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - path: - description: |- - Path defines a path rewrite. - - Support: Extended - properties: - replaceFullPath: - description: |- - ReplaceFullPath specifies the value with which to replace the full path - of a request during a rewrite or redirect. - maxLength: 1024 - type: string - replacePrefixMatch: - description: |- - ReplacePrefixMatch specifies the value with which to replace the prefix - match of a request during a rewrite or redirect. For example, a request - to "/foo/bar" with a prefix match of "/foo" and a ReplacePrefixMatch - of "/xyz" would be modified to "/xyz/bar". - - Note that this matches the behavior of the PathPrefix match type. This - matches full path elements. A path element refers to the list of labels - in the path split by the `/` separator. When specified, a trailing `/` is - ignored. For example, the paths `/abc`, `/abc/`, and `/abc/def` would all - match the prefix `/abc`, but the path `/abcd` would not. - - ReplacePrefixMatch is only compatible with a `PathPrefix` HTTPRouteMatch. - Using any other HTTPRouteMatch type on the same HTTPRouteRule will result in - the implementation setting the Accepted Condition for the Route to `status: False`. - - Request Path | Prefix Match | Replace Prefix | Modified Path - maxLength: 1024 - type: string - type: - description: |- - Type defines the type of path modifier. Additional types may be - added in a future release of the API. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - enum: - - ReplaceFullPath - - ReplacePrefixMatch - type: string - required: - - type - type: object - x-kubernetes-validations: - - message: replaceFullPath must be specified - when type is set to 'ReplaceFullPath' - rule: 'self.type == ''ReplaceFullPath'' ? - has(self.replaceFullPath) : true' - - message: type must be 'ReplaceFullPath' when - replaceFullPath is set - rule: 'has(self.replaceFullPath) ? self.type - == ''ReplaceFullPath'' : true' - - message: replacePrefixMatch must be specified - when type is set to 'ReplacePrefixMatch' - rule: 'self.type == ''ReplacePrefixMatch'' - ? has(self.replacePrefixMatch) : true' - - message: type must be 'ReplacePrefixMatch' - when replacePrefixMatch is set - rule: 'has(self.replacePrefixMatch) ? self.type - == ''ReplacePrefixMatch'' : true' - type: object - required: - - type - type: object - x-kubernetes-validations: - - message: filter.requestHeaderModifier must be nil - if the filter.type is not RequestHeaderModifier - rule: '!(has(self.requestHeaderModifier) && self.type - != ''RequestHeaderModifier'')' - - message: filter.requestHeaderModifier must be specified - for RequestHeaderModifier filter.type - rule: '!(!has(self.requestHeaderModifier) && self.type - == ''RequestHeaderModifier'')' - - message: filter.responseHeaderModifier must be nil - if the filter.type is not ResponseHeaderModifier - rule: '!(has(self.responseHeaderModifier) && self.type - != ''ResponseHeaderModifier'')' - - message: filter.responseHeaderModifier must be specified - for ResponseHeaderModifier filter.type - rule: '!(!has(self.responseHeaderModifier) && self.type - == ''ResponseHeaderModifier'')' - - message: filter.requestMirror must be nil if the filter.type - is not RequestMirror - rule: '!(has(self.requestMirror) && self.type != ''RequestMirror'')' - - message: filter.requestMirror must be specified for - RequestMirror filter.type - rule: '!(!has(self.requestMirror) && self.type == - ''RequestMirror'')' - - message: filter.requestRedirect must be nil if the - filter.type is not RequestRedirect - rule: '!(has(self.requestRedirect) && self.type != - ''RequestRedirect'')' - - message: filter.requestRedirect must be specified - for RequestRedirect filter.type - rule: '!(!has(self.requestRedirect) && self.type == - ''RequestRedirect'')' - - message: filter.urlRewrite must be nil if the filter.type - is not URLRewrite - rule: '!(has(self.urlRewrite) && self.type != ''URLRewrite'')' - - message: filter.urlRewrite must be specified for URLRewrite - filter.type - rule: '!(!has(self.urlRewrite) && self.type == ''URLRewrite'')' - - message: filter.extensionRef must be nil if the filter.type - is not ExtensionRef - rule: '!(has(self.extensionRef) && self.type != ''ExtensionRef'')' - - message: filter.extensionRef must be specified for - ExtensionRef filter.type - rule: '!(!has(self.extensionRef) && self.type == ''ExtensionRef'')' - maxItems: 16 - type: array - x-kubernetes-validations: - - message: May specify either httpRouteFilterRequestRedirect - or httpRouteFilterRequestRewrite, but not both - rule: '!(self.exists(f, f.type == ''RequestRedirect'') - && self.exists(f, f.type == ''URLRewrite''))' - - message: May specify either httpRouteFilterRequestRedirect - or httpRouteFilterRequestRewrite, but not both - rule: '!(self.exists(f, f.type == ''RequestRedirect'') - && self.exists(f, f.type == ''URLRewrite''))' - - message: RequestHeaderModifier filter cannot be repeated - rule: self.filter(f, f.type == 'RequestHeaderModifier').size() - <= 1 - - message: ResponseHeaderModifier filter cannot be repeated - rule: self.filter(f, f.type == 'ResponseHeaderModifier').size() - <= 1 - - message: RequestRedirect filter cannot be repeated - rule: self.filter(f, f.type == 'RequestRedirect').size() - <= 1 - - message: URLRewrite filter cannot be repeated - rule: self.filter(f, f.type == 'URLRewrite').size() - <= 1 - group: - default: "" - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Service - description: |- - Kind is the Kubernetes resource kind of the referent. For example - "Service". - - Defaults to "Service" when not specified. - - ExternalName services can refer to CNAME DNS records that may live - outside of the cluster and as such are difficult to reason about in - terms of conformance. They also may not be safe to forward to (see - CVE-2021-25740 for more information). Implementations SHOULD NOT - support ExternalName Services. - - Support: Core (Services with a type other than ExternalName) - - Support: Implementation-specific (Services with type ExternalName) - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the backend. When unspecified, the local - namespace is inferred. - - Note that when a namespace different than the local namespace is specified, - a ReferenceGrant object is required in the referent namespace to allow that - namespace's owner to accept the reference. See the ReferenceGrant - documentation for details. - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port specifies the destination port number to use for this resource. - Port is required when the referent is a Kubernetes Service. In this - case, the port number is the service port number, not the target port. - For other resources, destination port might be derived from the referent - resource or this field. - format: int32 - maximum: 65535 - minimum: 1 - type: integer - weight: - default: 1 - description: |- - Weight specifies the proportion of requests forwarded to the referenced - backend. This is computed as weight/(sum of all weights in this - BackendRefs list). For non-zero values, there may be some epsilon from - the exact proportion defined here depending on the precision an - implementation supports. Weight is not a percentage and the sum of - weights does not need to equal 100. - - If only one backend is specified and it has a weight greater than 0, 100% - of the traffic is forwarded to that backend. If weight is set to 0, no - traffic should be forwarded for this entry. If unspecified, weight - defaults to 1. - - Support for this field varies based on the context where used. - format: int32 - maximum: 1000000 - minimum: 0 - type: integer - required: - - name - type: object - x-kubernetes-validations: - - message: Must have port for Service reference - rule: '(size(self.group) == 0 && self.kind == ''Service'') - ? has(self.port) : true' - maxItems: 16 - type: array - filters: - description: |- - Filters define the filters that are applied to requests that match - this rule. - - Wherever possible, implementations SHOULD implement filters in the order - they are specified. - - Implementations MAY choose to implement this ordering strictly, rejecting - any combination or order of filters that can not be supported. If implementations - choose a strict interpretation of filter ordering, they MUST clearly document - that behavior. - - To reject an invalid combination or order of filters, implementations SHOULD - consider the Route Rules with this configuration invalid. If all Route Rules - in a Route are invalid, the entire Route would be considered invalid. If only - a portion of Route Rules are invalid, implementations MUST set the - "PartiallyInvalid" condition for the Route. - - Conformance-levels at this level are defined based on the type of filter: - - - ALL core filters MUST be supported by all implementations. - - Implementers are encouraged to support extended filters. - - Implementation-specific custom filters have no API guarantees across - implementations. - - Specifying the same filter multiple times is not supported unless explicitly - indicated in the filter. - - All filters are expected to be compatible with each other except for the - URLRewrite and RequestRedirect filters, which may not be combined. If an - implementation can not support other combinations of filters, they must clearly - document that limitation. In cases where incompatible or unsupported - filters are specified and cause the `Accepted` condition to be set to status - `False`, implementations may use the `IncompatibleFilters` reason to specify - this configuration error. - - Support: Core - items: - description: |- - HTTPRouteFilter defines processing steps that must be completed during the - request or response lifecycle. HTTPRouteFilters are meant as an extension - point to express processing that may be done in Gateway implementations. Some - examples include request or response modification, implementing - authentication strategies, rate-limiting, and traffic shaping. API - guarantee/conformance is defined based on the type of the filter. - properties: - extensionRef: - description: |- - ExtensionRef is an optional, implementation-specific extension to the - "filter" behavior. For example, resource "myroutefilter" in group - "networking.example.net"). ExtensionRef MUST NOT be used for core and - extended filters. - - This filter can be used multiple times within the same rule. - - Support: Implementation-specific - properties: - group: - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: Kind is kind of the referent. For example - "HTTPRoute" or "Service". - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - required: - - group - - kind - - name - type: object - requestHeaderModifier: - description: |- - RequestHeaderModifier defines a schema for a filter that modifies request - headers. - - Support: Core - properties: - add: - description: |- - Add adds the given header(s) (name, value) to the request - before the action. It appends to any existing values associated - with the header name. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - add: - - name: "my-header" - value: "bar,baz" - - Output: - GET /foo HTTP/1.1 - my-header: foo,bar,baz - items: - description: HTTPHeader represents an HTTP Header - name and value as defined by RFC 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP Header - to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - remove: - description: |- - Remove the given header(s) from the HTTP request before the action. The - value of Remove is a list of HTTP header names. Note that the header - names are case-insensitive (see - https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). - - Input: - GET /foo HTTP/1.1 - my-header1: foo - my-header2: bar - my-header3: baz - - Config: - remove: ["my-header1", "my-header3"] - - Output: - GET /foo HTTP/1.1 - my-header2: bar - items: - type: string - maxItems: 16 - type: array - x-kubernetes-list-type: set - set: - description: |- - Set overwrites the request with the given header (name, value) - before the action. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - set: - - name: "my-header" - value: "bar" - - Output: - GET /foo HTTP/1.1 - my-header: bar - items: - description: HTTPHeader represents an HTTP Header - name and value as defined by RFC 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP Header - to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - requestMirror: - description: |+ - RequestMirror defines a schema for a filter that mirrors requests. - Requests are sent to the specified destination, but responses from - that destination are ignored. - - This filter can be used multiple times within the same rule. Note that - not all implementations will be able to support mirroring to multiple - backends. - - Support: Extended - - properties: - backendRef: - description: |- - BackendRef references a resource where mirrored requests are sent. - - Mirrored requests must be sent only to a single destination endpoint - within this BackendRef, irrespective of how many endpoints are present - within this BackendRef. - - If the referent cannot be found, this BackendRef is invalid and must be - dropped from the Gateway. The controller must ensure the "ResolvedRefs" - condition on the Route status is set to `status: False` and not configure - this backend in the underlying implementation. - - If there is a cross-namespace reference to an *existing* object - that is not allowed by a ReferenceGrant, the controller must ensure the - "ResolvedRefs" condition on the Route is set to `status: False`, - with the "RefNotPermitted" reason and not configure this backend in the - underlying implementation. - - In either error case, the Message of the `ResolvedRefs` Condition - should be used to provide more detail about the problem. - - Support: Extended for Kubernetes Service - - Support: Implementation-specific for any other resource - properties: - group: - default: "" - description: |- - Group is the group of the referent. For example, "gateway.networking.k8s.io". - When unspecified or empty string, core API group is inferred. - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Service - description: |- - Kind is the Kubernetes resource kind of the referent. For example - "Service". - - Defaults to "Service" when not specified. - - ExternalName services can refer to CNAME DNS records that may live - outside of the cluster and as such are difficult to reason about in - terms of conformance. They also may not be safe to forward to (see - CVE-2021-25740 for more information). Implementations SHOULD NOT - support ExternalName Services. - - Support: Core (Services with a type other than ExternalName) - - Support: Implementation-specific (Services with type ExternalName) - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: Name is the name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the backend. When unspecified, the local - namespace is inferred. - - Note that when a namespace different than the local namespace is specified, - a ReferenceGrant object is required in the referent namespace to allow that - namespace's owner to accept the reference. See the ReferenceGrant - documentation for details. - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port specifies the destination port number to use for this resource. - Port is required when the referent is a Kubernetes Service. In this - case, the port number is the service port number, not the target port. - For other resources, destination port might be derived from the referent - resource or this field. - format: int32 - maximum: 65535 - minimum: 1 - type: integer - required: - - name - type: object - x-kubernetes-validations: - - message: Must have port for Service reference - rule: '(size(self.group) == 0 && self.kind == ''Service'') - ? has(self.port) : true' - required: - - backendRef - type: object - requestRedirect: - description: |- - RequestRedirect defines a schema for a filter that responds to the - request with an HTTP redirection. - - Support: Core - properties: - hostname: - description: |- - Hostname is the hostname to be used in the value of the `Location` - header in the response. - When empty, the hostname in the `Host` header of the request is used. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - path: - description: |- - Path defines parameters used to modify the path of the incoming request. - The modified path is then used to construct the `Location` header. When - empty, the request path is used as-is. - - Support: Extended - properties: - replaceFullPath: - description: |- - ReplaceFullPath specifies the value with which to replace the full path - of a request during a rewrite or redirect. - maxLength: 1024 - type: string - replacePrefixMatch: - description: |- - ReplacePrefixMatch specifies the value with which to replace the prefix - match of a request during a rewrite or redirect. For example, a request - to "/foo/bar" with a prefix match of "/foo" and a ReplacePrefixMatch - of "/xyz" would be modified to "/xyz/bar". - - Note that this matches the behavior of the PathPrefix match type. This - matches full path elements. A path element refers to the list of labels - in the path split by the `/` separator. When specified, a trailing `/` is - ignored. For example, the paths `/abc`, `/abc/`, and `/abc/def` would all - match the prefix `/abc`, but the path `/abcd` would not. - - ReplacePrefixMatch is only compatible with a `PathPrefix` HTTPRouteMatch. - Using any other HTTPRouteMatch type on the same HTTPRouteRule will result in - the implementation setting the Accepted Condition for the Route to `status: False`. - - Request Path | Prefix Match | Replace Prefix | Modified Path - maxLength: 1024 - type: string - type: - description: |- - Type defines the type of path modifier. Additional types may be - added in a future release of the API. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - enum: - - ReplaceFullPath - - ReplacePrefixMatch - type: string - required: - - type - type: object - x-kubernetes-validations: - - message: replaceFullPath must be specified when - type is set to 'ReplaceFullPath' - rule: 'self.type == ''ReplaceFullPath'' ? has(self.replaceFullPath) - : true' - - message: type must be 'ReplaceFullPath' when replaceFullPath - is set - rule: 'has(self.replaceFullPath) ? self.type == - ''ReplaceFullPath'' : true' - - message: replacePrefixMatch must be specified when - type is set to 'ReplacePrefixMatch' - rule: 'self.type == ''ReplacePrefixMatch'' ? has(self.replacePrefixMatch) - : true' - - message: type must be 'ReplacePrefixMatch' when - replacePrefixMatch is set - rule: 'has(self.replacePrefixMatch) ? self.type - == ''ReplacePrefixMatch'' : true' - port: - description: |- - Port is the port to be used in the value of the `Location` - header in the response. - - If no port is specified, the redirect port MUST be derived using the - following rules: - - * If redirect scheme is not-empty, the redirect port MUST be the well-known - port associated with the redirect scheme. Specifically "http" to port 80 - and "https" to port 443. If the redirect scheme does not have a - well-known port, the listener port of the Gateway SHOULD be used. - * If redirect scheme is empty, the redirect port MUST be the Gateway - Listener port. - - Implementations SHOULD NOT add the port number in the 'Location' - header in the following cases: - - * A Location header that will use HTTP (whether that is determined via - the Listener protocol or the Scheme field) _and_ use port 80. - * A Location header that will use HTTPS (whether that is determined via - the Listener protocol or the Scheme field) _and_ use port 443. - - Support: Extended - format: int32 - maximum: 65535 - minimum: 1 - type: integer - scheme: - description: |- - Scheme is the scheme to be used in the value of the `Location` header in - the response. When empty, the scheme of the request is used. - - Scheme redirects can affect the port of the redirect, for more information, - refer to the documentation for the port field of this filter. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - - Support: Extended - enum: - - http - - https - type: string - statusCode: - default: 302 - description: |- - StatusCode is the HTTP status code to be used in response. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - - Support: Core - enum: - - 301 - - 302 - type: integer - type: object - responseHeaderModifier: - description: |- - ResponseHeaderModifier defines a schema for a filter that modifies response - headers. - - Support: Extended - properties: - add: - description: |- - Add adds the given header(s) (name, value) to the request - before the action. It appends to any existing values associated - with the header name. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - add: - - name: "my-header" - value: "bar,baz" - - Output: - GET /foo HTTP/1.1 - my-header: foo,bar,baz - items: - description: HTTPHeader represents an HTTP Header - name and value as defined by RFC 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP Header - to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - remove: - description: |- - Remove the given header(s) from the HTTP request before the action. The - value of Remove is a list of HTTP header names. Note that the header - names are case-insensitive (see - https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). - - Input: - GET /foo HTTP/1.1 - my-header1: foo - my-header2: bar - my-header3: baz - - Config: - remove: ["my-header1", "my-header3"] - - Output: - GET /foo HTTP/1.1 - my-header2: bar - items: - type: string - maxItems: 16 - type: array - x-kubernetes-list-type: set - set: - description: |- - Set overwrites the request with the given header (name, value) - before the action. - - Input: - GET /foo HTTP/1.1 - my-header: foo - - Config: - set: - - name: "my-header" - value: "bar" - - Output: - GET /foo HTTP/1.1 - my-header: bar - items: - description: HTTPHeader represents an HTTP Header - name and value as defined by RFC 7230. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, the first entry with - an equivalent name MUST be considered for a match. Subsequent entries - with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - value: - description: Value is the value of HTTP Header - to be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: - description: |- - Type identifies the type of filter to apply. As with other API fields, - types are classified into three conformance levels: - - - Core: Filter types and their corresponding configuration defined by - "Support: Core" in this package, e.g. "RequestHeaderModifier". All - implementations must support core filters. - - - Extended: Filter types and their corresponding configuration defined by - "Support: Extended" in this package, e.g. "RequestMirror". Implementers - are encouraged to support extended filters. - - - Implementation-specific: Filters that are defined and supported by - specific vendors. - In the future, filters showing convergence in behavior across multiple - implementations will be considered for inclusion in extended or core - conformance levels. Filter-specific configuration for such filters - is specified using the ExtensionRef field. `Type` should be set to - "ExtensionRef" for custom filters. - - Implementers are encouraged to define custom implementation types to - extend the core API with implementation-specific behavior. - - If a reference to a custom filter type cannot be resolved, the filter - MUST NOT be skipped. Instead, requests that would have been processed by - that filter MUST receive a HTTP error response. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - enum: - - RequestHeaderModifier - - ResponseHeaderModifier - - RequestMirror - - RequestRedirect - - URLRewrite - - ExtensionRef - type: string - urlRewrite: - description: |- - URLRewrite defines a schema for a filter that modifies a request during forwarding. - - Support: Extended - properties: - hostname: - description: |- - Hostname is the value to be used to replace the Host header value during - forwarding. - - Support: Extended - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - path: - description: |- - Path defines a path rewrite. - - Support: Extended - properties: - replaceFullPath: - description: |- - ReplaceFullPath specifies the value with which to replace the full path - of a request during a rewrite or redirect. - maxLength: 1024 - type: string - replacePrefixMatch: - description: |- - ReplacePrefixMatch specifies the value with which to replace the prefix - match of a request during a rewrite or redirect. For example, a request - to "/foo/bar" with a prefix match of "/foo" and a ReplacePrefixMatch - of "/xyz" would be modified to "/xyz/bar". - - Note that this matches the behavior of the PathPrefix match type. This - matches full path elements. A path element refers to the list of labels - in the path split by the `/` separator. When specified, a trailing `/` is - ignored. For example, the paths `/abc`, `/abc/`, and `/abc/def` would all - match the prefix `/abc`, but the path `/abcd` would not. - - ReplacePrefixMatch is only compatible with a `PathPrefix` HTTPRouteMatch. - Using any other HTTPRouteMatch type on the same HTTPRouteRule will result in - the implementation setting the Accepted Condition for the Route to `status: False`. - - Request Path | Prefix Match | Replace Prefix | Modified Path - maxLength: 1024 - type: string - type: - description: |- - Type defines the type of path modifier. Additional types may be - added in a future release of the API. - - Note that values may be added to this enum, implementations - must ensure that unknown values will not cause a crash. - - Unknown values here must result in the implementation setting the - Accepted Condition for the Route to `status: False`, with a - Reason of `UnsupportedValue`. - enum: - - ReplaceFullPath - - ReplacePrefixMatch - type: string - required: - - type - type: object - x-kubernetes-validations: - - message: replaceFullPath must be specified when - type is set to 'ReplaceFullPath' - rule: 'self.type == ''ReplaceFullPath'' ? has(self.replaceFullPath) - : true' - - message: type must be 'ReplaceFullPath' when replaceFullPath - is set - rule: 'has(self.replaceFullPath) ? self.type == - ''ReplaceFullPath'' : true' - - message: replacePrefixMatch must be specified when - type is set to 'ReplacePrefixMatch' - rule: 'self.type == ''ReplacePrefixMatch'' ? has(self.replacePrefixMatch) - : true' - - message: type must be 'ReplacePrefixMatch' when - replacePrefixMatch is set - rule: 'has(self.replacePrefixMatch) ? self.type - == ''ReplacePrefixMatch'' : true' - type: object - required: - - type - type: object - x-kubernetes-validations: - - message: filter.requestHeaderModifier must be nil if the - filter.type is not RequestHeaderModifier - rule: '!(has(self.requestHeaderModifier) && self.type != - ''RequestHeaderModifier'')' - - message: filter.requestHeaderModifier must be specified - for RequestHeaderModifier filter.type - rule: '!(!has(self.requestHeaderModifier) && self.type == - ''RequestHeaderModifier'')' - - message: filter.responseHeaderModifier must be nil if the - filter.type is not ResponseHeaderModifier - rule: '!(has(self.responseHeaderModifier) && self.type != - ''ResponseHeaderModifier'')' - - message: filter.responseHeaderModifier must be specified - for ResponseHeaderModifier filter.type - rule: '!(!has(self.responseHeaderModifier) && self.type - == ''ResponseHeaderModifier'')' - - message: filter.requestMirror must be nil if the filter.type - is not RequestMirror - rule: '!(has(self.requestMirror) && self.type != ''RequestMirror'')' - - message: filter.requestMirror must be specified for RequestMirror - filter.type - rule: '!(!has(self.requestMirror) && self.type == ''RequestMirror'')' - - message: filter.requestRedirect must be nil if the filter.type - is not RequestRedirect - rule: '!(has(self.requestRedirect) && self.type != ''RequestRedirect'')' - - message: filter.requestRedirect must be specified for RequestRedirect - filter.type - rule: '!(!has(self.requestRedirect) && self.type == ''RequestRedirect'')' - - message: filter.urlRewrite must be nil if the filter.type - is not URLRewrite - rule: '!(has(self.urlRewrite) && self.type != ''URLRewrite'')' - - message: filter.urlRewrite must be specified for URLRewrite - filter.type - rule: '!(!has(self.urlRewrite) && self.type == ''URLRewrite'')' - - message: filter.extensionRef must be nil if the filter.type - is not ExtensionRef - rule: '!(has(self.extensionRef) && self.type != ''ExtensionRef'')' - - message: filter.extensionRef must be specified for ExtensionRef - filter.type - rule: '!(!has(self.extensionRef) && self.type == ''ExtensionRef'')' - maxItems: 16 - type: array - x-kubernetes-validations: - - message: May specify either httpRouteFilterRequestRedirect - or httpRouteFilterRequestRewrite, but not both - rule: '!(self.exists(f, f.type == ''RequestRedirect'') && - self.exists(f, f.type == ''URLRewrite''))' - - message: RequestHeaderModifier filter cannot be repeated - rule: self.filter(f, f.type == 'RequestHeaderModifier').size() - <= 1 - - message: ResponseHeaderModifier filter cannot be repeated - rule: self.filter(f, f.type == 'ResponseHeaderModifier').size() - <= 1 - - message: RequestRedirect filter cannot be repeated - rule: self.filter(f, f.type == 'RequestRedirect').size() <= - 1 - - message: URLRewrite filter cannot be repeated - rule: self.filter(f, f.type == 'URLRewrite').size() <= 1 - matches: - default: - - path: - type: PathPrefix - value: / - description: |- - Matches define conditions used for matching the rule against incoming - HTTP requests. Each match is independent, i.e. this rule will be matched - if **any** one of the matches is satisfied. - - For example, take the following matches configuration: - - ``` - matches: - - path: - value: "/foo" - headers: - - name: "version" - value: "v2" - - path: - value: "/v2/foo" - ``` - - For a request to match against this rule, a request must satisfy - EITHER of the two conditions: - - - path prefixed with `/foo` AND contains the header `version: v2` - - path prefix of `/v2/foo` - - See the documentation for HTTPRouteMatch on how to specify multiple - match conditions that should be ANDed together. - - If no matches are specified, the default is a prefix - path match on "/", which has the effect of matching every - HTTP request. - - Proxy or Load Balancer routing configuration generated from HTTPRoutes - MUST prioritize matches based on the following criteria, continuing on - ties. Across all rules specified on applicable Routes, precedence must be - given to the match having: - - * "Exact" path match. - * "Prefix" path match with largest number of characters. - * Method match. - * Largest number of header matches. - * Largest number of query param matches. - - Note: The precedence of RegularExpression path matches are implementation-specific. - - If ties still exist across multiple Routes, matching precedence MUST be - determined in order of the following criteria, continuing on ties: - - * The oldest Route based on creation timestamp. - * The Route appearing first in alphabetical order by - "{namespace}/{name}". - - If ties still exist within an HTTPRoute, matching precedence MUST be granted - to the FIRST matching rule (in list order) with a match meeting the above - criteria. - - When no rules matching a request have been successfully attached to the - parent a request is coming from, a HTTP 404 status code MUST be returned. - items: - description: "HTTPRouteMatch defines the predicate used to - match requests to a given\naction. Multiple match types - are ANDed together, i.e. the match will\nevaluate to true - only if all conditions are satisfied.\n\nFor example, the - match below will match a HTTP request only if its path\nstarts - with `/foo` AND it contains the `version: v1` header:\n\n```\nmatch:\n\n\tpath:\n\t - \ value: \"/foo\"\n\theaders:\n\t- name: \"version\"\n\t - \ value \"v1\"\n\n```" - properties: - headers: - description: |- - Headers specifies HTTP request header matchers. Multiple match values are - ANDed together, meaning, a request must match all the specified headers - to select the route. - items: - description: |- - HTTPHeaderMatch describes how to select a HTTP route by matching HTTP request - headers. - properties: - name: - description: |- - Name is the name of the HTTP Header to be matched. Name matching MUST be - case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). - - If multiple entries specify equivalent header names, only the first - entry with an equivalent name MUST be considered for a match. Subsequent - entries with an equivalent header name MUST be ignored. Due to the - case-insensitivity of header names, "foo" and "Foo" are considered - equivalent. - - When a header is repeated in an HTTP request, it is - implementation-specific behavior as to how this is represented. - Generally, proxies should follow the guidance from the RFC: - https://www.rfc-editor.org/rfc/rfc7230.html#section-3.2.2 regarding - processing a repeated header, with special handling for "Set-Cookie". - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - type: - default: Exact - description: |- - Type specifies how to match against the value of the header. - - Support: Core (Exact) - - Support: Implementation-specific (RegularExpression) - - Since RegularExpression HeaderMatchType has implementation-specific - conformance, implementations can support POSIX, PCRE or any other dialects - of regular expressions. Please read the implementation's documentation to - determine the supported dialect. - enum: - - Exact - - RegularExpression - type: string - value: - description: Value is the value of HTTP Header to - be matched. - maxLength: 4096 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - method: - description: |- - Method specifies HTTP method matcher. - When specified, this route will be matched only if the request has the - specified method. - - Support: Extended - enum: - - GET - - HEAD - - POST - - PUT - - DELETE - - CONNECT - - OPTIONS - - TRACE - - PATCH - type: string - path: - default: - type: PathPrefix - value: / - description: |- - Path specifies a HTTP request path matcher. If this field is not - specified, a default prefix match on the "/" path is provided. - properties: - type: - default: PathPrefix - description: |- - Type specifies how to match against the path Value. - - Support: Core (Exact, PathPrefix) - - Support: Implementation-specific (RegularExpression) - enum: - - Exact - - PathPrefix - - RegularExpression - type: string - value: - default: / - description: Value of the HTTP path to match against. - maxLength: 1024 - type: string - type: object - x-kubernetes-validations: - - message: value must be an absolute path and start with - '/' when type one of ['Exact', 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? self.value.startsWith(''/'') - : true' - - message: must not contain '//' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.contains(''//'') - : true' - - message: must not contain '/./' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.contains(''/./'') - : true' - - message: must not contain '/../' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.contains(''/../'') - : true' - - message: must not contain '%2f' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.contains(''%2f'') - : true' - - message: must not contain '%2F' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.contains(''%2F'') - : true' - - message: must not contain '#' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.contains(''#'') - : true' - - message: must not end with '/..' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.endsWith(''/..'') - : true' - - message: must not end with '/.' when type one of ['Exact', - 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? !self.value.endsWith(''/.'') - : true' - - message: type must be one of ['Exact', 'PathPrefix', - 'RegularExpression'] - rule: self.type in ['Exact','PathPrefix'] || self.type - == 'RegularExpression' - - message: must only contain valid characters (matching - ^(?:[-A-Za-z0-9/._~!$&'()*+,;=:@]|[%][0-9a-fA-F]{2})+$) - for types ['Exact', 'PathPrefix'] - rule: '(self.type in [''Exact'',''PathPrefix'']) ? self.value.matches(r"""^(?:[-A-Za-z0-9/._~!$&''()*+,;=:@]|[%][0-9a-fA-F]{2})+$""") - : true' - queryParams: - description: |- - QueryParams specifies HTTP query parameter matchers. Multiple match - values are ANDed together, meaning, a request must match all the - specified query parameters to select the route. - - Support: Extended - items: - description: |- - HTTPQueryParamMatch describes how to select a HTTP route by matching HTTP - query parameters. - properties: - name: - description: |- - Name is the name of the HTTP query param to be matched. This must be an - exact string match. (See - https://tools.ietf.org/html/rfc7230#section-2.7.3). - - If multiple entries specify equivalent query param names, only the first - entry with an equivalent name MUST be considered for a match. Subsequent - entries with an equivalent query param name MUST be ignored. - - If a query param is repeated in an HTTP request, the behavior is - purposely left undefined, since different data planes have different - capabilities. However, it is *recommended* that implementations should - match against the first value of the param if the data plane supports it, - as this behavior is expected in other load balancing contexts outside of - the Gateway API. - - Users SHOULD NOT route traffic based on repeated query params to guard - themselves against potential differences in the implementations. - maxLength: 256 - minLength: 1 - pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ - type: string - type: - default: Exact - description: |- - Type specifies how to match against the value of the query parameter. - - Support: Extended (Exact) - - Support: Implementation-specific (RegularExpression) - - Since RegularExpression QueryParamMatchType has Implementation-specific - conformance, implementations can support POSIX, PCRE or any other - dialects of regular expressions. Please read the implementation's - documentation to determine the supported dialect. - enum: - - Exact - - RegularExpression - type: string - value: - description: Value is the value of HTTP query param - to be matched. - maxLength: 1024 - minLength: 1 - type: string - required: - - name - - value - type: object - maxItems: 16 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - maxItems: 64 - type: array - timeouts: - description: |- - Timeouts defines the timeouts that can be configured for an HTTP request. - - Support: Extended - properties: - backendRequest: - description: |- - BackendRequest specifies a timeout for an individual request from the gateway - to a backend. This covers the time from when the request first starts being - sent from the gateway to when the full response has been received from the backend. - - Setting a timeout to the zero duration (e.g. "0s") SHOULD disable the timeout - completely. Implementations that cannot completely disable the timeout MUST - instead interpret the zero duration as the longest possible value to which - the timeout can be set. - - An entire client HTTP transaction with a gateway, covered by the Request timeout, - may result in more than one call from the gateway to the destination backend, - for example, if automatic retries are supported. - - The value of BackendRequest must be a Gateway API Duration string as defined by - GEP-2257. When this field is unspecified, its behavior is implementation-specific; - when specified, the value of BackendRequest must be no more than the value of the - Request timeout (since the Request timeout encompasses the BackendRequest timeout). - - Support: Extended - pattern: ^([0-9]{1,5}(h|m|s|ms)){1,4}$ - type: string - request: - description: |- - Request specifies the maximum duration for a gateway to respond to an HTTP request. - If the gateway has not been able to respond before this deadline is met, the gateway - MUST return a timeout error. - - For example, setting the `rules.timeouts.request` field to the value `10s` in an - `HTTPRoute` will cause a timeout if a client request is taking longer than 10 seconds - to complete. - - Setting a timeout to the zero duration (e.g. "0s") SHOULD disable the timeout - completely. Implementations that cannot completely disable the timeout MUST - instead interpret the zero duration as the longest possible value to which - the timeout can be set. - - This timeout is intended to cover as close to the whole request-response transaction - as possible although an implementation MAY choose to start the timeout after the entire - request stream has been received instead of immediately after the transaction is - initiated by the client. - - The value of Request is a Gateway API Duration string as defined by GEP-2257. When this - field is unspecified, request timeout behavior is implementation-specific. - - Support: Extended - pattern: ^([0-9]{1,5}(h|m|s|ms)){1,4}$ - type: string - type: object - x-kubernetes-validations: - - message: backendRequest timeout cannot be longer than request - timeout - rule: '!(has(self.request) && has(self.backendRequest) && - duration(self.request) != duration(''0s'') && duration(self.backendRequest) - > duration(self.request))' - type: object - x-kubernetes-validations: - - message: RequestRedirect filter must not be used together with - backendRefs - rule: '(has(self.backendRefs) && size(self.backendRefs) > 0) ? - (!has(self.filters) || self.filters.all(f, !has(f.requestRedirect))): - true' - - message: When using RequestRedirect filter with path.replacePrefixMatch, - exactly one PathPrefix match must be specified - rule: '(has(self.filters) && self.filters.exists_one(f, has(f.requestRedirect) - && has(f.requestRedirect.path) && f.requestRedirect.path.type - == ''ReplacePrefixMatch'' && has(f.requestRedirect.path.replacePrefixMatch))) - ? ((size(self.matches) != 1 || !has(self.matches[0].path) || - self.matches[0].path.type != ''PathPrefix'') ? false : true) - : true' - - message: When using URLRewrite filter with path.replacePrefixMatch, - exactly one PathPrefix match must be specified - rule: '(has(self.filters) && self.filters.exists_one(f, has(f.urlRewrite) - && has(f.urlRewrite.path) && f.urlRewrite.path.type == ''ReplacePrefixMatch'' - && has(f.urlRewrite.path.replacePrefixMatch))) ? ((size(self.matches) - != 1 || !has(self.matches[0].path) || self.matches[0].path.type - != ''PathPrefix'') ? false : true) : true' - - message: Within backendRefs, when using RequestRedirect filter - with path.replacePrefixMatch, exactly one PathPrefix match must - be specified - rule: '(has(self.backendRefs) && self.backendRefs.exists_one(b, - (has(b.filters) && b.filters.exists_one(f, has(f.requestRedirect) - && has(f.requestRedirect.path) && f.requestRedirect.path.type - == ''ReplacePrefixMatch'' && has(f.requestRedirect.path.replacePrefixMatch))) - )) ? ((size(self.matches) != 1 || !has(self.matches[0].path) - || self.matches[0].path.type != ''PathPrefix'') ? false : true) - : true' - - message: Within backendRefs, When using URLRewrite filter with - path.replacePrefixMatch, exactly one PathPrefix match must be - specified - rule: '(has(self.backendRefs) && self.backendRefs.exists_one(b, - (has(b.filters) && b.filters.exists_one(f, has(f.urlRewrite) - && has(f.urlRewrite.path) && f.urlRewrite.path.type == ''ReplacePrefixMatch'' - && has(f.urlRewrite.path.replacePrefixMatch))) )) ? ((size(self.matches) - != 1 || !has(self.matches[0].path) || self.matches[0].path.type - != ''PathPrefix'') ? false : true) : true' - maxItems: 16 - type: array - x-kubernetes-validations: - - message: While 16 rules and 64 matches per rule are allowed, the - total number of matches across all rules in a route must be less - than 128 - rule: '(self.size() > 0 ? self[0].matches.size() : 0) + (self.size() - > 1 ? self[1].matches.size() : 0) + (self.size() > 2 ? self[2].matches.size() - : 0) + (self.size() > 3 ? self[3].matches.size() : 0) + (self.size() - > 4 ? self[4].matches.size() : 0) + (self.size() > 5 ? self[5].matches.size() - : 0) + (self.size() > 6 ? self[6].matches.size() : 0) + (self.size() - > 7 ? self[7].matches.size() : 0) + (self.size() > 8 ? self[8].matches.size() - : 0) + (self.size() > 9 ? self[9].matches.size() : 0) + (self.size() - > 10 ? self[10].matches.size() : 0) + (self.size() > 11 ? self[11].matches.size() - : 0) + (self.size() > 12 ? self[12].matches.size() : 0) + (self.size() - > 13 ? self[13].matches.size() : 0) + (self.size() > 14 ? self[14].matches.size() - : 0) + (self.size() > 15 ? self[15].matches.size() : 0) <= 128' - type: object - status: - description: Status defines the current state of HTTPRoute. - properties: - parents: - description: |- - Parents is a list of parent resources (usually Gateways) that are - associated with the route, and the status of the route with respect to - each parent. When this route attaches to a parent, the controller that - manages the parent must add an entry to this list when the controller - first sees the route and should update the entry as appropriate when the - route or gateway is modified. - - Note that parent references that cannot be resolved by an implementation - of this API will not be added to this list. Implementations of this API - can only populate Route status for the Gateways/parent resources they are - responsible for. - - A maximum of 32 Gateways will be represented in this list. An empty list - means the route has not been attached to any Gateway. - items: - description: |- - RouteParentStatus describes the status of a route with respect to an - associated Parent. - properties: - conditions: - description: |- - Conditions describes the status of the route with respect to the Gateway. - Note that the route's availability is also subject to the Gateway's own - status conditions and listener status. - - If the Route's ParentRef specifies an existing Gateway that supports - Routes of this kind AND that Gateway's controller has sufficient access, - then that Gateway's controller MUST set the "Accepted" condition on the - Route, to indicate whether the route has been accepted or rejected by the - Gateway, and why. - - A Route MUST be considered "Accepted" if at least one of the Route's - rules is implemented by the Gateway. - - There are a number of cases where the "Accepted" condition may not be set - due to lack of controller visibility, that includes when: - - * The Route refers to a non-existent parent. - * The Route is of a type that the controller does not support. - * The Route is in a namespace the controller does not have access to. - items: - description: Condition contains details for one aspect of - the current state of this API Resource. - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, - Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: type of condition in CamelCase or in foo.example.com/CamelCase. - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - maxItems: 8 - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - type - x-kubernetes-list-type: map - controllerName: - description: |- - ControllerName is a domain/path string that indicates the name of the - controller that wrote this status. This corresponds with the - controllerName field on GatewayClass. - - Example: "example.net/gateway-controller". - - The format of this field is DOMAIN "/" PATH, where DOMAIN and PATH are - valid Kubernetes names - (https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names). - - Controllers MUST populate this field when writing status. Controllers should ensure that - entries to status populated with their ControllerName are cleaned up when they are no - longer necessary. - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ - type: string - parentRef: - description: |- - ParentRef corresponds with a ParentRef in the spec that this - RouteParentStatus struct describes the status of. - properties: - group: - default: gateway.networking.k8s.io - description: |- - Group is the group of the referent. - When unspecified, "gateway.networking.k8s.io" is inferred. - To set the core API group (such as for a "Service" kind referent), - Group must be explicitly set to "" (empty string). - - Support: Core - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - default: Gateway - description: |- - Kind is kind of the referent. - - There are two kinds of parent resources with "Core" support: - - * Gateway (Gateway conformance profile) - * Service (Mesh conformance profile, ClusterIP Services only) - - Support for other resources is Implementation-Specific. - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: |- - Name is the name of the referent. - - Support: Core - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace is the namespace of the referent. When unspecified, this refers - to the local namespace of the Route. - - Note that there are specific rules for ParentRefs which cross namespace - boundaries. Cross-namespace references are only valid if they are explicitly - allowed by something in the namespace they are referring to. For example: - Gateway has the AllowedRoutes field, and ReferenceGrant provides a - generic way to enable any other kind of cross-namespace reference. - - - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - port: - description: |- - Port is the network port this Route targets. It can be interpreted - differently based on the type of parent resource. - - When the parent resource is a Gateway, this targets all listeners - listening on the specified port that also support this kind of Route(and - select this Route). It's not recommended to set `Port` unless the - networking behaviors specified in a Route must apply to a specific port - as opposed to a listener(s) whose port(s) may be changed. When both Port - and SectionName are specified, the name and port of the selected listener - must match both specified values. - - - - Implementations MAY choose to support other parent resources. - Implementations supporting other types of parent resources MUST clearly - document how/if Port is interpreted. - - For the purpose of status, an attachment is considered successful as - long as the parent resource accepts it partially. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment - from the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, - the Route MUST be considered detached from the Gateway. - - Support: Extended - format: int32 - maximum: 65535 - minimum: 1 - type: integer - sectionName: - description: |- - SectionName is the name of a section within the target resource. In the - following resources, SectionName is interpreted as the following: - - * Gateway: Listener name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - * Service: Port name. When both Port (experimental) and SectionName - are specified, the name and port of the selected listener must match - both specified values. - - Implementations MAY choose to support attaching Routes to other resources. - If that is the case, they MUST clearly document how SectionName is - interpreted. - - When unspecified (empty string), this will reference the entire resource. - For the purpose of status, an attachment is considered successful if at - least one section in the parent resource accepts it. For example, Gateway - listeners can restrict which Routes can attach to them by Route kind, - namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from - the referencing Route, the Route MUST be considered successfully - attached. If no Gateway listeners accept attachment from this Route, the - Route MUST be considered detached from the Gateway. - - Support: Core - maxLength: 253 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - required: - - name - type: object - required: - - controllerName - - parentRef - type: object - maxItems: 32 - type: array - required: - - parents - type: object - required: - - spec - type: object - served: true - storage: false - subresources: - status: {} -status: - acceptedNames: - kind: "" - plural: "" - conditions: null - storedVersions: null ---- -# -# config/crd/standard/gateway.networking.k8s.io_referencegrants.yaml -# -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - api-approved.kubernetes.io: https://github.com/kubernetes-sigs/gateway-api/pull/3328 - gateway.networking.k8s.io/bundle-version: v1.2.0 - gateway.networking.k8s.io/channel: standard - creationTimestamp: null - name: referencegrants.gateway.networking.k8s.io -spec: - group: gateway.networking.k8s.io - names: - categories: - - gateway-api - kind: ReferenceGrant - listKind: ReferenceGrantList - plural: referencegrants - shortNames: - - refgrant - singular: referencegrant - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1beta1 - schema: - openAPIV3Schema: - description: |- - ReferenceGrant identifies kinds of resources in other namespaces that are - trusted to reference the specified kinds of resources in the same namespace - as the policy. - - Each ReferenceGrant can be used to represent a unique trust relationship. - Additional Reference Grants can be used to add to the set of trusted - sources of inbound references for the namespace they are defined within. - - All cross-namespace references in Gateway API (with the exception of cross-namespace - Gateway-route attachment) require a ReferenceGrant. - - ReferenceGrant is a form of runtime verification allowing users to assert - which cross-namespace object references are permitted. Implementations that - support ReferenceGrant MUST NOT permit cross-namespace references which have - no grant, and MUST respond to the removal of a grant by revoking the access - that the grant allowed. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: Spec defines the desired state of ReferenceGrant. - properties: - from: - description: |- - From describes the trusted namespaces and kinds that can reference the - resources described in "To". Each entry in this list MUST be considered - to be an additional place that references can be valid from, or to put - this another way, entries MUST be combined using OR. - - Support: Core - items: - description: ReferenceGrantFrom describes trusted namespaces and - kinds. - properties: - group: - description: |- - Group is the group of the referent. - When empty, the Kubernetes core API group is inferred. - - Support: Core - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: |- - Kind is the kind of the referent. Although implementations may support - additional resources, the following types are part of the "Core" - support level for this field. - - When used to permit a SecretObjectReference: - - * Gateway - - When used to permit a BackendObjectReference: - - * GRPCRoute - * HTTPRoute - * TCPRoute - * TLSRoute - * UDPRoute - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - namespace: - description: |- - Namespace is the namespace of the referent. - - Support: Core - maxLength: 63 - minLength: 1 - pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ - type: string - required: - - group - - kind - - namespace - type: object - maxItems: 16 - minItems: 1 - type: array - to: - description: |- - To describes the resources that may be referenced by the resources - described in "From". Each entry in this list MUST be considered to be an - additional place that references can be valid to, or to put this another - way, entries MUST be combined using OR. - - Support: Core - items: - description: |- - ReferenceGrantTo describes what Kinds are allowed as targets of the - references. - properties: - group: - description: |- - Group is the group of the referent. - When empty, the Kubernetes core API group is inferred. - - Support: Core - maxLength: 253 - pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ - type: string - kind: - description: |- - Kind is the kind of the referent. Although implementations may support - additional resources, the following types are part of the "Core" - support level for this field: - - * Secret when used to permit a SecretObjectReference - * Service when used to permit a BackendObjectReference - maxLength: 63 - minLength: 1 - pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ - type: string - name: - description: |- - Name is the name of the referent. When unspecified, this policy - refers to all resources of the specified Group and Kind in the local - namespace. - maxLength: 253 - minLength: 1 - type: string - required: - - group - - kind - type: object - maxItems: 16 - minItems: 1 - type: array - required: - - from - - to - type: object - type: object - served: true - storage: true - subresources: {} -status: - acceptedNames: - kind: "" - plural: "" - conditions: null - storedVersions: null diff --git a/charts/traefik/traefik/crds/hub.traefik.io_accesscontrolpolicies.yaml b/charts/traefik/traefik/crds/hub.traefik.io_accesscontrolpolicies.yaml deleted file mode 100644 index 821f969..0000000 --- a/charts/traefik/traefik/crds/hub.traefik.io_accesscontrolpolicies.yaml +++ /dev/null @@ -1,368 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: accesscontrolpolicies.hub.traefik.io -spec: - group: hub.traefik.io - names: - kind: AccessControlPolicy - listKind: AccessControlPolicyList - plural: accesscontrolpolicies - singular: accesscontrolpolicy - scope: Cluster - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: AccessControlPolicy defines an access control policy. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: AccessControlPolicySpec configures an access control policy. - properties: - apiKey: - description: AccessControlPolicyAPIKey configure an APIKey control - policy. - properties: - forwardHeaders: - additionalProperties: - type: string - description: ForwardHeaders instructs the middleware to forward - key metadata as header values upon successful authentication. - type: object - keySource: - description: KeySource defines how to extract API keys from requests. - properties: - cookie: - description: Cookie is the name of a cookie. - type: string - header: - description: Header is the name of a header. - type: string - headerAuthScheme: - description: |- - HeaderAuthScheme sets an optional auth scheme when Header is set to "Authorization". - If set, this scheme is removed from the token, and all requests not including it are dropped. - type: string - query: - description: Query is the name of a query parameter. - type: string - type: object - keys: - description: Keys define the set of authorized keys to access - a protected resource. - items: - description: AccessControlPolicyAPIKeyKey defines an API key. - properties: - id: - description: ID is the unique identifier of the key. - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds arbitrary metadata for this - key, can be used by ForwardHeaders. - type: object - value: - description: Value is the SHAKE-256 hash (using 64 bytes) - of the API key. - type: string - required: - - id - - value - type: object - type: array - required: - - keySource - type: object - basicAuth: - description: AccessControlPolicyBasicAuth holds the HTTP basic authentication - configuration. - properties: - forwardUsernameHeader: - type: string - realm: - type: string - stripAuthorizationHeader: - type: boolean - users: - items: - type: string - type: array - type: object - jwt: - description: AccessControlPolicyJWT configures a JWT access control - policy. - properties: - claims: - type: string - forwardHeaders: - additionalProperties: - type: string - type: object - jwksFile: - type: string - jwksUrl: - type: string - publicKey: - type: string - signingSecret: - type: string - signingSecretBase64Encoded: - type: boolean - stripAuthorizationHeader: - type: boolean - tokenQueryKey: - type: string - type: object - oAuthIntro: - description: AccessControlOAuthIntro configures an OAuth 2.0 Token - Introspection access control policy. - properties: - claims: - type: string - clientConfig: - description: AccessControlOAuthIntroClientConfig configures the - OAuth 2.0 client for issuing token introspection requests. - properties: - headers: - additionalProperties: - type: string - description: Headers to set when sending requests to the Authorization - Server. - type: object - maxRetries: - default: 3 - description: MaxRetries defines the number of retries for - introspection requests. - type: integer - timeoutSeconds: - default: 5 - description: TimeoutSeconds configures the maximum amount - of seconds to wait before giving up on requests. - type: integer - tls: - description: TLS configures TLS communication with the Authorization - Server. - properties: - ca: - description: CA sets the CA bundle used to sign the Authorization - Server certificate. - type: string - insecureSkipVerify: - description: |- - InsecureSkipVerify skips the Authorization Server certificate validation. - For testing purposes only, do not use in production. - type: boolean - type: object - tokenTypeHint: - description: |- - TokenTypeHint is a hint to pass to the Authorization Server. - See https://tools.ietf.org/html/rfc7662#section-2.1 for more information. - type: string - url: - description: URL of the Authorization Server. - type: string - required: - - url - type: object - forwardHeaders: - additionalProperties: - type: string - type: object - tokenSource: - description: |- - TokenSource describes how to extract tokens from HTTP requests. - If multiple sources are set, the order is the following: header > query > cookie. - properties: - cookie: - description: Cookie is the name of a cookie. - type: string - header: - description: Header is the name of a header. - type: string - headerAuthScheme: - description: |- - HeaderAuthScheme sets an optional auth scheme when Header is set to "Authorization". - If set, this scheme is removed from the token, and all requests not including it are dropped. - type: string - query: - description: Query is the name of a query parameter. - type: string - type: object - required: - - clientConfig - - tokenSource - type: object - oidc: - description: AccessControlPolicyOIDC holds the OIDC authentication - configuration. - properties: - authParams: - additionalProperties: - type: string - type: object - claims: - type: string - clientId: - type: string - disableAuthRedirectionPaths: - items: - type: string - type: array - forwardHeaders: - additionalProperties: - type: string - type: object - issuer: - type: string - logoutUrl: - type: string - redirectUrl: - type: string - scopes: - items: - type: string - type: array - secret: - description: |- - SecretReference represents a Secret Reference. It has enough information to retrieve secret - in any namespace - properties: - name: - description: name is unique within a namespace to reference - a secret resource. - type: string - namespace: - description: namespace defines the space within which the - secret name must be unique. - type: string - type: object - x-kubernetes-map-type: atomic - session: - description: Session holds session configuration. - properties: - domain: - type: string - path: - type: string - refresh: - type: boolean - sameSite: - type: string - secure: - type: boolean - type: object - stateCookie: - description: StateCookie holds state cookie configuration. - properties: - domain: - type: string - path: - type: string - sameSite: - type: string - secure: - type: boolean - type: object - type: object - oidcGoogle: - description: AccessControlPolicyOIDCGoogle holds the Google OIDC authentication - configuration. - properties: - authParams: - additionalProperties: - type: string - type: object - clientId: - type: string - emails: - description: Emails are the allowed emails to connect. - items: - type: string - minItems: 1 - type: array - forwardHeaders: - additionalProperties: - type: string - type: object - logoutUrl: - type: string - redirectUrl: - type: string - secret: - description: |- - SecretReference represents a Secret Reference. It has enough information to retrieve secret - in any namespace - properties: - name: - description: name is unique within a namespace to reference - a secret resource. - type: string - namespace: - description: namespace defines the space within which the - secret name must be unique. - type: string - type: object - x-kubernetes-map-type: atomic - session: - description: Session holds session configuration. - properties: - domain: - type: string - path: - type: string - refresh: - type: boolean - sameSite: - type: string - secure: - type: boolean - type: object - stateCookie: - description: StateCookie holds state cookie configuration. - properties: - domain: - type: string - path: - type: string - sameSite: - type: string - secure: - type: boolean - type: object - type: object - type: object - status: - description: The current status of this access control policy. - properties: - specHash: - type: string - syncedAt: - format: date-time - type: string - version: - type: string - type: object - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/hub.traefik.io_apiaccesses.yaml b/charts/traefik/traefik/crds/hub.traefik.io_apiaccesses.yaml deleted file mode 100644 index ee8ecaf..0000000 --- a/charts/traefik/traefik/crds/hub.traefik.io_apiaccesses.yaml +++ /dev/null @@ -1,188 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: apiaccesses.hub.traefik.io -spec: - group: hub.traefik.io - names: - kind: APIAccess - listKind: APIAccessList - plural: apiaccesses - singular: apiaccess - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: APIAccess defines who can access to a set of APIs. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: The desired behavior of this APIAccess. - properties: - apiBundles: - description: |- - APIBundles defines a set of APIBundle that will be accessible to the configured audience. - Multiple APIAccesses can select the same APIBundles. - items: - description: APIBundleReference references an APIBundle. - properties: - name: - description: Name of the APIBundle. - maxLength: 253 - type: string - required: - - name - type: object - maxItems: 100 - type: array - x-kubernetes-validations: - - message: duplicated apiBundles - rule: self.all(x, self.exists_one(y, x.name == y.name)) - apiPlan: - description: APIPlan defines which APIPlan will be used. - properties: - name: - description: Name of the APIPlan. - maxLength: 253 - type: string - required: - - name - type: object - apiSelector: - description: |- - APISelector selects the APIs that will be accessible to the configured audience. - Multiple APIAccesses can select the same set of APIs. - This field is optional and follows standard label selector semantics. - An empty APISelector matches any API. - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. - The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector applies - to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - required: - - key - - operator - type: object - type: array - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - apis: - description: |- - APIs defines a set of APIs that will be accessible to the configured audience. - Multiple APIAccesses can select the same APIs. - When combined with APISelector, this set of APIs is appended to the matching APIs. - items: - description: APIReference references an API. - properties: - name: - description: Name of the API. - maxLength: 253 - type: string - required: - - name - type: object - maxItems: 100 - type: array - x-kubernetes-validations: - - message: duplicated apis - rule: self.all(x, self.exists_one(y, x.name == y.name)) - everyone: - description: Everyone indicates that all users will have access to - the selected APIs. - type: boolean - groups: - description: Groups are the consumer groups that will gain access - to the selected APIs. - items: - type: string - type: array - operationFilter: - description: |- - OperationFilter specifies the allowed operations on APIs and APIVersions. - If not set, all operations are available. - An empty OperationFilter prohibits all operations. - properties: - include: - description: Include defines the names of OperationSets that will - be accessible. - items: - type: string - maxItems: 100 - type: array - type: object - weight: - description: Weight specifies the evaluation order of the plan. - type: integer - x-kubernetes-validations: - - message: must be a positive number - rule: self >= 0 - type: object - x-kubernetes-validations: - - message: groups and everyone are mutually exclusive - rule: '(has(self.everyone) && has(self.groups)) ? !(self.everyone && - self.groups.size() > 0) : true' - status: - description: The current status of this APIAccess. - properties: - hash: - description: Hash is a hash representing the APIAccess. - type: string - syncedAt: - format: date-time - type: string - version: - type: string - type: object - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/hub.traefik.io_apibundles.yaml b/charts/traefik/traefik/crds/hub.traefik.io_apibundles.yaml deleted file mode 100644 index a45a0b1..0000000 --- a/charts/traefik/traefik/crds/hub.traefik.io_apibundles.yaml +++ /dev/null @@ -1,125 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: apibundles.hub.traefik.io -spec: - group: hub.traefik.io - names: - kind: APIBundle - listKind: APIBundleList - plural: apibundles - singular: apibundle - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: APIBundle defines a set of APIs. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: The desired behavior of this APIBundle. - properties: - apiSelector: - description: |- - APISelector selects the APIs that will be accessible to the configured audience. - Multiple APIBundles can select the same set of APIs. - This field is optional and follows standard label selector semantics. - An empty APISelector matches any API. - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. - The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector applies - to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - required: - - key - - operator - type: object - type: array - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - apis: - description: |- - APIs defines a set of APIs that will be accessible to the configured audience. - Multiple APIBundles can select the same APIs. - When combined with APISelector, this set of APIs is appended to the matching APIs. - items: - description: APIReference references an API. - properties: - name: - description: Name of the API. - maxLength: 253 - type: string - required: - - name - type: object - maxItems: 100 - type: array - x-kubernetes-validations: - - message: duplicated apis - rule: self.all(x, self.exists_one(y, x.name == y.name)) - type: object - status: - description: The current status of this APIBundle. - properties: - hash: - description: Hash is a hash representing the APIBundle. - type: string - syncedAt: - format: date-time - type: string - version: - type: string - type: object - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/hub.traefik.io_apiplans.yaml b/charts/traefik/traefik/crds/hub.traefik.io_apiplans.yaml deleted file mode 100644 index 92e1b9b..0000000 --- a/charts/traefik/traefik/crds/hub.traefik.io_apiplans.yaml +++ /dev/null @@ -1,103 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: apiplans.hub.traefik.io -spec: - group: hub.traefik.io - names: - kind: APIPlan - listKind: APIPlanList - plural: apiplans - singular: apiplan - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: APIPlan defines API Plan policy. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: The desired behavior of this APIPlan. - properties: - description: - description: Description describes the plan. - type: string - quota: - description: Quota defines the quota policy. - properties: - limit: - description: Limit is the maximum number of token in the bucket. - type: integer - x-kubernetes-validations: - - message: must be a positive number - rule: self >= 0 - period: - description: Period is the unit of time for the Limit. - format: duration - type: string - x-kubernetes-validations: - - message: must be between 1s and 9999h - rule: self >= duration('1s') && self <= duration('9999h') - required: - - limit - type: object - rateLimit: - description: RateLimit defines the rate limit policy. - properties: - limit: - description: Limit is the maximum number of token in the bucket. - type: integer - x-kubernetes-validations: - - message: must be a positive number - rule: self >= 0 - period: - description: Period is the unit of time for the Limit. - format: duration - type: string - x-kubernetes-validations: - - message: must be between 1s and 1h - rule: self >= duration('1s') && self <= duration('1h') - required: - - limit - type: object - title: - description: Title is the human-readable name of the plan. - type: string - required: - - title - type: object - status: - description: The current status of this APIPlan. - properties: - hash: - description: Hash is a hash representing the APIPlan. - type: string - syncedAt: - format: date-time - type: string - version: - type: string - type: object - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/hub.traefik.io_apiportals.yaml b/charts/traefik/traefik/crds/hub.traefik.io_apiportals.yaml deleted file mode 100644 index bc04170..0000000 --- a/charts/traefik/traefik/crds/hub.traefik.io_apiportals.yaml +++ /dev/null @@ -1,139 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: apiportals.hub.traefik.io -spec: - group: hub.traefik.io - names: - kind: APIPortal - listKind: APIPortalList - plural: apiportals - singular: apiportal - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: APIPortal defines a developer portal for accessing the documentation - of APIs. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: The desired behavior of this APIPortal. - properties: - description: - description: Description of the APIPortal. - type: string - title: - description: Title is the public facing name of the APIPortal. - type: string - trustedUrls: - description: TrustedURLs are the urls that are trusted by the OAuth - 2.0 authorization server. - items: - type: string - maxItems: 1 - minItems: 1 - type: array - x-kubernetes-validations: - - message: must be a valid URLs - rule: self.all(x, isURL(x)) - ui: - description: UI holds the UI customization options. - properties: - logoUrl: - description: LogoURL is the public URL of the logo. - type: string - type: object - required: - - trustedUrls - type: object - status: - description: The current status of this APIPortal. - properties: - hash: - description: Hash is a hash representing the APIPortal. - type: string - oidc: - description: OIDC is the OIDC configuration for accessing the exposed - APIPortal WebUI. - properties: - clientId: - description: ClientID is the OIDC ClientID for accessing the exposed - APIPortal WebUI. - type: string - companyClaim: - description: CompanyClaim is the name of the JWT claim containing - the user company. - type: string - emailClaim: - description: EmailClaim is the name of the JWT claim containing - the user email. - type: string - firstnameClaim: - description: FirstnameClaim is the name of the JWT claim containing - the user firstname. - type: string - generic: - description: Generic indicates whether or not the APIPortal authentication - relies on Generic OIDC. - type: boolean - groupsClaim: - description: GroupsClaim is the name of the JWT claim containing - the user groups. - type: string - issuer: - description: Issuer is the OIDC issuer for accessing the exposed - APIPortal WebUI. - type: string - lastnameClaim: - description: LastnameClaim is the name of the JWT claim containing - the user lastname. - type: string - scopes: - description: Scopes is the OIDC scopes for getting user attributes - during the authentication to the exposed APIPortal WebUI. - type: string - secretName: - description: SecretName is the name of the secret containing the - OIDC ClientSecret for accessing the exposed APIPortal WebUI. - type: string - syncedAttributes: - description: SyncedAttributes configure the user attributes to - sync. - items: - type: string - type: array - userIdClaim: - description: UserIDClaim is the name of the JWT claim containing - the user ID. - type: string - type: object - syncedAt: - format: date-time - type: string - version: - type: string - type: object - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/hub.traefik.io_apiratelimits.yaml b/charts/traefik/traefik/crds/hub.traefik.io_apiratelimits.yaml deleted file mode 100644 index 8e328d3..0000000 --- a/charts/traefik/traefik/crds/hub.traefik.io_apiratelimits.yaml +++ /dev/null @@ -1,166 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: apiratelimits.hub.traefik.io -spec: - group: hub.traefik.io - names: - kind: APIRateLimit - listKind: APIRateLimitList - plural: apiratelimits - singular: apiratelimit - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: APIRateLimit defines how group of consumers are rate limited - on a set of APIs. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: The desired behavior of this APIRateLimit. - properties: - apiSelector: - description: |- - APISelector selects the APIs that will be rate limited. - Multiple APIRateLimits can select the same set of APIs. - This field is optional and follows standard label selector semantics. - An empty APISelector matches any API. - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. - The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector applies - to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - required: - - key - - operator - type: object - type: array - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - apis: - description: |- - APIs defines a set of APIs that will be rate limited. - Multiple APIRateLimits can select the same APIs. - When combined with APISelector, this set of APIs is appended to the matching APIs. - items: - description: APIReference references an API. - properties: - name: - description: Name of the API. - maxLength: 253 - type: string - required: - - name - type: object - maxItems: 100 - type: array - x-kubernetes-validations: - - message: duplicated apis - rule: self.all(x, self.exists_one(y, x.name == y.name)) - everyone: - description: |- - Everyone indicates that all users will, by default, be rate limited with this configuration. - If an APIRateLimit explicitly target a group, the default rate limit will be ignored. - type: boolean - groups: - description: |- - Groups are the consumer groups that will be rate limited. - Multiple APIRateLimits can target the same set of consumer groups, the most restrictive one applies. - When a consumer belongs to multiple groups, the least restrictive APIRateLimit applies. - items: - type: string - type: array - limit: - description: Limit is the maximum number of token in the bucket. - type: integer - x-kubernetes-validations: - - message: must be a positive number - rule: self >= 0 - period: - description: Period is the unit of time for the Limit. - format: duration - type: string - x-kubernetes-validations: - - message: must be between 1s and 1h - rule: self >= duration('1s') && self <= duration('1h') - strategy: - description: |- - Strategy defines how the bucket state will be synchronized between the different Traefik Hub instances. - It can be, either "local" or "distributed". - enum: - - local - - distributed - type: string - required: - - limit - type: object - x-kubernetes-validations: - - message: groups and everyone are mutually exclusive - rule: '(has(self.everyone) && has(self.groups)) ? !(self.everyone && - self.groups.size() > 0) : true' - status: - description: The current status of this APIRateLimit. - properties: - hash: - description: Hash is a hash representing the APIRateLimit. - type: string - syncedAt: - format: date-time - type: string - version: - type: string - type: object - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/hub.traefik.io_apis.yaml b/charts/traefik/traefik/crds/hub.traefik.io_apis.yaml deleted file mode 100644 index a7b9e5e..0000000 --- a/charts/traefik/traefik/crds/hub.traefik.io_apis.yaml +++ /dev/null @@ -1,190 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: apis.hub.traefik.io -spec: - group: hub.traefik.io - names: - kind: API - listKind: APIList - plural: apis - singular: api - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - API defines an HTTP interface that is exposed to external clients. It specifies the supported versions - and provides instructions for accessing its documentation. Once instantiated, an API object is associated - with an Ingress, IngressRoute, or HTTPRoute resource, enabling the exposure of the described API to the outside world. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: APISpec describes the API. - properties: - openApiSpec: - description: OpenAPISpec defines the API contract as an OpenAPI specification. - properties: - operationSets: - description: OperationSets defines the sets of operations to be - referenced for granular filtering in APIAccesses. - items: - description: |- - OperationSet gives a name to a set of matching OpenAPI operations. - This set of operations can then be referenced for granular filtering in APIAccesses. - properties: - matchers: - description: Matchers defines a list of alternative rules - for matching OpenAPI operations. - items: - description: OperationMatcher defines criteria for matching - an OpenAPI operation. - minProperties: 1 - properties: - methods: - description: Methods specifies the HTTP methods to - be included for selection. - items: - type: string - maxItems: 10 - type: array - path: - description: Path specifies the exact path of the - operations to select. - maxLength: 255 - type: string - x-kubernetes-validations: - - message: must start with a '/' - rule: self.startsWith('/') - - message: cannot contains '../' - rule: '!self.matches(r"""(\/\.\.\/)|(\/\.\.$)""")' - pathPrefix: - description: PathPrefix specifies the path prefix - of the operations to select. - maxLength: 255 - type: string - x-kubernetes-validations: - - message: must start with a '/' - rule: self.startsWith('/') - - message: cannot contains '../' - rule: '!self.matches(r"""(\/\.\.\/)|(\/\.\.$)""")' - pathRegex: - description: PathRegex specifies a regular expression - pattern for matching operations based on their paths. - type: string - type: object - x-kubernetes-validations: - - message: path, pathPrefix and pathRegex are mutually - exclusive - rule: '[has(self.path), has(self.pathPrefix), has(self.pathRegex)].filter(x, - x).size() <= 1' - maxItems: 100 - minItems: 1 - type: array - name: - description: Name is the name of the OperationSet to reference - in APIAccesses. - maxLength: 253 - type: string - required: - - matchers - - name - type: object - maxItems: 100 - type: array - override: - description: Override holds data used to override OpenAPI specification. - properties: - servers: - items: - properties: - url: - type: string - x-kubernetes-validations: - - message: must be a valid URL - rule: isURL(self) - required: - - url - type: object - maxItems: 100 - minItems: 1 - type: array - required: - - servers - type: object - path: - description: |- - Path specifies the endpoint path within the Kubernetes Service where the OpenAPI specification can be obtained. - The Service queried is determined by the associated Ingress, IngressRoute, or HTTPRoute resource to which the API is attached. - It's important to note that this option is incompatible if the Ingress or IngressRoute specifies multiple backend services. - The Path must be accessible via a GET request method and should serve a YAML or JSON document containing the OpenAPI specification. - maxLength: 255 - type: string - x-kubernetes-validations: - - message: must start with a '/' - rule: self.startsWith('/') - - message: cannot contains '../' - rule: '!self.matches(r"""(\/\.\.\/)|(\/\.\.$)""")' - url: - description: |- - URL is a Traefik Hub agent accessible URL for obtaining the OpenAPI specification. - The URL must be accessible via a GET request method and should serve a YAML or JSON document containing the OpenAPI specification. - type: string - x-kubernetes-validations: - - message: must be a valid URL - rule: isURL(self) - type: object - x-kubernetes-validations: - - message: path or url must be defined - rule: has(self.path) || has(self.url) - versions: - description: Versions are the different APIVersions available. - items: - description: APIVersionRef references an APIVersion. - properties: - name: - description: Name of the APIVersion. - maxLength: 253 - type: string - required: - - name - type: object - maxItems: 100 - minItems: 1 - type: array - type: object - status: - description: The current status of this API. - properties: - hash: - description: Hash is a hash representing the API. - type: string - syncedAt: - format: date-time - type: string - version: - type: string - type: object - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/hub.traefik.io_apiversions.yaml b/charts/traefik/traefik/crds/hub.traefik.io_apiversions.yaml deleted file mode 100644 index 97184ef..0000000 --- a/charts/traefik/traefik/crds/hub.traefik.io_apiversions.yaml +++ /dev/null @@ -1,194 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: apiversions.hub.traefik.io -spec: - group: hub.traefik.io - names: - kind: APIVersion - listKind: APIVersionList - plural: apiversions - singular: apiversion - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.title - name: Title - type: string - - jsonPath: .spec.release - name: Release - type: string - name: v1alpha1 - schema: - openAPIV3Schema: - description: APIVersion defines a version of an API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: The desired behavior of this APIVersion. - properties: - openApiSpec: - description: OpenAPISpec defines the API contract as an OpenAPI specification. - properties: - operationSets: - description: OperationSets defines the sets of operations to be - referenced for granular filtering in APIAccesses. - items: - description: |- - OperationSet gives a name to a set of matching OpenAPI operations. - This set of operations can then be referenced for granular filtering in APIAccesses. - properties: - matchers: - description: Matchers defines a list of alternative rules - for matching OpenAPI operations. - items: - description: OperationMatcher defines criteria for matching - an OpenAPI operation. - minProperties: 1 - properties: - methods: - description: Methods specifies the HTTP methods to - be included for selection. - items: - type: string - maxItems: 10 - type: array - path: - description: Path specifies the exact path of the - operations to select. - maxLength: 255 - type: string - x-kubernetes-validations: - - message: must start with a '/' - rule: self.startsWith('/') - - message: cannot contains '../' - rule: '!self.matches(r"""(\/\.\.\/)|(\/\.\.$)""")' - pathPrefix: - description: PathPrefix specifies the path prefix - of the operations to select. - maxLength: 255 - type: string - x-kubernetes-validations: - - message: must start with a '/' - rule: self.startsWith('/') - - message: cannot contains '../' - rule: '!self.matches(r"""(\/\.\.\/)|(\/\.\.$)""")' - pathRegex: - description: PathRegex specifies a regular expression - pattern for matching operations based on their paths. - type: string - type: object - x-kubernetes-validations: - - message: path, pathPrefix and pathRegex are mutually - exclusive - rule: '[has(self.path), has(self.pathPrefix), has(self.pathRegex)].filter(x, - x).size() <= 1' - maxItems: 100 - minItems: 1 - type: array - name: - description: Name is the name of the OperationSet to reference - in APIAccesses. - maxLength: 253 - type: string - required: - - matchers - - name - type: object - maxItems: 100 - type: array - override: - description: Override holds data used to override OpenAPI specification. - properties: - servers: - items: - properties: - url: - type: string - x-kubernetes-validations: - - message: must be a valid URL - rule: isURL(self) - required: - - url - type: object - maxItems: 100 - minItems: 1 - type: array - required: - - servers - type: object - path: - description: |- - Path specifies the endpoint path within the Kubernetes Service where the OpenAPI specification can be obtained. - The Service queried is determined by the associated Ingress, IngressRoute, or HTTPRoute resource to which the API is attached. - It's important to note that this option is incompatible if the Ingress or IngressRoute specifies multiple backend services. - The Path must be accessible via a GET request method and should serve a YAML or JSON document containing the OpenAPI specification. - maxLength: 255 - type: string - x-kubernetes-validations: - - message: must start with a '/' - rule: self.startsWith('/') - - message: cannot contains '../' - rule: '!self.matches(r"""(\/\.\.\/)|(\/\.\.$)""")' - url: - description: |- - URL is a Traefik Hub agent accessible URL for obtaining the OpenAPI specification. - The URL must be accessible via a GET request method and should serve a YAML or JSON document containing the OpenAPI specification. - type: string - x-kubernetes-validations: - - message: must be a valid URL - rule: isURL(self) - type: object - x-kubernetes-validations: - - message: path or url must be defined - rule: has(self.path) || has(self.url) - release: - description: |- - Release is the version number of the API. - This value must follow the SemVer format: https://semver.org/ - maxLength: 100 - type: string - x-kubernetes-validations: - - message: must be a valid semver version - rule: self.matches(r"""^v?(0|[1-9]\d*)\.(0|[1-9]\d*)\.(0|[1-9]\d*)(?:-((?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?$""") - title: - description: Title is the public facing name of the APIVersion. - type: string - required: - - release - type: object - status: - description: The current status of this APIVersion. - properties: - hash: - description: Hash is a hash representing the APIVersion. - type: string - syncedAt: - format: date-time - type: string - version: - type: string - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/traefik/traefik/crds/traefik.io_ingressroutes.yaml b/charts/traefik/traefik/crds/traefik.io_ingressroutes.yaml deleted file mode 100644 index ccb374a..0000000 --- a/charts/traefik/traefik/crds/traefik.io_ingressroutes.yaml +++ /dev/null @@ -1,366 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.16.1 - name: ingressroutes.traefik.io -spec: - group: traefik.io - names: - kind: IngressRoute - listKind: IngressRouteList - plural: ingressroutes - singular: ingressroute - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: IngressRoute is the CRD implementation of a Traefik HTTP Router. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: IngressRouteSpec defines the desired state of IngressRoute. - properties: - entryPoints: - description: |- - EntryPoints defines the list of entry point names to bind to. - Entry points have to be configured in the static configuration. - More info: https://doc.traefik.io/traefik/v3.2/routing/entrypoints/ - Default: all. - items: - type: string - type: array - routes: - description: Routes defines the list of routes. - items: - description: Route holds the HTTP route configuration. - properties: - kind: - description: |- - Kind defines the kind of the route. - Rule is the only supported kind. - enum: - - Rule - type: string - match: - description: |- - Match defines the router's rule. - More info: https://doc.traefik.io/traefik/v3.2/routing/routers/#rule - type: string - middlewares: - description: |- - Middlewares defines the list of references to Middleware resources. - More info: https://doc.traefik.io/traefik/v3.2/routing/providers/kubernetes-crd/#kind-middleware - items: - description: MiddlewareRef is a reference to a Middleware - resource. - properties: - name: - description: Name defines the name of the referenced Middleware - resource. - type: string - namespace: - description: Namespace defines the namespace of the referenced - Middleware resource. - type: string - required: - - name - type: object - type: array - priority: - description: |- - Priority defines the router's priority. - More info: https://doc.traefik.io/traefik/v3.2/routing/routers/#priority - type: integer - services: - description: |- - Services defines the list of Service. - It can contain any combination of TraefikService and/or reference to a Kubernetes Service. - items: - description: Service defines an upstream HTTP service to proxy - traffic to. - properties: - healthCheck: - description: Healthcheck defines health checks for ExternalName - services. - properties: - followRedirects: - description: |- - FollowRedirects defines whether redirects should be followed during the health check calls. - Default: true - type: boolean - headers: - additionalProperties: - type: string - description: Headers defines custom headers to be - sent to the health check endpoint. - type: object - hostname: - description: Hostname defines the value of hostname - in the Host header of the health check request. - type: string - interval: - anyOf: - - type: integer - - type: string - description: |- - Interval defines the frequency of the health check calls. - Default: 30s - x-kubernetes-int-or-string: true - method: - description: Method defines the healthcheck method. - type: string - mode: - description: |- - Mode defines the health check mode. - If defined to grpc, will use the gRPC health check protocol to probe the server. - Default: http - type: string - path: - description: Path defines the server URL path for - the health check endpoint. - type: string - port: - description: Port defines the server URL port for - the health check endpoint. - type: integer - scheme: - description: Scheme replaces the server URL scheme - for the health check endpoint. - type: string - status: - description: Status defines the expected HTTP status - code of the response to the health check request. - type: integer - timeout: - anyOf: - - type: integer - - type: string - description: |- - Timeout defines the maximum duration Traefik will wait for a health check request before considering the server unhealthy. - Default: 5s - x-kubernetes-int-or-string: true - type: object - kind: - description: Kind defines the kind of the Service. - enum: - - Service - - TraefikService - type: string - name: - description: |- - Name defines the name of the referenced Kubernetes Service or TraefikService. - The differentiation between the two is specified in the Kind field. - type: string - namespace: - description: Namespace defines the namespace of the referenced - Kubernetes Service or TraefikService. - type: string - nativeLB: - description: |- - NativeLB controls, when creating the load-balancer, - whether the LB's children are directly the pods IPs or if the only child is the Kubernetes Service clusterIP. - The Kubernetes Service itself does load-balance to the pods. - By default, NativeLB is false. - type: boolean - nodePortLB: - description: |- - NodePortLB controls, when creating the load-balancer, - whether the LB's children are directly the nodes internal IPs using the nodePort when the service type is NodePort. - It allows services to be reachable when Traefik runs externally from the Kubernetes cluster but within the same network of the nodes. - By default, NodePortLB is false. - type: boolean - passHostHeader: - description: |- - PassHostHeader defines whether the client Host header is forwarded to the upstream Kubernetes Service. - By default, passHostHeader is true. - type: boolean - port: - anyOf: - - type: integer - - type: string - description: |- - Port defines the port of a Kubernetes Service. - This can be a reference to a named port. - x-kubernetes-int-or-string: true - responseForwarding: - description: ResponseForwarding defines how Traefik forwards - the response from the upstream Kubernetes Service to - the client. - properties: - flushInterval: - description: |- - FlushInterval defines the interval, in milliseconds, in between flushes to the client while copying the response body. - A negative value means to flush immediately after each write to the client. - This configuration is ignored when ReverseProxy recognizes a response as a streaming response; - for such responses, writes are flushed to the client immediately. - Default: 100ms - type: string - type: object - scheme: - description: |- - Scheme defines the scheme to use for the request to the upstream Kubernetes Service. - It defaults to https when Kubernetes Service port is 443, http otherwise. - type: string - serversTransport: - description: |- - ServersTransport defines the name of ServersTransport resource to use. - It allows to configure the transport between Traefik and your servers. - Can only be used on a Kubernetes Service. - type: string - sticky: - description: |- - Sticky defines the sticky sessions configuration. - More info: https://doc.traefik.io/traefik/v3.2/routing/services/#sticky-sessions - properties: - cookie: - description: Cookie defines the sticky cookie configuration. - properties: - httpOnly: - description: HTTPOnly defines whether the cookie - can be accessed by client-side APIs, such as - JavaScript. - type: boolean - maxAge: - description: |- - MaxAge indicates the number of seconds until the cookie expires. - When set to a negative number, the cookie expires immediately. - When set to zero, the cookie never expires. - type: integer - name: - description: Name defines the Cookie name. - type: string - sameSite: - description: |- - SameSite defines the same site policy. - More info: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie/SameSite - type: string - secure: - description: Secure defines whether the cookie - can only be transmitted over an encrypted connection - (i.e. HTTPS). - type: boolean - type: object - type: object - strategy: - description: |- - Strategy defines the load balancing strategy between the servers. - RoundRobin is the only supported value at the moment. - type: string - weight: - description: |- - Weight defines the weight and should only be specified when Name references a TraefikService object - (and to be precise, one that embeds a Weighted Round Robin). - type: integer - required: - - name - type: object - type: array - syntax: - description: |- - Syntax defines the router's rule syntax. - More info: https://doc.traefik.io/traefik/v3.2/routing/routers/#rulesyntax - type: string - required: - - kind - - match - type: object - type: array - tls: - description: |- - TLS defines the TLS configuration. - More info: https://doc.traefik.io/traefik/v3.2/routing/routers/#tls - properties: - certResolver: - description: |- - CertResolver defines the name of the certificate resolver to use. - Cert resolvers have to be configured in the static configuration. - More info: https://doc.traefik.io/traefik/v3.2/https/acme/#certificate-resolvers - type: string - domains: - description: |- - Domains defines the list of domains that will be used to issue certificates. - More info: https://doc.traefik.io/traefik/v3.2/routing/routers/#domains - items: - description: Domain holds a domain name with SANs. - properties: - main: - description: Main defines the main domain name. - type: string - sans: - description: SANs defines the subject alternative domain - names. - items: - type: string - type: array - type: object - type: array - options: - description: |- - Options defines the reference to a TLSOption, that specifies the parameters of the TLS connection. - If not defined, the `default` TLSOption is used. - More info: https://doc.traefik.io/traefik/v3.2/https/tls/#tls-options - properties: - name: - description: |- - Name defines the name of the referenced TLSOption. - More info: https://doc.traefik.io/traefik/v3.2/routing/providers/kubernetes-crd/#kind-tlsoption - type: string - namespace: - description: |- - Namespace defines the namespace of the referenced TLSOption. - More info: https://doc.traefik.io/traefik/v3.2/routing/providers/kubernetes-crd/#kind-tlsoption - type: string - required: - - name - type: object - secretName: - description: SecretName is the name of the referenced Kubernetes - Secret to specify the certificate details. - type: string - store: - description: |- - Store defines the reference to the TLSStore, that will be used to store certificates. - Please note that only `default` TLSStore can be used. - properties: - name: - description: |- - Name defines the name of the referenced TLSStore. - More info: https://doc.traefik.io/traefik/v3.2/routing/providers/kubernetes-crd/#kind-tlsstore - type: string - namespace: - description: |- - Namespace defines the namespace of the referenced TLSStore. - More info: https://doc.traefik.io/traefik/v3.2/routing/providers/kubernetes-crd/#kind-tlsstore - type: string - required: - - name - type: object - type: object - required: - - routes - type: object - required: - - metadata - - spec - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/traefik.io_ingressroutetcps.yaml b/charts/traefik/traefik/crds/traefik.io_ingressroutetcps.yaml deleted file mode 100644 index ae675f6..0000000 --- a/charts/traefik/traefik/crds/traefik.io_ingressroutetcps.yaml +++ /dev/null @@ -1,247 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.16.1 - name: ingressroutetcps.traefik.io -spec: - group: traefik.io - names: - kind: IngressRouteTCP - listKind: IngressRouteTCPList - plural: ingressroutetcps - singular: ingressroutetcp - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: IngressRouteTCP is the CRD implementation of a Traefik TCP Router. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: IngressRouteTCPSpec defines the desired state of IngressRouteTCP. - properties: - entryPoints: - description: |- - EntryPoints defines the list of entry point names to bind to. - Entry points have to be configured in the static configuration. - More info: https://doc.traefik.io/traefik/v3.2/routing/entrypoints/ - Default: all. - items: - type: string - type: array - routes: - description: Routes defines the list of routes. - items: - description: RouteTCP holds the TCP route configuration. - properties: - match: - description: |- - Match defines the router's rule. - More info: https://doc.traefik.io/traefik/v3.2/routing/routers/#rule_1 - type: string - middlewares: - description: Middlewares defines the list of references to MiddlewareTCP - resources. - items: - description: ObjectReference is a generic reference to a Traefik - resource. - properties: - name: - description: Name defines the name of the referenced Traefik - resource. - type: string - namespace: - description: Namespace defines the namespace of the referenced - Traefik resource. - type: string - required: - - name - type: object - type: array - priority: - description: |- - Priority defines the router's priority. - More info: https://doc.traefik.io/traefik/v3.2/routing/routers/#priority_1 - type: integer - services: - description: Services defines the list of TCP services. - items: - description: ServiceTCP defines an upstream TCP service to - proxy traffic to. - properties: - name: - description: Name defines the name of the referenced Kubernetes - Service. - type: string - namespace: - description: Namespace defines the namespace of the referenced - Kubernetes Service. - type: string - nativeLB: - description: |- - NativeLB controls, when creating the load-balancer, - whether the LB's children are directly the pods IPs or if the only child is the Kubernetes Service clusterIP. - The Kubernetes Service itself does load-balance to the pods. - By default, NativeLB is false. - type: boolean - nodePortLB: - description: |- - NodePortLB controls, when creating the load-balancer, - whether the LB's children are directly the nodes internal IPs using the nodePort when the service type is NodePort. - It allows services to be reachable when Traefik runs externally from the Kubernetes cluster but within the same network of the nodes. - By default, NodePortLB is false. - type: boolean - port: - anyOf: - - type: integer - - type: string - description: |- - Port defines the port of a Kubernetes Service. - This can be a reference to a named port. - x-kubernetes-int-or-string: true - proxyProtocol: - description: |- - ProxyProtocol defines the PROXY protocol configuration. - More info: https://doc.traefik.io/traefik/v3.2/routing/services/#proxy-protocol - properties: - version: - description: Version defines the PROXY Protocol version - to use. - type: integer - type: object - serversTransport: - description: |- - ServersTransport defines the name of ServersTransportTCP resource to use. - It allows to configure the transport between Traefik and your servers. - Can only be used on a Kubernetes Service. - type: string - terminationDelay: - description: |- - TerminationDelay defines the deadline that the proxy sets, after one of its connected peers indicates - it has closed the writing capability of its connection, to close the reading capability as well, - hence fully terminating the connection. - It is a duration in milliseconds, defaulting to 100. - A negative value means an infinite deadline (i.e. the reading capability is never closed). - Deprecated: TerminationDelay will not be supported in future APIVersions, please use ServersTransport to configure the TerminationDelay instead. - type: integer - tls: - description: TLS determines whether to use TLS when dialing - with the backend. - type: boolean - weight: - description: Weight defines the weight used when balancing - requests between multiple Kubernetes Service. - type: integer - required: - - name - - port - type: object - type: array - syntax: - description: |- - Syntax defines the router's rule syntax. - More info: https://doc.traefik.io/traefik/v3.2/routing/routers/#rulesyntax_1 - type: string - required: - - match - type: object - type: array - tls: - description: |- - TLS defines the TLS configuration on a layer 4 / TCP Route. - More info: https://doc.traefik.io/traefik/v3.2/routing/routers/#tls_1 - properties: - certResolver: - description: |- - CertResolver defines the name of the certificate resolver to use. - Cert resolvers have to be configured in the static configuration. - More info: https://doc.traefik.io/traefik/v3.2/https/acme/#certificate-resolvers - type: string - domains: - description: |- - Domains defines the list of domains that will be used to issue certificates. - More info: https://doc.traefik.io/traefik/v3.2/routing/routers/#domains - items: - description: Domain holds a domain name with SANs. - properties: - main: - description: Main defines the main domain name. - type: string - sans: - description: SANs defines the subject alternative domain - names. - items: - type: string - type: array - type: object - type: array - options: - description: |- - Options defines the reference to a TLSOption, that specifies the parameters of the TLS connection. - If not defined, the `default` TLSOption is used. - More info: https://doc.traefik.io/traefik/v3.2/https/tls/#tls-options - properties: - name: - description: Name defines the name of the referenced Traefik - resource. - type: string - namespace: - description: Namespace defines the namespace of the referenced - Traefik resource. - type: string - required: - - name - type: object - passthrough: - description: Passthrough defines whether a TLS router will terminate - the TLS connection. - type: boolean - secretName: - description: SecretName is the name of the referenced Kubernetes - Secret to specify the certificate details. - type: string - store: - description: |- - Store defines the reference to the TLSStore, that will be used to store certificates. - Please note that only `default` TLSStore can be used. - properties: - name: - description: Name defines the name of the referenced Traefik - resource. - type: string - namespace: - description: Namespace defines the namespace of the referenced - Traefik resource. - type: string - required: - - name - type: object - type: object - required: - - routes - type: object - required: - - metadata - - spec - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/traefik.io_ingressrouteudps.yaml b/charts/traefik/traefik/crds/traefik.io_ingressrouteudps.yaml deleted file mode 100644 index a815d86..0000000 --- a/charts/traefik/traefik/crds/traefik.io_ingressrouteudps.yaml +++ /dev/null @@ -1,111 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.16.1 - name: ingressrouteudps.traefik.io -spec: - group: traefik.io - names: - kind: IngressRouteUDP - listKind: IngressRouteUDPList - plural: ingressrouteudps - singular: ingressrouteudp - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: IngressRouteUDP is a CRD implementation of a Traefik UDP Router. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: IngressRouteUDPSpec defines the desired state of a IngressRouteUDP. - properties: - entryPoints: - description: |- - EntryPoints defines the list of entry point names to bind to. - Entry points have to be configured in the static configuration. - More info: https://doc.traefik.io/traefik/v3.2/routing/entrypoints/ - Default: all. - items: - type: string - type: array - routes: - description: Routes defines the list of routes. - items: - description: RouteUDP holds the UDP route configuration. - properties: - services: - description: Services defines the list of UDP services. - items: - description: ServiceUDP defines an upstream UDP service to - proxy traffic to. - properties: - name: - description: Name defines the name of the referenced Kubernetes - Service. - type: string - namespace: - description: Namespace defines the namespace of the referenced - Kubernetes Service. - type: string - nativeLB: - description: |- - NativeLB controls, when creating the load-balancer, - whether the LB's children are directly the pods IPs or if the only child is the Kubernetes Service clusterIP. - The Kubernetes Service itself does load-balance to the pods. - By default, NativeLB is false. - type: boolean - nodePortLB: - description: |- - NodePortLB controls, when creating the load-balancer, - whether the LB's children are directly the nodes internal IPs using the nodePort when the service type is NodePort. - It allows services to be reachable when Traefik runs externally from the Kubernetes cluster but within the same network of the nodes. - By default, NodePortLB is false. - type: boolean - port: - anyOf: - - type: integer - - type: string - description: |- - Port defines the port of a Kubernetes Service. - This can be a reference to a named port. - x-kubernetes-int-or-string: true - weight: - description: Weight defines the weight used when balancing - requests between multiple Kubernetes Service. - type: integer - required: - - name - - port - type: object - type: array - type: object - type: array - required: - - routes - type: object - required: - - metadata - - spec - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/traefik.io_middlewares.yaml b/charts/traefik/traefik/crds/traefik.io_middlewares.yaml deleted file mode 100644 index f3ea9fc..0000000 --- a/charts/traefik/traefik/crds/traefik.io_middlewares.yaml +++ /dev/null @@ -1,1126 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.16.1 - name: middlewares.traefik.io -spec: - group: traefik.io - names: - kind: Middleware - listKind: MiddlewareList - plural: middlewares - singular: middleware - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - Middleware is the CRD implementation of a Traefik Middleware. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/overview/ - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: MiddlewareSpec defines the desired state of a Middleware. - properties: - addPrefix: - description: |- - AddPrefix holds the add prefix middleware configuration. - This middleware updates the path of a request before forwarding it. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/addprefix/ - properties: - prefix: - description: |- - Prefix is the string to add before the current path in the requested URL. - It should include a leading slash (/). - type: string - type: object - basicAuth: - description: |- - BasicAuth holds the basic auth middleware configuration. - This middleware restricts access to your services to known users. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/basicauth/ - properties: - headerField: - description: |- - HeaderField defines a header field to store the authenticated user. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/basicauth/#headerfield - type: string - realm: - description: |- - Realm allows the protected resources on a server to be partitioned into a set of protection spaces, each with its own authentication scheme. - Default: traefik. - type: string - removeHeader: - description: |- - RemoveHeader sets the removeHeader option to true to remove the authorization header before forwarding the request to your service. - Default: false. - type: boolean - secret: - description: Secret is the name of the referenced Kubernetes Secret - containing user credentials. - type: string - type: object - buffering: - description: |- - Buffering holds the buffering middleware configuration. - This middleware retries or limits the size of requests that can be forwarded to backends. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/buffering/#maxrequestbodybytes - properties: - maxRequestBodyBytes: - description: |- - MaxRequestBodyBytes defines the maximum allowed body size for the request (in bytes). - If the request exceeds the allowed size, it is not forwarded to the service, and the client gets a 413 (Request Entity Too Large) response. - Default: 0 (no maximum). - format: int64 - type: integer - maxResponseBodyBytes: - description: |- - MaxResponseBodyBytes defines the maximum allowed response size from the service (in bytes). - If the response exceeds the allowed size, it is not forwarded to the client. The client gets a 500 (Internal Server Error) response instead. - Default: 0 (no maximum). - format: int64 - type: integer - memRequestBodyBytes: - description: |- - MemRequestBodyBytes defines the threshold (in bytes) from which the request will be buffered on disk instead of in memory. - Default: 1048576 (1Mi). - format: int64 - type: integer - memResponseBodyBytes: - description: |- - MemResponseBodyBytes defines the threshold (in bytes) from which the response will be buffered on disk instead of in memory. - Default: 1048576 (1Mi). - format: int64 - type: integer - retryExpression: - description: |- - RetryExpression defines the retry conditions. - It is a logical combination of functions with operators AND (&&) and OR (||). - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/buffering/#retryexpression - type: string - type: object - chain: - description: |- - Chain holds the configuration of the chain middleware. - This middleware enables to define reusable combinations of other pieces of middleware. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/chain/ - properties: - middlewares: - description: Middlewares is the list of MiddlewareRef which composes - the chain. - items: - description: MiddlewareRef is a reference to a Middleware resource. - properties: - name: - description: Name defines the name of the referenced Middleware - resource. - type: string - namespace: - description: Namespace defines the namespace of the referenced - Middleware resource. - type: string - required: - - name - type: object - type: array - type: object - circuitBreaker: - description: CircuitBreaker holds the circuit breaker configuration. - properties: - checkPeriod: - anyOf: - - type: integer - - type: string - description: CheckPeriod is the interval between successive checks - of the circuit breaker condition (when in standby state). - x-kubernetes-int-or-string: true - expression: - description: Expression is the condition that triggers the tripped - state. - type: string - fallbackDuration: - anyOf: - - type: integer - - type: string - description: FallbackDuration is the duration for which the circuit - breaker will wait before trying to recover (from a tripped state). - x-kubernetes-int-or-string: true - recoveryDuration: - anyOf: - - type: integer - - type: string - description: RecoveryDuration is the duration for which the circuit - breaker will try to recover (as soon as it is in recovering - state). - x-kubernetes-int-or-string: true - responseCode: - description: ResponseCode is the status code that the circuit - breaker will return while it is in the open state. - type: integer - type: object - compress: - description: |- - Compress holds the compress middleware configuration. - This middleware compresses responses before sending them to the client, using gzip, brotli, or zstd compression. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/compress/ - properties: - defaultEncoding: - description: DefaultEncoding specifies the default encoding if - the `Accept-Encoding` header is not in the request or contains - a wildcard (`*`). - type: string - encodings: - description: Encodings defines the list of supported compression - algorithms. - items: - type: string - type: array - excludedContentTypes: - description: |- - ExcludedContentTypes defines the list of content types to compare the Content-Type header of the incoming requests and responses before compressing. - `application/grpc` is always excluded. - items: - type: string - type: array - includedContentTypes: - description: IncludedContentTypes defines the list of content - types to compare the Content-Type header of the responses before - compressing. - items: - type: string - type: array - minResponseBodyBytes: - description: |- - MinResponseBodyBytes defines the minimum amount of bytes a response body must have to be compressed. - Default: 1024. - type: integer - type: object - contentType: - description: |- - ContentType holds the content-type middleware configuration. - This middleware exists to enable the correct behavior until at least the default one can be changed in a future version. - properties: - autoDetect: - description: |- - AutoDetect specifies whether to let the `Content-Type` header, if it has not been set by the backend, - be automatically set to a value derived from the contents of the response. - Deprecated: AutoDetect option is deprecated, Content-Type middleware is only meant to be used to enable the content-type detection, please remove any usage of this option. - type: boolean - type: object - digestAuth: - description: |- - DigestAuth holds the digest auth middleware configuration. - This middleware restricts access to your services to known users. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/digestauth/ - properties: - headerField: - description: |- - HeaderField defines a header field to store the authenticated user. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/basicauth/#headerfield - type: string - realm: - description: |- - Realm allows the protected resources on a server to be partitioned into a set of protection spaces, each with its own authentication scheme. - Default: traefik. - type: string - removeHeader: - description: RemoveHeader defines whether to remove the authorization - header before forwarding the request to the backend. - type: boolean - secret: - description: Secret is the name of the referenced Kubernetes Secret - containing user credentials. - type: string - type: object - errors: - description: |- - ErrorPage holds the custom error middleware configuration. - This middleware returns a custom page in lieu of the default, according to configured ranges of HTTP Status codes. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/errorpages/ - properties: - query: - description: |- - Query defines the URL for the error page (hosted by service). - The {status} variable can be used in order to insert the status code in the URL. - type: string - service: - description: |- - Service defines the reference to a Kubernetes Service that will serve the error page. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/errorpages/#service - properties: - healthCheck: - description: Healthcheck defines health checks for ExternalName - services. - properties: - followRedirects: - description: |- - FollowRedirects defines whether redirects should be followed during the health check calls. - Default: true - type: boolean - headers: - additionalProperties: - type: string - description: Headers defines custom headers to be sent - to the health check endpoint. - type: object - hostname: - description: Hostname defines the value of hostname in - the Host header of the health check request. - type: string - interval: - anyOf: - - type: integer - - type: string - description: |- - Interval defines the frequency of the health check calls. - Default: 30s - x-kubernetes-int-or-string: true - method: - description: Method defines the healthcheck method. - type: string - mode: - description: |- - Mode defines the health check mode. - If defined to grpc, will use the gRPC health check protocol to probe the server. - Default: http - type: string - path: - description: Path defines the server URL path for the - health check endpoint. - type: string - port: - description: Port defines the server URL port for the - health check endpoint. - type: integer - scheme: - description: Scheme replaces the server URL scheme for - the health check endpoint. - type: string - status: - description: Status defines the expected HTTP status code - of the response to the health check request. - type: integer - timeout: - anyOf: - - type: integer - - type: string - description: |- - Timeout defines the maximum duration Traefik will wait for a health check request before considering the server unhealthy. - Default: 5s - x-kubernetes-int-or-string: true - type: object - kind: - description: Kind defines the kind of the Service. - enum: - - Service - - TraefikService - type: string - name: - description: |- - Name defines the name of the referenced Kubernetes Service or TraefikService. - The differentiation between the two is specified in the Kind field. - type: string - namespace: - description: Namespace defines the namespace of the referenced - Kubernetes Service or TraefikService. - type: string - nativeLB: - description: |- - NativeLB controls, when creating the load-balancer, - whether the LB's children are directly the pods IPs or if the only child is the Kubernetes Service clusterIP. - The Kubernetes Service itself does load-balance to the pods. - By default, NativeLB is false. - type: boolean - nodePortLB: - description: |- - NodePortLB controls, when creating the load-balancer, - whether the LB's children are directly the nodes internal IPs using the nodePort when the service type is NodePort. - It allows services to be reachable when Traefik runs externally from the Kubernetes cluster but within the same network of the nodes. - By default, NodePortLB is false. - type: boolean - passHostHeader: - description: |- - PassHostHeader defines whether the client Host header is forwarded to the upstream Kubernetes Service. - By default, passHostHeader is true. - type: boolean - port: - anyOf: - - type: integer - - type: string - description: |- - Port defines the port of a Kubernetes Service. - This can be a reference to a named port. - x-kubernetes-int-or-string: true - responseForwarding: - description: ResponseForwarding defines how Traefik forwards - the response from the upstream Kubernetes Service to the - client. - properties: - flushInterval: - description: |- - FlushInterval defines the interval, in milliseconds, in between flushes to the client while copying the response body. - A negative value means to flush immediately after each write to the client. - This configuration is ignored when ReverseProxy recognizes a response as a streaming response; - for such responses, writes are flushed to the client immediately. - Default: 100ms - type: string - type: object - scheme: - description: |- - Scheme defines the scheme to use for the request to the upstream Kubernetes Service. - It defaults to https when Kubernetes Service port is 443, http otherwise. - type: string - serversTransport: - description: |- - ServersTransport defines the name of ServersTransport resource to use. - It allows to configure the transport between Traefik and your servers. - Can only be used on a Kubernetes Service. - type: string - sticky: - description: |- - Sticky defines the sticky sessions configuration. - More info: https://doc.traefik.io/traefik/v3.2/routing/services/#sticky-sessions - properties: - cookie: - description: Cookie defines the sticky cookie configuration. - properties: - httpOnly: - description: HTTPOnly defines whether the cookie can - be accessed by client-side APIs, such as JavaScript. - type: boolean - maxAge: - description: |- - MaxAge indicates the number of seconds until the cookie expires. - When set to a negative number, the cookie expires immediately. - When set to zero, the cookie never expires. - type: integer - name: - description: Name defines the Cookie name. - type: string - sameSite: - description: |- - SameSite defines the same site policy. - More info: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie/SameSite - type: string - secure: - description: Secure defines whether the cookie can - only be transmitted over an encrypted connection - (i.e. HTTPS). - type: boolean - type: object - type: object - strategy: - description: |- - Strategy defines the load balancing strategy between the servers. - RoundRobin is the only supported value at the moment. - type: string - weight: - description: |- - Weight defines the weight and should only be specified when Name references a TraefikService object - (and to be precise, one that embeds a Weighted Round Robin). - type: integer - required: - - name - type: object - status: - description: |- - Status defines which status or range of statuses should result in an error page. - It can be either a status code as a number (500), - as multiple comma-separated numbers (500,502), - as ranges by separating two codes with a dash (500-599), - or a combination of the two (404,418,500-599). - items: - type: string - type: array - type: object - forwardAuth: - description: |- - ForwardAuth holds the forward auth middleware configuration. - This middleware delegates the request authentication to a Service. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/forwardauth/ - properties: - addAuthCookiesToResponse: - description: AddAuthCookiesToResponse defines the list of cookies - to copy from the authentication server response to the response. - items: - type: string - type: array - address: - description: Address defines the authentication server address. - type: string - authRequestHeaders: - description: |- - AuthRequestHeaders defines the list of the headers to copy from the request to the authentication server. - If not set or empty then all request headers are passed. - items: - type: string - type: array - authResponseHeaders: - description: AuthResponseHeaders defines the list of headers to - copy from the authentication server response and set on forwarded - request, replacing any existing conflicting headers. - items: - type: string - type: array - authResponseHeadersRegex: - description: |- - AuthResponseHeadersRegex defines the regex to match headers to copy from the authentication server response and set on forwarded request, after stripping all headers that match the regex. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/forwardauth/#authresponseheadersregex - type: string - tls: - description: TLS defines the configuration used to secure the - connection to the authentication server. - properties: - caOptional: - description: 'Deprecated: TLS client authentication is a server - side option (see https://github.com/golang/go/blob/740a490f71d026bb7d2d13cb8fa2d6d6e0572b70/src/crypto/tls/common.go#L634).' - type: boolean - caSecret: - description: |- - CASecret is the name of the referenced Kubernetes Secret containing the CA to validate the server certificate. - The CA certificate is extracted from key `tls.ca` or `ca.crt`. - type: string - certSecret: - description: |- - CertSecret is the name of the referenced Kubernetes Secret containing the client certificate. - The client certificate is extracted from the keys `tls.crt` and `tls.key`. - type: string - insecureSkipVerify: - description: InsecureSkipVerify defines whether the server - certificates should be validated. - type: boolean - type: object - trustForwardHeader: - description: 'TrustForwardHeader defines whether to trust (ie: - forward) all X-Forwarded-* headers.' - type: boolean - type: object - grpcWeb: - description: |- - GrpcWeb holds the gRPC web middleware configuration. - This middleware converts a gRPC web request to an HTTP/2 gRPC request. - properties: - allowOrigins: - description: |- - AllowOrigins is a list of allowable origins. - Can also be a wildcard origin "*". - items: - type: string - type: array - type: object - headers: - description: |- - Headers holds the headers middleware configuration. - This middleware manages the requests and responses headers. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/headers/#customrequestheaders - properties: - accessControlAllowCredentials: - description: AccessControlAllowCredentials defines whether the - request can include user credentials. - type: boolean - accessControlAllowHeaders: - description: AccessControlAllowHeaders defines the Access-Control-Request-Headers - values sent in preflight response. - items: - type: string - type: array - accessControlAllowMethods: - description: AccessControlAllowMethods defines the Access-Control-Request-Method - values sent in preflight response. - items: - type: string - type: array - accessControlAllowOriginList: - description: AccessControlAllowOriginList is a list of allowable - origins. Can also be a wildcard origin "*". - items: - type: string - type: array - accessControlAllowOriginListRegex: - description: AccessControlAllowOriginListRegex is a list of allowable - origins written following the Regular Expression syntax (https://golang.org/pkg/regexp/). - items: - type: string - type: array - accessControlExposeHeaders: - description: AccessControlExposeHeaders defines the Access-Control-Expose-Headers - values sent in preflight response. - items: - type: string - type: array - accessControlMaxAge: - description: AccessControlMaxAge defines the time that a preflight - request may be cached. - format: int64 - type: integer - addVaryHeader: - description: AddVaryHeader defines whether the Vary header is - automatically added/updated when the AccessControlAllowOriginList - is set. - type: boolean - allowedHosts: - description: AllowedHosts defines the fully qualified list of - allowed domain names. - items: - type: string - type: array - browserXssFilter: - description: BrowserXSSFilter defines whether to add the X-XSS-Protection - header with the value 1; mode=block. - type: boolean - contentSecurityPolicy: - description: ContentSecurityPolicy defines the Content-Security-Policy - header value. - type: string - contentSecurityPolicyReportOnly: - description: ContentSecurityPolicyReportOnly defines the Content-Security-Policy-Report-Only - header value. - type: string - contentTypeNosniff: - description: ContentTypeNosniff defines whether to add the X-Content-Type-Options - header with the nosniff value. - type: boolean - customBrowserXSSValue: - description: |- - CustomBrowserXSSValue defines the X-XSS-Protection header value. - This overrides the BrowserXssFilter option. - type: string - customFrameOptionsValue: - description: |- - CustomFrameOptionsValue defines the X-Frame-Options header value. - This overrides the FrameDeny option. - type: string - customRequestHeaders: - additionalProperties: - type: string - description: CustomRequestHeaders defines the header names and - values to apply to the request. - type: object - customResponseHeaders: - additionalProperties: - type: string - description: CustomResponseHeaders defines the header names and - values to apply to the response. - type: object - featurePolicy: - description: 'Deprecated: FeaturePolicy option is deprecated, - please use PermissionsPolicy instead.' - type: string - forceSTSHeader: - description: ForceSTSHeader defines whether to add the STS header - even when the connection is HTTP. - type: boolean - frameDeny: - description: FrameDeny defines whether to add the X-Frame-Options - header with the DENY value. - type: boolean - hostsProxyHeaders: - description: HostsProxyHeaders defines the header keys that may - hold a proxied hostname value for the request. - items: - type: string - type: array - isDevelopment: - description: |- - IsDevelopment defines whether to mitigate the unwanted effects of the AllowedHosts, SSL, and STS options when developing. - Usually testing takes place using HTTP, not HTTPS, and on localhost, not your production domain. - If you would like your development environment to mimic production with complete Host blocking, SSL redirects, - and STS headers, leave this as false. - type: boolean - permissionsPolicy: - description: |- - PermissionsPolicy defines the Permissions-Policy header value. - This allows sites to control browser features. - type: string - publicKey: - description: PublicKey is the public key that implements HPKP - to prevent MITM attacks with forged certificates. - type: string - referrerPolicy: - description: |- - ReferrerPolicy defines the Referrer-Policy header value. - This allows sites to control whether browsers forward the Referer header to other sites. - type: string - sslForceHost: - description: 'Deprecated: SSLForceHost option is deprecated, please - use RedirectRegex instead.' - type: boolean - sslHost: - description: 'Deprecated: SSLHost option is deprecated, please - use RedirectRegex instead.' - type: string - sslProxyHeaders: - additionalProperties: - type: string - description: |- - SSLProxyHeaders defines the header keys with associated values that would indicate a valid HTTPS request. - It can be useful when using other proxies (example: "X-Forwarded-Proto": "https"). - type: object - sslRedirect: - description: 'Deprecated: SSLRedirect option is deprecated, please - use EntryPoint redirection or RedirectScheme instead.' - type: boolean - sslTemporaryRedirect: - description: 'Deprecated: SSLTemporaryRedirect option is deprecated, - please use EntryPoint redirection or RedirectScheme instead.' - type: boolean - stsIncludeSubdomains: - description: STSIncludeSubdomains defines whether the includeSubDomains - directive is appended to the Strict-Transport-Security header. - type: boolean - stsPreload: - description: STSPreload defines whether the preload flag is appended - to the Strict-Transport-Security header. - type: boolean - stsSeconds: - description: |- - STSSeconds defines the max-age of the Strict-Transport-Security header. - If set to 0, the header is not set. - format: int64 - type: integer - type: object - inFlightReq: - description: |- - InFlightReq holds the in-flight request middleware configuration. - This middleware limits the number of requests being processed and served concurrently. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/inflightreq/ - properties: - amount: - description: |- - Amount defines the maximum amount of allowed simultaneous in-flight request. - The middleware responds with HTTP 429 Too Many Requests if there are already amount requests in progress (based on the same sourceCriterion strategy). - format: int64 - type: integer - sourceCriterion: - description: |- - SourceCriterion defines what criterion is used to group requests as originating from a common source. - If several strategies are defined at the same time, an error will be raised. - If none are set, the default is to use the requestHost. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/inflightreq/#sourcecriterion - properties: - ipStrategy: - description: |- - IPStrategy holds the IP strategy configuration used by Traefik to determine the client IP. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/ipallowlist/#ipstrategy - properties: - depth: - description: Depth tells Traefik to use the X-Forwarded-For - header and take the IP located at the depth position - (starting from the right). - type: integer - excludedIPs: - description: ExcludedIPs configures Traefik to scan the - X-Forwarded-For header and select the first IP not in - the list. - items: - type: string - type: array - ipv6Subnet: - description: IPv6Subnet configures Traefik to consider - all IPv6 addresses from the defined subnet as originating - from the same IP. Applies to RemoteAddrStrategy and - DepthStrategy. - type: integer - type: object - requestHeaderName: - description: RequestHeaderName defines the name of the header - used to group incoming requests. - type: string - requestHost: - description: RequestHost defines whether to consider the request - Host as the source. - type: boolean - type: object - type: object - ipAllowList: - description: |- - IPAllowList holds the IP allowlist middleware configuration. - This middleware limits allowed requests based on the client IP. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/ipallowlist/ - properties: - ipStrategy: - description: |- - IPStrategy holds the IP strategy configuration used by Traefik to determine the client IP. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/ipallowlist/#ipstrategy - properties: - depth: - description: Depth tells Traefik to use the X-Forwarded-For - header and take the IP located at the depth position (starting - from the right). - type: integer - excludedIPs: - description: ExcludedIPs configures Traefik to scan the X-Forwarded-For - header and select the first IP not in the list. - items: - type: string - type: array - ipv6Subnet: - description: IPv6Subnet configures Traefik to consider all - IPv6 addresses from the defined subnet as originating from - the same IP. Applies to RemoteAddrStrategy and DepthStrategy. - type: integer - type: object - rejectStatusCode: - description: |- - RejectStatusCode defines the HTTP status code used for refused requests. - If not set, the default is 403 (Forbidden). - type: integer - sourceRange: - description: SourceRange defines the set of allowed IPs (or ranges - of allowed IPs by using CIDR notation). - items: - type: string - type: array - type: object - ipWhiteList: - description: 'Deprecated: please use IPAllowList instead.' - properties: - ipStrategy: - description: |- - IPStrategy holds the IP strategy configuration used by Traefik to determine the client IP. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/ipallowlist/#ipstrategy - properties: - depth: - description: Depth tells Traefik to use the X-Forwarded-For - header and take the IP located at the depth position (starting - from the right). - type: integer - excludedIPs: - description: ExcludedIPs configures Traefik to scan the X-Forwarded-For - header and select the first IP not in the list. - items: - type: string - type: array - ipv6Subnet: - description: IPv6Subnet configures Traefik to consider all - IPv6 addresses from the defined subnet as originating from - the same IP. Applies to RemoteAddrStrategy and DepthStrategy. - type: integer - type: object - sourceRange: - description: SourceRange defines the set of allowed IPs (or ranges - of allowed IPs by using CIDR notation). Required. - items: - type: string - type: array - type: object - passTLSClientCert: - description: |- - PassTLSClientCert holds the pass TLS client cert middleware configuration. - This middleware adds the selected data from the passed client TLS certificate to a header. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/passtlsclientcert/ - properties: - info: - description: Info selects the specific client certificate details - you want to add to the X-Forwarded-Tls-Client-Cert-Info header. - properties: - issuer: - description: Issuer defines the client certificate issuer - details to add to the X-Forwarded-Tls-Client-Cert-Info header. - properties: - commonName: - description: CommonName defines whether to add the organizationalUnit - information into the issuer. - type: boolean - country: - description: Country defines whether to add the country - information into the issuer. - type: boolean - domainComponent: - description: DomainComponent defines whether to add the - domainComponent information into the issuer. - type: boolean - locality: - description: Locality defines whether to add the locality - information into the issuer. - type: boolean - organization: - description: Organization defines whether to add the organization - information into the issuer. - type: boolean - province: - description: Province defines whether to add the province - information into the issuer. - type: boolean - serialNumber: - description: SerialNumber defines whether to add the serialNumber - information into the issuer. - type: boolean - type: object - notAfter: - description: NotAfter defines whether to add the Not After - information from the Validity part. - type: boolean - notBefore: - description: NotBefore defines whether to add the Not Before - information from the Validity part. - type: boolean - sans: - description: Sans defines whether to add the Subject Alternative - Name information from the Subject Alternative Name part. - type: boolean - serialNumber: - description: SerialNumber defines whether to add the client - serialNumber information. - type: boolean - subject: - description: Subject defines the client certificate subject - details to add to the X-Forwarded-Tls-Client-Cert-Info header. - properties: - commonName: - description: CommonName defines whether to add the organizationalUnit - information into the subject. - type: boolean - country: - description: Country defines whether to add the country - information into the subject. - type: boolean - domainComponent: - description: DomainComponent defines whether to add the - domainComponent information into the subject. - type: boolean - locality: - description: Locality defines whether to add the locality - information into the subject. - type: boolean - organization: - description: Organization defines whether to add the organization - information into the subject. - type: boolean - organizationalUnit: - description: OrganizationalUnit defines whether to add - the organizationalUnit information into the subject. - type: boolean - province: - description: Province defines whether to add the province - information into the subject. - type: boolean - serialNumber: - description: SerialNumber defines whether to add the serialNumber - information into the subject. - type: boolean - type: object - type: object - pem: - description: PEM sets the X-Forwarded-Tls-Client-Cert header with - the certificate. - type: boolean - type: object - plugin: - additionalProperties: - x-kubernetes-preserve-unknown-fields: true - description: |- - Plugin defines the middleware plugin configuration. - More info: https://doc.traefik.io/traefik/plugins/ - type: object - rateLimit: - description: |- - RateLimit holds the rate limit configuration. - This middleware ensures that services will receive a fair amount of requests, and allows one to define what fair is. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/ratelimit/ - properties: - average: - description: |- - Average is the maximum rate, by default in requests/s, allowed for the given source. - It defaults to 0, which means no rate limiting. - The rate is actually defined by dividing Average by Period. So for a rate below 1req/s, - one needs to define a Period larger than a second. - format: int64 - type: integer - burst: - description: |- - Burst is the maximum number of requests allowed to arrive in the same arbitrarily small period of time. - It defaults to 1. - format: int64 - type: integer - period: - anyOf: - - type: integer - - type: string - description: |- - Period, in combination with Average, defines the actual maximum rate, such as: - r = Average / Period. It defaults to a second. - x-kubernetes-int-or-string: true - sourceCriterion: - description: |- - SourceCriterion defines what criterion is used to group requests as originating from a common source. - If several strategies are defined at the same time, an error will be raised. - If none are set, the default is to use the request's remote address field (as an ipStrategy). - properties: - ipStrategy: - description: |- - IPStrategy holds the IP strategy configuration used by Traefik to determine the client IP. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/ipallowlist/#ipstrategy - properties: - depth: - description: Depth tells Traefik to use the X-Forwarded-For - header and take the IP located at the depth position - (starting from the right). - type: integer - excludedIPs: - description: ExcludedIPs configures Traefik to scan the - X-Forwarded-For header and select the first IP not in - the list. - items: - type: string - type: array - ipv6Subnet: - description: IPv6Subnet configures Traefik to consider - all IPv6 addresses from the defined subnet as originating - from the same IP. Applies to RemoteAddrStrategy and - DepthStrategy. - type: integer - type: object - requestHeaderName: - description: RequestHeaderName defines the name of the header - used to group incoming requests. - type: string - requestHost: - description: RequestHost defines whether to consider the request - Host as the source. - type: boolean - type: object - type: object - redirectRegex: - description: |- - RedirectRegex holds the redirect regex middleware configuration. - This middleware redirects a request using regex matching and replacement. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/redirectregex/#regex - properties: - permanent: - description: Permanent defines whether the redirection is permanent - (301). - type: boolean - regex: - description: Regex defines the regex used to match and capture - elements from the request URL. - type: string - replacement: - description: Replacement defines how to modify the URL to have - the new target URL. - type: string - type: object - redirectScheme: - description: |- - RedirectScheme holds the redirect scheme middleware configuration. - This middleware redirects requests from a scheme/port to another. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/redirectscheme/ - properties: - permanent: - description: Permanent defines whether the redirection is permanent - (301). - type: boolean - port: - description: Port defines the port of the new URL. - type: string - scheme: - description: Scheme defines the scheme of the new URL. - type: string - type: object - replacePath: - description: |- - ReplacePath holds the replace path middleware configuration. - This middleware replaces the path of the request URL and store the original path in an X-Replaced-Path header. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/replacepath/ - properties: - path: - description: Path defines the path to use as replacement in the - request URL. - type: string - type: object - replacePathRegex: - description: |- - ReplacePathRegex holds the replace path regex middleware configuration. - This middleware replaces the path of a URL using regex matching and replacement. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/replacepathregex/ - properties: - regex: - description: Regex defines the regular expression used to match - and capture the path from the request URL. - type: string - replacement: - description: Replacement defines the replacement path format, - which can include captured variables. - type: string - type: object - retry: - description: |- - Retry holds the retry middleware configuration. - This middleware reissues requests a given number of times to a backend server if that server does not reply. - As soon as the server answers, the middleware stops retrying, regardless of the response status. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/retry/ - properties: - attempts: - description: Attempts defines how many times the request should - be retried. - type: integer - initialInterval: - anyOf: - - type: integer - - type: string - description: |- - InitialInterval defines the first wait time in the exponential backoff series. - The maximum interval is calculated as twice the initialInterval. - If unspecified, requests will be retried immediately. - The value of initialInterval should be provided in seconds or as a valid duration format, - see https://pkg.go.dev/time#ParseDuration. - x-kubernetes-int-or-string: true - type: object - stripPrefix: - description: |- - StripPrefix holds the strip prefix middleware configuration. - This middleware removes the specified prefixes from the URL path. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/stripprefix/ - properties: - forceSlash: - description: |- - Deprecated: ForceSlash option is deprecated, please remove any usage of this option. - ForceSlash ensures that the resulting stripped path is not the empty string, by replacing it with / when necessary. - Default: true. - type: boolean - prefixes: - description: Prefixes defines the prefixes to strip from the request - URL. - items: - type: string - type: array - type: object - stripPrefixRegex: - description: |- - StripPrefixRegex holds the strip prefix regex middleware configuration. - This middleware removes the matching prefixes from the URL path. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/http/stripprefixregex/ - properties: - regex: - description: Regex defines the regular expression to match the - path prefix from the request URL. - items: - type: string - type: array - type: object - type: object - required: - - metadata - - spec - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/traefik.io_middlewaretcps.yaml b/charts/traefik/traefik/crds/traefik.io_middlewaretcps.yaml deleted file mode 100644 index fc23e11..0000000 --- a/charts/traefik/traefik/crds/traefik.io_middlewaretcps.yaml +++ /dev/null @@ -1,87 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.16.1 - name: middlewaretcps.traefik.io -spec: - group: traefik.io - names: - kind: MiddlewareTCP - listKind: MiddlewareTCPList - plural: middlewaretcps - singular: middlewaretcp - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - MiddlewareTCP is the CRD implementation of a Traefik TCP middleware. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/overview/ - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: MiddlewareTCPSpec defines the desired state of a MiddlewareTCP. - properties: - inFlightConn: - description: InFlightConn defines the InFlightConn middleware configuration. - properties: - amount: - description: |- - Amount defines the maximum amount of allowed simultaneous connections. - The middleware closes the connection if there are already amount connections opened. - format: int64 - type: integer - type: object - ipAllowList: - description: |- - IPAllowList defines the IPAllowList middleware configuration. - This middleware accepts/refuses connections based on the client IP. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/tcp/ipallowlist/ - properties: - sourceRange: - description: SourceRange defines the allowed IPs (or ranges of - allowed IPs by using CIDR notation). - items: - type: string - type: array - type: object - ipWhiteList: - description: |- - IPWhiteList defines the IPWhiteList middleware configuration. - This middleware accepts/refuses connections based on the client IP. - Deprecated: please use IPAllowList instead. - More info: https://doc.traefik.io/traefik/v3.2/middlewares/tcp/ipwhitelist/ - properties: - sourceRange: - description: SourceRange defines the allowed IPs (or ranges of - allowed IPs by using CIDR notation). - items: - type: string - type: array - type: object - type: object - required: - - metadata - - spec - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/traefik.io_serverstransports.yaml b/charts/traefik/traefik/crds/traefik.io_serverstransports.yaml deleted file mode 100644 index fe2f129..0000000 --- a/charts/traefik/traefik/crds/traefik.io_serverstransports.yaml +++ /dev/null @@ -1,139 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.16.1 - name: serverstransports.traefik.io -spec: - group: traefik.io - names: - kind: ServersTransport - listKind: ServersTransportList - plural: serverstransports - singular: serverstransport - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - ServersTransport is the CRD implementation of a ServersTransport. - If no serversTransport is specified, the default@internal will be used. - The default@internal serversTransport is created from the static configuration. - More info: https://doc.traefik.io/traefik/v3.2/routing/services/#serverstransport_1 - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ServersTransportSpec defines the desired state of a ServersTransport. - properties: - certificatesSecrets: - description: CertificatesSecrets defines a list of secret storing - client certificates for mTLS. - items: - type: string - type: array - disableHTTP2: - description: DisableHTTP2 disables HTTP/2 for connections with backend - servers. - type: boolean - forwardingTimeouts: - description: ForwardingTimeouts defines the timeouts for requests - forwarded to the backend servers. - properties: - dialTimeout: - anyOf: - - type: integer - - type: string - description: DialTimeout is the amount of time to wait until a - connection to a backend server can be established. - x-kubernetes-int-or-string: true - idleConnTimeout: - anyOf: - - type: integer - - type: string - description: IdleConnTimeout is the maximum period for which an - idle HTTP keep-alive connection will remain open before closing - itself. - x-kubernetes-int-or-string: true - pingTimeout: - anyOf: - - type: integer - - type: string - description: PingTimeout is the timeout after which the HTTP/2 - connection will be closed if a response to ping is not received. - x-kubernetes-int-or-string: true - readIdleTimeout: - anyOf: - - type: integer - - type: string - description: ReadIdleTimeout is the timeout after which a health - check using ping frame will be carried out if no frame is received - on the HTTP/2 connection. - x-kubernetes-int-or-string: true - responseHeaderTimeout: - anyOf: - - type: integer - - type: string - description: ResponseHeaderTimeout is the amount of time to wait - for a server's response headers after fully writing the request - (including its body, if any). - x-kubernetes-int-or-string: true - type: object - insecureSkipVerify: - description: InsecureSkipVerify disables SSL certificate verification. - type: boolean - maxIdleConnsPerHost: - description: MaxIdleConnsPerHost controls the maximum idle (keep-alive) - to keep per-host. - type: integer - peerCertURI: - description: PeerCertURI defines the peer cert URI used to match against - SAN URI during the peer certificate verification. - type: string - rootCAsSecrets: - description: RootCAsSecrets defines a list of CA secret used to validate - self-signed certificate. - items: - type: string - type: array - serverName: - description: ServerName defines the server name used to contact the - server. - type: string - spiffe: - description: Spiffe defines the SPIFFE configuration. - properties: - ids: - description: IDs defines the allowed SPIFFE IDs (takes precedence - over the SPIFFE TrustDomain). - items: - type: string - type: array - trustDomain: - description: TrustDomain defines the allowed SPIFFE trust domain. - type: string - type: object - type: object - required: - - metadata - - spec - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/traefik.io_serverstransporttcps.yaml b/charts/traefik/traefik/crds/traefik.io_serverstransporttcps.yaml deleted file mode 100644 index 2f24c84..0000000 --- a/charts/traefik/traefik/crds/traefik.io_serverstransporttcps.yaml +++ /dev/null @@ -1,120 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.16.1 - name: serverstransporttcps.traefik.io -spec: - group: traefik.io - names: - kind: ServersTransportTCP - listKind: ServersTransportTCPList - plural: serverstransporttcps - singular: serverstransporttcp - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - ServersTransportTCP is the CRD implementation of a TCPServersTransport. - If no tcpServersTransport is specified, a default one named default@internal will be used. - The default@internal tcpServersTransport can be configured in the static configuration. - More info: https://doc.traefik.io/traefik/v3.2/routing/services/#serverstransport_3 - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ServersTransportTCPSpec defines the desired state of a ServersTransportTCP. - properties: - dialKeepAlive: - anyOf: - - type: integer - - type: string - description: DialKeepAlive is the interval between keep-alive probes - for an active network connection. If zero, keep-alive probes are - sent with a default value (currently 15 seconds), if supported by - the protocol and operating system. Network protocols or operating - systems that do not support keep-alives ignore this field. If negative, - keep-alive probes are disabled. - x-kubernetes-int-or-string: true - dialTimeout: - anyOf: - - type: integer - - type: string - description: DialTimeout is the amount of time to wait until a connection - to a backend server can be established. - x-kubernetes-int-or-string: true - terminationDelay: - anyOf: - - type: integer - - type: string - description: TerminationDelay defines the delay to wait before fully - terminating the connection, after one connected peer has closed - its writing capability. - x-kubernetes-int-or-string: true - tls: - description: TLS defines the TLS configuration - properties: - certificatesSecrets: - description: CertificatesSecrets defines a list of secret storing - client certificates for mTLS. - items: - type: string - type: array - insecureSkipVerify: - description: InsecureSkipVerify disables TLS certificate verification. - type: boolean - peerCertURI: - description: |- - MaxIdleConnsPerHost controls the maximum idle (keep-alive) to keep per-host. - PeerCertURI defines the peer cert URI used to match against SAN URI during the peer certificate verification. - type: string - rootCAsSecrets: - description: RootCAsSecrets defines a list of CA secret used to - validate self-signed certificates. - items: - type: string - type: array - serverName: - description: ServerName defines the server name used to contact - the server. - type: string - spiffe: - description: Spiffe defines the SPIFFE configuration. - properties: - ids: - description: IDs defines the allowed SPIFFE IDs (takes precedence - over the SPIFFE TrustDomain). - items: - type: string - type: array - trustDomain: - description: TrustDomain defines the allowed SPIFFE trust - domain. - type: string - type: object - type: object - type: object - required: - - metadata - - spec - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/traefik.io_tlsoptions.yaml b/charts/traefik/traefik/crds/traefik.io_tlsoptions.yaml deleted file mode 100644 index 498fc3c..0000000 --- a/charts/traefik/traefik/crds/traefik.io_tlsoptions.yaml +++ /dev/null @@ -1,114 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.16.1 - name: tlsoptions.traefik.io -spec: - group: traefik.io - names: - kind: TLSOption - listKind: TLSOptionList - plural: tlsoptions - singular: tlsoption - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - TLSOption is the CRD implementation of a Traefik TLS Option, allowing to configure some parameters of the TLS connection. - More info: https://doc.traefik.io/traefik/v3.2/https/tls/#tls-options - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: TLSOptionSpec defines the desired state of a TLSOption. - properties: - alpnProtocols: - description: |- - ALPNProtocols defines the list of supported application level protocols for the TLS handshake, in order of preference. - More info: https://doc.traefik.io/traefik/v3.2/https/tls/#alpn-protocols - items: - type: string - type: array - cipherSuites: - description: |- - CipherSuites defines the list of supported cipher suites for TLS versions up to TLS 1.2. - More info: https://doc.traefik.io/traefik/v3.2/https/tls/#cipher-suites - items: - type: string - type: array - clientAuth: - description: ClientAuth defines the server's policy for TLS Client - Authentication. - properties: - clientAuthType: - description: ClientAuthType defines the client authentication - type to apply. - enum: - - NoClientCert - - RequestClientCert - - RequireAnyClientCert - - VerifyClientCertIfGiven - - RequireAndVerifyClientCert - type: string - secretNames: - description: SecretNames defines the names of the referenced Kubernetes - Secret storing certificate details. - items: - type: string - type: array - type: object - curvePreferences: - description: |- - CurvePreferences defines the preferred elliptic curves in a specific order. - More info: https://doc.traefik.io/traefik/v3.2/https/tls/#curve-preferences - items: - type: string - type: array - maxVersion: - description: |- - MaxVersion defines the maximum TLS version that Traefik will accept. - Possible values: VersionTLS10, VersionTLS11, VersionTLS12, VersionTLS13. - Default: None. - type: string - minVersion: - description: |- - MinVersion defines the minimum TLS version that Traefik will accept. - Possible values: VersionTLS10, VersionTLS11, VersionTLS12, VersionTLS13. - Default: VersionTLS10. - type: string - preferServerCipherSuites: - description: |- - PreferServerCipherSuites defines whether the server chooses a cipher suite among his own instead of among the client's. - It is enabled automatically when minVersion or maxVersion is set. - Deprecated: https://github.com/golang/go/issues/45430 - type: boolean - sniStrict: - description: SniStrict defines whether Traefik allows connections - from clients connections that do not specify a server_name extension. - type: boolean - type: object - required: - - metadata - - spec - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/traefik.io_tlsstores.yaml b/charts/traefik/traefik/crds/traefik.io_tlsstores.yaml deleted file mode 100644 index 7eacb77..0000000 --- a/charts/traefik/traefik/crds/traefik.io_tlsstores.yaml +++ /dev/null @@ -1,97 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.16.1 - name: tlsstores.traefik.io -spec: - group: traefik.io - names: - kind: TLSStore - listKind: TLSStoreList - plural: tlsstores - singular: tlsstore - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - TLSStore is the CRD implementation of a Traefik TLS Store. - For the time being, only the TLSStore named default is supported. - This means that you cannot have two stores that are named default in different Kubernetes namespaces. - More info: https://doc.traefik.io/traefik/v3.2/https/tls/#certificates-stores - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: TLSStoreSpec defines the desired state of a TLSStore. - properties: - certificates: - description: Certificates is a list of secret names, each secret holding - a key/certificate pair to add to the store. - items: - description: Certificate holds a secret name for the TLSStore resource. - properties: - secretName: - description: SecretName is the name of the referenced Kubernetes - Secret to specify the certificate details. - type: string - required: - - secretName - type: object - type: array - defaultCertificate: - description: DefaultCertificate defines the default certificate configuration. - properties: - secretName: - description: SecretName is the name of the referenced Kubernetes - Secret to specify the certificate details. - type: string - required: - - secretName - type: object - defaultGeneratedCert: - description: DefaultGeneratedCert defines the default generated certificate - configuration. - properties: - domain: - description: Domain is the domain definition for the DefaultCertificate. - properties: - main: - description: Main defines the main domain name. - type: string - sans: - description: SANs defines the subject alternative domain names. - items: - type: string - type: array - type: object - resolver: - description: Resolver is the name of the resolver that will be - used to issue the DefaultCertificate. - type: string - type: object - type: object - required: - - metadata - - spec - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/crds/traefik.io_traefikservices.yaml b/charts/traefik/traefik/crds/traefik.io_traefikservices.yaml deleted file mode 100644 index 01e28fc..0000000 --- a/charts/traefik/traefik/crds/traefik.io_traefikservices.yaml +++ /dev/null @@ -1,644 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.16.1 - name: traefikservices.traefik.io -spec: - group: traefik.io - names: - kind: TraefikService - listKind: TraefikServiceList - plural: traefikservices - singular: traefikservice - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - TraefikService is the CRD implementation of a Traefik Service. - TraefikService object allows to: - - Apply weight to Services on load-balancing - - Mirror traffic on services - More info: https://doc.traefik.io/traefik/v3.2/routing/providers/kubernetes-crd/#kind-traefikservice - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: TraefikServiceSpec defines the desired state of a TraefikService. - properties: - mirroring: - description: Mirroring defines the Mirroring service configuration. - properties: - healthCheck: - description: Healthcheck defines health checks for ExternalName - services. - properties: - followRedirects: - description: |- - FollowRedirects defines whether redirects should be followed during the health check calls. - Default: true - type: boolean - headers: - additionalProperties: - type: string - description: Headers defines custom headers to be sent to - the health check endpoint. - type: object - hostname: - description: Hostname defines the value of hostname in the - Host header of the health check request. - type: string - interval: - anyOf: - - type: integer - - type: string - description: |- - Interval defines the frequency of the health check calls. - Default: 30s - x-kubernetes-int-or-string: true - method: - description: Method defines the healthcheck method. - type: string - mode: - description: |- - Mode defines the health check mode. - If defined to grpc, will use the gRPC health check protocol to probe the server. - Default: http - type: string - path: - description: Path defines the server URL path for the health - check endpoint. - type: string - port: - description: Port defines the server URL port for the health - check endpoint. - type: integer - scheme: - description: Scheme replaces the server URL scheme for the - health check endpoint. - type: string - status: - description: Status defines the expected HTTP status code - of the response to the health check request. - type: integer - timeout: - anyOf: - - type: integer - - type: string - description: |- - Timeout defines the maximum duration Traefik will wait for a health check request before considering the server unhealthy. - Default: 5s - x-kubernetes-int-or-string: true - type: object - kind: - description: Kind defines the kind of the Service. - enum: - - Service - - TraefikService - type: string - maxBodySize: - description: |- - MaxBodySize defines the maximum size allowed for the body of the request. - If the body is larger, the request is not mirrored. - Default value is -1, which means unlimited size. - format: int64 - type: integer - mirrorBody: - description: |- - MirrorBody defines whether the body of the request should be mirrored. - Default value is true. - type: boolean - mirrors: - description: Mirrors defines the list of mirrors where Traefik - will duplicate the traffic. - items: - description: MirrorService holds the mirror configuration. - properties: - healthCheck: - description: Healthcheck defines health checks for ExternalName - services. - properties: - followRedirects: - description: |- - FollowRedirects defines whether redirects should be followed during the health check calls. - Default: true - type: boolean - headers: - additionalProperties: - type: string - description: Headers defines custom headers to be sent - to the health check endpoint. - type: object - hostname: - description: Hostname defines the value of hostname - in the Host header of the health check request. - type: string - interval: - anyOf: - - type: integer - - type: string - description: |- - Interval defines the frequency of the health check calls. - Default: 30s - x-kubernetes-int-or-string: true - method: - description: Method defines the healthcheck method. - type: string - mode: - description: |- - Mode defines the health check mode. - If defined to grpc, will use the gRPC health check protocol to probe the server. - Default: http - type: string - path: - description: Path defines the server URL path for the - health check endpoint. - type: string - port: - description: Port defines the server URL port for the - health check endpoint. - type: integer - scheme: - description: Scheme replaces the server URL scheme for - the health check endpoint. - type: string - status: - description: Status defines the expected HTTP status - code of the response to the health check request. - type: integer - timeout: - anyOf: - - type: integer - - type: string - description: |- - Timeout defines the maximum duration Traefik will wait for a health check request before considering the server unhealthy. - Default: 5s - x-kubernetes-int-or-string: true - type: object - kind: - description: Kind defines the kind of the Service. - enum: - - Service - - TraefikService - type: string - name: - description: |- - Name defines the name of the referenced Kubernetes Service or TraefikService. - The differentiation between the two is specified in the Kind field. - type: string - namespace: - description: Namespace defines the namespace of the referenced - Kubernetes Service or TraefikService. - type: string - nativeLB: - description: |- - NativeLB controls, when creating the load-balancer, - whether the LB's children are directly the pods IPs or if the only child is the Kubernetes Service clusterIP. - The Kubernetes Service itself does load-balance to the pods. - By default, NativeLB is false. - type: boolean - nodePortLB: - description: |- - NodePortLB controls, when creating the load-balancer, - whether the LB's children are directly the nodes internal IPs using the nodePort when the service type is NodePort. - It allows services to be reachable when Traefik runs externally from the Kubernetes cluster but within the same network of the nodes. - By default, NodePortLB is false. - type: boolean - passHostHeader: - description: |- - PassHostHeader defines whether the client Host header is forwarded to the upstream Kubernetes Service. - By default, passHostHeader is true. - type: boolean - percent: - description: |- - Percent defines the part of the traffic to mirror. - Supported values: 0 to 100. - type: integer - port: - anyOf: - - type: integer - - type: string - description: |- - Port defines the port of a Kubernetes Service. - This can be a reference to a named port. - x-kubernetes-int-or-string: true - responseForwarding: - description: ResponseForwarding defines how Traefik forwards - the response from the upstream Kubernetes Service to the - client. - properties: - flushInterval: - description: |- - FlushInterval defines the interval, in milliseconds, in between flushes to the client while copying the response body. - A negative value means to flush immediately after each write to the client. - This configuration is ignored when ReverseProxy recognizes a response as a streaming response; - for such responses, writes are flushed to the client immediately. - Default: 100ms - type: string - type: object - scheme: - description: |- - Scheme defines the scheme to use for the request to the upstream Kubernetes Service. - It defaults to https when Kubernetes Service port is 443, http otherwise. - type: string - serversTransport: - description: |- - ServersTransport defines the name of ServersTransport resource to use. - It allows to configure the transport between Traefik and your servers. - Can only be used on a Kubernetes Service. - type: string - sticky: - description: |- - Sticky defines the sticky sessions configuration. - More info: https://doc.traefik.io/traefik/v3.2/routing/services/#sticky-sessions - properties: - cookie: - description: Cookie defines the sticky cookie configuration. - properties: - httpOnly: - description: HTTPOnly defines whether the cookie - can be accessed by client-side APIs, such as JavaScript. - type: boolean - maxAge: - description: |- - MaxAge indicates the number of seconds until the cookie expires. - When set to a negative number, the cookie expires immediately. - When set to zero, the cookie never expires. - type: integer - name: - description: Name defines the Cookie name. - type: string - sameSite: - description: |- - SameSite defines the same site policy. - More info: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie/SameSite - type: string - secure: - description: Secure defines whether the cookie can - only be transmitted over an encrypted connection - (i.e. HTTPS). - type: boolean - type: object - type: object - strategy: - description: |- - Strategy defines the load balancing strategy between the servers. - RoundRobin is the only supported value at the moment. - type: string - weight: - description: |- - Weight defines the weight and should only be specified when Name references a TraefikService object - (and to be precise, one that embeds a Weighted Round Robin). - type: integer - required: - - name - type: object - type: array - name: - description: |- - Name defines the name of the referenced Kubernetes Service or TraefikService. - The differentiation between the two is specified in the Kind field. - type: string - namespace: - description: Namespace defines the namespace of the referenced - Kubernetes Service or TraefikService. - type: string - nativeLB: - description: |- - NativeLB controls, when creating the load-balancer, - whether the LB's children are directly the pods IPs or if the only child is the Kubernetes Service clusterIP. - The Kubernetes Service itself does load-balance to the pods. - By default, NativeLB is false. - type: boolean - nodePortLB: - description: |- - NodePortLB controls, when creating the load-balancer, - whether the LB's children are directly the nodes internal IPs using the nodePort when the service type is NodePort. - It allows services to be reachable when Traefik runs externally from the Kubernetes cluster but within the same network of the nodes. - By default, NodePortLB is false. - type: boolean - passHostHeader: - description: |- - PassHostHeader defines whether the client Host header is forwarded to the upstream Kubernetes Service. - By default, passHostHeader is true. - type: boolean - port: - anyOf: - - type: integer - - type: string - description: |- - Port defines the port of a Kubernetes Service. - This can be a reference to a named port. - x-kubernetes-int-or-string: true - responseForwarding: - description: ResponseForwarding defines how Traefik forwards the - response from the upstream Kubernetes Service to the client. - properties: - flushInterval: - description: |- - FlushInterval defines the interval, in milliseconds, in between flushes to the client while copying the response body. - A negative value means to flush immediately after each write to the client. - This configuration is ignored when ReverseProxy recognizes a response as a streaming response; - for such responses, writes are flushed to the client immediately. - Default: 100ms - type: string - type: object - scheme: - description: |- - Scheme defines the scheme to use for the request to the upstream Kubernetes Service. - It defaults to https when Kubernetes Service port is 443, http otherwise. - type: string - serversTransport: - description: |- - ServersTransport defines the name of ServersTransport resource to use. - It allows to configure the transport between Traefik and your servers. - Can only be used on a Kubernetes Service. - type: string - sticky: - description: |- - Sticky defines the sticky sessions configuration. - More info: https://doc.traefik.io/traefik/v3.2/routing/services/#sticky-sessions - properties: - cookie: - description: Cookie defines the sticky cookie configuration. - properties: - httpOnly: - description: HTTPOnly defines whether the cookie can be - accessed by client-side APIs, such as JavaScript. - type: boolean - maxAge: - description: |- - MaxAge indicates the number of seconds until the cookie expires. - When set to a negative number, the cookie expires immediately. - When set to zero, the cookie never expires. - type: integer - name: - description: Name defines the Cookie name. - type: string - sameSite: - description: |- - SameSite defines the same site policy. - More info: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie/SameSite - type: string - secure: - description: Secure defines whether the cookie can only - be transmitted over an encrypted connection (i.e. HTTPS). - type: boolean - type: object - type: object - strategy: - description: |- - Strategy defines the load balancing strategy between the servers. - RoundRobin is the only supported value at the moment. - type: string - weight: - description: |- - Weight defines the weight and should only be specified when Name references a TraefikService object - (and to be precise, one that embeds a Weighted Round Robin). - type: integer - required: - - name - type: object - weighted: - description: Weighted defines the Weighted Round Robin configuration. - properties: - services: - description: Services defines the list of Kubernetes Service and/or - TraefikService to load-balance, with weight. - items: - description: Service defines an upstream HTTP service to proxy - traffic to. - properties: - healthCheck: - description: Healthcheck defines health checks for ExternalName - services. - properties: - followRedirects: - description: |- - FollowRedirects defines whether redirects should be followed during the health check calls. - Default: true - type: boolean - headers: - additionalProperties: - type: string - description: Headers defines custom headers to be sent - to the health check endpoint. - type: object - hostname: - description: Hostname defines the value of hostname - in the Host header of the health check request. - type: string - interval: - anyOf: - - type: integer - - type: string - description: |- - Interval defines the frequency of the health check calls. - Default: 30s - x-kubernetes-int-or-string: true - method: - description: Method defines the healthcheck method. - type: string - mode: - description: |- - Mode defines the health check mode. - If defined to grpc, will use the gRPC health check protocol to probe the server. - Default: http - type: string - path: - description: Path defines the server URL path for the - health check endpoint. - type: string - port: - description: Port defines the server URL port for the - health check endpoint. - type: integer - scheme: - description: Scheme replaces the server URL scheme for - the health check endpoint. - type: string - status: - description: Status defines the expected HTTP status - code of the response to the health check request. - type: integer - timeout: - anyOf: - - type: integer - - type: string - description: |- - Timeout defines the maximum duration Traefik will wait for a health check request before considering the server unhealthy. - Default: 5s - x-kubernetes-int-or-string: true - type: object - kind: - description: Kind defines the kind of the Service. - enum: - - Service - - TraefikService - type: string - name: - description: |- - Name defines the name of the referenced Kubernetes Service or TraefikService. - The differentiation between the two is specified in the Kind field. - type: string - namespace: - description: Namespace defines the namespace of the referenced - Kubernetes Service or TraefikService. - type: string - nativeLB: - description: |- - NativeLB controls, when creating the load-balancer, - whether the LB's children are directly the pods IPs or if the only child is the Kubernetes Service clusterIP. - The Kubernetes Service itself does load-balance to the pods. - By default, NativeLB is false. - type: boolean - nodePortLB: - description: |- - NodePortLB controls, when creating the load-balancer, - whether the LB's children are directly the nodes internal IPs using the nodePort when the service type is NodePort. - It allows services to be reachable when Traefik runs externally from the Kubernetes cluster but within the same network of the nodes. - By default, NodePortLB is false. - type: boolean - passHostHeader: - description: |- - PassHostHeader defines whether the client Host header is forwarded to the upstream Kubernetes Service. - By default, passHostHeader is true. - type: boolean - port: - anyOf: - - type: integer - - type: string - description: |- - Port defines the port of a Kubernetes Service. - This can be a reference to a named port. - x-kubernetes-int-or-string: true - responseForwarding: - description: ResponseForwarding defines how Traefik forwards - the response from the upstream Kubernetes Service to the - client. - properties: - flushInterval: - description: |- - FlushInterval defines the interval, in milliseconds, in between flushes to the client while copying the response body. - A negative value means to flush immediately after each write to the client. - This configuration is ignored when ReverseProxy recognizes a response as a streaming response; - for such responses, writes are flushed to the client immediately. - Default: 100ms - type: string - type: object - scheme: - description: |- - Scheme defines the scheme to use for the request to the upstream Kubernetes Service. - It defaults to https when Kubernetes Service port is 443, http otherwise. - type: string - serversTransport: - description: |- - ServersTransport defines the name of ServersTransport resource to use. - It allows to configure the transport between Traefik and your servers. - Can only be used on a Kubernetes Service. - type: string - sticky: - description: |- - Sticky defines the sticky sessions configuration. - More info: https://doc.traefik.io/traefik/v3.2/routing/services/#sticky-sessions - properties: - cookie: - description: Cookie defines the sticky cookie configuration. - properties: - httpOnly: - description: HTTPOnly defines whether the cookie - can be accessed by client-side APIs, such as JavaScript. - type: boolean - maxAge: - description: |- - MaxAge indicates the number of seconds until the cookie expires. - When set to a negative number, the cookie expires immediately. - When set to zero, the cookie never expires. - type: integer - name: - description: Name defines the Cookie name. - type: string - sameSite: - description: |- - SameSite defines the same site policy. - More info: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie/SameSite - type: string - secure: - description: Secure defines whether the cookie can - only be transmitted over an encrypted connection - (i.e. HTTPS). - type: boolean - type: object - type: object - strategy: - description: |- - Strategy defines the load balancing strategy between the servers. - RoundRobin is the only supported value at the moment. - type: string - weight: - description: |- - Weight defines the weight and should only be specified when Name references a TraefikService object - (and to be precise, one that embeds a Weighted Round Robin). - type: integer - required: - - name - type: object - type: array - sticky: - description: |- - Sticky defines whether sticky sessions are enabled. - More info: https://doc.traefik.io/traefik/v3.2/routing/providers/kubernetes-crd/#stickiness-and-load-balancing - properties: - cookie: - description: Cookie defines the sticky cookie configuration. - properties: - httpOnly: - description: HTTPOnly defines whether the cookie can be - accessed by client-side APIs, such as JavaScript. - type: boolean - maxAge: - description: |- - MaxAge indicates the number of seconds until the cookie expires. - When set to a negative number, the cookie expires immediately. - When set to zero, the cookie never expires. - type: integer - name: - description: Name defines the Cookie name. - type: string - sameSite: - description: |- - SameSite defines the same site policy. - More info: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie/SameSite - type: string - secure: - description: Secure defines whether the cookie can only - be transmitted over an encrypted connection (i.e. HTTPS). - type: boolean - type: object - type: object - type: object - type: object - required: - - metadata - - spec - type: object - served: true - storage: true diff --git a/charts/traefik/traefik/templates/NOTES.txt b/charts/traefik/traefik/templates/NOTES.txt deleted file mode 100644 index a1a10bf..0000000 --- a/charts/traefik/traefik/templates/NOTES.txt +++ /dev/null @@ -1,36 +0,0 @@ - - -{{ .Release.Name }} with {{ .Values.image.registry }}/{{ .Values.image.repository }}:{{ .Values.image.tag | default .Chart.AppVersion }} has been deployed successfully on {{ template "traefik.namespace" . }} namespace ! - -{{- if .Values.persistence }} -{{- if and .Values.persistence.enabled (empty .Values.deployment.initContainer)}} - -🚨 When enabling persistence for certificates, permissions on acme.json can be -lost when Traefik restarts. You can ensure correct permissions with an -initContainer. See https://github.com/traefik/traefik-helm-chart/blob/master/EXAMPLES.md#use-traefik-native-lets-encrypt-integration-without-cert-manager -for more info. 🚨 - -{{- end }} -{{- end }} -{{- with .Values.providers.kubernetesCRD.labelSelector }} - {{- $labelsApplied := include "traefik.labels" $ }} - {{- $labelSelectors := regexSplit "," . -1 }} - {{- range $labelSelectors }} - {{- $labelSelectorRaw := regexSplit "=" . -1 }} - {{- $labelSelector := printf "%s: %s" (first $labelSelectorRaw) (last $labelSelectorRaw) }} - {{- if not (contains $labelSelector $labelsApplied) }} -🚨 Resources populated with this chart don't match with labelSelector `{{.}}` applied on kubernetesCRD provider 🚨 - {{- end }} - {{- end }} -{{- end }} -{{- with .Values.providers.kubernetesIngress.labelSelector }} - {{- $labelsApplied := include "traefik.labels" $ }} - {{- $labelSelectors := regexSplit "," . -1 }} - {{- range $labelSelectors }} - {{- $labelSelectorRaw := regexSplit "=" . -1 }} - {{- $labelSelector := printf "%s: %s" (first $labelSelectorRaw) (last $labelSelectorRaw) }} - {{- if not (contains $labelSelector $labelsApplied) }} -🚨 Resources populated with this chart don't match with labelSelector `{{.}}` applied on kubernetesIngress provider 🚨 - {{- end }} - {{- end }} -{{- end }} diff --git a/charts/traefik/traefik/templates/_helpers.tpl b/charts/traefik/traefik/templates/_helpers.tpl deleted file mode 100644 index 284e912..0000000 --- a/charts/traefik/traefik/templates/_helpers.tpl +++ /dev/null @@ -1,178 +0,0 @@ -{{/* vim: set filetype=mustache: */}} - -{{/* -Expand the name of the chart. -*/}} -{{- define "traefik.name" -}} -{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Create chart name and version as used by the chart label. -*/}} -{{- define "traefik.chart" -}} -{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Create the chart image name. -*/}} -{{- define "traefik.image-name" -}} -{{- printf "%s/%s:%s" .Values.image.registry .Values.image.repository (.Values.image.tag | default .Chart.AppVersion) }} -{{- end -}} - -{{/* -Create a default fully qualified app name. -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -If release name contains chart name it will be used as a full name. -*/}} -{{- define "traefik.fullname" -}} -{{- if .Values.fullnameOverride -}} -{{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" -}} -{{- else -}} -{{- $name := default .Chart.Name .Values.nameOverride -}} -{{- if contains $name .Release.Name -}} -{{- .Release.Name | trunc 63 | trimSuffix "-" -}} -{{- else -}} -{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} -{{- end -}} -{{- end -}} -{{- end -}} - -{{/* -Allow customization of the instance label value. -*/}} -{{- define "traefik.instance-name" -}} -{{- default (printf "%s-%s" .Release.Name .Release.Namespace) .Values.instanceLabelOverride | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* Shared labels used for selector*/}} -{{/* This is an immutable field: this should not change between upgrade */}} -{{- define "traefik.labelselector" -}} -app.kubernetes.io/name: {{ template "traefik.name" . }} -app.kubernetes.io/instance: {{ template "traefik.instance-name" . }} -{{- end }} - -{{/* Shared labels used in metada */}} -{{- define "traefik.labels" -}} -{{ include "traefik.labelselector" . }} -helm.sh/chart: {{ template "traefik.chart" . }} -app.kubernetes.io/managed-by: {{ .Release.Service }} -{{- with .Values.commonLabels }} -{{ toYaml . }} -{{- end }} -{{- end }} - -{{/* -Construct the namespace for all namespaced resources -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -Preserve the default behavior of the Release namespace if no override is provided -*/}} -{{- define "traefik.namespace" -}} -{{- if .Values.namespaceOverride -}} -{{- .Values.namespaceOverride | trunc 63 | trimSuffix "-" -}} -{{- else -}} -{{- .Release.Namespace -}} -{{- end -}} -{{- end -}} - -{{/* -The name of the service account to use -*/}} -{{- define "traefik.serviceAccountName" -}} -{{- default (include "traefik.fullname" .) .Values.serviceAccount.name -}} -{{- end -}} - -{{/* -The name of the ClusterRole and ClusterRoleBinding to use. -Adds the namespace to name to prevent duplicate resource names when there -are multiple namespaced releases with the same release name. -*/}} -{{- define "traefik.clusterRoleName" -}} -{{- (printf "%s-%s" (include "traefik.fullname" .) .Release.Namespace) | trunc 63 | trimSuffix "-" }} -{{- end -}} - -{{/* -Construct the path for the providers.kubernetesingress.ingressendpoint.publishedservice. -By convention this will simply use the / to match the name of the -service generated. -Users can provide an override for an explicit service they want bound via `.Values.providers.kubernetesIngress.publishedService.pathOverride` -*/}} -{{- define "providers.kubernetesIngress.publishedServicePath" -}} -{{- $defServiceName := printf "%s/%s" .Release.Namespace (include "traefik.fullname" .) -}} -{{- $servicePath := default $defServiceName .Values.providers.kubernetesIngress.publishedService.pathOverride }} -{{- print $servicePath | trimSuffix "-" -}} -{{- end -}} - -{{/* -Construct a comma-separated list of whitelisted namespaces -*/}} -{{- define "providers.kubernetesCRD.namespaces" -}} -{{- default (include "traefik.namespace" .) (join "," .Values.providers.kubernetesCRD.namespaces) }} -{{- end -}} -{{- define "providers.kubernetesGateway.namespaces" -}} -{{- default (include "traefik.namespace" .) (join "," .Values.providers.kubernetesGateway.namespaces) }} -{{- end -}} -{{- define "providers.kubernetesIngress.namespaces" -}} -{{- default (include "traefik.namespace" .) (join "," .Values.providers.kubernetesIngress.namespaces) }} -{{- end -}} - -{{/* -Renders a complete tree, even values that contains template. -*/}} -{{- define "traefik.render" -}} - {{- if typeIs "string" .value }} - {{- tpl .value .context }} - {{ else }} - {{- tpl (.value | toYaml) .context }} - {{- end }} -{{- end -}} - -{{- define "imageVersion" -}} -{{/* -Traefik hub is based on v3.1 (v3.0 before v3.3.1) of traefik proxy, so this is a hack to avoid to much complexity in RBAC management which are -based on semverCompare -*/}} -{{- if $.Values.hub.token -}} -{{ if and (regexMatch "v[0-9]+.[0-9]+.[0-9]+" (default "" $.Values.image.tag)) (semverCompare "=3.1.2-0" $version) }} - - "--providers.kubernetescrd.disableClusterScopeResources=true" - {{- end }} - {{- if .Values.providers.kubernetesCRD.nativeLBByDefault }} - - "--providers.kubernetescrd.nativeLBByDefault=true" - {{- end }} - {{- end }} - {{- if .Values.providers.kubernetesIngress.enabled }} - - "--providers.kubernetesingress" - {{- if .Values.providers.kubernetesIngress.allowExternalNameServices }} - - "--providers.kubernetesingress.allowExternalNameServices=true" - {{- end }} - {{- if .Values.providers.kubernetesIngress.allowEmptyServices }} - - "--providers.kubernetesingress.allowEmptyServices=true" - {{- end }} - {{- if and .Values.service.enabled .Values.providers.kubernetesIngress.publishedService.enabled }} - - "--providers.kubernetesingress.ingressendpoint.publishedservice={{ template "providers.kubernetesIngress.publishedServicePath" . }}" - {{- end }} - {{- if .Values.providers.kubernetesIngress.labelSelector }} - - "--providers.kubernetesingress.labelSelector={{ .Values.providers.kubernetesIngress.labelSelector }}" - {{- end }} - {{- if .Values.providers.kubernetesIngress.ingressClass }} - - "--providers.kubernetesingress.ingressClass={{ .Values.providers.kubernetesIngress.ingressClass }}" - {{- end }} - {{- if .Values.rbac.namespaced }} - {{- if semverCompare "<3.1.5-0" $version }} - - "--providers.kubernetesingress.disableIngressClassLookup=true" - {{- if semverCompare ">=3.1.2-0" $version }} - - "--providers.kubernetesingress.disableClusterScopeResources=true" - {{- end }} - {{- else }} - - "--providers.kubernetesingress.disableClusterScopeResources=true" - {{- end }} - {{- end }} - {{- if .Values.providers.kubernetesIngress.nativeLBByDefault }} - - "--providers.kubernetesingress.nativeLBByDefault=true" - {{- end }} - {{- end }} - {{- if .Values.experimental.kubernetesGateway.enabled }} - - "--experimental.kubernetesgateway" - {{- end }} - {{- with .Values.providers.kubernetesCRD }} - {{- if (and .enabled (or .namespaces (and $.Values.rbac.enabled $.Values.rbac.namespaced))) }} - - "--providers.kubernetescrd.namespaces={{ template "providers.kubernetesCRD.namespaces" $ }}" - {{- end }} - {{- end }} - {{- with .Values.providers.kubernetesGateway }} - {{- if .enabled }} - - "--providers.kubernetesgateway" - {{- with .statusAddress }} - {{- with .ip }} - - "--providers.kubernetesgateway.statusaddress.ip={{ . }}" - {{- end }} - {{- with .hostname }} - - "--providers.kubernetesgateway.statusaddress.hostname={{ . }}" - {{- end }} - {{- with .service }} - - "--providers.kubernetesgateway.statusaddress.service.name={{ tpl .name $ }}" - - "--providers.kubernetesgateway.statusaddress.service.namespace={{ tpl .namespace $ }}" - {{- end }} - {{- end }} - {{- if or .namespaces (and $.Values.rbac.enabled $.Values.rbac.namespaced) }} - - "--providers.kubernetesgateway.namespaces={{ template "providers.kubernetesGateway.namespaces" $ }}" - {{- end }} - {{- if .experimentalChannel }} - - "--providers.kubernetesgateway.experimentalchannel=true" - {{- end }} - {{- with .labelselector }} - - "--providers.kubernetesgateway.labelselector={{ . }}" - {{- end }} - {{- end }} - {{- end }} - {{- with .Values.providers.kubernetesIngress }} - {{- if (and .enabled (or .namespaces (and $.Values.rbac.enabled $.Values.rbac.namespaced))) }} - - "--providers.kubernetesingress.namespaces={{ template "providers.kubernetesIngress.namespaces" $ }}" - {{- end }} - {{- end }} - {{- with .Values.providers.file }} - {{- if .enabled }} - - "--providers.file.directory=/etc/traefik/dynamic" - {{- if .watch }} - - "--providers.file.watch=true" - {{- end }} - {{- end }} - {{- end }} - {{- range $entrypoint, $config := $.Values.ports }} - {{- if $config }} - {{- if $config.redirectTo }} - {{- $toPort := index $.Values.ports $config.redirectTo.port }} - - "--entryPoints.{{ $entrypoint }}.http.redirections.entryPoint.to=:{{ $toPort.exposedPort }}" - - "--entryPoints.{{ $entrypoint }}.http.redirections.entryPoint.scheme=https" - {{- if $config.redirectTo.priority }} - - "--entryPoints.{{ $entrypoint }}.http.redirections.entryPoint.priority={{ $config.redirectTo.priority }}" - {{- end }} - {{- if $config.redirectTo.permanent }} - - "--entryPoints.{{ $entrypoint }}.http.redirections.entryPoint.permanent=true" - {{- end }} - {{- end }} - {{- if $config.middlewares }} - - "--entryPoints.{{ $entrypoint }}.http.middlewares={{ join "," $config.middlewares }}" - {{- end }} - {{- if $config.tls }} - {{- if $config.tls.enabled }} - - "--entryPoints.{{ $entrypoint }}.http.tls=true" - {{- if $config.tls.options }} - - "--entryPoints.{{ $entrypoint }}.http.tls.options={{ $config.tls.options }}" - {{- end }} - {{- if $config.tls.certResolver }} - - "--entryPoints.{{ $entrypoint }}.http.tls.certResolver={{ $config.tls.certResolver }}" - {{- end }} - {{- if $config.tls.domains }} - {{- range $index, $domain := $config.tls.domains }} - {{- if $domain.main }} - - "--entryPoints.{{ $entrypoint }}.http.tls.domains[{{ $index }}].main={{ $domain.main }}" - {{- end }} - {{- if $domain.sans }} - - "--entryPoints.{{ $entrypoint }}.http.tls.domains[{{ $index }}].sans={{ join "," $domain.sans }}" - {{- end }} - {{- end }} - {{- end }} - {{- if $config.http3 }} - {{- if $config.http3.enabled }} - - "--entryPoints.{{ $entrypoint }}.http3" - {{- if $config.http3.advertisedPort }} - - "--entryPoints.{{ $entrypoint }}.http3.advertisedPort={{ $config.http3.advertisedPort }}" - {{- end }} - {{- end }} - {{- end }} - {{- end }} - {{- end }} - {{- if $config.allowACMEByPass }} - {{- if (semverCompare "<3.1.3-0" $version) }} - {{- fail "ERROR: allowACMEByPass has been introduced with Traefik v3.1.3+" -}} - {{- end }} - - "--entryPoints.name.allowACMEByPass=true" - {{- end }} - {{- if $config.forwardedHeaders }} - {{- if $config.forwardedHeaders.trustedIPs }} - - "--entryPoints.{{ $entrypoint }}.forwardedHeaders.trustedIPs={{ join "," $config.forwardedHeaders.trustedIPs }}" - {{- end }} - {{- if $config.forwardedHeaders.insecure }} - - "--entryPoints.{{ $entrypoint }}.forwardedHeaders.insecure" - {{- end }} - {{- end }} - {{- if $config.proxyProtocol }} - {{- if $config.proxyProtocol.trustedIPs }} - - "--entryPoints.{{ $entrypoint }}.proxyProtocol.trustedIPs={{ join "," $config.proxyProtocol.trustedIPs }}" - {{- end }} - {{- if $config.proxyProtocol.insecure }} - - "--entryPoints.{{ $entrypoint }}.proxyProtocol.insecure" - {{- end }} - {{- end }} - {{- with $config.transport }} - {{- with .respondingTimeouts }} - {{- if and (ne .readTimeout nil) (toString .readTimeout) }} - - "--entryPoints.{{ $entrypoint }}.transport.respondingTimeouts.readTimeout={{ .readTimeout }}" - {{- end }} - {{- if and (ne .writeTimeout nil) (toString .writeTimeout) }} - - "--entryPoints.{{ $entrypoint }}.transport.respondingTimeouts.writeTimeout={{ .writeTimeout }}" - {{- end }} - {{- if and (ne .idleTimeout nil) (toString .idleTimeout) }} - - "--entryPoints.{{ $entrypoint }}.transport.respondingTimeouts.idleTimeout={{ .idleTimeout }}" - {{- end }} - {{- end }} - {{- with .lifeCycle }} - {{- if and (ne .requestAcceptGraceTimeout nil) (toString .requestAcceptGraceTimeout) }} - - "--entryPoints.{{ $entrypoint }}.transport.lifeCycle.requestAcceptGraceTimeout={{ .requestAcceptGraceTimeout }}" - {{- end }} - {{- if and (ne .graceTimeOut nil) (toString .graceTimeOut) }} - - "--entryPoints.{{ $entrypoint }}.transport.lifeCycle.graceTimeOut={{ .graceTimeOut }}" - {{- end }} - {{- end }} - {{- if and (ne .keepAliveMaxRequests nil) (toString .keepAliveMaxRequests) }} - - "--entryPoints.{{ $entrypoint }}.transport.keepAliveMaxRequests={{ .keepAliveMaxRequests }}" - {{- end }} - {{- if and (ne .keepAliveMaxTime nil) (toString .keepAliveMaxTime) }} - - "--entryPoints.{{ $entrypoint }}.transport.keepAliveMaxTime={{ .keepAliveMaxTime }}" - {{- end }} - {{- end }} - {{- end }} - {{- end }} - {{- with .Values.logs }} - {{- if and .general.format (not (has .general.format (list "common" "json"))) }} - {{- fail "ERROR: .Values.logs.general.format must be either common or json" }} - {{- end }} - {{- with .general.format }} - - "--log.format={{ . }}" - {{- end }} - {{- with .general.filePath }} - - "--log.filePath={{ . }}" - {{- end }} - {{- if and (or (eq .general.format "common") (not .general.format)) (eq .general.noColor true) }} - - "--log.noColor={{ .general.noColor }}" - {{- end }} - {{- with .general.level }} - - "--log.level={{ . | upper }}" - {{- end }} - {{- if .access.enabled }} - - "--accesslog=true" - {{- with .access.format }} - - "--accesslog.format={{ . }}" - {{- end }} - {{- with .access.filePath }} - - "--accesslog.filepath={{ . }}" - {{- end }} - {{- if .access.addInternals }} - - "--accesslog.addinternals" - {{- end }} - {{- with .access.bufferingSize }} - - "--accesslog.bufferingsize={{ . }}" - {{- end }} - {{- with .access.filters }} - {{- with .statuscodes }} - - "--accesslog.filters.statuscodes={{ . }}" - {{- end }} - {{- if .retryattempts }} - - "--accesslog.filters.retryattempts" - {{- end }} - {{- with .minduration }} - - "--accesslog.filters.minduration={{ . }}" - {{- end }} - {{- end }} - - "--accesslog.fields.defaultmode={{ .access.fields.general.defaultmode }}" - {{- range $fieldname, $fieldaction := .access.fields.general.names }} - - "--accesslog.fields.names.{{ $fieldname }}={{ $fieldaction }}" - {{- end }} - - "--accesslog.fields.headers.defaultmode={{ .access.fields.headers.defaultmode }}" - {{- range $fieldname, $fieldaction := .access.fields.headers.names }} - - "--accesslog.fields.headers.names.{{ $fieldname }}={{ $fieldaction }}" - {{- end }} - {{- end }} - {{- end }} - {{- include "traefik.yaml2CommandLineArgs" (dict "path" "certificatesresolvers" "content" $.Values.certificatesResolvers) | nindent 10 }} - {{- with .Values.additionalArguments }} - {{- range . }} - - {{ . | quote }} - {{- end }} - {{- end }} - {{- with .Values.hub }} - {{- if .token }} - - "--hub.token=$(HUB_TOKEN)" - {{- if and (not .apimanagement.enabled) ($.Values.hub.apimanagement.admission.listenAddr) }} - {{- fail "ERROR: Cannot configure admission without enabling hub.apimanagement" }} - {{- end }} - {{- with .apimanagement }} - {{- if .enabled }} - {{- $listenAddr := default ":9943" .admission.listenAddr }} - - "--hub.apimanagement" - - "--hub.apimanagement.admission.listenAddr={{ $listenAddr }}" - {{- with .admission.secretName }} - - "--hub.apimanagement.admission.secretName={{ . }}" - {{- end }} - {{- end }} - {{- end }} - {{- with .platformUrl }} - - "--hub.platformUrl={{ . }}" - {{- end -}} - {{- range $field, $value := .redis }} - {{- if has $field (list "cluster" "database" "endpoints" "username" "password" "timeout") -}} - {{- with $value }} - - "--hub.redis.{{ $field }}={{ $value }}" - {{- end }} - {{- end }} - {{- end }} - {{- range $field, $value := .redis.sentinel }} - {{- if has $field (list "masterset" "password" "username") -}} - {{- with $value }} - - "--hub.redis.sentinel.{{ $field }}={{ $value }}" - {{- end }} - {{- end }} - {{- end }} - {{- range $field, $value := .redis.tls }} - {{- if has $field (list "ca" "cert" "insecureSkipVerify" "key") -}} - {{- with $value }} - - "--hub.redis.tls.{{ $field }}={{ $value }}" - {{- end }} - {{- end }} - {{- end }} - {{- with .sendlogs }} - - "--hub.sendlogs={{ . }}" - {{- end }} - {{- end }} - {{- end }} - env: - - name: POD_NAME - valueFrom: - fieldRef: - fieldPath: metadata.name - - name: POD_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - {{- if ($.Values.resources.limits).cpu }} - - name: GOMAXPROCS - valueFrom: - resourceFieldRef: - resource: limits.cpu - divisor: '1' - {{- end }} - {{- if ($.Values.resources.limits).memory }} - - name: GOMEMLIMIT - valueFrom: - resourceFieldRef: - resource: limits.memory - divisor: '1' - {{- end }} - {{- with .Values.hub.token }} - - name: HUB_TOKEN - valueFrom: - secretKeyRef: - name: {{ . }} - key: token - {{- end }} - {{- with .Values.env }} - {{- toYaml . | nindent 10 }} - {{- end }} - {{- with .Values.envFrom }} - envFrom: - {{- toYaml . | nindent 10 }} - {{- end }} - {{- if .Values.deployment.additionalContainers }} - {{- toYaml .Values.deployment.additionalContainers | nindent 6 }} - {{- end }} - volumes: - - name: {{ .Values.persistence.name }} - {{- if .Values.persistence.enabled }} - persistentVolumeClaim: - claimName: {{ default (include "traefik.fullname" .) .Values.persistence.existingClaim }} - {{- else }} - emptyDir: {} - {{- end }} - - name: tmp - emptyDir: {} - {{- $root := . }} - {{- range .Values.volumes }} - - name: {{ tpl (.name) $root | replace "." "-" }} - {{- if eq .type "secret" }} - secret: - secretName: {{ tpl (.name) $root }} - {{- else if eq .type "configMap" }} - configMap: - name: {{ tpl (.name) $root }} - {{- end }} - {{- end }} - {{- if .Values.deployment.additionalVolumes }} - {{- toYaml .Values.deployment.additionalVolumes | nindent 8 }} - {{- end }} - {{- if gt (len .Values.experimental.plugins) 0 }} - - name: plugins - emptyDir: {} - {{- end }} - {{- if .Values.providers.file.enabled }} - - name: traefik-extra-config - configMap: - name: {{ template "traefik.fullname" . }}-file-provider - {{- end }} - {{- if .Values.affinity }} - affinity: - {{- tpl (toYaml .Values.affinity) . | nindent 8 }} - {{- end }} - {{- with .Values.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- if .Values.priorityClassName }} - priorityClassName: {{ .Values.priorityClassName }} - {{- end }} - {{- with .Values.podSecurityContext }} - securityContext: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- if .Values.topologySpreadConstraints }} - {{- if (semverCompare "<1.19.0-0" .Capabilities.KubeVersion.Version) }} - {{- fail "ERROR: topologySpreadConstraints are supported only on kubernetes >= v1.19" -}} - {{- end }} - topologySpreadConstraints: - {{- tpl (toYaml .Values.topologySpreadConstraints) . | nindent 8 }} - {{- end }} -{{ end -}} diff --git a/charts/traefik/traefik/templates/_service-metrics.tpl b/charts/traefik/traefik/templates/_service-metrics.tpl deleted file mode 100644 index d16a362..0000000 --- a/charts/traefik/traefik/templates/_service-metrics.tpl +++ /dev/null @@ -1,25 +0,0 @@ -{{- define "traefik.metrics-service-metadata" }} - labels: - {{- include "traefik.metricsservicelabels" . | nindent 4 -}} - {{- with .Values.metrics.prometheus.service.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end }} - -{{/* Labels used for metrics-relevant selector*/}} -{{/* This is an immutable field: this should not change between upgrade */}} -{{- define "traefik.metricslabelselector" -}} -{{- include "traefik.labelselector" . }} -app.kubernetes.io/component: metrics -{{- end }} - -{{/* Shared labels used in metadata of metrics-service and servicemonitor */}} -{{- define "traefik.metricsservicelabels" -}} -{{ include "traefik.metricslabelselector" . }} -helm.sh/chart: {{ template "traefik.chart" . }} -app.kubernetes.io/managed-by: {{ .Release.Service }} -{{- with .Values.commonLabels }} -{{ toYaml . }} -{{- end }} -{{- end }} - diff --git a/charts/traefik/traefik/templates/_service.tpl b/charts/traefik/traefik/templates/_service.tpl deleted file mode 100644 index 27d5bc4..0000000 --- a/charts/traefik/traefik/templates/_service.tpl +++ /dev/null @@ -1,84 +0,0 @@ -{{- define "traefik.service-name" -}} -{{- $fullname := printf "%s-%s" (include "traefik.fullname" .root) .name -}} -{{- if eq .name "default" -}} -{{- $fullname = include "traefik.fullname" .root -}} -{{- end -}} - -{{- if ge (len $fullname) 60 -}} # 64 - 4 (udp-postfix) = 60 - {{- fail "ERROR: Cannot create a service whose full name contains more than 60 characters" -}} -{{- end -}} - -{{- $fullname -}} -{{- end -}} - -{{- define "traefik.service-metadata" }} - labels: - {{- include "traefik.labels" .root | nindent 4 -}} - {{- with .service.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end }} - -{{- define "traefik.service-spec" -}} - {{- $type := default "LoadBalancer" .service.type }} - type: {{ $type }} - {{- with .service.loadBalancerClass }} - loadBalancerClass: {{ . }} - {{- end}} - {{- with .service.spec }} - {{- toYaml . | nindent 2 }} - {{- end }} - selector: - {{- include "traefik.labelselector" .root | nindent 4 }} - {{- if eq $type "LoadBalancer" }} - {{- with .service.loadBalancerSourceRanges }} - loadBalancerSourceRanges: - {{- toYaml . | nindent 2 }} - {{- end -}} - {{- end -}} - {{- with .service.externalIPs }} - externalIPs: - {{- toYaml . | nindent 2 }} - {{- end -}} - {{- with .service.ipFamilyPolicy }} - ipFamilyPolicy: {{ . }} - {{- end }} - {{- with .service.ipFamilies }} - ipFamilies: - {{- toYaml . | nindent 2 }} - {{- end -}} -{{- end }} - -{{- define "traefik.service-ports" }} - {{- range $name, $config := .ports }} - {{- if (index (default dict $config.expose) $.serviceName) }} - {{- $port := default $config.port $config.exposedPort }} - {{- if empty $port }} - {{- fail (print "ERROR: Cannot create " (trim $name) " port on Service without .port or .exposedPort") }} - {{- end }} - - port: {{ $port }} - name: {{ $name | quote }} - targetPort: {{ default $name $config.targetPort }} - protocol: {{ default "TCP" $config.protocol }} - {{- if $config.nodePort }} - nodePort: {{ $config.nodePort }} - {{- end }} - {{- if $config.appProtocol }} - appProtocol: {{ $config.appProtocol }} - {{- end }} - {{- if and ($config.http3).enabled ($config.single) }} - {{- $http3Port := default $config.exposedPort $config.http3.advertisedPort }} - - port: {{ $http3Port }} - name: "{{ $name }}-http3" - targetPort: "{{ $name }}-http3" - protocol: UDP - {{- if $config.nodePort }} - nodePort: {{ $config.nodePort }} - {{- end }} - {{- if $config.appProtocol }} - appProtocol: {{ $config.appProtocol }} - {{- end }} - {{- end }} - {{- end }} - {{- end }} -{{- end }} diff --git a/charts/traefik/traefik/templates/daemonset.yaml b/charts/traefik/traefik/templates/daemonset.yaml deleted file mode 100644 index b370c22..0000000 --- a/charts/traefik/traefik/templates/daemonset.yaml +++ /dev/null @@ -1,58 +0,0 @@ -{{- if and .Values.deployment.enabled (eq .Values.deployment.kind "DaemonSet") -}} - {{- with .Values.additionalArguments -}} - {{- range . -}} - {{- if contains ".acme." . -}} - {{- fail (printf "ACME functionality is not supported when running Traefik as a DaemonSet") -}} - {{- end -}} - {{- end -}} - {{- end -}} - {{- if eq (default .Chart.AppVersion .Values.image.tag) "latest" }} - {{- fail "\n\n ERROR: latest tag should not be used" }} - {{- end }} - {{- with .Values.updateStrategy }} - {{- if and (eq (.type) "RollingUpdate") (.rollingUpdate) }} - {{- if not (contains "%" (toString .rollingUpdate.maxUnavailable)) }} - {{- if and ($.Values.hostNetwork) (lt (float64 .rollingUpdate.maxUnavailable) 1.0) }} - {{- fail "maxUnavailable should be greater than 1 when using hostNetwork." }} - {{- end }} - {{- end }} - {{- end }} - {{- end }} - ---- -apiVersion: apps/v1 -kind: DaemonSet -metadata: - name: {{ template "traefik.fullname" . }} - namespace: {{ template "traefik.namespace" . }} - labels: - {{- include "traefik.labels" . | nindent 4 }} - {{- with .Values.deployment.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} - annotations: - {{- if and .Values.providers.file.enabled (not .Values.providers.file.watch) }} - checksum/traefik-dynamic-conf: {{ include (print $.Template.BasePath "/provider-file-cm.yaml") . | sha256sum }} - {{- end }} - {{- with .Values.deployment.annotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - selector: - matchLabels: - {{- include "traefik.labelselector" . | nindent 6 }} - {{- with .Values.updateStrategy }} - updateStrategy: - type: {{ .type }} - {{- if (eq .type "RollingUpdate") }} - rollingUpdate: - maxUnavailable: {{ .rollingUpdate.maxUnavailable }} - maxSurge: {{ .rollingUpdate.maxSurge }} - {{- end }} - {{- end }} - minReadySeconds: {{ .Values.deployment.minReadySeconds }} - {{- if .Values.deployment.revisionHistoryLimit }} - revisionHistoryLimit: {{ .Values.deployment.revisionHistoryLimit }} - {{- end }} - template: {{ template "traefik.podTemplate" . }} -{{- end -}} diff --git a/charts/traefik/traefik/templates/deployment.yaml b/charts/traefik/traefik/templates/deployment.yaml deleted file mode 100644 index 4b3a1ae..0000000 --- a/charts/traefik/traefik/templates/deployment.yaml +++ /dev/null @@ -1,58 +0,0 @@ -{{/* check helm version */}} -{{- if (semverCompare "= 3.9.0 is required" -}} -{{- end -}} - -{{- if and .Values.deployment.enabled (eq .Values.deployment.kind "Deployment") -}} - {{- if gt (int .Values.deployment.replicas) 1 -}} - {{- with .Values.additionalArguments -}} - {{- range . -}} - {{- if contains ".acme." . -}} - {{- fail (printf "You can not enable acme if you set more than one traefik replica") -}} - {{- end -}} - {{- end -}} - {{- end -}} - {{- end -}} - {{- if eq (default .Chart.AppVersion .Values.image.tag) "latest" }} - {{- fail "\n\n ERROR: latest tag should not be used" }} - {{- end }} ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - name: {{ template "traefik.fullname" . }} - namespace: {{ template "traefik.namespace" . }} - labels: - {{- include "traefik.labels" . | nindent 4 }} - {{- with .Values.deployment.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} - annotations: - {{- if and .Values.providers.file.enabled (not .Values.providers.file.watch) }} - checksum/traefik-dynamic-conf: {{ include (print $.Template.BasePath "/provider-file-cm.yaml") . | sha256sum }} - {{- end }} - {{- with .Values.deployment.annotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - {{- if not .Values.autoscaling.enabled }} - replicas: {{ default 1 .Values.deployment.replicas }} - {{- end }} - {{- if .Values.deployment.revisionHistoryLimit }} - revisionHistoryLimit: {{ .Values.deployment.revisionHistoryLimit }} - {{- end }} - selector: - matchLabels: - {{- include "traefik.labelselector" . | nindent 6 }} - {{- with .Values.updateStrategy }} - strategy: - type: {{ .type }} - {{- if (eq .type "RollingUpdate") }} - rollingUpdate: - maxUnavailable: {{ .rollingUpdate.maxUnavailable }} - maxSurge: {{ .rollingUpdate.maxSurge }} - {{- end }} - {{- end }} - minReadySeconds: {{ .Values.deployment.minReadySeconds }} - template: {{ template "traefik.podTemplate" . }} -{{- end -}} diff --git a/charts/traefik/traefik/templates/extra-objects.yaml b/charts/traefik/traefik/templates/extra-objects.yaml deleted file mode 100644 index fb38e97..0000000 --- a/charts/traefik/traefik/templates/extra-objects.yaml +++ /dev/null @@ -1,4 +0,0 @@ -{{- range .Values.extraObjects }} ---- -{{ include "traefik.render" (dict "value" . "context" $) }} -{{- end }} diff --git a/charts/traefik/traefik/templates/gateway.yaml b/charts/traefik/traefik/templates/gateway.yaml deleted file mode 100644 index 42bc33d..0000000 --- a/charts/traefik/traefik/templates/gateway.yaml +++ /dev/null @@ -1,62 +0,0 @@ -{{- if and (.Values.gateway).enabled (.Values.providers.kubernetesGateway).enabled }} - {{- if not .Values.gateway.listeners }} - {{- fail "ERROR: gateway must have at least one listener or should be disabled" }} - {{- end }} ---- -apiVersion: gateway.networking.k8s.io/v1 -kind: Gateway -metadata: - name: {{ default "traefik-gateway" .Values.gateway.name }} - namespace: {{ template "traefik.namespace" . }} - labels: - {{- include "traefik.labels" . | nindent 4 }} - {{- with .Values.gateway.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - gatewayClassName: {{ default "traefik" .Values.gatewayClass.name }} - {{- with .Values.gateway.infrastructure }} - infrastructure: - {{ toYaml . | nindent 4 }} - {{- end }} - listeners: - {{- range $name, $config := .Values.gateway.listeners }} - - name: {{ $name }} - {{ if not .port }} - {{- fail "ERROR: port needs to be specified" }} - {{- end -}} - {{ $found := false }} - {{- range $portName, $portConfig := $.Values.ports -}} - {{- if eq $portConfig.port $config.port -}} - {{ $found = true }} - {{- end -}} - {{- end -}} - {{ if not $found }} - {{- fail (printf "ERROR: port %0.f is not declared in ports" .port ) }} - {{- end -}} - port: {{ .port }} - protocol: {{ .protocol }} - {{- with .hostname }} - hostname: {{ . | toYaml }} - {{- end }} - {{- with .namespacePolicy }} - allowedRoutes: - namespaces: - from: {{ . }} - {{- end }} - {{ if and (eq .protocol "HTTPS") (not .certificateRefs) }} - {{- fail "ERROR: certificateRefs needs to be specified using HTTPS" }} - {{- end }} - {{ if or .certificateRefs .mode }} - tls: - {{ with .mode }} - mode: {{ . }} - {{- end }} - {{ with .certificateRefs }} - certificateRefs: - {{- toYaml . | nindent 10 }} - {{- end }} - {{- end }} - {{- end }} -{{- end }} diff --git a/charts/traefik/traefik/templates/gatewayclass.yaml b/charts/traefik/traefik/templates/gatewayclass.yaml deleted file mode 100644 index 7f98c1e..0000000 --- a/charts/traefik/traefik/templates/gatewayclass.yaml +++ /dev/null @@ -1,14 +0,0 @@ -{{- if and (.Values.gatewayClass).enabled (.Values.providers.kubernetesGateway).enabled }} ---- -apiVersion: gateway.networking.k8s.io/v1 -kind: GatewayClass -metadata: - name: {{ default "traefik" .Values.gatewayClass.name }} - labels: - {{- include "traefik.labels" . | nindent 4 }} - {{- with .Values.gatewayClass.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - controllerName: traefik.io/gateway-controller -{{- end }} diff --git a/charts/traefik/traefik/templates/hpa.yaml b/charts/traefik/traefik/templates/hpa.yaml deleted file mode 100644 index cfa1e5a..0000000 --- a/charts/traefik/traefik/templates/hpa.yaml +++ /dev/null @@ -1,35 +0,0 @@ -{{- if .Values.autoscaling.enabled }} - -{{- if not .Values.autoscaling.maxReplicas }} - {{- fail "ERROR: maxReplicas is required on HPA" }} -{{- end }} - -{{- if semverCompare ">=1.23.0-0" .Capabilities.KubeVersion.Version }} -apiVersion: autoscaling/v2 -{{- else }} -apiVersion: autoscaling/v2beta2 -{{- end }} -kind: HorizontalPodAutoscaler -metadata: - name: {{ template "traefik.fullname" . }} - namespace: {{ template "traefik.namespace" . }} - labels: - {{- include "traefik.labels" . | nindent 4 }} -spec: - scaleTargetRef: - apiVersion: apps/v1 - kind: Deployment - name: {{ template "traefik.fullname" . }} -{{- if .Values.autoscaling.minReplicas }} - minReplicas: {{ .Values.autoscaling.minReplicas }} -{{- end }} - maxReplicas: {{ .Values.autoscaling.maxReplicas }} -{{- if .Values.autoscaling.metrics }} - metrics: -{{ toYaml .Values.autoscaling.metrics | indent 4 }} -{{- end }} -{{- if .Values.autoscaling.behavior }} - behavior: -{{ toYaml .Values.autoscaling.behavior | indent 4 }} -{{- end }} -{{- end }} diff --git a/charts/traefik/traefik/templates/hub-admission-controller.yaml b/charts/traefik/traefik/templates/hub-admission-controller.yaml deleted file mode 100644 index 37b2314..0000000 --- a/charts/traefik/traefik/templates/hub-admission-controller.yaml +++ /dev/null @@ -1,198 +0,0 @@ -{{- if .Values.hub.token -}} -{{- if .Values.hub.apimanagement.enabled }} -{{- $cert := include "traefik-hub.webhook_cert" . | fromYaml }} ---- -apiVersion: v1 -kind: Secret -type: kubernetes.io/tls -metadata: - name: hub-agent-cert - namespace: {{ template "traefik.namespace" . }} - labels: - {{- include "traefik.labels" . | nindent 4 }} -data: - tls.crt: {{ $cert.Cert }} - tls.key: {{ $cert.Key }} - ---- -apiVersion: admissionregistration.k8s.io/v1 -kind: MutatingWebhookConfiguration -metadata: - name: hub-acp - labels: - {{- include "traefik.labels" . | nindent 4 }} -webhooks: - - name: admission.traefik.svc - clientConfig: - service: - name: admission - namespace: {{ template "traefik.namespace" . }} - path: /acp - caBundle: {{ $cert.Cert }} - sideEffects: None - admissionReviewVersions: - - v1 - rules: - - operations: - - CREATE - - UPDATE - - DELETE - apiGroups: - - hub.traefik.io - apiVersions: - - v1alpha1 - resources: - - accesscontrolpolicies - ---- -apiVersion: admissionregistration.k8s.io/v1 -kind: MutatingWebhookConfiguration -metadata: - name: hub-api - labels: - {{- include "traefik.labels" . | nindent 4 }} -webhooks: - - name: hub-agent.traefik.portal - clientConfig: - service: - name: admission - namespace: {{ template "traefik.namespace" . }} - path: /api-portal - caBundle: {{ $cert.Cert }} - sideEffects: None - admissionReviewVersions: - - v1 - rules: - - operations: - - CREATE - - UPDATE - - DELETE - apiGroups: - - hub.traefik.io - apiVersions: - - v1alpha1 - resources: - - apiportals - - name: hub-agent.traefik.api - clientConfig: - service: - name: admission - namespace: {{ template "traefik.namespace" . }} - path: /api - caBundle: {{ $cert.Cert }} - sideEffects: None - admissionReviewVersions: - - v1 - rules: - - operations: - - CREATE - - UPDATE - - DELETE - apiGroups: - - hub.traefik.io - apiVersions: - - v1alpha1 - resources: - - apis - - name: hub-agent.traefik.access - clientConfig: - service: - name: admission - namespace: {{ template "traefik.namespace" . }} - path: /api-access - caBundle: {{ $cert.Cert }} - sideEffects: None - admissionReviewVersions: - - v1 - rules: - - operations: - - CREATE - - UPDATE - - DELETE - apiGroups: - - hub.traefik.io - apiVersions: - - v1alpha1 - resources: - - apiaccesses - - name: hub-agent.traefik.plan - clientConfig: - service: - name: admission - namespace: {{ template "traefik.namespace" . }} - path: /api-plan - caBundle: {{ $cert.Cert }} - sideEffects: None - admissionReviewVersions: - - v1 - rules: - - operations: - - CREATE - - UPDATE - - DELETE - apiGroups: - - hub.traefik.io - apiVersions: - - v1alpha1 - resources: - - apiplans - - name: hub-agent.traefik.bundle - clientConfig: - service: - name: admission - namespace: {{ template "traefik.namespace" . }} - path: /api-bundle - caBundle: {{ $cert.Cert }} - sideEffects: None - admissionReviewVersions: - - v1 - rules: - - operations: - - CREATE - - UPDATE - - DELETE - apiGroups: - - hub.traefik.io - apiVersions: - - v1alpha1 - resources: - - apibundles - - name: hub-agent.traefik.version - clientConfig: - service: - name: admission - namespace: {{ template "traefik.namespace" . }} - path: /api-version - caBundle: {{ $cert.Cert }} - sideEffects: None - admissionReviewVersions: - - v1 - rules: - - operations: - - CREATE - - UPDATE - - DELETE - apiGroups: - - hub.traefik.io - apiVersions: - - v1alpha1 - resources: - - apiversions - ---- -apiVersion: v1 -kind: Service -metadata: - name: admission - namespace: {{ template "traefik.namespace" . }} - labels: - {{- include "traefik.labels" . | nindent 4 }} -spec: - ports: - - name: https - port: 443 - targetPort: admission - selector: - {{- include "traefik.labelselector" . | nindent 4 }} -{{- end -}} -{{- end -}} diff --git a/charts/traefik/traefik/templates/hub-apiportal.yaml b/charts/traefik/traefik/templates/hub-apiportal.yaml deleted file mode 100644 index 246b127..0000000 --- a/charts/traefik/traefik/templates/hub-apiportal.yaml +++ /dev/null @@ -1,19 +0,0 @@ -{{- if .Values.hub.apimanagement.enabled }} ---- -apiVersion: v1 -kind: Service -metadata: - name: apiportal - namespace: {{ template "traefik.namespace" . }} - labels: - {{- include "traefik.labels" . | nindent 4 }} -spec: - ports: - - name: apiportal - port: 9903 - protocol: TCP - targetPort: apiportal - selector: - {{- include "traefik.labelselector" . | nindent 4 }} -{{- end -}} - diff --git a/charts/traefik/traefik/templates/ingressclass.yaml b/charts/traefik/traefik/templates/ingressclass.yaml deleted file mode 100644 index 6a8ff81..0000000 --- a/charts/traefik/traefik/templates/ingressclass.yaml +++ /dev/null @@ -1,12 +0,0 @@ -{{- if .Values.ingressClass.enabled -}} -apiVersion: networking.k8s.io/v1 -kind: IngressClass -metadata: - annotations: - ingressclass.kubernetes.io/is-default-class: {{ .Values.ingressClass.isDefaultClass | quote }} - labels: - {{- include "traefik.labels" . | nindent 4 }} - name: {{ .Values.ingressClass.name | default (include "traefik.fullname" .) }} -spec: - controller: traefik.io/ingress-controller -{{- end -}} diff --git a/charts/traefik/traefik/templates/ingressroute.yaml b/charts/traefik/traefik/templates/ingressroute.yaml deleted file mode 100644 index 2f35abb..0000000 --- a/charts/traefik/traefik/templates/ingressroute.yaml +++ /dev/null @@ -1,43 +0,0 @@ -{{ range $name, $config := .Values.ingressRoute }} -{{ if $config.enabled }} ---- -apiVersion: traefik.io/v1alpha1 -kind: IngressRoute -metadata: - name: {{ $.Release.Name }}-{{ $name }} - namespace: {{ template "traefik.namespace" $ }} - annotations: - {{- if and $.Values.ingressClass.enabled $.Values.providers.kubernetesCRD.enabled $.Values.providers.kubernetesCRD.ingressClass }} - kubernetes.io/ingress.class: {{ $.Values.providers.kubernetesCRD.ingressClass }} - {{- end }} - {{- with $config.annotations }} - {{- toYaml . | nindent 4 }} - {{- end }} - labels: - {{- include "traefik.labels" $ | nindent 4 }} - {{- with $config.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - entryPoints: - {{- range $config.entryPoints }} - - {{ . }} - {{- end }} - routes: - - match: {{ $config.matchRule }} - kind: Rule - {{- with $config.services }} - services: - {{- toYaml . | nindent 6 }} - {{- end -}} - {{- with $config.middlewares }} - middlewares: - {{- toYaml . | nindent 6 }} - {{- end -}} - - {{- with $config.tls }} - tls: - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end -}} -{{ end }} diff --git a/charts/traefik/traefik/templates/poddisruptionbudget.yaml b/charts/traefik/traefik/templates/poddisruptionbudget.yaml deleted file mode 100644 index f171639..0000000 --- a/charts/traefik/traefik/templates/poddisruptionbudget.yaml +++ /dev/null @@ -1,23 +0,0 @@ -{{- if .Values.podDisruptionBudget.enabled -}} -{{- if .Capabilities.APIVersions.Has "policy/v1/PodDisruptionBudget" }} -apiVersion: policy/v1 -{{- else }} -apiVersion: policy/v1beta1 -{{- end }} -kind: PodDisruptionBudget -metadata: - name: {{ template "traefik.fullname" . }} - namespace: {{ template "traefik.namespace" . }} - labels: - {{- include "traefik.labels" . | nindent 4 }} -spec: - selector: - matchLabels: - {{- include "traefik.labelselector" . | nindent 6 }} - {{- if .Values.podDisruptionBudget.minAvailable }} - minAvailable: {{ .Values.podDisruptionBudget.minAvailable }} - {{- end }} - {{- if .Values.podDisruptionBudget.maxUnavailable }} - maxUnavailable: {{ .Values.podDisruptionBudget.maxUnavailable }} - {{- end }} -{{- end -}} diff --git a/charts/traefik/traefik/templates/prometheusrules.yaml b/charts/traefik/traefik/templates/prometheusrules.yaml deleted file mode 100644 index 3231aba..0000000 --- a/charts/traefik/traefik/templates/prometheusrules.yaml +++ /dev/null @@ -1,28 +0,0 @@ -{{- if .Values.metrics.prometheus }} -{{- if (.Values.metrics.prometheus.prometheusRule).enabled }} - {{- if (not (.Capabilities.APIVersions.Has "monitoring.coreos.com/v1")) }} - {{- if (not (.Values.metrics.prometheus.disableAPICheck)) }} - {{- fail "ERROR: You have to deploy monitoring.coreos.com/v1 first" }} - {{- end }} - {{- end }} -apiVersion: monitoring.coreos.com/v1 -kind: PrometheusRule -metadata: - name: {{ template "traefik.fullname" . }} - namespace: {{ .Values.metrics.prometheus.prometheusRule.namespace | default (include "traefik.namespace" .) }} - labels: - {{- include "traefik.labels" . | nindent 4 }} - {{- with .Values.metrics.prometheus.prometheusRule.additionalLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - {{- if .Values.metrics.prometheus.prometheusRule.rules }} - groups: - - name: {{ template "traefik.name" $ }} - rules: - {{- with .Values.metrics.prometheus.prometheusRule.rules }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- end }} -{{- end }} -{{- end }} diff --git a/charts/traefik/traefik/templates/provider-file-cm.yaml b/charts/traefik/traefik/templates/provider-file-cm.yaml deleted file mode 100644 index 139a5a6..0000000 --- a/charts/traefik/traefik/templates/provider-file-cm.yaml +++ /dev/null @@ -1,12 +0,0 @@ -{{- if .Values.providers.file.enabled -}} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ template "traefik.fullname" . }}-file-provider - namespace: {{ template "traefik.namespace" . }} - labels: - {{- include "traefik.labels" . | nindent 4 }} -data: - config.yml: - {{ toYaml .Values.providers.file.content | nindent 4 }} -{{- end -}} diff --git a/charts/traefik/traefik/templates/pvc.yaml b/charts/traefik/traefik/templates/pvc.yaml deleted file mode 100644 index 7ab96f9..0000000 --- a/charts/traefik/traefik/templates/pvc.yaml +++ /dev/null @@ -1,26 +0,0 @@ -{{- if and .Values.persistence.enabled (not .Values.persistence.existingClaim) -}} -apiVersion: v1 -kind: PersistentVolumeClaim -metadata: - name: {{ template "traefik.fullname" . }} - namespace: {{ template "traefik.namespace" . }} - annotations: - {{- with .Values.persistence.annotations }} - {{ toYaml . | nindent 4 }} - {{- end }} - helm.sh/resource-policy: keep - labels: - {{- include "traefik.labels" . | nindent 4 }} -spec: - accessModes: - - {{ .Values.persistence.accessMode | quote }} - resources: - requests: - storage: {{ .Values.persistence.size | quote }} - {{- if .Values.persistence.storageClass }} - storageClassName: {{ .Values.persistence.storageClass | quote }} - {{- end }} - {{- if .Values.persistence.volumeName }} - volumeName: {{ .Values.persistence.volumeName | quote }} - {{- end }} -{{- end -}} diff --git a/charts/traefik/traefik/templates/rbac/clusterrole.yaml b/charts/traefik/traefik/templates/rbac/clusterrole.yaml deleted file mode 100644 index 9e52d53..0000000 --- a/charts/traefik/traefik/templates/rbac/clusterrole.yaml +++ /dev/null @@ -1,271 +0,0 @@ -{{- $version := include "imageVersion" $ }} -{{- if and .Values.rbac.enabled (not .Values.rbac.namespaced) }} ---- -kind: ClusterRole -apiVersion: rbac.authorization.k8s.io/v1 -metadata: - name: {{ template "traefik.clusterRoleName" . }} - labels: - {{- include "traefik.labels" . | nindent 4 }} - {{- range .Values.rbac.aggregateTo }} - rbac.authorization.k8s.io/aggregate-to-{{ . }}: "true" - {{- end }} -rules: - {{- if semverCompare ">=v3.1.0-0" $version }} - - apiGroups: - - "" - resources: - - nodes - verbs: - - get - - list - - watch - {{- end }} - {{- if (semverCompare "=v3.2.0-0" $version }} - - configmaps - {{- end }} - verbs: - - get - - list - - watch - {{- if (semverCompare ">=v3.1.0-0" $version) }} - - apiGroups: - - discovery.k8s.io - resources: - - endpointslices - verbs: - - list - - watch - {{- end }} - - apiGroups: - - gateway.networking.k8s.io - resources: - {{- if semverCompare ">=v3.2.0-0" $version }} - - backendtlspolicies - {{- end }} - - gatewayclasses - - gateways - {{- if semverCompare ">=v3.2.0-0" $version }} - - grpcroutes - {{- end }} - - httproutes - - referencegrants - - tcproutes - - tlsroutes - verbs: - - get - - list - - watch - - apiGroups: - - gateway.networking.k8s.io - resources: - {{- if semverCompare ">=v3.2.0-0" $version }} - - backendtlspolicies/status - {{- end }} - - gatewayclasses/status - - gateways/status - {{- if semverCompare ">=v3.2.0-0" $version }} - - grpcroutes/status - {{- end }} - - httproutes/status - - tcproutes/status - - tlsroutes/status - verbs: - - update - {{- end }} - {{- if .Values.hub.token }} - - apiGroups: - - coordination.k8s.io - resources: - - leases - verbs: - - get - - list - - watch - - create - - update - - patch - - delete - {{- end }} - {{- if .Values.hub.token }} - {{- if or (semverCompare ">=v3.1.0-0" $version) .Values.hub.apimanagement.enabled }} - - apiGroups: - - "" - resources: - - endpoints - verbs: - - list - - watch - {{- end }} - - apiGroups: - - "" - resources: - - namespaces - {{- if .Values.hub.apimanagement.enabled }} - - pods - {{- end }} - verbs: - - get - - list - {{- if .Values.hub.apimanagement.enabled }} - - watch - {{- end }} - {{- if .Values.hub.apimanagement.enabled }} - - apiGroups: - - hub.traefik.io - resources: - - accesscontrolpolicies - - apiaccesses - - apiportals - - apiratelimits - - apis - - apiversions - - apibundles - - apiplans - verbs: - - list - - watch - - create - - update - - patch - - delete - - get - - apiGroups: - - "" - resources: - - events - verbs: - - create - - patch - - apiGroups: - - apps - resources: - - replicasets - verbs: - - get - - list - - watch - {{- if (semverCompare "=1.25.0-0" .Capabilities.KubeVersion.Version }} - {{- fail "ERROR: PodSecurityPolicy has been removed in Kubernetes v1.25+" }} -{{- end }} ---- -apiVersion: policy/v1beta1 -kind: PodSecurityPolicy -metadata: - annotations: - seccomp.security.alpha.kubernetes.io/allowedProfileNames: runtime/default - seccomp.security.alpha.kubernetes.io/defaultProfileName: runtime/default - name: {{ template "traefik.fullname" . }} - labels: - {{- include "traefik.labels" . | nindent 4 }} -spec: - privileged: false - allowPrivilegeEscalation: false - requiredDropCapabilities: - - ALL -{{- if not .Values.securityContext.runAsNonRoot }} - allowedCapabilities: - - NET_BIND_SERVICE -{{- end }} - hostNetwork: {{ .Values.hostNetwork }} - hostIPC: false - hostPID: false - fsGroup: -{{- if .Values.securityContext.runAsNonRoot }} - ranges: - - max: 65535 - min: 1 - rule: MustRunAs -{{- else }} - rule: RunAsAny -{{- end }} -{{- if .Values.hostNetwork }} - hostPorts: - - max: 65535 - min: 1 -{{- end }} - readOnlyRootFilesystem: true - runAsUser: -{{- if .Values.securityContext.runAsNonRoot }} - rule: MustRunAsNonRoot -{{- else }} - rule: RunAsAny -{{- end }} - seLinux: - rule: RunAsAny - supplementalGroups: -{{- if .Values.securityContext.runAsNonRoot }} - ranges: - - max: 65535 - min: 1 - rule: MustRunAs -{{- else }} - rule: RunAsAny -{{- end }} - volumes: - - configMap - - downwardAPI - - secret - - emptyDir - - projected -{{- if .Values.persistence.enabled }} - - persistentVolumeClaim -{{- end -}} -{{- end -}} diff --git a/charts/traefik/traefik/templates/rbac/role.yaml b/charts/traefik/traefik/templates/rbac/role.yaml deleted file mode 100644 index e81aaa8..0000000 --- a/charts/traefik/traefik/templates/rbac/role.yaml +++ /dev/null @@ -1,143 +0,0 @@ -{{- $version := include "imageVersion" $ }} -{{- $ingressNamespaces := concat (include "traefik.namespace" . | list) .Values.providers.kubernetesIngress.namespaces -}} -{{- $CRDNamespaces := concat (include "traefik.namespace" . | list) .Values.providers.kubernetesCRD.namespaces -}} -{{- $allNamespaces := sortAlpha (uniq (concat $ingressNamespaces $CRDNamespaces)) -}} - -{{- if and .Values.rbac.enabled .Values.rbac.namespaced -}} -{{- range $allNamespaces }} ---- -kind: Role -apiVersion: rbac.authorization.k8s.io/v1 -metadata: - name: {{ template "traefik.fullname" $ }} - namespace: {{ . }} - labels: - {{- include "traefik.labels" $ | nindent 4 }} -rules: - {{- if (semverCompare "://:. Default: http://localhost:4318/v1/metrics - endpoint: "" - # -- Additional headers sent with metrics by the reporter to the OpenTelemetry Collector. - headers: {} - ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. - tls: - # -- The path to the certificate authority, it defaults to the system bundle. - ca: "" - # -- The path to the public certificate. When using this option, setting the key option is required. - cert: "" - # -- The path to the private key. When using this option, setting the cert option is required. - key: "" - # -- When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. - insecureSkipVerify: # @schema type:[boolean, null] - grpc: - # -- Set to true in order to send metrics to the OpenTelemetry Collector using gRPC - enabled: false - # -- Format: ://:. Default: http://localhost:4318/v1/metrics - endpoint: "" - # -- Allows reporter to send metrics to the OpenTelemetry Collector without using a secured protocol. - insecure: false - ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. - tls: - # -- The path to the certificate authority, it defaults to the system bundle. - ca: "" - # -- The path to the public certificate. When using this option, setting the key option is required. - cert: "" - # -- The path to the private key. When using this option, setting the cert option is required. - key: "" - # -- When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. - insecureSkipVerify: false - -## Tracing -# -- https://doc.traefik.io/traefik/observability/tracing/overview/ -tracing: # @schema additionalProperties: false - # -- Enables tracing for internal resources. Default: false. - addInternals: false - otlp: - # -- See https://doc.traefik.io/traefik/v3.0/observability/tracing/opentelemetry/ - enabled: false - http: - # -- Set to true in order to send metrics to the OpenTelemetry Collector using HTTP. - enabled: false - # -- Format: ://:. Default: http://localhost:4318/v1/metrics - endpoint: "" - # -- Additional headers sent with metrics by the reporter to the OpenTelemetry Collector. - headers: {} - ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. - tls: - # -- The path to the certificate authority, it defaults to the system bundle. - ca: "" - # -- The path to the public certificate. When using this option, setting the key option is required. - cert: "" - # -- The path to the private key. When using this option, setting the cert option is required. - key: "" - # -- When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. - insecureSkipVerify: false - grpc: - # -- Set to true in order to send metrics to the OpenTelemetry Collector using gRPC - enabled: false - # -- Format: ://:. Default: http://localhost:4318/v1/metrics - endpoint: "" - # -- Allows reporter to send metrics to the OpenTelemetry Collector without using a secured protocol. - insecure: false - ## Defines the TLS configuration used by the reporter to send metrics to the OpenTelemetry Collector. - tls: - # -- The path to the certificate authority, it defaults to the system bundle. - ca: "" - # -- The path to the public certificate. When using this option, setting the key option is required. - cert: "" - # -- The path to the private key. When using this option, setting the cert option is required. - key: "" - # -- When set to true, the TLS connection accepts any certificate presented by the server regardless of the hostnames it covers. - insecureSkipVerify: false - -# -- Global command arguments to be passed to all traefik's pods -globalArguments: -- "--global.checknewversion" -- "--global.sendanonymoususage" - -# -- Additional arguments to be passed at Traefik's binary -# See [CLI Reference](https://docs.traefik.io/reference/static-configuration/cli/) -# Use curly braces to pass values: `helm install --set="additionalArguments={--providers.kubernetesingress.ingressclass=traefik-internal,--log.level=DEBUG}"` -additionalArguments: [] -# - "--providers.kubernetesingress.ingressclass=traefik-internal" -# - "--log.level=DEBUG" - -# -- Additional Environment variables to be passed to Traefik's binary -# @default -- See _values.yaml_ -env: [] - -# -- Environment variables to be passed to Traefik's binary from configMaps or secrets -envFrom: [] - -ports: - traefik: - port: 8080 - # -- Use hostPort if set. - hostPort: # @schema type:[integer, null]; minimum:0 - # -- Use hostIP if set. If not set, Kubernetes will default to 0.0.0.0, which - # means it's listening on all your interfaces and all your IPs. You may want - # to set this value if you need traefik to listen on specific interface - # only. - hostIP: # @schema type:[string, null] - - # Defines whether the port is exposed if service.type is LoadBalancer or - # NodePort. - # - # -- You SHOULD NOT expose the traefik port on production deployments. - # If you want to access it from outside your cluster, - # use `kubectl port-forward` or create a secure ingress - expose: - default: false - # -- The exposed port for this service - exposedPort: 8080 - # -- The port protocol (TCP/UDP) - protocol: TCP - web: - ## -- Enable this entrypoint as a default entrypoint. When a service doesn't explicitly set an entrypoint it will only use this entrypoint. - # asDefault: true - port: 8000 - # hostPort: 8000 - # containerPort: 8000 - expose: - default: true - exposedPort: 80 - ## -- Different target traefik port on the cluster, useful for IP type LB - targetPort: # @schema type:[string, integer, null]; minimum:0 - # The port protocol (TCP/UDP) - protocol: TCP - # -- See [upstream documentation](https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport) - nodePort: # @schema type:[integer, null]; minimum:0 - # Port Redirections - # Added in 2.2, you can make permanent redirects via entrypoints. - # https://docs.traefik.io/routing/entrypoints/#redirection - redirectTo: {} - forwardedHeaders: - # -- Trust forwarded headers information (X-Forwarded-*). - trustedIPs: [] - insecure: false - proxyProtocol: - # -- Enable the Proxy Protocol header parsing for the entry point - trustedIPs: [] - insecure: false - # -- Set transport settings for the entrypoint; see also - # https://doc.traefik.io/traefik/routing/entrypoints/#transport - transport: - respondingTimeouts: - readTimeout: # @schema type:[string, integer, null] - writeTimeout: # @schema type:[string, integer, null] - idleTimeout: # @schema type:[string, integer, null] - lifeCycle: - requestAcceptGraceTimeout: # @schema type:[string, integer, null] - graceTimeOut: # @schema type:[string, integer, null] - keepAliveMaxRequests: # @schema type:[integer, null]; minimum:0 - keepAliveMaxTime: # @schema type:[string, integer, null] - websecure: - ## -- Enable this entrypoint as a default entrypoint. When a service doesn't explicitly set an entrypoint it will only use this entrypoint. - # asDefault: true - port: 8443 - hostPort: # @schema type:[integer, null]; minimum:0 - containerPort: # @schema type:[integer, null]; minimum:0 - expose: - default: true - exposedPort: 443 - ## -- Different target traefik port on the cluster, useful for IP type LB - targetPort: # @schema type:[string, integer, null]; minimum:0 - ## -- The port protocol (TCP/UDP) - protocol: TCP - # -- See [upstream documentation](https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport) - nodePort: # @schema type:[integer, null]; minimum:0 - # -- See [upstream documentation](https://kubernetes.io/docs/concepts/services-networking/service/#application-protocol) - appProtocol: # @schema type:[string, null] - # -- See [upstream documentation](https://doc.traefik.io/traefik/routing/entrypoints/#allowacmebypass) - allowACMEByPass: false - http3: - ## -- Enable HTTP/3 on the entrypoint - ## Enabling it will also enable http3 experimental feature - ## https://doc.traefik.io/traefik/routing/entrypoints/#http3 - ## There are known limitations when trying to listen on same ports for - ## TCP & UDP (Http3). There is a workaround in this chart using dual Service. - ## https://github.com/kubernetes/kubernetes/issues/47249#issuecomment-587960741 - enabled: false - advertisedPort: # @schema type:[integer, null]; minimum:0 - forwardedHeaders: - # -- Trust forwarded headers information (X-Forwarded-*). - trustedIPs: [] - insecure: false - proxyProtocol: - # -- Enable the Proxy Protocol header parsing for the entry point - trustedIPs: [] - insecure: false - # -- See [upstream documentation](https://doc.traefik.io/traefik/routing/entrypoints/#transport) - transport: - respondingTimeouts: - readTimeout: # @schema type:[string, integer, null] - writeTimeout: # @schema type:[string, integer, null] - idleTimeout: # @schema type:[string, integer, null] - lifeCycle: - requestAcceptGraceTimeout: # @schema type:[string, integer, null] - graceTimeOut: # @schema type:[string, integer, null] - keepAliveMaxRequests: # @schema type:[integer, null]; minimum:0 - keepAliveMaxTime: # @schema type:[string, integer, null] - # -- See [upstream documentation](https://doc.traefik.io/traefik/routing/entrypoints/#tls) - tls: - enabled: true - options: "" - certResolver: "" - domains: [] - # -- One can apply Middlewares on an entrypoint - # https://doc.traefik.io/traefik/middlewares/overview/ - # https://doc.traefik.io/traefik/routing/entrypoints/#middlewares - # -- /!\ It introduces here a link between your static configuration and your dynamic configuration /!\ - # It follows the provider naming convention: https://doc.traefik.io/traefik/providers/overview/#provider-namespace - # - namespace-name1@kubernetescrd - # - namespace-name2@kubernetescrd - middlewares: [] - metrics: - # -- When using hostNetwork, use another port to avoid conflict with node exporter: - # https://github.com/prometheus/prometheus/wiki/Default-port-allocations - port: 9100 - # -- You may not want to expose the metrics port on production deployments. - # If you want to access it from outside your cluster, - # use `kubectl port-forward` or create a secure ingress - expose: - default: false - # -- The exposed port for this service - exposedPort: 9100 - # -- The port protocol (TCP/UDP) - protocol: TCP - -# -- TLS Options are created as [TLSOption CRDs](https://doc.traefik.io/traefik/https/tls/#tls-options) -# When using `labelSelector`, you'll need to set labels on tlsOption accordingly. -# See EXAMPLE.md for details. -tlsOptions: {} - -# -- TLS Store are created as [TLSStore CRDs](https://doc.traefik.io/traefik/https/tls/#default-certificate). This is useful if you want to set a default certificate. See EXAMPLE.md for details. -tlsStore: {} - -service: - enabled: true - ## -- Single service is using `MixedProtocolLBService` feature gate. - ## -- When set to false, it will create two Service, one for TCP and one for UDP. - single: true - type: LoadBalancer - # -- Additional annotations applied to both TCP and UDP services (e.g. for cloud provider specific config) - annotations: {} - # -- Additional annotations for TCP service only - annotationsTCP: {} - # -- Additional annotations for UDP service only - annotationsUDP: {} - # -- Additional service labels (e.g. for filtering Service by custom labels) - labels: {} - # -- Additional entries here will be added to the service spec. - # -- Cannot contain type, selector or ports entries. - spec: {} - # externalTrafficPolicy: Cluster - # loadBalancerIP: "1.2.3.4" - # clusterIP: "2.3.4.5" - loadBalancerSourceRanges: [] - # - 192.168.0.1/32 - # - 172.16.0.0/16 - ## -- Class of the load balancer implementation - # loadBalancerClass: service.k8s.aws/nlb - externalIPs: [] - # - 1.2.3.4 - ## One of SingleStack, PreferDualStack, or RequireDualStack. - # ipFamilyPolicy: SingleStack - ## List of IP families (e.g. IPv4 and/or IPv6). - ## ref: https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services - # ipFamilies: - # - IPv4 - # - IPv6 - ## - additionalServices: {} - ## -- An additional and optional internal Service. - ## Same parameters as external Service - # internal: - # type: ClusterIP - # # labels: {} - # # annotations: {} - # # spec: {} - # # loadBalancerSourceRanges: [] - # # externalIPs: [] - # # ipFamilies: [ "IPv4","IPv6" ] - -autoscaling: - # -- Create HorizontalPodAutoscaler object. - # See EXAMPLES.md for more details. - enabled: false - -persistence: - # -- Enable persistence using Persistent Volume Claims - # ref: http://kubernetes.io/docs/user-guide/persistent-volumes/. - # It can be used to store TLS certificates along with `certificatesResolvers..acme.storage` option - enabled: false - name: data - existingClaim: "" - accessMode: ReadWriteOnce - size: 128Mi - storageClass: "" - volumeName: "" - path: /data - annotations: {} - # -- Only mount a subpath of the Volume into the pod - subPath: "" - -# -- Certificates resolvers configuration. -# Ref: https://doc.traefik.io/traefik/https/acme/#certificate-resolvers -# See EXAMPLES.md for more details. -certificatesResolvers: {} - -# -- If hostNetwork is true, runs traefik in the host network namespace -# To prevent unschedulabel pods due to port collisions, if hostNetwork=true -# and replicas>1, a pod anti-affinity is recommended and will be set if the -# affinity is left as default. -hostNetwork: false - -# -- Whether Role Based Access Control objects like roles and rolebindings should be created -rbac: # @schema additionalProperties: false - enabled: true - # When set to true: - # 1. It switches respectively the use of `ClusterRole` and `ClusterRoleBinding` to `Role` and `RoleBinding`. - # 2. It adds `disableIngressClassLookup` on Kubernetes Ingress with Traefik Proxy v3 until v3.1.4 - # 3. It adds `disableClusterScopeResources` on Ingress and CRD (Kubernetes) providers with Traefik Proxy v3.1.2+ - # **NOTE**: `IngressClass`, `NodePortLB` and **Gateway** provider cannot be used with namespaced RBAC. - # See [upstream documentation](https://doc.traefik.io/traefik/providers/kubernetes-ingress/#disableclusterscoperesources) for more details. - namespaced: false - # Enable user-facing roles - # https://kubernetes.io/docs/reference/access-authn-authz/rbac/#user-facing-roles - aggregateTo: [] - # List of Kubernetes secrets that are accessible for Traefik. If empty, then access is granted to every secret. - secretResourceNames: [] - -# -- Enable to create a PodSecurityPolicy and assign it to the Service Account via RoleBinding or ClusterRoleBinding -podSecurityPolicy: - enabled: false - -# -- The service account the pods will use to interact with the Kubernetes API -serviceAccount: # @schema additionalProperties: false - # If set, an existing service account is used - # If not set, a service account is created automatically using the fullname template - name: "" - -# -- Additional serviceAccount annotations (e.g. for oidc authentication) -serviceAccountAnnotations: {} - -# -- [Resources](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/) for `traefik` container. -resources: {} - -# -- This example pod anti-affinity forces the scheduler to put traefik pods -# -- on nodes where no other traefik pods are scheduled. -# It should be used when hostNetwork: true to prevent port conflicts -affinity: {} -# podAntiAffinity: -# requiredDuringSchedulingIgnoredDuringExecution: -# - labelSelector: -# matchLabels: -# app.kubernetes.io/name: '{{ template "traefik.name" . }}' -# app.kubernetes.io/instance: '{{ .Release.Name }}-{{ .Release.Namespace }}' -# topologyKey: kubernetes.io/hostname - -# -- nodeSelector is the simplest recommended form of node selection constraint. -nodeSelector: {} -# -- Tolerations allow the scheduler to schedule pods with matching taints. -tolerations: [] -# -- You can use topology spread constraints to control -# how Pods are spread across your cluster among failure-domains. -topologySpreadConstraints: [] -# This example topologySpreadConstraints forces the scheduler to put traefik pods -# on nodes where no other traefik pods are scheduled. -# - labelSelector: -# matchLabels: -# app.kubernetes.io/name: '{{ template "traefik.name" . }}' -# maxSkew: 1 -# topologyKey: kubernetes.io/hostname -# whenUnsatisfiable: DoNotSchedule - -# -- [Pod Priority and Preemption](https://kubernetes.io/docs/concepts/scheduling-eviction/pod-priority-preemption/) -priorityClassName: "" - -# -- [SecurityContext](https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/pod-v1/#security-context-1) -# @default -- See _values.yaml_ -securityContext: - allowPrivilegeEscalation: false - capabilities: - drop: [ALL] - readOnlyRootFilesystem: true - -# -- [Pod Security Context](https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/pod-v1/#security-context) -# @default -- See _values.yaml_ -podSecurityContext: - runAsGroup: 65532 - runAsNonRoot: true - runAsUser: 65532 - -# -# -- Extra objects to deploy (value evaluated as a template) -# -# In some cases, it can avoid the need for additional, extended or adhoc deployments. -# See #595 for more details and traefik/tests/values/extra.yaml for example. -extraObjects: [] - -# -- This field override the default Release Namespace for Helm. -# It will not affect optional CRDs such as `ServiceMonitor` and `PrometheusRules` -namespaceOverride: "" - -## -- This field override the default app.kubernetes.io/instance label for all Objects. -instanceLabelOverride: "" - -# Traefik Hub configuration. See https://doc.traefik.io/traefik-hub/ -hub: - # -- Name of `Secret` with key 'token' set to a valid license token. - # It enables API Gateway. - token: "" - apimanagement: - # -- Set to true in order to enable API Management. Requires a valid license token. - enabled: false - admission: - # -- WebHook admission server listen address. Default: "0.0.0.0:9943". - listenAddr: "" - # -- Certificate of the WebHook admission server. Default: "hub-agent-cert". - secretName: "" - - redis: - # -- Enable Redis Cluster. Default: true. - cluster: # @schema type:[boolean, null] - # -- Database used to store information. Default: "0". - database: # @schema type:[string, null] - # -- Endpoints of the Redis instances to connect to. Default: "". - endpoints: "" - # -- The username to use when connecting to Redis endpoints. Default: "". - username: "" - # -- The password to use when connecting to Redis endpoints. Default: "". - password: "" - sentinel: - # -- Name of the set of main nodes to use for main selection. Required when using Sentinel. Default: "". - masterset: "" - # -- Username to use for sentinel authentication (can be different from endpoint username). Default: "". - username: "" - # -- Password to use for sentinel authentication (can be different from endpoint password). Default: "". - password: "" - # -- Timeout applied on connection with redis. Default: "0s". - timeout: "" - tls: - # -- Path to the certificate authority used for the secured connection. - ca: "" - # -- Path to the public certificate used for the secure connection. - cert: "" - # -- Path to the private key used for the secure connection. - key: "" - # -- When insecureSkipVerify is set to true, the TLS connection accepts any certificate presented by the server. Default: false. - insecureSkipVerify: false - # Enable export of errors logs to the platform. Default: true. - sendlogs: # @schema type:[boolean, null] diff --git a/charts/traefik/values-overrides.yaml b/charts/traefik/values-overrides.yaml deleted file mode 100644 index 5c5ceb9..0000000 --- a/charts/traefik/values-overrides.yaml +++ /dev/null @@ -1,39 +0,0 @@ - - -fullnameOverride: traefik - -globalArguments: - - --global.sendanonymoususage=false - - --api.insecure=true # api.insecure=true is required for the dashboard to be reachable via traefik entrypoint -service: - enabled: true - annotations: - # chisel-operator.io/exit-node-name: "vault-exit-node" - type: ClusterIP # this needs to be LoadBalancer in prod, but here in dev we want ClusterIP - -logs: - access: - enabled: true - format: json - general: - level: DEBUG - format: json -providers: - kubernetesCRD: - enabled: false - kubernetesIngress: - enabled: false - publishedService: - enabled: false - kubernetesGateway: - enabled: true - -# ingressRoute: -# dashboard: -# # we disable dashboard via the chart and enable it ourselves in traefik.yaml -# enabled: false -# matchRule: PathPrefix(`/doshboard`) -# # # -- The internal service used for the dashboard ingressRoute -# services: -# - name: dashboard@internal -# kind: TraefikService \ No newline at end of file diff --git a/charts/velero/values.yaml b/charts/velero/values.yaml deleted file mode 100644 index 2ddae71..0000000 --- a/charts/velero/values.yaml +++ /dev/null @@ -1,34 +0,0 @@ -credentials: - useSecret: true - existingSecret: velero - -backupsEnabled: true -snapshotsEnabled: true - -initContainers: - - name: velero-plugin-for-aws - image: velero/velero-plugin-for-aws:v1.10.1 - imagePullPolicy: IfNotPresent - volumeMounts: - - mountPath: /target - name: plugins - -configuration: - backupStorageLocation: - - name: futureporn-db-backup-dev - provider: aws - bucket: futureporn-db-backup-dev - default: true - s3ForcePathStyle: true - config: - region: us-west-000 - s3Url: https://s3.us-west-000.backblazeb2.com - checksumAlgorithm: "" - - volumeSnapshotLocation: - - name: futureporn-db-backup-dev - provider: aws - bucket: futureporn-db-backup-dev - s3ForcePathStyle: true - config: - region: us-west-000 diff --git a/charts/velero/velero/.helmignore b/charts/velero/velero/.helmignore deleted file mode 100644 index f0c1319..0000000 --- a/charts/velero/velero/.helmignore +++ /dev/null @@ -1,21 +0,0 @@ -# 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/charts/velero/velero/Chart.yaml b/charts/velero/velero/Chart.yaml deleted file mode 100644 index 2eb64a4..0000000 --- a/charts/velero/velero/Chart.yaml +++ /dev/null @@ -1,19 +0,0 @@ -apiVersion: v2 -appVersion: 1.14.1 -description: A Helm chart for velero -home: https://github.com/vmware-tanzu/velero -icon: https://cdn-images-1.medium.com/max/1600/1*-9mb3AKnKdcL_QD3CMnthQ.png -kubeVersion: '>=1.16.0-0' -maintainers: -- email: hsiaoairplane@gmail.com - name: jenting -- email: jiangd@vmware.com - name: reasonerjt -- email: mqiu@vmware.com - name: qiuming-best -- email: yinw@vmware.com - name: ywk253100 -name: velero -sources: -- https://github.com/vmware-tanzu/velero -version: 7.2.1 diff --git a/charts/velero/velero/OWNERS b/charts/velero/velero/OWNERS deleted file mode 100644 index d705d91..0000000 --- a/charts/velero/velero/OWNERS +++ /dev/null @@ -1,10 +0,0 @@ -approvers: -- jenting -- reasonerjt -- qiuming-best -- ywk253100 -reviewers: -- jenting -- reasonerjt -- qiuming-best -- ywk253100 diff --git a/charts/velero/velero/README.md b/charts/velero/velero/README.md deleted file mode 100644 index e6176d8..0000000 --- a/charts/velero/velero/README.md +++ /dev/null @@ -1,173 +0,0 @@ -# Velero - -Velero is an open source tool to safely backup and restore, perform disaster recovery, and migrate Kubernetes cluster resources and persistent volumes. - -Velero has two main components: a CLI, and a server-side Kubernetes deployment. - -## Installing the Velero CLI - -See the different options for installing the [Velero CLI](https://velero.io/docs/v1.13/basic-install/#install-the-cli). - -## Installing the Velero server - -### Installation Requirements - -Kubernetes v1.16+, because this helm chart uses CustomResourceDefinition `apiextensions.k8s.io/v1`. This API version was introduced in Kubernetes v1.16. - -### Velero version - -This helm chart installs Velero version v1.14 https://velero.io/docs/v1.14/. See the [#Upgrading](#upgrading) section for information on how to upgrade from other versions. - -### Provider credentials - -When installing using the Helm chart, the provider's credential information will need to be appended into your values. The easiest way to do this is with the `--set-file` argument, available in Helm 2.10 and higher. See your cloud provider's documentation for the contents and creation of the `credentials-velero` file. - -### Azure resources - -When using the Azure plug-in, requests and limits must be set. See https://github.com/vmware-tanzu/velero/issues/3234 and https://github.com/vmware-tanzu/helm-charts/issues/469 for details. - -### Installing - -The default configuration values for this chart are listed in values.yaml. - -See Velero's full [official documentation](https://velero.io/docs/v1.13/basic-install/). More specifically, find your provider in the Velero list of [supported providers](https://velero.io/docs/v1.13/supported-providers/) for specific configuration information and examples. - -#### Set up Helm - -See the main [README.md](https://github.com/vmware-tanzu/helm-charts#kubernetes-helm-charts-for-vmware-tanzu). - -#### Using Helm 3 - -##### Option 1) CLI commands - -Note: You may add the flag `--set cleanUpCRDs=true` if you want to delete the Velero CRDs after deleting a release. -Please note that cleaning up CRDs will also delete any CRD instance, such as BackupStorageLocation and VolumeSnapshotLocation, which would have to be reconfigured when reinstalling Velero. The backup data in object storage will not be deleted, even though the backup instances in the cluster will. - -Specify the necessary values using the --set key=value[,key=value] argument to helm install. For example, - -```bash -helm install velero vmware-tanzu/velero \ ---namespace \ ---create-namespace \ ---set-file credentials.secretContents.cloud= \ ---set configuration.backupStorageLocation[0].name= \ ---set configuration.backupStorageLocation[0].provider= \ ---set configuration.backupStorageLocation[0].bucket= \ ---set configuration.backupStorageLocation[0].config.region= \ ---set configuration.volumeSnapshotLocation[0].name= \ ---set configuration.volumeSnapshotLocation[0].provider= \ ---set configuration.volumeSnapshotLocation[0].config.region= \ ---set initContainers[0].name=velero-plugin-for- \ ---set initContainers[0].image=velero/velero-plugin-for-: \ ---set initContainers[0].volumeMounts[0].mountPath=/target \ ---set initContainers[0].volumeMounts[0].name=plugins -``` - -Users of zsh might need to put quotes around key/value pairs. - -##### Option 2) YAML file - -Add/update the necessary values by changing the values.yaml from this repository, then run: - -```bash -helm install vmware-tanzu/velero --namespace -f values.yaml --generate-name -``` -##### Upgrade the configuration - -If a value needs to be added or changed, you may do so with the `upgrade` command. An example: - -```bash -helm upgrade vmware-tanzu/velero --namespace --reuse-values --set configuration.backupStorageLocation[0].provider= -``` - -#### Using Helm 2 - -We're no longer supporting Helm v2 since it was deprecated in November 2020. - -##### Upgrade the configuration - -If a value needs to be added or changed, you may do so with the `upgrade` command. An example: - -```bash -helm upgrade vmware-tanzu/velero --reuse-values --set configuration.backupStorageLocation[0].provider= -``` -## Upgrading Chart - -### Upgrading to 7.0.0 - -Delete the CSI plugin. Because the Velero CSI plugin is already merged into the Velero, need to remove the existing CSI plugin InitContainer. Otherwise, the Velero server plugin would fail to start due to same plugin registered twice. -CSI plugin has been merged into velero repo in v1.14 release. It will be installed by default as an internal plugin. - -### Upgrading to 6.0.0 - -This version removes the `nodeAgent.privileged` field, you should use `nodeAgent.containerSecurityContext.privileged` instead - -## Upgrading Velero - -### Upgrading to v1.14 - -The [instructions found here](https://velero.io/docs/v1.14/upgrade-to-1.14/) will assist you in upgrading from version v1.13.x to v1.14. - -### Upgrading to v1.13 - -The [instructions found here](https://velero.io/docs/v1.13/upgrade-to-1.13/) will assist you in upgrading from version v1.12.x to v1.13. - -### Upgrading to v1.12 - -The [instructions found here](https://velero.io/docs/v1.12/upgrade-to-1.12/) will assist you in upgrading from version v1.11.x to v1.12. - -### Upgrading to v1.11 - -The [instructions found here](https://velero.io/docs/v1.11/upgrade-to-1.11/) will assist you in upgrading from version v1.10.x to v1.11. - -### Upgrading to v1.10 - -The [instructions found here](https://velero.io/docs/v1.10/upgrade-to-1.10/) will assist you in upgrading from version v1.9.x to v1.10. - -### Upgrading to v1.9 - -The [instructions found here](https://velero.io/docs/v1.9/upgrade-to-1.9/) will assist you in upgrading from version v1.8.x to v1.9. - -### Upgrading to v1.8 - -The [instructions found here](https://velero.io/docs/v1.8/upgrade-to-1.8/) will assist you in upgrading from version v1.7.x to v1.8. - -### Upgrading to v1.7 - -The [instructions found here](https://velero.io/docs/v1.7/upgrade-to-1.7/) will assist you in upgrading from version v1.6.x to v1.7. - -### Upgrading to v1.6 - -The [instructions found here](https://velero.io/docs/v1.6/upgrade-to-1.6/) will assist you in upgrading from version v1.5.x to v1.6. - -### Upgrading to v1.5 - -The [instructions found here](https://velero.io/docs/v1.5/upgrade-to-1.5/) will assist you in upgrading from version v1.4.x to v1.5. - -### Upgrading to v1.4 - -The [instructions found here](https://velero.io/docs/v1.4/upgrade-to-1.4/) will assist you in upgrading from version v1.3.x to v1.4. - -### Upgrading to v1.3.1 - -The [instructions found here](https://velero.io/docs/v1.3.1/upgrade-to-1.3/) will assist you in upgrading from version v1.2.0 or v1.3.0 to v1.3.1. - -### Upgrading to v1.2.0 - -The [instructions found here](https://velero.io/docs/v1.2.0/upgrade-to-1.2/) will assist you in upgrading from version v1.0.0 or v1.1.0 to v1.2.0. - -### Upgrading to v1.1.0 - -The [instructions found here](https://velero.io/docs/v1.1.0/upgrade-to-1.1/) will assist you in upgrading from version v1.0.0 to v1.1.0. - -## Uninstall Velero - -Note: when you uninstall the Velero server, all backups remain untouched. - -### Using Helm 3 - -```bash -helm uninstall -n -``` -### Note -Since from velero v1.10.0, it has supported both Restic and Kopia to do file-system level backup and restore, some configuration that contains the keyword Restic is not suitable anymore, which means from chart version 3.0.0 is not backward compatible, and we've done a configure filed name validation. diff --git a/charts/velero/velero/ci/test-values.yaml b/charts/velero/velero/ci/test-values.yaml deleted file mode 100644 index 80d1fbf..0000000 --- a/charts/velero/velero/ci/test-values.yaml +++ /dev/null @@ -1,119 +0,0 @@ -# Set provider name and backup storage location bucket name -configuration: - backupStorageLocation: - - name: default - bucket: velero-backups - default: true - provider: aws - credential: - name: test-credential - key: test-key - config: - region: us-east-1 - profile: us-east-1-profile - - name: backups-secondary - bucket: velero-backups - provider: aws - config: - region: us-west-1 - profile: us-west-1-profile - volumeSnapshotLocation: - - name: ebs-us-east-1 - provider: aws - config: - region: us-east-1 - - name: portworx-cloud - provider: portworx - config: - type: cloud - -schedules: - mybackup: - labels: - myenv: foo - schedule: "0 0 * * *" - template: - ttl: "240h" - includedNamespaces: - - foo - -# Set a service account so that the CRD clean up job has proper permissions to delete CRDs -serviceAccount: - server: - name: velero - -# The Velero server -# Annotations to Velero deployment -annotations: - annotation: velero - foo: bar - -# Labels to Velero deployment -labels: - label: velero - foo: bar - -# Annotations to Velero deployment's template -podAnnotations: - pod-annotation: velero - foo: bar - -# Labels to Velero deployment's template -podLabels: - pod-label: velero - foo: bar - -# Resources to Velero deployment -resources: - requests: - cpu: 100m - memory: 128Mi - limits: - cpu: 100m - memory: 128Mi - -# The node-agent daemonset -deployNodeAgent: true - -nodeAgent: - # Annotations to node-agent daemonset - annotations: - annotation: node-agent - foo: bar - # Labels to node-agent daemonset - labels: - label: node-agent - foo: bar - # Resources to node-agent daemonset - resources: - requests: - cpu: 100m - memory: 128Mi - limits: - cpu: 100m - memory: 128Mi - -# The kubectl upgrade/cleanup job -kubectl: - # Annotations to kubectl job - annotations: - annotation: kubectl - foo: bar - # Labels to kubectl job - labels: - label: kubectl - foo: bar - # Resources to kubectl job - resources: - requests: - cpu: 100m - memory: 128Mi - limits: - cpu: 100m - memory: 128Mi - -# Whether or not to clean up CustomResourceDefintions when deleting a release. -# Cleaning up CRDs will delete the BackupStorageLocation and VolumeSnapshotLocation instances, which would have to be reconfigured. -# Backup data in object storage will _not_ be deleted, however Backup instances in the Kubernetes API will. -# Always clean up CRDs in CI. -cleanUpCRDs: true diff --git a/charts/velero/velero/crds/backuprepositories.yaml b/charts/velero/velero/crds/backuprepositories.yaml deleted file mode 100644 index 4453dca..0000000 --- a/charts/velero/velero/crds/backuprepositories.yaml +++ /dev/null @@ -1,106 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - labels: - component: velero - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: backuprepositories.velero.io -spec: - group: velero.io - names: - kind: BackupRepository - listKind: BackupRepositoryList - plural: backuprepositories - singular: backuprepository - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .spec.repositoryType - name: Repository Type - type: string - name: v1 - schema: - openAPIV3Schema: - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: BackupRepositorySpec is the specification for a BackupRepository. - properties: - backupStorageLocation: - description: |- - BackupStorageLocation is the name of the BackupStorageLocation - that should contain this repository. - type: string - maintenanceFrequency: - description: MaintenanceFrequency is how often maintenance should - be run. - type: string - repositoryType: - description: RepositoryType indicates the type of the backend repository - enum: - - kopia - - restic - - "" - type: string - resticIdentifier: - description: |- - ResticIdentifier is the full restic-compatible string for identifying - this repository. - type: string - volumeNamespace: - description: |- - VolumeNamespace is the namespace this backup repository contains - pod volume backups for. - type: string - required: - - backupStorageLocation - - maintenanceFrequency - - resticIdentifier - - volumeNamespace - type: object - status: - description: BackupRepositoryStatus is the current status of a BackupRepository. - properties: - lastMaintenanceTime: - description: LastMaintenanceTime is the last time maintenance was - run. - format: date-time - nullable: true - type: string - message: - description: Message is a message about the current status of the - BackupRepository. - type: string - phase: - description: Phase is the current state of the BackupRepository. - enum: - - New - - Ready - - NotReady - type: string - type: object - type: object - served: true - storage: true - subresources: {} \ No newline at end of file diff --git a/charts/velero/velero/crds/backups.yaml b/charts/velero/velero/crds/backups.yaml deleted file mode 100644 index fd08085..0000000 --- a/charts/velero/velero/crds/backups.yaml +++ /dev/null @@ -1,662 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - labels: - component: velero - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: backups.velero.io -spec: - group: velero.io - names: - kind: Backup - listKind: BackupList - plural: backups - singular: backup - scope: Namespaced - versions: - - name: v1 - schema: - openAPIV3Schema: - description: |- - Backup is a Velero resource that represents the capture of Kubernetes - cluster state at a point in time (API objects and associated volume state). - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: BackupSpec defines the specification for a Velero backup. - properties: - csiSnapshotTimeout: - description: |- - CSISnapshotTimeout specifies the time used to wait for CSI VolumeSnapshot status turns to - ReadyToUse during creation, before returning error as timeout. - The default value is 10 minute. - type: string - datamover: - description: |- - DataMover specifies the data mover to be used by the backup. - If DataMover is "" or "velero", the built-in data mover will be used. - type: string - defaultVolumesToFsBackup: - description: |- - DefaultVolumesToFsBackup specifies whether pod volume file system backup should be used - for all volumes by default. - nullable: true - type: boolean - defaultVolumesToRestic: - description: |- - DefaultVolumesToRestic specifies whether restic should be used to take a - backup of all pod volumes by default. - - - Deprecated: this field is no longer used and will be removed entirely in future. Use DefaultVolumesToFsBackup instead. - nullable: true - type: boolean - excludedClusterScopedResources: - description: |- - ExcludedClusterScopedResources is a slice of cluster-scoped - resource type names to exclude from the backup. - If set to "*", all cluster-scoped resource types are excluded. - The default value is empty. - items: - type: string - nullable: true - type: array - excludedNamespaceScopedResources: - description: |- - ExcludedNamespaceScopedResources is a slice of namespace-scoped - resource type names to exclude from the backup. - If set to "*", all namespace-scoped resource types are excluded. - The default value is empty. - items: - type: string - nullable: true - type: array - excludedNamespaces: - description: |- - ExcludedNamespaces contains a list of namespaces that are not - included in the backup. - items: - type: string - nullable: true - type: array - excludedResources: - description: |- - ExcludedResources is a slice of resource names that are not - included in the backup. - items: - type: string - nullable: true - type: array - hooks: - description: Hooks represent custom behaviors that should be executed - at different phases of the backup. - properties: - resources: - description: Resources are hooks that should be executed when - backing up individual instances of a resource. - items: - description: |- - BackupResourceHookSpec defines one or more BackupResourceHooks that should be executed based on - the rules defined for namespaces, resources, and label selector. - properties: - excludedNamespaces: - description: ExcludedNamespaces specifies the namespaces - to which this hook spec does not apply. - items: - type: string - nullable: true - type: array - excludedResources: - description: ExcludedResources specifies the resources to - which this hook spec does not apply. - items: - type: string - nullable: true - type: array - includedNamespaces: - description: |- - IncludedNamespaces specifies the namespaces to which this hook spec applies. If empty, it applies - to all namespaces. - items: - type: string - nullable: true - type: array - includedResources: - description: |- - IncludedResources specifies the resources to which this hook spec applies. If empty, it applies - to all resources. - items: - type: string - nullable: true - type: array - labelSelector: - description: LabelSelector, if specified, filters the resources - to which this hook spec applies. - nullable: true - properties: - matchExpressions: - description: matchExpressions is a list of label selector - requirements. The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector - applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - required: - - key - - operator - type: object - type: array - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - name: - description: Name is the name of this hook. - type: string - post: - description: |- - PostHooks is a list of BackupResourceHooks to execute after storing the item in the backup. - These are executed after all "additional items" from item actions are processed. - items: - description: BackupResourceHook defines a hook for a resource. - properties: - exec: - description: Exec defines an exec hook. - properties: - command: - description: Command is the command and arguments - to execute. - items: - type: string - minItems: 1 - type: array - container: - description: |- - Container is the container in the pod where the command should be executed. If not specified, - the pod's first container is used. - type: string - onError: - description: OnError specifies how Velero should - behave if it encounters an error executing this - hook. - enum: - - Continue - - Fail - type: string - timeout: - description: |- - Timeout defines the maximum amount of time Velero should wait for the hook to complete before - considering the execution a failure. - type: string - required: - - command - type: object - required: - - exec - type: object - type: array - pre: - description: |- - PreHooks is a list of BackupResourceHooks to execute prior to storing the item in the backup. - These are executed before any "additional items" from item actions are processed. - items: - description: BackupResourceHook defines a hook for a resource. - properties: - exec: - description: Exec defines an exec hook. - properties: - command: - description: Command is the command and arguments - to execute. - items: - type: string - minItems: 1 - type: array - container: - description: |- - Container is the container in the pod where the command should be executed. If not specified, - the pod's first container is used. - type: string - onError: - description: OnError specifies how Velero should - behave if it encounters an error executing this - hook. - enum: - - Continue - - Fail - type: string - timeout: - description: |- - Timeout defines the maximum amount of time Velero should wait for the hook to complete before - considering the execution a failure. - type: string - required: - - command - type: object - required: - - exec - type: object - type: array - required: - - name - type: object - nullable: true - type: array - type: object - includeClusterResources: - description: |- - IncludeClusterResources specifies whether cluster-scoped resources - should be included for consideration in the backup. - nullable: true - type: boolean - includedClusterScopedResources: - description: |- - IncludedClusterScopedResources is a slice of cluster-scoped - resource type names to include in the backup. - If set to "*", all cluster-scoped resource types are included. - The default value is empty, which means only related - cluster-scoped resources are included. - items: - type: string - nullable: true - type: array - includedNamespaceScopedResources: - description: |- - IncludedNamespaceScopedResources is a slice of namespace-scoped - resource type names to include in the backup. - The default value is "*". - items: - type: string - nullable: true - type: array - includedNamespaces: - description: |- - IncludedNamespaces is a slice of namespace names to include objects - from. If empty, all namespaces are included. - items: - type: string - nullable: true - type: array - includedResources: - description: |- - IncludedResources is a slice of resource names to include - in the backup. If empty, all resources are included. - items: - type: string - nullable: true - type: array - itemOperationTimeout: - description: |- - ItemOperationTimeout specifies the time used to wait for asynchronous BackupItemAction operations - The default value is 4 hour. - type: string - labelSelector: - description: |- - LabelSelector is a metav1.LabelSelector to filter with - when adding individual objects to the backup. If empty - or nil, all objects are included. Optional. - nullable: true - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. - The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector applies - to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - required: - - key - - operator - type: object - type: array - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - metadata: - properties: - labels: - additionalProperties: - type: string - type: object - type: object - orLabelSelectors: - description: |- - OrLabelSelectors is list of metav1.LabelSelector to filter with - when adding individual objects to the backup. If multiple provided - they will be joined by the OR operator. LabelSelector as well as - OrLabelSelectors cannot co-exist in backup request, only one of them - can be used. - items: - description: |- - A label selector is a label query over a set of resources. The result of matchLabels and - matchExpressions are ANDed. An empty label selector matches all objects. A null - label selector matches no objects. - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. - The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector applies - to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - required: - - key - - operator - type: object - type: array - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - nullable: true - type: array - orderedResources: - additionalProperties: - type: string - description: |- - OrderedResources specifies the backup order of resources of specific Kind. - The map key is the resource name and value is a list of object names separated by commas. - Each resource name has format "namespace/objectname". For cluster resources, simply use "objectname". - nullable: true - type: object - resourcePolicy: - description: ResourcePolicy specifies the referenced resource policies - that backup should follow - properties: - apiGroup: - description: |- - APIGroup is the group for the resource being referenced. - If APIGroup is not specified, the specified Kind must be in the core API group. - For any other third-party types, APIGroup is required. - type: string - kind: - description: Kind is the type of resource being referenced - type: string - name: - description: Name is the name of resource being referenced - type: string - required: - - kind - - name - type: object - x-kubernetes-map-type: atomic - snapshotMoveData: - description: SnapshotMoveData specifies whether snapshot data should - be moved - nullable: true - type: boolean - snapshotVolumes: - description: |- - SnapshotVolumes specifies whether to take snapshots - of any PV's referenced in the set of objects included - in the Backup. - nullable: true - type: boolean - storageLocation: - description: StorageLocation is a string containing the name of a - BackupStorageLocation where the backup should be stored. - type: string - ttl: - description: |- - TTL is a time.Duration-parseable string describing how long - the Backup should be retained for. - type: string - uploaderConfig: - description: UploaderConfig specifies the configuration for the uploader. - nullable: true - properties: - parallelFilesUpload: - description: ParallelFilesUpload is the number of files parallel - uploads to perform when using the uploader. - type: integer - type: object - volumeSnapshotLocations: - description: VolumeSnapshotLocations is a list containing names of - VolumeSnapshotLocations associated with this backup. - items: - type: string - type: array - type: object - status: - description: BackupStatus captures the current status of a Velero backup. - properties: - backupItemOperationsAttempted: - description: |- - BackupItemOperationsAttempted is the total number of attempted - async BackupItemAction operations for this backup. - type: integer - backupItemOperationsCompleted: - description: |- - BackupItemOperationsCompleted is the total number of successfully completed - async BackupItemAction operations for this backup. - type: integer - backupItemOperationsFailed: - description: |- - BackupItemOperationsFailed is the total number of async - BackupItemAction operations for this backup which ended with an error. - type: integer - completionTimestamp: - description: |- - CompletionTimestamp records the time a backup was completed. - Completion time is recorded even on failed backups. - Completion time is recorded before uploading the backup object. - The server's time is used for CompletionTimestamps - format: date-time - nullable: true - type: string - csiVolumeSnapshotsAttempted: - description: |- - CSIVolumeSnapshotsAttempted is the total number of attempted - CSI VolumeSnapshots for this backup. - type: integer - csiVolumeSnapshotsCompleted: - description: |- - CSIVolumeSnapshotsCompleted is the total number of successfully - completed CSI VolumeSnapshots for this backup. - type: integer - errors: - description: |- - Errors is a count of all error messages that were generated during - execution of the backup. The actual errors are in the backup's log - file in object storage. - type: integer - expiration: - description: Expiration is when this Backup is eligible for garbage-collection. - format: date-time - nullable: true - type: string - failureReason: - description: FailureReason is an error that caused the entire backup - to fail. - type: string - formatVersion: - description: FormatVersion is the backup format version, including - major, minor, and patch version. - type: string - hookStatus: - description: HookStatus contains information about the status of the - hooks. - nullable: true - properties: - hooksAttempted: - description: |- - HooksAttempted is the total number of attempted hooks - Specifically, HooksAttempted represents the number of hooks that failed to execute - and the number of hooks that executed successfully. - type: integer - hooksFailed: - description: HooksFailed is the total number of hooks which ended - with an error - type: integer - type: object - phase: - description: Phase is the current state of the Backup. - enum: - - New - - FailedValidation - - InProgress - - WaitingForPluginOperations - - WaitingForPluginOperationsPartiallyFailed - - Finalizing - - FinalizingPartiallyFailed - - Completed - - PartiallyFailed - - Failed - - Deleting - type: string - progress: - description: |- - Progress contains information about the backup's execution progress. Note - that this information is best-effort only -- if Velero fails to update it - during a backup for any reason, it may be inaccurate/stale. - nullable: true - properties: - itemsBackedUp: - description: |- - ItemsBackedUp is the number of items that have actually been written to the - backup tarball so far. - type: integer - totalItems: - description: |- - TotalItems is the total number of items to be backed up. This number may change - throughout the execution of the backup due to plugins that return additional related - items to back up, the velero.io/exclude-from-backup label, and various other - filters that happen as items are processed. - type: integer - type: object - startTimestamp: - description: |- - StartTimestamp records the time a backup was started. - Separate from CreationTimestamp, since that value changes - on restores. - The server's time is used for StartTimestamps - format: date-time - nullable: true - type: string - validationErrors: - description: |- - ValidationErrors is a slice of all validation errors (if - applicable). - items: - type: string - nullable: true - type: array - version: - description: |- - Version is the backup format major version. - Deprecated: Please see FormatVersion - type: integer - volumeSnapshotsAttempted: - description: |- - VolumeSnapshotsAttempted is the total number of attempted - volume snapshots for this backup. - type: integer - volumeSnapshotsCompleted: - description: |- - VolumeSnapshotsCompleted is the total number of successfully - completed volume snapshots for this backup. - type: integer - warnings: - description: |- - Warnings is a count of all warning messages that were generated during - execution of the backup. The actual warnings are in the backup's log - file in object storage. - type: integer - type: object - type: object - served: true - storage: true \ No newline at end of file diff --git a/charts/velero/velero/crds/backupstoragelocations.yaml b/charts/velero/velero/crds/backupstoragelocations.yaml deleted file mode 100644 index 05251b5..0000000 --- a/charts/velero/velero/crds/backupstoragelocations.yaml +++ /dev/null @@ -1,190 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - labels: - component: velero - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: backupstoragelocations.velero.io -spec: - group: velero.io - names: - kind: BackupStorageLocation - listKind: BackupStorageLocationList - plural: backupstoragelocations - shortNames: - - bsl - singular: backupstoragelocation - scope: Namespaced - versions: - - additionalPrinterColumns: - - description: Backup Storage Location status such as Available/Unavailable - jsonPath: .status.phase - name: Phase - type: string - - description: LastValidationTime is the last time the backup store location was - validated - jsonPath: .status.lastValidationTime - name: Last Validated - type: date - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - description: Default backup storage location - jsonPath: .spec.default - name: Default - type: boolean - name: v1 - schema: - openAPIV3Schema: - description: BackupStorageLocation is a location where Velero stores backup - objects - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: BackupStorageLocationSpec defines the desired state of a - Velero BackupStorageLocation - properties: - accessMode: - description: AccessMode defines the permissions for the backup storage - location. - enum: - - ReadOnly - - ReadWrite - type: string - backupSyncPeriod: - description: BackupSyncPeriod defines how frequently to sync backup - API objects from object storage. A value of 0 disables sync. - nullable: true - type: string - config: - additionalProperties: - type: string - description: Config is for provider-specific configuration fields. - type: object - credential: - description: Credential contains the credential information intended - to be used with this location - properties: - key: - description: The key of the secret to select from. Must be a - valid secret key. - type: string - name: - description: |- - Name of the referent. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - TODO: Add other useful fields. apiVersion, kind, uid? - type: string - optional: - description: Specify whether the Secret or its key must be defined - type: boolean - required: - - key - type: object - x-kubernetes-map-type: atomic - default: - description: Default indicates this location is the default backup - storage location. - type: boolean - objectStorage: - description: ObjectStorageLocation specifies the settings necessary - to connect to a provider's object storage. - properties: - bucket: - description: Bucket is the bucket to use for object storage. - type: string - caCert: - description: CACert defines a CA bundle to use when verifying - TLS connections to the provider. - format: byte - type: string - prefix: - description: Prefix is the path inside a bucket to use for Velero - storage. Optional. - type: string - required: - - bucket - type: object - provider: - description: Provider is the provider of the backup storage. - type: string - validationFrequency: - description: ValidationFrequency defines how frequently to validate - the corresponding object storage. A value of 0 disables validation. - nullable: true - type: string - required: - - objectStorage - - provider - type: object - status: - description: BackupStorageLocationStatus defines the observed state of - BackupStorageLocation - properties: - accessMode: - description: |- - AccessMode is an unused field. - - - Deprecated: there is now an AccessMode field on the Spec and this field - will be removed entirely as of v2.0. - enum: - - ReadOnly - - ReadWrite - type: string - lastSyncedRevision: - description: |- - LastSyncedRevision is the value of the `metadata/revision` file in the backup - storage location the last time the BSL's contents were synced into the cluster. - - - Deprecated: this field is no longer updated or used for detecting changes to - the location's contents and will be removed entirely in v2.0. - type: string - lastSyncedTime: - description: |- - LastSyncedTime is the last time the contents of the location were synced into - the cluster. - format: date-time - nullable: true - type: string - lastValidationTime: - description: |- - LastValidationTime is the last time the backup store location was validated - the cluster. - format: date-time - nullable: true - type: string - message: - description: Message is a message about the backup storage location's - status. - type: string - phase: - description: Phase is the current state of the BackupStorageLocation. - enum: - - Available - - Unavailable - type: string - type: object - type: object - served: true - storage: true - subresources: {} \ No newline at end of file diff --git a/charts/velero/velero/crds/datadownloads.yaml b/charts/velero/velero/crds/datadownloads.yaml deleted file mode 100644 index ecbca2c..0000000 --- a/charts/velero/velero/crds/datadownloads.yaml +++ /dev/null @@ -1,191 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - labels: - component: velero - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: datadownloads.velero.io -spec: - group: velero.io - names: - kind: DataDownload - listKind: DataDownloadList - plural: datadownloads - singular: datadownload - scope: Namespaced - versions: - - additionalPrinterColumns: - - description: DataDownload status such as New/InProgress - jsonPath: .status.phase - name: Status - type: string - - description: Time duration since this DataDownload was started - jsonPath: .status.startTimestamp - name: Started - type: date - - description: Completed bytes - format: int64 - jsonPath: .status.progress.bytesDone - name: Bytes Done - type: integer - - description: Total bytes - format: int64 - jsonPath: .status.progress.totalBytes - name: Total Bytes - type: integer - - description: Name of the Backup Storage Location where the backup data is stored - jsonPath: .spec.backupStorageLocation - name: Storage Location - type: string - - description: Time duration since this DataDownload was created - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - description: Name of the node where the DataDownload is processed - jsonPath: .status.node - name: Node - type: string - name: v2alpha1 - schema: - openAPIV3Schema: - description: DataDownload acts as the protocol between data mover plugins - and data mover controller for the datamover restore operation - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: DataDownloadSpec is the specification for a DataDownload. - properties: - backupStorageLocation: - description: |- - BackupStorageLocation is the name of the backup storage location - where the backup repository is stored. - type: string - cancel: - description: |- - Cancel indicates request to cancel the ongoing DataDownload. It can be set - when the DataDownload is in InProgress phase - type: boolean - dataMoverConfig: - additionalProperties: - type: string - description: DataMoverConfig is for data-mover-specific configuration - fields. - type: object - datamover: - description: |- - DataMover specifies the data mover to be used by the backup. - If DataMover is "" or "velero", the built-in data mover will be used. - type: string - operationTimeout: - description: |- - OperationTimeout specifies the time used to wait internal operations, - before returning error as timeout. - type: string - snapshotID: - description: SnapshotID is the ID of the Velero backup snapshot to - be restored from. - type: string - sourceNamespace: - description: |- - SourceNamespace is the original namespace where the volume is backed up from. - It may be different from SourcePVC's namespace if namespace is remapped during restore. - type: string - targetVolume: - description: TargetVolume is the information of the target PVC and - PV. - properties: - namespace: - description: Namespace is the target namespace - type: string - pv: - description: PV is the name of the target PV that is created by - Velero restore - type: string - pvc: - description: PVC is the name of the target PVC that is created - by Velero restore - type: string - required: - - namespace - - pv - - pvc - type: object - required: - - backupStorageLocation - - operationTimeout - - snapshotID - - sourceNamespace - - targetVolume - type: object - status: - description: DataDownloadStatus is the current status of a DataDownload. - properties: - completionTimestamp: - description: |- - CompletionTimestamp records the time a restore was completed. - Completion time is recorded even on failed restores. - The server's time is used for CompletionTimestamps - format: date-time - nullable: true - type: string - message: - description: Message is a message about the DataDownload's status. - type: string - node: - description: Node is name of the node where the DataDownload is processed. - type: string - phase: - description: Phase is the current state of the DataDownload. - enum: - - New - - Accepted - - Prepared - - InProgress - - Canceling - - Canceled - - Completed - - Failed - type: string - progress: - description: |- - Progress holds the total number of bytes of the snapshot and the current - number of restored bytes. This can be used to display progress information - about the restore operation. - properties: - bytesDone: - format: int64 - type: integer - totalBytes: - format: int64 - type: integer - type: object - startTimestamp: - description: |- - StartTimestamp records the time a restore was started. - The server's time is used for StartTimestamps - format: date-time - nullable: true - type: string - type: object - type: object - served: true - storage: true - subresources: {} \ No newline at end of file diff --git a/charts/velero/velero/crds/datauploads.yaml b/charts/velero/velero/crds/datauploads.yaml deleted file mode 100644 index f0dd8cf..0000000 --- a/charts/velero/velero/crds/datauploads.yaml +++ /dev/null @@ -1,216 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - labels: - component: velero - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: datauploads.velero.io -spec: - group: velero.io - names: - kind: DataUpload - listKind: DataUploadList - plural: datauploads - singular: dataupload - scope: Namespaced - versions: - - additionalPrinterColumns: - - description: DataUpload status such as New/InProgress - jsonPath: .status.phase - name: Status - type: string - - description: Time duration since this DataUpload was started - jsonPath: .status.startTimestamp - name: Started - type: date - - description: Completed bytes - format: int64 - jsonPath: .status.progress.bytesDone - name: Bytes Done - type: integer - - description: Total bytes - format: int64 - jsonPath: .status.progress.totalBytes - name: Total Bytes - type: integer - - description: Name of the Backup Storage Location where this backup should be - stored - jsonPath: .spec.backupStorageLocation - name: Storage Location - type: string - - description: Time duration since this DataUpload was created - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - description: Name of the node where the DataUpload is processed - jsonPath: .status.node - name: Node - type: string - name: v2alpha1 - schema: - openAPIV3Schema: - description: DataUpload acts as the protocol between data mover plugins and - data mover controller for the datamover backup operation - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: DataUploadSpec is the specification for a DataUpload. - properties: - backupStorageLocation: - description: |- - BackupStorageLocation is the name of the backup storage location - where the backup repository is stored. - type: string - cancel: - description: |- - Cancel indicates request to cancel the ongoing DataUpload. It can be set - when the DataUpload is in InProgress phase - type: boolean - csiSnapshot: - description: If SnapshotType is CSI, CSISnapshot provides the information - of the CSI snapshot. - nullable: true - properties: - snapshotClass: - description: SnapshotClass is the name of the snapshot class that - the volume snapshot is created with - type: string - storageClass: - description: StorageClass is the name of the storage class of - the PVC that the volume snapshot is created from - type: string - volumeSnapshot: - description: VolumeSnapshot is the name of the volume snapshot - to be backed up - type: string - required: - - storageClass - - volumeSnapshot - type: object - dataMoverConfig: - additionalProperties: - type: string - description: DataMoverConfig is for data-mover-specific configuration - fields. - nullable: true - type: object - datamover: - description: |- - DataMover specifies the data mover to be used by the backup. - If DataMover is "" or "velero", the built-in data mover will be used. - type: string - operationTimeout: - description: |- - OperationTimeout specifies the time used to wait internal operations, - before returning error as timeout. - type: string - snapshotType: - description: SnapshotType is the type of the snapshot to be backed - up. - type: string - sourceNamespace: - description: |- - SourceNamespace is the original namespace where the volume is backed up from. - It is the same namespace for SourcePVC and CSI namespaced objects. - type: string - sourcePVC: - description: SourcePVC is the name of the PVC which the snapshot is - taken for. - type: string - required: - - backupStorageLocation - - operationTimeout - - snapshotType - - sourceNamespace - - sourcePVC - type: object - status: - description: DataUploadStatus is the current status of a DataUpload. - properties: - completionTimestamp: - description: |- - CompletionTimestamp records the time a backup was completed. - Completion time is recorded even on failed backups. - Completion time is recorded before uploading the backup object. - The server's time is used for CompletionTimestamps - format: date-time - nullable: true - type: string - dataMoverResult: - additionalProperties: - type: string - description: DataMoverResult stores data-mover-specific information - as a result of the DataUpload. - nullable: true - type: object - message: - description: Message is a message about the DataUpload's status. - type: string - node: - description: Node is name of the node where the DataUpload is processed. - type: string - path: - description: Path is the full path of the snapshot volume being backed - up. - type: string - phase: - description: Phase is the current state of the DataUpload. - enum: - - New - - Accepted - - Prepared - - InProgress - - Canceling - - Canceled - - Completed - - Failed - type: string - progress: - description: |- - Progress holds the total number of bytes of the volume and the current - number of backed up bytes. This can be used to display progress information - about the backup operation. - properties: - bytesDone: - format: int64 - type: integer - totalBytes: - format: int64 - type: integer - type: object - snapshotID: - description: SnapshotID is the identifier for the snapshot in the - backup repository. - type: string - startTimestamp: - description: |- - StartTimestamp records the time a backup was started. - Separate from CreationTimestamp, since that value changes - on restores. - The server's time is used for StartTimestamps - format: date-time - nullable: true - type: string - type: object - type: object - served: true - storage: true - subresources: {} \ No newline at end of file diff --git a/charts/velero/velero/crds/deletebackuprequests.yaml b/charts/velero/velero/crds/deletebackuprequests.yaml deleted file mode 100644 index 62c8431..0000000 --- a/charts/velero/velero/crds/deletebackuprequests.yaml +++ /dev/null @@ -1,80 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - labels: - component: velero - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: deletebackuprequests.velero.io -spec: - group: velero.io - names: - kind: DeleteBackupRequest - listKind: DeleteBackupRequestList - plural: deletebackuprequests - singular: deletebackuprequest - scope: Namespaced - versions: - - additionalPrinterColumns: - - description: The name of the backup to be deleted - jsonPath: .spec.backupName - name: BackupName - type: string - - description: The status of the deletion request - jsonPath: .status.phase - name: Status - type: string - name: v1 - schema: - openAPIV3Schema: - description: DeleteBackupRequest is a request to delete one or more backups. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: DeleteBackupRequestSpec is the specification for which backups - to delete. - properties: - backupName: - type: string - required: - - backupName - type: object - status: - description: DeleteBackupRequestStatus is the current status of a DeleteBackupRequest. - properties: - errors: - description: Errors contains any errors that were encountered during - the deletion process. - items: - type: string - nullable: true - type: array - phase: - description: Phase is the current state of the DeleteBackupRequest. - enum: - - New - - InProgress - - Processed - type: string - type: object - type: object - served: true - storage: true - subresources: {} \ No newline at end of file diff --git a/charts/velero/velero/crds/downloadrequests.yaml b/charts/velero/velero/crds/downloadrequests.yaml deleted file mode 100644 index 6582379..0000000 --- a/charts/velero/velero/crds/downloadrequests.yaml +++ /dev/null @@ -1,100 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - labels: - component: velero - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: downloadrequests.velero.io -spec: - group: velero.io - names: - kind: DownloadRequest - listKind: DownloadRequestList - plural: downloadrequests - singular: downloadrequest - scope: Namespaced - versions: - - name: v1 - schema: - openAPIV3Schema: - description: |- - DownloadRequest is a request to download an artifact from backup object storage, such as a backup - log file. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: DownloadRequestSpec is the specification for a download request. - properties: - target: - description: Target is what to download (e.g. logs for a backup). - properties: - kind: - description: Kind is the type of file to download. - enum: - - BackupLog - - BackupContents - - BackupVolumeSnapshots - - BackupItemOperations - - BackupResourceList - - BackupResults - - RestoreLog - - RestoreResults - - RestoreResourceList - - RestoreItemOperations - - CSIBackupVolumeSnapshots - - CSIBackupVolumeSnapshotContents - - BackupVolumeInfos - - RestoreVolumeInfo - type: string - name: - description: Name is the name of the Kubernetes resource with - which the file is associated. - type: string - required: - - kind - - name - type: object - required: - - target - type: object - status: - description: DownloadRequestStatus is the current status of a DownloadRequest. - properties: - downloadURL: - description: DownloadURL contains the pre-signed URL for the target - file. - type: string - expiration: - description: Expiration is when this DownloadRequest expires and can - be deleted by the system. - format: date-time - nullable: true - type: string - phase: - description: Phase is the current state of the DownloadRequest. - enum: - - New - - Processed - type: string - type: object - type: object - served: true - storage: true \ No newline at end of file diff --git a/charts/velero/velero/crds/podvolumebackups.yaml b/charts/velero/velero/crds/podvolumebackups.yaml deleted file mode 100644 index 7727b8c..0000000 --- a/charts/velero/velero/crds/podvolumebackups.yaml +++ /dev/null @@ -1,225 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - labels: - component: velero - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: podvolumebackups.velero.io -spec: - group: velero.io - names: - kind: PodVolumeBackup - listKind: PodVolumeBackupList - plural: podvolumebackups - singular: podvolumebackup - scope: Namespaced - versions: - - additionalPrinterColumns: - - description: Pod Volume Backup status such as New/InProgress - jsonPath: .status.phase - name: Status - type: string - - description: Time when this backup was started - jsonPath: .status.startTimestamp - name: Created - type: date - - description: Namespace of the pod containing the volume to be backed up - jsonPath: .spec.pod.namespace - name: Namespace - type: string - - description: Name of the pod containing the volume to be backed up - jsonPath: .spec.pod.name - name: Pod - type: string - - description: Name of the volume to be backed up - jsonPath: .spec.volume - name: Volume - type: string - - description: The type of the uploader to handle data transfer - jsonPath: .spec.uploaderType - name: Uploader Type - type: string - - description: Name of the Backup Storage Location where this backup should be - stored - jsonPath: .spec.backupStorageLocation - name: Storage Location - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1 - schema: - openAPIV3Schema: - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: PodVolumeBackupSpec is the specification for a PodVolumeBackup. - properties: - backupStorageLocation: - description: |- - BackupStorageLocation is the name of the backup storage location - where the backup repository is stored. - type: string - node: - description: Node is the name of the node that the Pod is running - on. - type: string - pod: - description: Pod is a reference to the pod containing the volume to - be backed up. - properties: - apiVersion: - description: API version of the referent. - type: string - fieldPath: - description: |- - If referring to a piece of an object instead of an entire object, this string - should contain a valid JSON/Go field access statement, such as desiredState.manifest.containers[2]. - For example, if the object reference is to a container within a pod, this would take on a value like: - "spec.containers{name}" (where "name" refers to the name of the container that triggered - the event) or if no container name is specified "spec.containers[2]" (container with - index 2 in this pod). This syntax is chosen only to have some well-defined way of - referencing a part of an object. - TODO: this design is not final and this field is subject to change in the future. - type: string - kind: - description: |- - Kind of the referent. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - name: - description: |- - Name of the referent. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - namespace: - description: |- - Namespace of the referent. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/ - type: string - resourceVersion: - description: |- - Specific resourceVersion to which this reference is made, if any. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#concurrency-control-and-consistency - type: string - uid: - description: |- - UID of the referent. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#uids - type: string - type: object - x-kubernetes-map-type: atomic - repoIdentifier: - description: RepoIdentifier is the backup repository identifier. - type: string - tags: - additionalProperties: - type: string - description: |- - Tags are a map of key-value pairs that should be applied to the - volume backup as tags. - type: object - uploaderSettings: - additionalProperties: - type: string - description: |- - UploaderSettings are a map of key-value pairs that should be applied to the - uploader configuration. - nullable: true - type: object - uploaderType: - description: UploaderType is the type of the uploader to handle the - data transfer. - enum: - - kopia - - restic - - "" - type: string - volume: - description: |- - Volume is the name of the volume within the Pod to be backed - up. - type: string - required: - - backupStorageLocation - - node - - pod - - repoIdentifier - - volume - type: object - status: - description: PodVolumeBackupStatus is the current status of a PodVolumeBackup. - properties: - completionTimestamp: - description: |- - CompletionTimestamp records the time a backup was completed. - Completion time is recorded even on failed backups. - Completion time is recorded before uploading the backup object. - The server's time is used for CompletionTimestamps - format: date-time - nullable: true - type: string - message: - description: Message is a message about the pod volume backup's status. - type: string - path: - description: Path is the full path within the controller pod being - backed up. - type: string - phase: - description: Phase is the current state of the PodVolumeBackup. - enum: - - New - - InProgress - - Completed - - Failed - type: string - progress: - description: |- - Progress holds the total number of bytes of the volume and the current - number of backed up bytes. This can be used to display progress information - about the backup operation. - properties: - bytesDone: - format: int64 - type: integer - totalBytes: - format: int64 - type: integer - type: object - snapshotID: - description: SnapshotID is the identifier for the snapshot of the - pod volume. - type: string - startTimestamp: - description: |- - StartTimestamp records the time a backup was started. - Separate from CreationTimestamp, since that value changes - on restores. - The server's time is used for StartTimestamps - format: date-time - nullable: true - type: string - type: object - type: object - served: true - storage: true - subresources: {} \ No newline at end of file diff --git a/charts/velero/velero/crds/podvolumerestores.yaml b/charts/velero/velero/crds/podvolumerestores.yaml deleted file mode 100644 index 48296ac..0000000 --- a/charts/velero/velero/crds/podvolumerestores.yaml +++ /dev/null @@ -1,211 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - labels: - component: velero - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: podvolumerestores.velero.io -spec: - group: velero.io - names: - kind: PodVolumeRestore - listKind: PodVolumeRestoreList - plural: podvolumerestores - singular: podvolumerestore - scope: Namespaced - versions: - - additionalPrinterColumns: - - description: Namespace of the pod containing the volume to be restored - jsonPath: .spec.pod.namespace - name: Namespace - type: string - - description: Name of the pod containing the volume to be restored - jsonPath: .spec.pod.name - name: Pod - type: string - - description: The type of the uploader to handle data transfer - jsonPath: .spec.uploaderType - name: Uploader Type - type: string - - description: Name of the volume to be restored - jsonPath: .spec.volume - name: Volume - type: string - - description: Pod Volume Restore status such as New/InProgress - jsonPath: .status.phase - name: Status - type: string - - description: Pod Volume Restore status such as New/InProgress - format: int64 - jsonPath: .status.progress.totalBytes - name: TotalBytes - type: integer - - description: Pod Volume Restore status such as New/InProgress - format: int64 - jsonPath: .status.progress.bytesDone - name: BytesDone - type: integer - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1 - schema: - openAPIV3Schema: - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: PodVolumeRestoreSpec is the specification for a PodVolumeRestore. - properties: - backupStorageLocation: - description: |- - BackupStorageLocation is the name of the backup storage location - where the backup repository is stored. - type: string - pod: - description: Pod is a reference to the pod containing the volume to - be restored. - properties: - apiVersion: - description: API version of the referent. - type: string - fieldPath: - description: |- - If referring to a piece of an object instead of an entire object, this string - should contain a valid JSON/Go field access statement, such as desiredState.manifest.containers[2]. - For example, if the object reference is to a container within a pod, this would take on a value like: - "spec.containers{name}" (where "name" refers to the name of the container that triggered - the event) or if no container name is specified "spec.containers[2]" (container with - index 2 in this pod). This syntax is chosen only to have some well-defined way of - referencing a part of an object. - TODO: this design is not final and this field is subject to change in the future. - type: string - kind: - description: |- - Kind of the referent. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - name: - description: |- - Name of the referent. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - type: string - namespace: - description: |- - Namespace of the referent. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/ - type: string - resourceVersion: - description: |- - Specific resourceVersion to which this reference is made, if any. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#concurrency-control-and-consistency - type: string - uid: - description: |- - UID of the referent. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#uids - type: string - type: object - x-kubernetes-map-type: atomic - repoIdentifier: - description: RepoIdentifier is the backup repository identifier. - type: string - snapshotID: - description: SnapshotID is the ID of the volume snapshot to be restored. - type: string - sourceNamespace: - description: SourceNamespace is the original namespace for namaspace - mapping. - type: string - uploaderSettings: - additionalProperties: - type: string - description: |- - UploaderSettings are a map of key-value pairs that should be applied to the - uploader configuration. - nullable: true - type: object - uploaderType: - description: UploaderType is the type of the uploader to handle the - data transfer. - enum: - - kopia - - restic - - "" - type: string - volume: - description: Volume is the name of the volume within the Pod to be - restored. - type: string - required: - - backupStorageLocation - - pod - - repoIdentifier - - snapshotID - - sourceNamespace - - volume - type: object - status: - description: PodVolumeRestoreStatus is the current status of a PodVolumeRestore. - properties: - completionTimestamp: - description: |- - CompletionTimestamp records the time a restore was completed. - Completion time is recorded even on failed restores. - The server's time is used for CompletionTimestamps - format: date-time - nullable: true - type: string - message: - description: Message is a message about the pod volume restore's status. - type: string - phase: - description: Phase is the current state of the PodVolumeRestore. - enum: - - New - - InProgress - - Completed - - Failed - type: string - progress: - description: |- - Progress holds the total number of bytes of the snapshot and the current - number of restored bytes. This can be used to display progress information - about the restore operation. - properties: - bytesDone: - format: int64 - type: integer - totalBytes: - format: int64 - type: integer - type: object - startTimestamp: - description: |- - StartTimestamp records the time a restore was started. - The server's time is used for StartTimestamps - format: date-time - nullable: true - type: string - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/velero/velero/crds/restores.yaml b/charts/velero/velero/crds/restores.yaml deleted file mode 100644 index 43b52b7..0000000 --- a/charts/velero/velero/crds/restores.yaml +++ /dev/null @@ -1,558 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - labels: - component: velero - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: restores.velero.io -spec: - group: velero.io - names: - kind: Restore - listKind: RestoreList - plural: restores - singular: restore - scope: Namespaced - versions: - - name: v1 - schema: - openAPIV3Schema: - description: |- - Restore is a Velero resource that represents the application of - resources from a Velero backup to a target Kubernetes cluster. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: RestoreSpec defines the specification for a Velero restore. - properties: - backupName: - description: |- - BackupName is the unique name of the Velero backup to restore - from. - type: string - excludedNamespaces: - description: |- - ExcludedNamespaces contains a list of namespaces that are not - included in the restore. - items: - type: string - nullable: true - type: array - excludedResources: - description: |- - ExcludedResources is a slice of resource names that are not - included in the restore. - items: - type: string - nullable: true - type: array - existingResourcePolicy: - description: ExistingResourcePolicy specifies the restore behavior - for the Kubernetes resource to be restored - nullable: true - type: string - hooks: - description: Hooks represent custom behaviors that should be executed - during or post restore. - properties: - resources: - items: - description: |- - RestoreResourceHookSpec defines one or more RestoreResrouceHooks that should be executed based on - the rules defined for namespaces, resources, and label selector. - properties: - excludedNamespaces: - description: ExcludedNamespaces specifies the namespaces - to which this hook spec does not apply. - items: - type: string - nullable: true - type: array - excludedResources: - description: ExcludedResources specifies the resources to - which this hook spec does not apply. - items: - type: string - nullable: true - type: array - includedNamespaces: - description: |- - IncludedNamespaces specifies the namespaces to which this hook spec applies. If empty, it applies - to all namespaces. - items: - type: string - nullable: true - type: array - includedResources: - description: |- - IncludedResources specifies the resources to which this hook spec applies. If empty, it applies - to all resources. - items: - type: string - nullable: true - type: array - labelSelector: - description: LabelSelector, if specified, filters the resources - to which this hook spec applies. - nullable: true - properties: - matchExpressions: - description: matchExpressions is a list of label selector - requirements. The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector - applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - required: - - key - - operator - type: object - type: array - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - name: - description: Name is the name of this hook. - type: string - postHooks: - description: PostHooks is a list of RestoreResourceHooks - to execute during and after restoring a resource. - items: - description: RestoreResourceHook defines a restore hook - for a resource. - properties: - exec: - description: Exec defines an exec restore hook. - properties: - command: - description: Command is the command and arguments - to execute from within a container after a pod - has been restored. - items: - type: string - minItems: 1 - type: array - container: - description: |- - Container is the container in the pod where the command should be executed. If not specified, - the pod's first container is used. - type: string - execTimeout: - description: |- - ExecTimeout defines the maximum amount of time Velero should wait for the hook to complete before - considering the execution a failure. - type: string - onError: - description: OnError specifies how Velero should - behave if it encounters an error executing this - hook. - enum: - - Continue - - Fail - type: string - waitForReady: - description: WaitForReady ensures command will - be launched when container is Ready instead - of Running. - nullable: true - type: boolean - waitTimeout: - description: |- - WaitTimeout defines the maximum amount of time Velero should wait for the container to be Ready - before attempting to run the command. - type: string - required: - - command - type: object - init: - description: Init defines an init restore hook. - properties: - initContainers: - description: InitContainers is list of init containers - to be added to a pod during its restore. - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - x-kubernetes-preserve-unknown-fields: true - timeout: - description: Timeout defines the maximum amount - of time Velero should wait for the initContainers - to complete. - type: string - type: object - type: object - type: array - required: - - name - type: object - type: array - type: object - includeClusterResources: - description: |- - IncludeClusterResources specifies whether cluster-scoped resources - should be included for consideration in the restore. If null, defaults - to true. - nullable: true - type: boolean - includedNamespaces: - description: |- - IncludedNamespaces is a slice of namespace names to include objects - from. If empty, all namespaces are included. - items: - type: string - nullable: true - type: array - includedResources: - description: |- - IncludedResources is a slice of resource names to include - in the restore. If empty, all resources in the backup are included. - items: - type: string - nullable: true - type: array - itemOperationTimeout: - description: |- - ItemOperationTimeout specifies the time used to wait for RestoreItemAction operations - The default value is 4 hour. - type: string - labelSelector: - description: |- - LabelSelector is a metav1.LabelSelector to filter with - when restoring individual objects from the backup. If empty - or nil, all objects are included. Optional. - nullable: true - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. - The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector applies - to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - required: - - key - - operator - type: object - type: array - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - namespaceMapping: - additionalProperties: - type: string - description: |- - NamespaceMapping is a map of source namespace names - to target namespace names to restore into. Any source - namespaces not included in the map will be restored into - namespaces of the same name. - type: object - orLabelSelectors: - description: |- - OrLabelSelectors is list of metav1.LabelSelector to filter with - when restoring individual objects from the backup. If multiple provided - they will be joined by the OR operator. LabelSelector as well as - OrLabelSelectors cannot co-exist in restore request, only one of them - can be used - items: - description: |- - A label selector is a label query over a set of resources. The result of matchLabels and - matchExpressions are ANDed. An empty label selector matches all objects. A null - label selector matches no objects. - properties: - matchExpressions: - description: matchExpressions is a list of label selector requirements. - The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector applies - to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - required: - - key - - operator - type: object - type: array - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - nullable: true - type: array - preserveNodePorts: - description: PreserveNodePorts specifies whether to restore old nodePorts - from backup. - nullable: true - type: boolean - resourceModifier: - description: ResourceModifier specifies the reference to JSON resource - patches that should be applied to resources before restoration. - nullable: true - properties: - apiGroup: - description: |- - APIGroup is the group for the resource being referenced. - If APIGroup is not specified, the specified Kind must be in the core API group. - For any other third-party types, APIGroup is required. - type: string - kind: - description: Kind is the type of resource being referenced - type: string - name: - description: Name is the name of resource being referenced - type: string - required: - - kind - - name - type: object - x-kubernetes-map-type: atomic - restorePVs: - description: |- - RestorePVs specifies whether to restore all included - PVs from snapshot - nullable: true - type: boolean - restoreStatus: - description: |- - RestoreStatus specifies which resources we should restore the status - field. If nil, no objects are included. Optional. - nullable: true - properties: - excludedResources: - description: ExcludedResources specifies the resources to which - will not restore the status. - items: - type: string - nullable: true - type: array - includedResources: - description: |- - IncludedResources specifies the resources to which will restore the status. - If empty, it applies to all resources. - items: - type: string - nullable: true - type: array - type: object - scheduleName: - description: |- - ScheduleName is the unique name of the Velero schedule to restore - from. If specified, and BackupName is empty, Velero will restore - from the most recent successful backup created from this schedule. - type: string - uploaderConfig: - description: UploaderConfig specifies the configuration for the restore. - nullable: true - properties: - parallelFilesDownload: - description: ParallelFilesDownload is the concurrency number setting - for restore. - type: integer - writeSparseFiles: - description: WriteSparseFiles is a flag to indicate whether write - files sparsely or not. - nullable: true - type: boolean - type: object - type: object - status: - description: RestoreStatus captures the current status of a Velero restore - properties: - completionTimestamp: - description: |- - CompletionTimestamp records the time the restore operation was completed. - Completion time is recorded even on failed restore. - The server's time is used for StartTimestamps - format: date-time - nullable: true - type: string - errors: - description: |- - Errors is a count of all error messages that were generated during - execution of the restore. The actual errors are stored in object storage. - type: integer - failureReason: - description: FailureReason is an error that caused the entire restore - to fail. - type: string - hookStatus: - description: HookStatus contains information about the status of the - hooks. - nullable: true - properties: - hooksAttempted: - description: |- - HooksAttempted is the total number of attempted hooks - Specifically, HooksAttempted represents the number of hooks that failed to execute - and the number of hooks that executed successfully. - type: integer - hooksFailed: - description: HooksFailed is the total number of hooks which ended - with an error - type: integer - type: object - phase: - description: Phase is the current state of the Restore - enum: - - New - - FailedValidation - - InProgress - - WaitingForPluginOperations - - WaitingForPluginOperationsPartiallyFailed - - Completed - - PartiallyFailed - - Failed - - Finalizing - - FinalizingPartiallyFailed - type: string - progress: - description: |- - Progress contains information about the restore's execution progress. Note - that this information is best-effort only -- if Velero fails to update it - during a restore for any reason, it may be inaccurate/stale. - nullable: true - properties: - itemsRestored: - description: ItemsRestored is the number of items that have actually - been restored so far - type: integer - totalItems: - description: |- - TotalItems is the total number of items to be restored. This number may change - throughout the execution of the restore due to plugins that return additional related - items to restore - type: integer - type: object - restoreItemOperationsAttempted: - description: |- - RestoreItemOperationsAttempted is the total number of attempted - async RestoreItemAction operations for this restore. - type: integer - restoreItemOperationsCompleted: - description: |- - RestoreItemOperationsCompleted is the total number of successfully completed - async RestoreItemAction operations for this restore. - type: integer - restoreItemOperationsFailed: - description: |- - RestoreItemOperationsFailed is the total number of async - RestoreItemAction operations for this restore which ended with an error. - type: integer - startTimestamp: - description: |- - StartTimestamp records the time the restore operation was started. - The server's time is used for StartTimestamps - format: date-time - nullable: true - type: string - validationErrors: - description: |- - ValidationErrors is a slice of all validation errors (if - applicable) - items: - type: string - nullable: true - type: array - warnings: - description: |- - Warnings is a count of all warning messages that were generated during - execution of the restore. The actual warnings are stored in object storage. - type: integer - type: object - type: object - served: true - storage: true \ No newline at end of file diff --git a/charts/velero/velero/crds/schedules.yaml b/charts/velero/velero/crds/schedules.yaml deleted file mode 100644 index 4d6889d..0000000 --- a/charts/velero/velero/crds/schedules.yaml +++ /dev/null @@ -1,599 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - labels: - component: velero - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: schedules.velero.io -spec: - group: velero.io - names: - kind: Schedule - listKind: ScheduleList - plural: schedules - singular: schedule - scope: Namespaced - versions: - - additionalPrinterColumns: - - description: Status of the schedule - jsonPath: .status.phase - name: Status - type: string - - description: A Cron expression defining when to run the Backup - jsonPath: .spec.schedule - name: Schedule - type: string - - description: The last time a Backup was run for this schedule - jsonPath: .status.lastBackup - name: LastBackup - type: date - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .spec.paused - name: Paused - type: boolean - name: v1 - schema: - openAPIV3Schema: - description: |- - Schedule is a Velero resource that represents a pre-scheduled or - periodic Backup that should be run. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ScheduleSpec defines the specification for a Velero schedule - properties: - paused: - description: Paused specifies whether the schedule is paused or not - type: boolean - schedule: - description: |- - Schedule is a Cron expression defining when to run - the Backup. - type: string - skipImmediately: - description: |- - SkipImmediately specifies whether to skip backup if schedule is due immediately from `schedule.status.lastBackup` timestamp when schedule is unpaused or if schedule is new. - If true, backup will be skipped immediately when schedule is unpaused if it is due based on .Status.LastBackupTimestamp or schedule is new, and will run at next schedule time. - If false, backup will not be skipped immediately when schedule is unpaused, but will run at next schedule time. - If empty, will follow server configuration (default: false). - type: boolean - template: - description: |- - Template is the definition of the Backup to be run - on the provided schedule - properties: - csiSnapshotTimeout: - description: |- - CSISnapshotTimeout specifies the time used to wait for CSI VolumeSnapshot status turns to - ReadyToUse during creation, before returning error as timeout. - The default value is 10 minute. - type: string - datamover: - description: |- - DataMover specifies the data mover to be used by the backup. - If DataMover is "" or "velero", the built-in data mover will be used. - type: string - defaultVolumesToFsBackup: - description: |- - DefaultVolumesToFsBackup specifies whether pod volume file system backup should be used - for all volumes by default. - nullable: true - type: boolean - defaultVolumesToRestic: - description: |- - DefaultVolumesToRestic specifies whether restic should be used to take a - backup of all pod volumes by default. - - - Deprecated: this field is no longer used and will be removed entirely in future. Use DefaultVolumesToFsBackup instead. - nullable: true - type: boolean - excludedClusterScopedResources: - description: |- - ExcludedClusterScopedResources is a slice of cluster-scoped - resource type names to exclude from the backup. - If set to "*", all cluster-scoped resource types are excluded. - The default value is empty. - items: - type: string - nullable: true - type: array - excludedNamespaceScopedResources: - description: |- - ExcludedNamespaceScopedResources is a slice of namespace-scoped - resource type names to exclude from the backup. - If set to "*", all namespace-scoped resource types are excluded. - The default value is empty. - items: - type: string - nullable: true - type: array - excludedNamespaces: - description: |- - ExcludedNamespaces contains a list of namespaces that are not - included in the backup. - items: - type: string - nullable: true - type: array - excludedResources: - description: |- - ExcludedResources is a slice of resource names that are not - included in the backup. - items: - type: string - nullable: true - type: array - hooks: - description: Hooks represent custom behaviors that should be executed - at different phases of the backup. - properties: - resources: - description: Resources are hooks that should be executed when - backing up individual instances of a resource. - items: - description: |- - BackupResourceHookSpec defines one or more BackupResourceHooks that should be executed based on - the rules defined for namespaces, resources, and label selector. - properties: - excludedNamespaces: - description: ExcludedNamespaces specifies the namespaces - to which this hook spec does not apply. - items: - type: string - nullable: true - type: array - excludedResources: - description: ExcludedResources specifies the resources - to which this hook spec does not apply. - items: - type: string - nullable: true - type: array - includedNamespaces: - description: |- - IncludedNamespaces specifies the namespaces to which this hook spec applies. If empty, it applies - to all namespaces. - items: - type: string - nullable: true - type: array - includedResources: - description: |- - IncludedResources specifies the resources to which this hook spec applies. If empty, it applies - to all resources. - items: - type: string - nullable: true - type: array - labelSelector: - description: LabelSelector, if specified, filters the - resources to which this hook spec applies. - nullable: true - properties: - matchExpressions: - description: matchExpressions is a list of label - selector requirements. The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the - selector applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - required: - - key - - operator - type: object - type: array - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - name: - description: Name is the name of this hook. - type: string - post: - description: |- - PostHooks is a list of BackupResourceHooks to execute after storing the item in the backup. - These are executed after all "additional items" from item actions are processed. - items: - description: BackupResourceHook defines a hook for - a resource. - properties: - exec: - description: Exec defines an exec hook. - properties: - command: - description: Command is the command and arguments - to execute. - items: - type: string - minItems: 1 - type: array - container: - description: |- - Container is the container in the pod where the command should be executed. If not specified, - the pod's first container is used. - type: string - onError: - description: OnError specifies how Velero - should behave if it encounters an error - executing this hook. - enum: - - Continue - - Fail - type: string - timeout: - description: |- - Timeout defines the maximum amount of time Velero should wait for the hook to complete before - considering the execution a failure. - type: string - required: - - command - type: object - required: - - exec - type: object - type: array - pre: - description: |- - PreHooks is a list of BackupResourceHooks to execute prior to storing the item in the backup. - These are executed before any "additional items" from item actions are processed. - items: - description: BackupResourceHook defines a hook for - a resource. - properties: - exec: - description: Exec defines an exec hook. - properties: - command: - description: Command is the command and arguments - to execute. - items: - type: string - minItems: 1 - type: array - container: - description: |- - Container is the container in the pod where the command should be executed. If not specified, - the pod's first container is used. - type: string - onError: - description: OnError specifies how Velero - should behave if it encounters an error - executing this hook. - enum: - - Continue - - Fail - type: string - timeout: - description: |- - Timeout defines the maximum amount of time Velero should wait for the hook to complete before - considering the execution a failure. - type: string - required: - - command - type: object - required: - - exec - type: object - type: array - required: - - name - type: object - nullable: true - type: array - type: object - includeClusterResources: - description: |- - IncludeClusterResources specifies whether cluster-scoped resources - should be included for consideration in the backup. - nullable: true - type: boolean - includedClusterScopedResources: - description: |- - IncludedClusterScopedResources is a slice of cluster-scoped - resource type names to include in the backup. - If set to "*", all cluster-scoped resource types are included. - The default value is empty, which means only related - cluster-scoped resources are included. - items: - type: string - nullable: true - type: array - includedNamespaceScopedResources: - description: |- - IncludedNamespaceScopedResources is a slice of namespace-scoped - resource type names to include in the backup. - The default value is "*". - items: - type: string - nullable: true - type: array - includedNamespaces: - description: |- - IncludedNamespaces is a slice of namespace names to include objects - from. If empty, all namespaces are included. - items: - type: string - nullable: true - type: array - includedResources: - description: |- - IncludedResources is a slice of resource names to include - in the backup. If empty, all resources are included. - items: - type: string - nullable: true - type: array - itemOperationTimeout: - description: |- - ItemOperationTimeout specifies the time used to wait for asynchronous BackupItemAction operations - The default value is 4 hour. - type: string - labelSelector: - description: |- - LabelSelector is a metav1.LabelSelector to filter with - when adding individual objects to the backup. If empty - or nil, all objects are included. Optional. - nullable: true - properties: - matchExpressions: - description: matchExpressions is a list of label selector - requirements. The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector - applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - required: - - key - - operator - type: object - type: array - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - metadata: - properties: - labels: - additionalProperties: - type: string - type: object - type: object - orLabelSelectors: - description: |- - OrLabelSelectors is list of metav1.LabelSelector to filter with - when adding individual objects to the backup. If multiple provided - they will be joined by the OR operator. LabelSelector as well as - OrLabelSelectors cannot co-exist in backup request, only one of them - can be used. - items: - description: |- - A label selector is a label query over a set of resources. The result of matchLabels and - matchExpressions are ANDed. An empty label selector matches all objects. A null - label selector matches no objects. - properties: - matchExpressions: - description: matchExpressions is a list of label selector - requirements. The requirements are ANDed. - items: - description: |- - A label selector requirement is a selector that contains values, a key, and an operator that - relates the key and values. - properties: - key: - description: key is the label key that the selector - applies to. - type: string - operator: - description: |- - operator represents a key's relationship to a set of values. - Valid operators are In, NotIn, Exists and DoesNotExist. - type: string - values: - description: |- - values is an array of string values. If the operator is In or NotIn, - the values array must be non-empty. If the operator is Exists or DoesNotExist, - the values array must be empty. This array is replaced during a strategic - merge patch. - items: - type: string - type: array - required: - - key - - operator - type: object - type: array - matchLabels: - additionalProperties: - type: string - description: |- - matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - x-kubernetes-map-type: atomic - nullable: true - type: array - orderedResources: - additionalProperties: - type: string - description: |- - OrderedResources specifies the backup order of resources of specific Kind. - The map key is the resource name and value is a list of object names separated by commas. - Each resource name has format "namespace/objectname". For cluster resources, simply use "objectname". - nullable: true - type: object - resourcePolicy: - description: ResourcePolicy specifies the referenced resource - policies that backup should follow - properties: - apiGroup: - description: |- - APIGroup is the group for the resource being referenced. - If APIGroup is not specified, the specified Kind must be in the core API group. - For any other third-party types, APIGroup is required. - type: string - kind: - description: Kind is the type of resource being referenced - type: string - name: - description: Name is the name of resource being referenced - type: string - required: - - kind - - name - type: object - x-kubernetes-map-type: atomic - snapshotMoveData: - description: SnapshotMoveData specifies whether snapshot data - should be moved - nullable: true - type: boolean - snapshotVolumes: - description: |- - SnapshotVolumes specifies whether to take snapshots - of any PV's referenced in the set of objects included - in the Backup. - nullable: true - type: boolean - storageLocation: - description: StorageLocation is a string containing the name of - a BackupStorageLocation where the backup should be stored. - type: string - ttl: - description: |- - TTL is a time.Duration-parseable string describing how long - the Backup should be retained for. - type: string - uploaderConfig: - description: UploaderConfig specifies the configuration for the - uploader. - nullable: true - properties: - parallelFilesUpload: - description: ParallelFilesUpload is the number of files parallel - uploads to perform when using the uploader. - type: integer - type: object - volumeSnapshotLocations: - description: VolumeSnapshotLocations is a list containing names - of VolumeSnapshotLocations associated with this backup. - items: - type: string - type: array - type: object - useOwnerReferencesInBackup: - description: |- - UseOwnerReferencesBackup specifies whether to use - OwnerReferences on backups created by this Schedule. - nullable: true - type: boolean - required: - - schedule - - template - type: object - status: - description: ScheduleStatus captures the current state of a Velero schedule - properties: - lastBackup: - description: |- - LastBackup is the last time a Backup was run for this - Schedule schedule - format: date-time - nullable: true - type: string - lastSkipped: - description: LastSkipped is the last time a Schedule was skipped - format: date-time - nullable: true - type: string - phase: - description: Phase is the current phase of the Schedule - enum: - - New - - Enabled - - FailedValidation - type: string - validationErrors: - description: |- - ValidationErrors is a slice of all validation errors (if - applicable) - items: - type: string - type: array - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/velero/velero/crds/serverstatusrequests.yaml b/charts/velero/velero/crds/serverstatusrequests.yaml deleted file mode 100644 index 234d40c..0000000 --- a/charts/velero/velero/crds/serverstatusrequests.yaml +++ /dev/null @@ -1,86 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - labels: - component: velero - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: serverstatusrequests.velero.io -spec: - group: velero.io - names: - kind: ServerStatusRequest - listKind: ServerStatusRequestList - plural: serverstatusrequests - shortNames: - - ssr - singular: serverstatusrequest - scope: Namespaced - versions: - - name: v1 - schema: - openAPIV3Schema: - description: |- - ServerStatusRequest is a request to access current status information about - the Velero server. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ServerStatusRequestSpec is the specification for a ServerStatusRequest. - type: object - status: - description: ServerStatusRequestStatus is the current status of a ServerStatusRequest. - properties: - phase: - description: Phase is the current lifecycle phase of the ServerStatusRequest. - enum: - - New - - Processed - type: string - plugins: - description: Plugins list information about the plugins running on - the Velero server - items: - description: PluginInfo contains attributes of a Velero plugin - properties: - kind: - type: string - name: - type: string - required: - - kind - - name - type: object - nullable: true - type: array - processedTimestamp: - description: |- - ProcessedTimestamp is when the ServerStatusRequest was processed - by the ServerStatusRequestController. - format: date-time - nullable: true - type: string - serverVersion: - description: ServerVersion is the Velero server version. - type: string - type: object - type: object - served: true - storage: true diff --git a/charts/velero/velero/crds/volumesnapshotlocations.yaml b/charts/velero/velero/crds/volumesnapshotlocations.yaml deleted file mode 100644 index cb349fe..0000000 --- a/charts/velero/velero/crds/volumesnapshotlocations.yaml +++ /dev/null @@ -1,94 +0,0 @@ ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - labels: - component: velero - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - name: volumesnapshotlocations.velero.io -spec: - group: velero.io - names: - kind: VolumeSnapshotLocation - listKind: VolumeSnapshotLocationList - plural: volumesnapshotlocations - shortNames: - - vsl - singular: volumesnapshotlocation - scope: Namespaced - versions: - - name: v1 - schema: - openAPIV3Schema: - description: VolumeSnapshotLocation is a location where Velero stores volume - snapshots. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: VolumeSnapshotLocationSpec defines the specification for - a Velero VolumeSnapshotLocation. - properties: - config: - additionalProperties: - type: string - description: Config is for provider-specific configuration fields. - type: object - credential: - description: Credential contains the credential information intended - to be used with this location - properties: - key: - description: The key of the secret to select from. Must be a - valid secret key. - type: string - name: - description: |- - Name of the referent. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names - TODO: Add other useful fields. apiVersion, kind, uid? - type: string - optional: - description: Specify whether the Secret or its key must be defined - type: boolean - required: - - key - type: object - x-kubernetes-map-type: atomic - provider: - description: Provider is the provider of the volume storage. - type: string - required: - - provider - type: object - status: - description: VolumeSnapshotLocationStatus describes the current status - of a Velero VolumeSnapshotLocation. - properties: - phase: - description: VolumeSnapshotLocationPhase is the lifecycle phase of - a Velero VolumeSnapshotLocation. - enum: - - Available - - Unavailable - type: string - type: object - type: object - served: true - storage: true \ No newline at end of file diff --git a/charts/velero/velero/templates/NOTES.txt b/charts/velero/velero/templates/NOTES.txt deleted file mode 100644 index 940c6be..0000000 --- a/charts/velero/velero/templates/NOTES.txt +++ /dev/null @@ -1,79 +0,0 @@ -Check that the velero is up and running: - - kubectl get deployment/{{ include "velero.fullname" . }} -n {{ .Release.Namespace }} - -Check that the secret has been created: - - kubectl get secret/{{ include "velero.fullname" . }} -n {{ .Release.Namespace }} - -Once velero server is up and running you need the client before you can use it -1. wget https://github.com/vmware-tanzu/velero/releases/download/{{ .Values.image.tag }}/velero-{{ .Values.image.tag }}-darwin-amd64.tar.gz -2. tar -xvf velero-{{ .Values.image.tag }}-darwin-amd64.tar.gz -C velero-client - -More info on the official site: https://velero.io/docs - -{{- /* - Breaking changes. -*/}} - -{{- $breaking := "" }} -{{- $breaking_title := "\n" }} -{{- $breaking_title = print $breaking_title "\n#################################################################################" }} -{{- $breaking_title = print $breaking_title "\n###### BREAKING: The config values passed contained no longer accepted #####" }} -{{- $breaking_title = print $breaking_title "\n###### options. See the messages below for more details. #####" }} -{{- $breaking_title = print $breaking_title "\n###### #####" }} -{{- $breaking_title = print $breaking_title "\n###### To verify your updated config is accepted, you can use #####" }} -{{- $breaking_title = print $breaking_title "\n###### the `helm template` command. #####" }} -{{- $breaking_title = print $breaking_title "\n#################################################################################" }} - -{{- if typeIs "map[string]interface {}" .Values.configuration.backupStorageLocation }} -{{- $breaking = print $breaking "\n\nERROR: Please make .configuration.backupStorageLocation from map to slice" }} -{{- end }} - -{{- if typeIs "map[string]interface {}" .Values.configuration.volumeSnapshotLocation }} -{{- $breaking = print $breaking "\n\nERROR: Please make .configuration.volumeSnapshotLocation from map to slice" }} -{{- end }} - -{{- if hasKey .Values.configuration "provider" }} -{{- $breaking = print $breaking "\n\nREMOVED: .configuration.provider has been removed, instead each backupStorageLocation and volumeSnapshotLocation has a provider configured" }} -{{- end }} - -{{- if hasKey .Values "resticTimeout" }} -{{- $breaking = print $breaking "\n\nREMOVED: resticTimeout has been removed, and it is named fsBackupTimeout" }} -{{- end }} - -{{- if hasKey .Values "defaultVolumesToRestic" }} -{{- $breaking = print $breaking "\n\nREMOVED: defaultVolumesToRestic has been removed, and it is named defaultVolumesToFsBackup" }} -{{- end }} - -{{- if hasKey .Values "defaultResticPruneFrequency" }} -{{- $breaking = print $breaking "\n\nREMOVED: defaultResticPruneFrequency has been removed, and it is named defaultRepoMaintainFrequency" }} -{{- end }} - -{{- if hasKey .Values "deployRestic" }} -{{- $breaking = print $breaking "\n\nREMOVED: deployRestic has been removed, and it is named deployNodeAgent" }} -{{- end }} - -{{- if hasKey .Values "restic" }} -{{- $breaking = print $breaking "\n\nREMOVED: restic has been removed, and it is named nodeAgent" }} -{{- end }} - -{{- if hasKey .Values.configMaps "restic-restore-action-config" }} -{{- $breaking = print $breaking "\n\nREMOVED: restic-restore-action-config has been removed, and it is named fs-restore-action-config" }} -{{- end }} - -{{- range $key, $value := .Values.configMaps }} -{{- eq $key "fs-restore-action-config" }} -{{- if hasKey $value.labels "velero.io/restic" }} -{{- $breaking = print $breaking "\n\nREMOVED: velero.io/restic has been removed, and it is named velero.io/pod-volume-restore" }} -{{- end }} -{{- if and $value.data.image }} -{{- if contains "velero-restic-restore-helper" $value.data.image }} -{{- $breaking = print $breaking "\n\nREMOVED: restore helper image velero-restic-restore-helper has been changed to velero-restore-helper" }} -{{- end }} -{{- end }} -{{- end }} - -{{- if $breaking }} -{{- fail (print $breaking_title $breaking) }} -{{- end }} diff --git a/charts/velero/velero/templates/_helpers.tpl b/charts/velero/velero/templates/_helpers.tpl deleted file mode 100644 index 00d97db..0000000 --- a/charts/velero/velero/templates/_helpers.tpl +++ /dev/null @@ -1,96 +0,0 @@ -{{/* vim: set filetype=mustache: */}} -{{/* -Expand the name of the chart. -*/}} -{{- define "velero.name" -}} -{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Create a default fully qualified app name. -We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). -If release name contains chart name it will be used as a full name. -*/}} -{{- define "velero.fullname" -}} -{{- if .Values.fullnameOverride -}} -{{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" -}} -{{- else -}} -{{- $name := default .Chart.Name .Values.nameOverride -}} -{{- if contains $name .Release.Name -}} -{{- .Release.Name | trunc 63 | trimSuffix "-" -}} -{{- else -}} -{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} -{{- end -}} -{{- end -}} -{{- end -}} - -{{/* -Create chart name and version as used by the chart label. -*/}} -{{- define "velero.chart" -}} -{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} -{{- end -}} - -{{/* -Create the name of the service account to use for creating or deleting the velero server -*/}} -{{- define "velero.serverServiceAccount" -}} -{{- if .Values.serviceAccount.server.create -}} - {{ default (printf "%s-%s" (include "velero.fullname" .) "server") .Values.serviceAccount.server.name }} -{{- else -}} - {{ default "default" .Values.serviceAccount.server.name }} -{{- end -}} -{{- end -}} - -{{/* -Create the name for the credentials secret. -*/}} -{{- define "velero.secretName" -}} -{{- if .Values.credentials.existingSecret -}} - {{- .Values.credentials.existingSecret -}} -{{- else -}} - {{ default (include "velero.fullname" .) .Values.credentials.name }} -{{- end -}} -{{- end -}} - -{{/* -Create the Velero priority class name. -*/}} -{{- define "velero.priorityClassName" -}} -{{- if .Values.priorityClassName -}} - {{- .Values.priorityClassName -}} -{{- else -}} - {{- include "velero.fullname" . -}} -{{- end -}} -{{- end -}} - -{{/* -Create the node-Agent priority class name. -*/}} -{{- define "velero.nodeAgent.priorityClassName" -}} -{{- if .Values.nodeAgent.priorityClassName -}} - {{- .Values.nodeAgent.priorityClassName -}} -{{- else -}} - {{- include "velero.fullname" . -}} -{{- end -}} -{{- end -}} - -{{/* -Kubernetes version -Built-in object .Capabilities.KubeVersion.Minor can provide non-number output -For examples: -- on GKE it returns "18+" instead of "18" -- on EKS it returns "20+" instead of "20" -*/}} -{{- define "chart.KubernetesVersion" -}} -{{- $minorVersion := .Capabilities.KubeVersion.Minor | regexFind "[0-9]+" -}} -{{- printf "%s.%s" .Capabilities.KubeVersion.Major $minorVersion -}} -{{- end -}} - - -{{/* -Calculate the checksum of the credentials secret. -*/}} -{{- define "chart.config-checksum" -}} -{{- tpl (print .Values.credentials.secretContents .Values.credentials.extraEnvVars ) $ | sha256sum -}} -{{- end -}} diff --git a/charts/velero/velero/templates/backupstoragelocation.yaml b/charts/velero/velero/templates/backupstoragelocation.yaml deleted file mode 100644 index b86cf19..0000000 --- a/charts/velero/velero/templates/backupstoragelocation.yaml +++ /dev/null @@ -1,56 +0,0 @@ -{{- if .Values.backupsEnabled }} - -{{- if typeIs "[]interface {}" .Values.configuration.backupStorageLocation }} -{{- range .Values.configuration.backupStorageLocation }} ---- -apiVersion: velero.io/v1 -kind: BackupStorageLocation -metadata: - name: {{ .name | default "default" }} - namespace: {{ $.Release.Namespace }} - {{- with .annotations }} - annotations: - {{- range $key, $value := . }} - {{- $key | nindent 4 }}: {{ $value | quote }} - {{- end }} - {{- end }} - labels: - app.kubernetes.io/name: {{ include "velero.name" $ }} - app.kubernetes.io/instance: {{ $.Release.Name }} - app.kubernetes.io/managed-by: {{ $.Release.Service }} - helm.sh/chart: {{ include "velero.chart" $ }} -spec: - {{- if not (empty .credential) }} - credential: - {{- with .credential.name }} - name: {{ . }} - {{- end }} - {{- with .credential.key }} - key: {{ . }} - {{- end }} - {{- end }} - provider: {{ .provider }} - accessMode: {{ .accessMode | default "ReadWrite" }} - {{- with .default }} - default: {{ . }} - {{- end }} - {{- with .validationFrequency }} - validationFrequency: {{ . }} - {{- end }} - objectStorage: - bucket: {{ .bucket | quote }} - {{- with .prefix }} - prefix: {{ . | quote }} - {{- end }} - {{- with .caCert }} - caCert: {{ . }} - {{- end }} -{{- with .config }} - config: -{{- range $key, $value := . }} -{{- $key | nindent 4 }}: {{ $value | quote }} -{{- end }} -{{- end }} -{{- end }} -{{- end }} -{{- end }} diff --git a/charts/velero/velero/templates/cleanup-crds.yaml b/charts/velero/velero/templates/cleanup-crds.yaml deleted file mode 100644 index 4e86320..0000000 --- a/charts/velero/velero/templates/cleanup-crds.yaml +++ /dev/null @@ -1,85 +0,0 @@ -{{- if .Values.cleanUpCRDs }} -# This job is meant primarily for cleaning up on CI systems. -# Using this on production systems, especially those that have multiple releases of Velero, will be destructive. -{{/* 'securityContext' got renamed to 'podSecurityContext', merge both dicts into one for backward compatibility */}} -{{- $podSecurityContext := merge (.Values.podSecurityContext | default dict) (.Values.securityContext | default dict) -}} -apiVersion: batch/v1 -kind: Job -metadata: - name: {{ template "velero.fullname" . }}-cleanup-crds - namespace: {{ .Release.Namespace }} - annotations: - "helm.sh/hook": pre-delete - "helm.sh/hook-delete-policy": hook-succeeded - labels: - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} -spec: - backoffLimit: 3 - template: - metadata: - name: velero-cleanup-crds - {{- with .Values.kubectl.labels }} - labels: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.kubectl.annotations }} - annotations: - {{- toYaml . | nindent 8 }} - {{- end }} - spec: - {{- if .Values.image.imagePullSecrets }} - imagePullSecrets: - {{- range .Values.image.imagePullSecrets }} - - name: {{ . }} - {{- end }} - {{- end }} - serviceAccountName: {{ include "velero.serverServiceAccount" . }} - containers: - - name: kubectl - {{- if .Values.kubectl.image.digest }} - image: "{{ .Values.kubectl.image.repository }}@{{ .Values.kubectl.image.digest }}" - {{- else if .Values.kubectl.image.tag }} - image: "{{ .Values.kubectl.image.repository }}:{{ .Values.kubectl.image.tag }}" - {{- else }} - image: "{{ .Values.kubectl.image.repository }}:{{ template "chart.KubernetesVersion" . }}" - {{- end }} - imagePullPolicy: {{ .Values.image.pullPolicy }} - command: - - /bin/sh - - -c - - > - kubectl delete restore --all; - kubectl delete backup --all; - kubectl delete backupstoragelocation --all; - kubectl delete volumesnapshotlocation --all; - kubectl delete podvolumerestore --all; - kubectl delete crd -l component=velero; - {{- with .Values.kubectl.containerSecurityContext }} - securityContext: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- with .Values.kubectl.resources }} - resources: - {{- toYaml . | nindent 12 }} - {{- end }} - restartPolicy: OnFailure - {{- with $podSecurityContext }} - securityContext: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} -{{- end }} diff --git a/charts/velero/velero/templates/clusterrolebinding.yaml b/charts/velero/velero/templates/clusterrolebinding.yaml deleted file mode 100644 index 669289c..0000000 --- a/charts/velero/velero/templates/clusterrolebinding.yaml +++ /dev/null @@ -1,20 +0,0 @@ -{{- if and .Values.rbac.create .Values.rbac.clusterAdministrator }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ include "velero.fullname" . }}-server - labels: - app.kubernetes.io/component: server - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} -subjects: - - kind: ServiceAccount - namespace: {{ .Release.Namespace }} - name: {{ include "velero.serverServiceAccount" . }} -roleRef: - kind: ClusterRole - name: {{ .Values.rbac.clusterAdministratorName }} - apiGroup: rbac.authorization.k8s.io -{{- end }} diff --git a/charts/velero/velero/templates/configmaps.yaml b/charts/velero/velero/templates/configmaps.yaml deleted file mode 100644 index 5840681..0000000 --- a/charts/velero/velero/templates/configmaps.yaml +++ /dev/null @@ -1,18 +0,0 @@ -{{- range $configMapName, $configMap := .Values.configMaps }} -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ include "velero.fullname" $ }}-{{ $configMapName }} - namespace: {{ $.Release.Namespace }} - labels: - app.kubernetes.io/name: {{ include "velero.name" $ }} - app.kubernetes.io/instance: {{ $.Release.Name }} - app.kubernetes.io/managed-by: {{ $.Release.Service }} - helm.sh/chart: {{ include "velero.chart" $ }} - {{- with $configMap.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} -data: - {{- toYaml $configMap.data | nindent 2 }} ---- -{{- end }} diff --git a/charts/velero/velero/templates/deployment.yaml b/charts/velero/velero/templates/deployment.yaml deleted file mode 100644 index 93f8061..0000000 --- a/charts/velero/velero/templates/deployment.yaml +++ /dev/null @@ -1,315 +0,0 @@ -{{/* 'securityContext' got renamed to 'podSecurityContext', merge both dicts into one for backward compatibility */}} -{{- $podSecurityContext := merge (.Values.podSecurityContext | default dict) (.Values.securityContext | default dict) -}} -apiVersion: apps/v1 -kind: Deployment -metadata: - name: {{ include "velero.fullname" . }} - namespace: {{ .Release.Namespace }} - {{- with .Values.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} - labels: - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} - component: velero - {{- with .Values.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - replicas: 1 - {{- if .Values.revisionHistoryLimit }} - revisionHistoryLimit: {{ .Values.revisionHistoryLimit }} - {{- end }} - strategy: - type: Recreate - selector: - matchLabels: - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/name: {{ include "velero.name" . }} - template: - metadata: - labels: - name: velero - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} - {{- if .Values.podLabels }} - {{- toYaml .Values.podLabels | nindent 8 }} - {{- end }} - {{- if or .Values.podAnnotations .Values.metrics.enabled (and .Values.credentials.useSecret (not .Values.credentials.existingSecret)) }} - annotations: - {{- with .Values.podAnnotations }} - {{- toYaml . | nindent 8 }} - {{- end }} - {{- if and (.Values.metrics.enabled) (not .Values.metrics.serviceMonitor.enabled) }} - {{- with .Values.metrics.podAnnotations }} - {{- toYaml . | nindent 8 }} - {{- end }} - {{- end }} - {{- if and .Values.credentials.useSecret (not .Values.credentials.existingSecret) }} - checksum/secret: {{ template "chart.config-checksum" . }} - {{- end }} - {{- end }} - spec: - {{- if .Values.image.imagePullSecrets }} - imagePullSecrets: - {{- range .Values.image.imagePullSecrets }} - - name: {{ . }} - {{- end }} - {{- end }} - restartPolicy: Always - serviceAccountName: {{ include "velero.serverServiceAccount" . }} - {{- if .Values.priorityClassName }} - priorityClassName: {{ include "velero.priorityClassName" . }} - {{- end }} - terminationGracePeriodSeconds: {{ .Values.terminationGracePeriodSeconds }} - containers: - - name: velero - {{- if .Values.image.digest }} - image: "{{ .Values.image.repository }}@{{ .Values.image.digest }}" - {{- else }} - image: "{{ .Values.image.repository }}:{{ .Values.image.tag }}" - {{- end }} - imagePullPolicy: {{ .Values.image.pullPolicy }} - {{- if .Values.metrics.enabled }} - ports: - - name: http-monitoring - containerPort: 8085 - {{- end }} - command: - - /velero - args: - - server - ### Flags - {{- with .Values.configuration }} - - --uploader-type={{ default "kopia" .uploaderType }} - {{- with .backupSyncPeriod }} - - --backup-sync-period={{ . }} - {{- end }} - {{- with .fsBackupTimeout }} - - --fs-backup-timeout={{ . }} - {{- end }} - {{- with .clientBurst }} - - --client-burst={{ . }} - {{- end }} - {{- with .clientPageSize }} - - --client-page-size={{ . }} - {{- end }} - {{- with .clientQPS }} - - --client-qps={{ . }} - {{- end }} - {{- with .defaultBackupStorageLocation }} - - --default-backup-storage-location={{ . }} - {{- end }} - {{- with .defaultBackupTTL }} - - --default-backup-ttl={{ . }} - {{- end }} - {{- with .defaultItemOperationTimeout }} - - --default-item-operation-timeout={{ . }} - {{- end }} - {{- with .defaultVolumeSnapshotLocations }} - - --default-volume-snapshot-locations={{ . }} - {{- end }} - {{- if .defaultVolumesToFsBackup }} - - --default-volumes-to-fs-backup - {{- end }} - {{- with .defaultRepoMaintainFrequency }} - - --default-repo-maintain-frequency={{ . }} - {{- end }} - {{- with .disableControllers }} - - --disable-controllers={{ . }} - {{- end }} - {{- with .disableInformerCache }} - - --disable-informer-cache={{ . }} - {{- end }} - {{- with .garbageCollectionFrequency }} - - --garbage-collection-frequency={{ . }} - {{- end }} - {{- with .logFormat }} - - --log-format={{ . }} - {{- end }} - {{- with .logLevel }} - - --log-level={{ . }} - {{- end }} - {{- with .metricsAddress }} - - --metrics-address={{ . }} - {{- end }} - {{- with .pluginDir }} - - --plugin-dir={{ . }} - {{- end }} - {{- with .profilerAddress }} - - --profiler-address={{ . }} - {{- end }} - {{- if .restoreOnlyMode }} - - --restore-only - {{- end }} - {{- with .restoreResourcePriorities }} - - --restore-resource-priorities={{ . }} - {{- end }} - {{- with .storeValidationFrequency }} - - --store-validation-frequency={{ . }} - {{- end }} - {{- with .terminatingResourceTimeout }} - - --terminating-resource-timeout={{ . }} - {{- end }} - {{- with .defaultSnapshotMoveData }} - - --default-snapshot-move-data={{ . }} - {{- end }} - ### Global Flags - {{- with .features }} - - --features={{ . }} - {{- end }} - {{- with .namespace }} - - --namespace={{ . }} - {{- end }} - {{- with .repositoryMaintenanceJob }} - {{- with .requests }} - {{- with .cpu }} - - --maintenance-job-cpu-request={{ . }} - {{- end }} - {{- with .memory }} - - --maintenance-job-mem-request={{ . }} - {{- end }} - {{- end }} - {{- with .limits }} - {{- with .cpu }} - - --maintenance-job-cpu-limit={{ . }} - {{- end }} - {{- with .memory }} - - --maintenance-job-mem-limit={{ . }} - {{- end }} - {{- end }} - {{- with .latestJobsCount }} - - --keep-latest-maintenance-jobs={{ . }} - {{- end }} - {{- end }} - {{- with .extraArgs }} - ### User-supplied overwrite flags - {{- toYaml . | nindent 12 }} - {{- end }} - {{- end }} - {{- with .Values.resources }} - resources: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- if .Values.metrics.enabled }} - {{- with .Values.livenessProbe }} - livenessProbe: {{- toYaml . | nindent 12 }} - {{- end }} - {{- with .Values.readinessProbe }} - readinessProbe: {{- toYaml . | nindent 12 }} - {{- end }} - {{- end }} - {{- with .Values.containerSecurityContext }} - securityContext: - {{- toYaml . | nindent 12 }} - {{- end }} - volumeMounts: - - name: plugins - mountPath: /plugins - {{- if .Values.credentials.useSecret }} - - name: cloud-credentials - mountPath: /credentials - {{- end }} - - name: scratch - mountPath: /scratch - {{- if .Values.containerSecurityContext.readOnlyRootFilesystem }} - - name: tmpdir - mountPath: /tmp - {{- end }} - {{- if .Values.extraVolumeMounts }} - {{- toYaml .Values.extraVolumeMounts | nindent 12 }} - {{- end }} - {{- if .Values.credentials.extraSecretRef }} - envFrom: - - secretRef: - name: {{ .Values.credentials.extraSecretRef }} - {{- end }} - env: - - name: VELERO_SCRATCH_DIR - value: /scratch - - name: VELERO_NAMESPACE - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.namespace - - name: LD_LIBRARY_PATH - value: /plugins - {{- if .Values.credentials.useSecret }} - - name: AWS_SHARED_CREDENTIALS_FILE - value: /credentials/cloud - - name: GOOGLE_APPLICATION_CREDENTIALS - value: /credentials/cloud - - name: AZURE_CREDENTIALS_FILE - value: /credentials/cloud - - name: ALIBABA_CLOUD_CREDENTIALS_FILE - value: /credentials/cloud - {{- end }} - {{- with .Values.configuration.extraEnvVars }} - {{- range $key, $value := . }} - - name: {{ default "none" $key }} - value: {{ tpl (default "none" $value) $ | quote }} - {{- end }} - {{- end }} - {{- with .Values.credentials.extraEnvVars }} - {{- range $key, $value := . }} - - name: {{ default "none" $key }} - valueFrom: - secretKeyRef: - name: {{ include "velero.secretName" $ }} - key: {{ default "none" $key }} - {{- end }} - {{- end }} - {{- if .Values.lifecycle }} - lifecycle: {{ toYaml .Values.lifecycle | nindent 12 }} - {{- end }} - dnsPolicy: {{ .Values.dnsPolicy }} -{{- if .Values.initContainers }} - initContainers: - {{- if eq (typeOf .Values.initContainers) "string" }} - {{- tpl .Values.initContainers . | nindent 8 }} - {{- else }} - {{- toYaml .Values.initContainers | nindent 8 }} - {{- end }} -{{- end }} - volumes: - {{- if .Values.credentials.useSecret }} - - name: cloud-credentials - secret: - secretName: {{ include "velero.secretName" . }} - {{- end }} - - name: plugins - emptyDir: {} - - name: scratch - emptyDir: {} - {{- if .Values.containerSecurityContext.readOnlyRootFilesystem }} - - name: tmpdir - emptyDir: {} - {{- end }} - {{- if .Values.extraVolumes }} - {{- toYaml .Values.extraVolumes | nindent 8 }} - {{- end }} - {{- with $podSecurityContext }} - securityContext: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.dnsConfig }} - dnsConfig: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} diff --git a/charts/velero/velero/templates/extra-manifests.yaml b/charts/velero/velero/templates/extra-manifests.yaml deleted file mode 100644 index a9bb3b6..0000000 --- a/charts/velero/velero/templates/extra-manifests.yaml +++ /dev/null @@ -1,4 +0,0 @@ -{{ range .Values.extraObjects }} ---- -{{ tpl (toYaml .) $ }} -{{ end }} diff --git a/charts/velero/velero/templates/label-namespace/labelnamespace.yaml b/charts/velero/velero/templates/label-namespace/labelnamespace.yaml deleted file mode 100644 index 47b63f1..0000000 --- a/charts/velero/velero/templates/label-namespace/labelnamespace.yaml +++ /dev/null @@ -1,39 +0,0 @@ -{{- if .Values.namespace }} -{{- if gt (len .Values.namespace.labels) 0 }} -apiVersion: batch/v1 -kind: Job -metadata: - name: {{ template "velero.fullname" . }}-label-namespace - namespace: {{ .Release.Namespace }} - annotations: - "helm.sh/hook": post-install,post-upgrade,post-rollback - "helm.sh/hook-delete-policy": hook-succeeded - labels: - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} -spec: - template: - spec: - serviceAccountName: {{ include "velero.serverServiceAccount" . }} - containers: - - name: label-namespace - {{- if .Values.kubectl.image.digest }} - image: "{{ .Values.kubectl.image.repository }}@{{ .Values.kubectl.image.digest }}" - {{- else if .Values.kubectl.image.tag }} - image: "{{ .Values.kubectl.image.repository }}:{{ .Values.kubectl.image.tag }}" - {{- else }} - image: "{{ .Values.kubectl.image.repository }}:{{ template "chart.KubernetesVersion" . }}" - {{- end }} - command: - - /bin/sh - - -c - - | - {{- range .Values.namespace.labels }} - kubectl label namespace {{ $.Release.Namespace }} {{ .key }}={{ .value }} - {{- end }} - restartPolicy: Never - backoffLimit: 3 -{{- end }} -{{- end }} \ No newline at end of file diff --git a/charts/velero/velero/templates/node-agent-daemonset.yaml b/charts/velero/velero/templates/node-agent-daemonset.yaml deleted file mode 100644 index 68fedf3..0000000 --- a/charts/velero/velero/templates/node-agent-daemonset.yaml +++ /dev/null @@ -1,204 +0,0 @@ -{{- if .Values.deployNodeAgent }} -{{/* 'nodeAgent.securityContext' got renamed to 'nodeAgent.containerSecurityContext', merge both dicts into one for backward compatibility */}} -{{- $containerSecurityContext := merge (.Values.nodeAgent.containerSecurityContext | default dict) (.Values.nodeAgent.securityContext | default dict) -}} -apiVersion: apps/v1 -kind: DaemonSet -metadata: - name: node-agent - namespace: {{ .Release.Namespace }} - {{- with .Values.nodeAgent.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} - labels: - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} - {{- with .Values.nodeAgent.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - selector: - matchLabels: - name: node-agent - template: - metadata: - labels: - name: node-agent - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} - {{- if .Values.podLabels }} - {{- toYaml .Values.podLabels | nindent 8 }} - {{- end }} - {{- if or .Values.podAnnotations .Values.metrics.enabled (and .Values.credentials.useSecret (not .Values.credentials.existingSecret)) }} - annotations: - {{- with .Values.podAnnotations }} - {{- toYaml . | nindent 8 }} - {{- end }} - {{- if and (.Values.metrics.enabled) (not .Values.metrics.nodeAgentPodMonitor.enabled) }} - {{- with .Values.metrics.podAnnotations }} - {{- toYaml . | nindent 8 }} - {{- end }} - {{- end }} - {{- if and .Values.credentials.useSecret (not .Values.credentials.existingSecret) }} - checksum/secret: {{ template "chart.config-checksum" . }} - {{- end }} - {{- end }} - spec: - {{- if .Values.image.imagePullSecrets }} - imagePullSecrets: - {{- range .Values.image.imagePullSecrets }} - - name: {{ . }} - {{- end }} - {{- end }} - serviceAccountName: {{ include "velero.serverServiceAccount" . }} - {{- with .Values.nodeAgent.podSecurityContext }} - securityContext: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- if .Values.nodeAgent.priorityClassName }} - priorityClassName: {{ include "velero.nodeAgent.priorityClassName" . }} - {{- end }} - terminationGracePeriodSeconds: {{ .Values.terminationGracePeriodSeconds }} - volumes: - {{- if .Values.credentials.useSecret }} - - name: cloud-credentials - secret: - secretName: {{ include "velero.secretName" . }} - {{- end }} - - name: host-pods - hostPath: - path: {{ .Values.nodeAgent.podVolumePath }} - {{- if .Values.nodeAgent.useScratchEmptyDir }} - - name: scratch - emptyDir: {} - {{- end }} - {{- if .Values.nodeAgent.extraVolumes }} - {{- toYaml .Values.nodeAgent.extraVolumes | nindent 8 }} - {{- end }} - dnsPolicy: {{ .Values.nodeAgent.dnsPolicy }} - containers: - - name: node-agent - {{- if .Values.image.digest }} - image: "{{ .Values.image.repository }}@{{ .Values.image.digest }}" - {{- else }} - image: "{{ .Values.image.repository }}:{{ .Values.image.tag }}" - {{- end }} - imagePullPolicy: {{ .Values.image.pullPolicy }} - {{- if .Values.metrics.enabled }} - ports: - - name: http-monitoring - containerPort: 8085 - {{- end }} - command: - - /velero - args: - - node-agent - - server - {{- with .Values.configuration }} - {{- with .features }} - - --features={{ . }} - {{- end }} - {{- with .logLevel }} - - --log-level={{ . }} - {{- end }} - {{- with .logFormat }} - - --log-format={{ . }} - {{- end }} - {{- end }} - {{- with .Values.nodeAgent.extraArgs }} - {{- toYaml . | nindent 12 }} - {{- end }} - volumeMounts: - {{- if .Values.credentials.useSecret }} - - name: cloud-credentials - mountPath: /credentials - {{- end }} - - name: host-pods - mountPath: /host_pods - mountPropagation: HostToContainer - {{- if .Values.nodeAgent.useScratchEmptyDir }} - - name: scratch - mountPath: /scratch - {{- end }} - {{- if .Values.nodeAgent.extraVolumeMounts }} - {{- toYaml .Values.nodeAgent.extraVolumeMounts | nindent 12 }} - {{- end }} - {{- if .Values.credentials.extraSecretRef }} - envFrom: - - secretRef: - name: {{ .Values.credentials.extraSecretRef }} - {{- end }} - env: - - name: VELERO_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - - name: NODE_NAME - valueFrom: - fieldRef: - fieldPath: spec.nodeName - - name: VELERO_SCRATCH_DIR - value: /scratch - {{- if .Values.credentials.useSecret }} - - name: AWS_SHARED_CREDENTIALS_FILE - value: /credentials/cloud - - name: GOOGLE_APPLICATION_CREDENTIALS - value: /credentials/cloud - - name: AZURE_CREDENTIALS_FILE - value: /credentials/cloud - - name: ALIBABA_CLOUD_CREDENTIALS_FILE - value: /credentials/cloud - {{- end }} - {{- with .Values.configuration.extraEnvVars }} - {{- range $key, $value := . }} - - name: {{ default "none" $key }} - value: {{ tpl (default "none" $value) $ | quote }} - {{- end }} - {{- end }} - {{- with .Values.credentials.extraEnvVars }} - {{- range $key, $value := . }} - - name: {{ default "none" $key }} - valueFrom: - secretKeyRef: - name: {{ include "velero.secretName" $ }} - key: {{ default "none" $key }} - {{- end }} - {{- end }} - {{- with .Values.nodeAgent.extraEnvVars }} - {{- range $key, $value := . }} - - name: {{ default "none" $key }} - value: {{ default "none" $value | quote }} - {{- end }} - {{- end }} - {{- if .Values.lifecycle }} - lifecycle: {{ toYaml .Values.nodeAgent.lifecycle | nindent 12 }} - {{- end }} - securityContext: - {{- with $containerSecurityContext }} - {{- toYaml . | nindent 12 }} - {{- end }} - {{- with .Values.nodeAgent.resources }} - resources: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- with .Values.nodeAgent.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.nodeAgent.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.nodeAgent.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.nodeAgent.dnsConfig }} - dnsConfig: - {{- toYaml . | nindent 8 }} - {{- end }} -{{- end }} diff --git a/charts/velero/velero/templates/podmonitor.yaml b/charts/velero/velero/templates/podmonitor.yaml deleted file mode 100644 index 3e6a3b9..0000000 --- a/charts/velero/velero/templates/podmonitor.yaml +++ /dev/null @@ -1,44 +0,0 @@ -{{ if and (and .Values.metrics.enabled .Values.metrics.nodeAgentPodMonitor.enabled) (or (not .Values.metrics.nodeAgentPodMonitor.autodetect) (.Capabilities.APIVersions.Has "monitoring.coreos.com/v1")) }} -apiVersion: monitoring.coreos.com/v1 -kind: PodMonitor -metadata: - name: node-agent - {{- if .Values.metrics.nodeAgentPodMonitor.namespace }} - namespace: {{ .Values.metrics.nodeAgentPodMonitor.namespace }} - {{- end }} - {{- with .Values.metrics.nodeAgentPodMonitor.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} - labels: - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} - {{- with .Values.metrics.nodeAgentPodMonitor.additionalLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - namespaceSelector: - matchNames: - - {{ .Release.Namespace }} - selector: - matchLabels: - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - name: node-agent - {{- with .Values.nodeAgent.labels }} - {{- toYaml . | nindent 6 }} - {{- end }} - podMetricsEndpoints: - - port: http-monitoring - interval: {{ .Values.metrics.scrapeInterval }} - scrapeTimeout: {{ .Values.metrics.scrapeTimeout }} - {{- if .Values.metrics.nodeAgentPodMonitor.scheme }} - scheme: {{ .Values.metrics.nodeAgentPodMonitor.scheme }} - {{- end }} - {{- if .Values.metrics.nodeAgentPodMonitor.tlsConfig }} - tlsConfig: - {{- toYaml .Values.metrics.nodeAgentPodMonitor.tlsConfig | nindent 6 }} - {{- end }} -{{- end }} diff --git a/charts/velero/velero/templates/prometheusrule.yaml b/charts/velero/velero/templates/prometheusrule.yaml deleted file mode 100644 index 60bcfc3..0000000 --- a/charts/velero/velero/templates/prometheusrule.yaml +++ /dev/null @@ -1,21 +0,0 @@ -{{- if and (and .Values.metrics.enabled .Values.metrics.prometheusRule.enabled) (or (not .Values.metrics.prometheusRule.autodetect) (.Capabilities.APIVersions.Has "monitoring.coreos.com/v1")) (.Values.metrics.prometheusRule.spec) }} -apiVersion: monitoring.coreos.com/v1 -kind: PrometheusRule -metadata: - name: {{ include "velero.fullname" . }} - {{- if .Values.metrics.prometheusRule.namespace }} - namespace: {{ .Values.metrics.prometheusRule.namespace }} - {{- end }} - labels: - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - {{- if .Values.metrics.prometheusRule.additionalLabels }} - {{- toYaml .Values.metrics.prometheusRule.additionalLabels | nindent 4 }} - {{- end }} -spec: - groups: - - name: {{ include "velero.name" . }} - rules: - {{- toYaml .Values.metrics.prometheusRule.spec | nindent 4 }} -{{- end }} diff --git a/charts/velero/velero/templates/role.yaml b/charts/velero/velero/templates/role.yaml deleted file mode 100644 index f6bc87c..0000000 --- a/charts/velero/velero/templates/role.yaml +++ /dev/null @@ -1,21 +0,0 @@ -{{- if .Values.rbac.create }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: {{ include "velero.fullname" . }}-server - namespace: {{ .Release.Namespace }} - labels: - app.kubernetes.io/component: server - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} -rules: -- apiGroups: - - "*" - resources: - - "*" - verbs: - - "*" - -{{- end }} diff --git a/charts/velero/velero/templates/rolebinding.yaml b/charts/velero/velero/templates/rolebinding.yaml deleted file mode 100644 index d77bea4..0000000 --- a/charts/velero/velero/templates/rolebinding.yaml +++ /dev/null @@ -1,21 +0,0 @@ -{{- if .Values.rbac.create }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: {{ include "velero.fullname" . }}-server - namespace: {{ .Release.Namespace }} - labels: - app.kubernetes.io/component: server - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} -subjects: - - kind: ServiceAccount - namespace: {{ .Release.Namespace }} - name: {{ include "velero.serverServiceAccount" . }} -roleRef: - kind: Role - name: {{ include "velero.fullname" . }}-server - apiGroup: rbac.authorization.k8s.io -{{- end }} diff --git a/charts/velero/velero/templates/schedule.yaml b/charts/velero/velero/templates/schedule.yaml deleted file mode 100644 index d09d900..0000000 --- a/charts/velero/velero/templates/schedule.yaml +++ /dev/null @@ -1,34 +0,0 @@ -{{- range $scheduleName, $schedule := .Values.schedules }} -{{- if (not $schedule.disabled) }} -apiVersion: velero.io/v1 -kind: Schedule -metadata: - name: {{ include "velero.fullname" $ }}-{{ $scheduleName }} - namespace: {{ $.Release.Namespace }} - {{- if $schedule.annotations }} - annotations: - {{- toYaml $schedule.annotations | nindent 4 }} - {{- end }} - labels: - app.kubernetes.io/name: {{ include "velero.name" $ }} - app.kubernetes.io/instance: {{ $.Release.Name }} - app.kubernetes.io/managed-by: {{ $.Release.Service }} - helm.sh/chart: {{ include "velero.chart" $ }} - {{- if $schedule.labels }} - {{- toYaml $schedule.labels | nindent 4 }} - {{- end }} -spec: -{{- if $schedule.paused }} - paused: {{ $schedule.paused }} -{{- end }} -{{- if $schedule.useOwnerReferencesInBackup }} - useOwnerReferencesInBackup: {{ $schedule.useOwnerReferencesInBackup }} -{{- end }} - schedule: {{ $schedule.schedule | quote }} -{{- with $schedule.template }} - template: - {{- toYaml . | nindent 4 }} -{{- end }} ---- -{{- end }} -{{- end }} diff --git a/charts/velero/velero/templates/secret.yaml b/charts/velero/velero/templates/secret.yaml deleted file mode 100644 index 0cd9d57..0000000 --- a/charts/velero/velero/templates/secret.yaml +++ /dev/null @@ -1,24 +0,0 @@ -{{- if and .Values.credentials.useSecret (not .Values.credentials.existingSecret) -}} -apiVersion: v1 -kind: Secret -metadata: - name: {{ include "velero.secretName" . }} - namespace: {{ .Release.Namespace }} - {{- with .Values.secretAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} - labels: - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} -type: Opaque -data: -{{- range $key, $value := .Values.credentials.secretContents }} - {{ $key }}: {{ tpl $value $ | b64enc | quote }} -{{- end }} -{{- range $key, $value := .Values.credentials.extraEnvVars }} - {{ $key }}: {{ tpl $value $ | b64enc | quote }} -{{- end }} -{{- end -}} diff --git a/charts/velero/velero/templates/service.yaml b/charts/velero/velero/templates/service.yaml deleted file mode 100644 index bd66c31..0000000 --- a/charts/velero/velero/templates/service.yaml +++ /dev/null @@ -1,29 +0,0 @@ -{{- if .Values.metrics.enabled }} -apiVersion: v1 -kind: Service -metadata: - name: {{ include "velero.fullname" . }} - namespace: {{ .Release.Namespace }} - {{- with .Values.metrics.service.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} - labels: - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} - {{- with .Values.metrics.service.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - type: ClusterIP - ports: - - name: http-monitoring - port: 8085 - targetPort: http-monitoring - selector: - name: velero - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} -{{- end }} diff --git a/charts/velero/velero/templates/serviceaccount-server.yaml b/charts/velero/velero/templates/serviceaccount-server.yaml deleted file mode 100644 index 76feca0..0000000 --- a/charts/velero/velero/templates/serviceaccount-server.yaml +++ /dev/null @@ -1,25 +0,0 @@ -{{- if .Values.serviceAccount.server.create }} -apiVersion: v1 -kind: ServiceAccount -metadata: - name: {{ include "velero.serverServiceAccount" . }} - namespace: {{ .Release.Namespace }} -{{- if .Values.serviceAccount.server.annotations }} - annotations: -{{ toYaml .Values.serviceAccount.server.annotations | nindent 4 }} -{{- end }} - labels: - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} -{{- with .Values.serviceAccount.server.labels }} - {{- toYaml . | nindent 4 }} -{{- end }} -{{- if .Values.serviceAccount.server.imagePullSecrets }} -imagePullSecrets: -{{- range .Values.serviceAccount.server.imagePullSecrets }} - - name: {{ . }} -{{- end }} -{{- end }} -{{- end }} diff --git a/charts/velero/velero/templates/servicemonitor.yaml b/charts/velero/velero/templates/servicemonitor.yaml deleted file mode 100644 index 777ff9f..0000000 --- a/charts/velero/velero/templates/servicemonitor.yaml +++ /dev/null @@ -1,46 +0,0 @@ -{{ if and (and .Values.metrics.enabled .Values.metrics.serviceMonitor.enabled) (or (not .Values.metrics.serviceMonitor.autodetect) (.Capabilities.APIVersions.Has "monitoring.coreos.com/v1")) }} -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ include "velero.fullname" . }} - {{- if .Values.metrics.serviceMonitor.namespace }} - namespace: {{ .Values.metrics.serviceMonitor.namespace }} - {{- end }} - {{- with .Values.metrics.serviceMonitor.annotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} - labels: - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} - {{- with .Values.metrics.serviceMonitor.additionalLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - namespaceSelector: - matchNames: - - {{ .Release.Namespace }} - selector: - matchLabels: - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - endpoints: - - port: http-monitoring - interval: {{ .Values.metrics.scrapeInterval }} - scrapeTimeout: {{ .Values.metrics.scrapeTimeout }} - {{- if .Values.metrics.serviceMonitor.scheme }} - scheme: {{ .Values.metrics.serviceMonitor.scheme }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.metricRelabelings }} - metricRelabelings: {{- toYaml .Values.metrics.serviceMonitor.metricRelabelings | nindent 6 }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.relabelings }} - relabelings: {{ toYaml .Values.metrics.serviceMonitor.relabelings | nindent 6 }} - {{- end }} - {{- if .Values.metrics.serviceMonitor.tlsConfig }} - tlsConfig: - {{- toYaml .Values.metrics.serviceMonitor.tlsConfig | nindent 6 }} - {{- end }} -{{- end }} diff --git a/charts/velero/velero/templates/upgrade-crds/clusterrole-upgrade.yaml b/charts/velero/velero/templates/upgrade-crds/clusterrole-upgrade.yaml deleted file mode 100644 index 426d1cb..0000000 --- a/charts/velero/velero/templates/upgrade-crds/clusterrole-upgrade.yaml +++ /dev/null @@ -1,27 +0,0 @@ -{{- if .Values.upgradeCRDs }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ include "velero.fullname" . }}-upgrade-crds - annotations: - "helm.sh/hook": pre-install,pre-upgrade,pre-rollback - "helm.sh/hook-weight": "-5" - "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded - labels: - app.kubernetes.io/component: upgrade-crds - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} -rules: -- apiGroups: - - "apiextensions.k8s.io" - resources: - - "customresourcedefinitions" - verbs: - - create - - patch - - update - - get - - list -{{- end }} diff --git a/charts/velero/velero/templates/upgrade-crds/clusterrolebinding-upgrade.yaml b/charts/velero/velero/templates/upgrade-crds/clusterrolebinding-upgrade.yaml deleted file mode 100644 index 07d802c..0000000 --- a/charts/velero/velero/templates/upgrade-crds/clusterrolebinding-upgrade.yaml +++ /dev/null @@ -1,25 +0,0 @@ -{{- if .Values.upgradeCRDs }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ include "velero.fullname" . }}-upgrade-crds - labels: - app.kubernetes.io/component: upgrade-crds - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} - annotations: - "helm.sh/hook": pre-install,pre-upgrade,pre-rollback - "helm.sh/hook-weight": "-3" - "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded -subjects: - - kind: ServiceAccount - namespace: {{ .Release.Namespace }} - name: {{ include "velero.serverServiceAccount" . }}-upgrade-crds -roleRef: - kind: ClusterRole - name: {{ include "velero.fullname" . }}-upgrade-crds - apiGroup: rbac.authorization.k8s.io -{{- end }} - diff --git a/charts/velero/velero/templates/upgrade-crds/serviceaccount-upgrade.yaml b/charts/velero/velero/templates/upgrade-crds/serviceaccount-upgrade.yaml deleted file mode 100644 index 93e3f1e..0000000 --- a/charts/velero/velero/templates/upgrade-crds/serviceaccount-upgrade.yaml +++ /dev/null @@ -1,22 +0,0 @@ -{{- if .Values.upgradeCRDs }} -apiVersion: v1 -kind: ServiceAccount -metadata: - name: {{ include "velero.serverServiceAccount" . }}-upgrade-crds - namespace: {{ .Release.Namespace }} - annotations: - "helm.sh/hook": pre-install,pre-upgrade,pre-rollback - "helm.sh/hook-weight": "-4" - "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded -{{- if .Values.serviceAccount.server.annotations }} -{{ toYaml .Values.serviceAccount.server.annotations | nindent 4 }} -{{- end }} - labels: - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} -{{- with .Values.serviceAccount.server.labels }} - {{- toYaml . | nindent 4 }} -{{- end }} -{{- end }} diff --git a/charts/velero/velero/templates/upgrade-crds/upgrade-crds.yaml b/charts/velero/velero/templates/upgrade-crds/upgrade-crds.yaml deleted file mode 100644 index 59d3b2e..0000000 --- a/charts/velero/velero/templates/upgrade-crds/upgrade-crds.yaml +++ /dev/null @@ -1,127 +0,0 @@ -{{- if .Values.upgradeCRDs }} -{{/* 'securityContext' got renamed to 'podSecurityContext', merge both dicts into one for backward compatibility */}} -{{- $podSecurityContext := merge (.Values.podSecurityContext | default dict) (.Values.securityContext | default dict) -}} -apiVersion: batch/v1 -kind: Job -metadata: - name: {{ template "velero.fullname" . }}-upgrade-crds - namespace: {{ .Release.Namespace }} - annotations: - "helm.sh/hook": pre-install,pre-upgrade,pre-rollback - "helm.sh/hook-weight": "5" - "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded - labels: - app.kubernetes.io/name: {{ include "velero.name" . }} - app.kubernetes.io/instance: {{ .Release.Name }} - app.kubernetes.io/managed-by: {{ .Release.Service }} - helm.sh/chart: {{ include "velero.chart" . }} - {{- with .Values.kubectl.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - backoffLimit: 3 - template: - metadata: - name: velero-upgrade-crds - {{- with .Values.kubectl.labels }} - labels: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.kubectl.annotations }} - annotations: - {{- toYaml . | nindent 8 }} - {{- end }} - spec: - {{- if .Values.image.imagePullSecrets }} - imagePullSecrets: - {{- range .Values.image.imagePullSecrets }} - - name: {{ . }} - {{- end }} - {{- end }} - serviceAccountName: {{ include "velero.serverServiceAccount" . }}-upgrade-crds - initContainers: - - name: kubectl - {{- if .Values.kubectl.image.digest }} - image: "{{ .Values.kubectl.image.repository }}@{{ .Values.kubectl.image.digest }}" - {{- else if .Values.kubectl.image.tag }} - image: "{{ .Values.kubectl.image.repository }}:{{ .Values.kubectl.image.tag }}" - {{- else }} - image: "{{ .Values.kubectl.image.repository }}:{{ template "chart.KubernetesVersion" . }}" - {{- end }} - imagePullPolicy: {{ .Values.image.pullPolicy }} - command: - - /bin/sh - args: - - -c - - cp `which sh` /tmp && cp `which kubectl` /tmp - {{- with .Values.kubectl.resources }} - resources: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- with .Values.kubectl.containerSecurityContext }} - securityContext: - {{- toYaml . | nindent 12 }} - {{- end }} - volumeMounts: - - mountPath: /tmp - name: crds - containers: - - name: velero - {{- if .Values.image.digest }} - image: "{{ .Values.image.repository }}@{{ .Values.image.digest }}" - {{- else }} - image: "{{ .Values.image.repository }}:{{ .Values.image.tag }}" - {{- end }} - imagePullPolicy: {{ .Values.image.pullPolicy }} - command: - - /tmp/sh - args: - - -c - - /velero install --crds-only --dry-run -o yaml | /tmp/kubectl apply -f - - {{- with .Values.upgradeJobResources }} - resources: - {{- toYaml . | nindent 12 }} - {{- end }} - {{- with .Values.containerSecurityContext }} - securityContext: - {{- toYaml . | nindent 12 }} - {{- end }} - volumeMounts: - - mountPath: /tmp - name: crds - {{- if (.Values.upgradeCRDsJob).extraVolumeMounts }} - {{- toYaml .Values.upgradeCRDsJob.extraVolumeMounts | nindent 12 }} - {{- end }} - {{- if (.Values.upgradeCRDsJob).extraEnvVars }} - env: - {{- with .Values.upgradeCRDsJob.extraEnvVars }} - {{- range $key, $value := . }} - - name: {{ default "none" $key }} - value: {{ default "none" $value | quote }} - {{- end }} - {{- end }} - {{- end }} - volumes: - - name: crds - emptyDir: {} - {{- if (.Values.upgradeCRDsJob).extraVolumes }} - {{- toYaml .Values.upgradeCRDsJob.extraVolumes | nindent 8 }} - {{- end }} - restartPolicy: OnFailure - {{- with $podSecurityContext }} - securityContext: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} -{{- end }} diff --git a/charts/velero/velero/templates/volumesnapshotlocation.yaml b/charts/velero/velero/templates/volumesnapshotlocation.yaml deleted file mode 100644 index 8319968..0000000 --- a/charts/velero/velero/templates/volumesnapshotlocation.yaml +++ /dev/null @@ -1,41 +0,0 @@ -{{- if .Values.snapshotsEnabled }} - -{{- if typeIs "[]interface {}" .Values.configuration.volumeSnapshotLocation }} -{{- range .Values.configuration.volumeSnapshotLocation }} ---- -apiVersion: velero.io/v1 -kind: VolumeSnapshotLocation -metadata: - name: {{ .name | default "default" }} - namespace: {{ $.Release.Namespace }} - {{- with .annotations }} - annotations: - {{- range $key, $value := . }} - {{- $key | nindent 4 }}: {{ $value | quote }} - {{- end }} - {{- end }} - labels: - app.kubernetes.io/name: {{ include "velero.name" $ }} - app.kubernetes.io/instance: {{ $.Release.Name }} - app.kubernetes.io/managed-by: {{ $.Release.Service }} - helm.sh/chart: {{ include "velero.chart" $ }} -spec: - {{- if not (empty .credential) }} - credential: - {{- with .credential.name }} - name: {{ . }} - {{- end }} - {{- with .credential.key }} - key: {{ . }} - {{- end }} - {{- end }} - provider: {{ .provider }} -{{- with .config }} - config: -{{- range $key, $value := . }} -{{- $key | nindent 4 }}: {{ $value | quote }} -{{- end }} -{{- end -}} -{{- end }} -{{- end }} -{{- end }} diff --git a/charts/velero/velero/values.yaml b/charts/velero/velero/values.yaml deleted file mode 100644 index d38a268..0000000 --- a/charts/velero/velero/values.yaml +++ /dev/null @@ -1,656 +0,0 @@ -## -## Configuration settings related to Velero installation namespace -## - -# Labels settings in namespace -namespace: - labels: {} - # Enforce Pod Security Standards with Namespace Labels - # https://kubernetes.io/docs/tasks/configure-pod-container/enforce-standards-namespace-labels/ - # - key: pod-security.kubernetes.io/enforce - # value: privileged - # - key: pod-security.kubernetes.io/enforce-version - # value: latest - # - key: pod-security.kubernetes.io/audit - # value: privileged - # - key: pod-security.kubernetes.io/audit-version - # value: latest - # - key: pod-security.kubernetes.io/warn - # value: privileged - # - key: pod-security.kubernetes.io/warn-version - # value: latest - -## -## End of namespace-related settings. -## - - -## -## Configuration settings that directly affect the Velero deployment YAML. -## - -# Details of the container image to use in the Velero deployment & daemonset (if -# enabling node-agent). Required. -image: - repository: velero/velero - tag: v1.14.1 - # Digest value example: sha256:d238835e151cec91c6a811fe3a89a66d3231d9f64d09e5f3c49552672d271f38. - # If used, it will take precedence over the image.tag. - # digest: - pullPolicy: IfNotPresent - # One or more secrets to be used when pulling images - imagePullSecrets: [] - # - registrySecretName - -nameOverride: "" -fullnameOverride: "" - -# Annotations to add to the Velero deployment's. Optional. -# -# If you are using reloader use the following annotation with your VELERO_SECRET_NAME -annotations: {} -# secret.reloader.stakater.com/reload: "" - -# Annotations to add to secret -secretAnnotations: {} - -# Labels to add to the Velero deployment's. Optional. -labels: {} - -# Annotations to add to the Velero deployment's pod template. Optional. -# -# If using kube2iam or kiam, use the following annotation with your AWS_ACCOUNT_ID -# and VELERO_ROLE_NAME filled in: -podAnnotations: {} - # iam.amazonaws.com/role: "arn:aws:iam:::role/" - -# Additional pod labels for Velero deployment's template. Optional -# ref: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/ -podLabels: {} - -# Number of old history to retain to allow rollback (If not set, default Kubernetes value is set to 10) -# revisionHistoryLimit: 1 - -# Resource requests/limits to specify for the Velero deployment. -# https://velero.io/docs/v1.6/customize-installation/#customize-resource-requests-and-limits -resources: {} - # requests: - # cpu: 500m - # memory: 128Mi - # limits: - # cpu: 1000m - # memory: 512Mi - -# Resource requests/limits to specify for the upgradeCRDs job pod. Need to be adjusted by user accordingly. -upgradeJobResources: {} -# requests: -# cpu: 50m -# memory: 128Mi -# limits: -# cpu: 100m -# memory: 256Mi -upgradeCRDsJob: - # Extra volumes for the Upgrade CRDs Job. Optional. - extraVolumes: [] - # Extra volumeMounts for the Upgrade CRDs Job. Optional. - extraVolumeMounts: [] - # Extra key/value pairs to be used as environment variables. Optional. - extraEnvVars: {} - - -# Configure the dnsPolicy of the Velero deployment -# See: https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/#pod-s-dns-policy -dnsPolicy: ClusterFirst - -# Init containers to add to the Velero deployment's pod spec. At least one plugin provider image is required. -# If the value is a string then it is evaluated as a template. -initContainers: - # - name: velero-plugin-for-aws - # image: velero/velero-plugin-for-aws:v1.10.0 - # imagePullPolicy: IfNotPresent - # volumeMounts: - # - mountPath: /target - # name: plugins - -# SecurityContext to use for the Velero deployment. Optional. -# Set fsGroup for `AWS IAM Roles for Service Accounts` -# see more informations at: https://docs.aws.amazon.com/eks/latest/userguide/iam-roles-for-service-accounts.html -podSecurityContext: {} - # fsGroup: 1337 - -# Container Level Security Context for the 'velero' container of the Velero deployment. Optional. -# See: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container -containerSecurityContext: {} - # allowPrivilegeEscalation: false - # capabilities: - # drop: ["ALL"] - # add: [] - # readOnlyRootFilesystem: true - -# Container Lifecycle Hooks to use for the Velero deployment. Optional. -lifecycle: {} - -# Pod priority class name to use for the Velero deployment. Optional. -priorityClassName: "" - -# The number of seconds to allow for graceful termination of the pod. Optional. -terminationGracePeriodSeconds: 3600 - -# Liveness probe of the pod -livenessProbe: - httpGet: - path: /metrics - port: http-monitoring - scheme: HTTP - initialDelaySeconds: 10 - periodSeconds: 30 - timeoutSeconds: 5 - successThreshold: 1 - failureThreshold: 5 - -# Readiness probe of the pod -readinessProbe: - httpGet: - path: /metrics - port: http-monitoring - scheme: HTTP - initialDelaySeconds: 10 - periodSeconds: 30 - timeoutSeconds: 5 - successThreshold: 1 - failureThreshold: 5 - -# Tolerations to use for the Velero deployment. Optional. -tolerations: [] - -# Affinity to use for the Velero deployment. Optional. -affinity: {} - -# Node selector to use for the Velero deployment. Optional. -nodeSelector: {} - -# DNS configuration to use for the Velero deployment. Optional. -dnsConfig: {} - -# Extra volumes for the Velero deployment. Optional. -extraVolumes: [] - -# Extra volumeMounts for the Velero deployment. Optional. -extraVolumeMounts: [] - -# Extra K8s manifests to deploy -extraObjects: [] - # - apiVersion: secrets-store.csi.x-k8s.io/v1 - # kind: SecretProviderClass - # metadata: - # name: velero-secrets-store - # spec: - # provider: aws - # parameters: - # objects: | - # - objectName: "velero" - # objectType: "secretsmanager" - # jmesPath: - # - path: "access_key" - # objectAlias: "access_key" - # - path: "secret_key" - # objectAlias: "secret_key" - # secretObjects: - # - data: - # - key: access_key - # objectName: client-id - # - key: client-secret - # objectName: client-secret - # secretName: velero-secrets-store - # type: Opaque - -# Settings for Velero's prometheus metrics. Enabled by default. -metrics: - enabled: true - scrapeInterval: 30s - scrapeTimeout: 10s - - # service metdata if metrics are enabled - service: - annotations: {} - labels: {} - - # Pod annotations for Prometheus - podAnnotations: - prometheus.io/scrape: "true" - prometheus.io/port: "8085" - prometheus.io/path: "/metrics" - - serviceMonitor: - autodetect: true - enabled: false - annotations: {} - additionalLabels: {} - - # metrics.serviceMonitor.metricRelabelings Specify Metric Relabelings to add to the scrape endpoint - # ref: https://github.com/coreos/prometheus-operator/blob/master/Documentation/api.md#relabelconfig - # metricRelabelings: [] - # metrics.serviceMonitor.relabelings [array] Prometheus relabeling rules - # relabelings: [] - # ServiceMonitor namespace. Default to Velero namespace. - # namespace: - # ServiceMonitor connection scheme. Defaults to HTTP. - # scheme: "" - # ServiceMonitor connection tlsConfig. Defaults to {}. - # tlsConfig: {} - nodeAgentPodMonitor: - autodetect: true - enabled: false - annotations: {} - additionalLabels: {} - # ServiceMonitor namespace. Default to Velero namespace. - # namespace: - # ServiceMonitor connection scheme. Defaults to HTTP. - # scheme: "" - # ServiceMonitor connection tlsConfig. Defaults to {}. - # tlsConfig: {} - - prometheusRule: - autodetect: true - enabled: false - # Additional labels to add to deployed PrometheusRule - additionalLabels: {} - # PrometheusRule namespace. Defaults to Velero namespace. - # namespace: "" - # Rules to be deployed - spec: [] - # - alert: VeleroBackupPartialFailures - # annotations: - # message: Velero backup {{ $labels.schedule }} has {{ $value | humanizePercentage }} partialy failed backups. - # expr: |- - # velero_backup_partial_failure_total{schedule!=""} / velero_backup_attempt_total{schedule!=""} > 0.25 - # for: 15m - # labels: - # severity: warning - # - alert: VeleroBackupFailures - # annotations: - # message: Velero backup {{ $labels.schedule }} has {{ $value | humanizePercentage }} failed backups. - # expr: |- - # velero_backup_failure_total{schedule!=""} / velero_backup_attempt_total{schedule!=""} > 0.25 - # for: 15m - # labels: - # severity: warning - -kubectl: - image: - repository: docker.io/bitnami/kubectl - # Digest value example: sha256:d238835e151cec91c6a811fe3a89a66d3231d9f64d09e5f3c49552672d271f38. - # If used, it will take precedence over the kubectl.image.tag. - # digest: - # kubectl image tag. If used, it will take precedence over the cluster Kubernetes version. - # tag: 1.16.15 - # Container Level Security Context for the 'kubectl' container of the crd jobs. Optional. - # See: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container - containerSecurityContext: {} - # Resource requests/limits to specify for the upgrade/cleanup job. Optional - resources: {} - # Annotations to set for the upgrade/cleanup job. Optional. - annotations: {} - # Labels to set for the upgrade/cleanup job. Optional. - labels: {} - -# This job upgrades the CRDs. -upgradeCRDs: true - -# This job is meant primarily for cleaning up CRDs on CI systems. -# Using this on production systems, especially those that have multiple releases of Velero, will be destructive. -cleanUpCRDs: false - -## -## End of deployment-related settings. -## - - -## -## Parameters for the `default` BackupStorageLocation and VolumeSnapshotLocation, -## and additional server settings. -## -configuration: - # Parameters for the BackupStorageLocation(s). Configure multiple by adding other element(s) to the backupStorageLocation slice. - # See https://velero.io/docs/v1.6/api-types/backupstoragelocation/ - backupStorageLocation: - # name is the name of the backup storage location where backups should be stored. If a name is not provided, - # a backup storage location will be created with the name "default". Optional. - - name: - # provider is the name for the backup storage location provider. - provider: - # bucket is the name of the bucket to store backups in. Required. - bucket: - # caCert defines a base64 encoded CA bundle to use when verifying TLS connections to the provider. Optional. - caCert: - # prefix is the directory under which all Velero data should be stored within the bucket. Optional. - prefix: - # default indicates this location is the default backup storage location. Optional. - default: - # validationFrequency defines how frequently Velero should validate the object storage. Optional. - validationFrequency: - # accessMode determines if velero can write to this backup storage location. Optional. - # default to ReadWrite, ReadOnly is used during migrations and restores. - accessMode: ReadWrite - credential: - # name of the secret used by this backupStorageLocation. - name: - # name of key that contains the secret data to be used. - key: - # Additional provider-specific configuration. See link above - # for details of required/optional fields for your provider. - config: {} - # region: - # s3ForcePathStyle: - # s3Url: - # kmsKeyId: - # resourceGroup: - # The ID of the subscription containing the storage account, if different from the cluster’s subscription. (Azure only) - # subscriptionId: - # storageAccount: - # publicUrl: - # Name of the GCP service account to use for this backup storage location. Specify the - # service account here if you want to use workload identity instead of providing the key file.(GCP only) - # serviceAccount: - # Option to skip certificate validation or not if insecureSkipTLSVerify is set to be true, the client side should set the - # flag. For Velero client Command like velero backup describe, velero backup logs needs to add the flag --insecure-skip-tls-verify - # insecureSkipTLSVerify: - - # annotations allows adding arbitrary annotations to this BackupStorageLocation resource. Optional. - annotations: {} - - # Parameters for the VolumeSnapshotLocation(s). Configure multiple by adding other element(s) to the volumeSnapshotLocation slice. - # See https://velero.io/docs/v1.6/api-types/volumesnapshotlocation/ - volumeSnapshotLocation: - # name is the name of the volume snapshot location where snapshots are being taken. Required. - - name: - # provider is the name for the volume snapshot provider. - provider: - credential: - # name of the secret used by this volumeSnapshotLocation. - name: - # name of key that contains the secret data to be used. - key: - # Additional provider-specific configuration. See link above - # for details of required/optional fields for your provider. - config: {} - # region: - # apiTimeout: - # resourceGroup: - # The ID of the subscription where volume snapshots should be stored, if different from the cluster’s subscription. If specified, also requires `configuration.volumeSnapshotLocation.config.resourceGroup`to be set. (Azure only) - # subscriptionId: - # incremental: - # snapshotLocation: - # project: - - # annotations allows adding arbitrary annotations to this VolumeSnapshotLocation resource. Optional. - annotations: {} - - # These are server-level settings passed as CLI flags to the `velero server` command. Velero - # uses default values if they're not passed in, so they only need to be explicitly specified - # here if using a non-default value. The `velero server` default values are shown in the - # comments below. - # -------------------- - # `velero server` default: kopia - uploaderType: - # `velero server` default: 1m - backupSyncPeriod: - # `velero server` default: 4h - fsBackupTimeout: - # `velero server` default: 30 - clientBurst: - # `velero server` default: 500 - clientPageSize: - # `velero server` default: 20.0 - clientQPS: - # Name of the default backup storage location. Default: default - defaultBackupStorageLocation: - # The default duration any single item operation can take before timing out, especially important for large volume schedules. Default 4h - defaultItemOperationTimeout: - # How long to wait by default before backups can be garbage collected. Default: 72h - defaultBackupTTL: - # Name of the default volume snapshot location. - defaultVolumeSnapshotLocations: - # `velero server` default: empty - disableControllers: - # `velero server` default: false - disableInformerCache: false - # `velero server` default: 1h - garbageCollectionFrequency: - # Set log-format for Velero pod. Default: text. Other option: json. - logFormat: - # Set log-level for Velero pod. Default: info. Other options: debug, warning, error, fatal, panic. - logLevel: - # The address to expose prometheus metrics. Default: :8085 - metricsAddress: - # Directory containing Velero plugins. Default: /plugins - pluginDir: - # The address to expose the pprof profiler. Default: localhost:6060 - profilerAddress: - # `velero server` default: false - restoreOnlyMode: - # `velero server` default: customresourcedefinitions,namespaces,storageclasses,volumesnapshotclass.snapshot.storage.k8s.io,volumesnapshotcontents.snapshot.storage.k8s.io,volumesnapshots.snapshot.storage.k8s.io,persistentvolumes,persistentvolumeclaims,secrets,configmaps,serviceaccounts,limitranges,pods,replicasets.apps,clusterclasses.cluster.x-k8s.io,clusters.cluster.x-k8s.io,clusterresourcesets.addons.cluster.x-k8s.io - restoreResourcePriorities: - # `velero server` default: 1m - storeValidationFrequency: - # How long to wait on persistent volumes and namespaces to terminate during a restore before timing out. Default: 10m - terminatingResourceTimeout: - # Bool flag to configure Velero server to move data by default for all snapshots supporting data movement. Default: false - defaultSnapshotMoveData: - # Comma separated list of velero feature flags. default: empty - # features: EnableCSI - features: - # Resource requests/limits to specify for the repository-maintenance job. Optional. - # https://velero.io/docs/v1.14/repository-maintenance/#resource-limitation - repositoryMaintenanceJob: - requests: - # cpu: 500m - # memory: 512Mi - limits: - # cpu: 1000m - # memory: 1024Mi - # Number of latest maintenance jobs to keep for each repository - latestJobsCount: 3 - # `velero server` default: velero - namespace: - # additional command-line arguments that will be passed to the `velero server` - # e.g.: extraArgs: ["--foo=bar"] - extraArgs: [] - - # additional key/value pairs to be used as environment variables such as "AWS_CLUSTER_NAME: 'yourcluster.domain.tld'" - extraEnvVars: {} - - # Set true for backup all pod volumes without having to apply annotation on the pod when used file system backup Default: false. - defaultVolumesToFsBackup: - - # How often repository maintain is run for repositories by default. - defaultRepoMaintainFrequency: - -## -## End of backup/snapshot location settings. -## - - -## -## Settings for additional Velero resources. -## - -rbac: - # Whether to create the Velero role and role binding to give all permissions to the namespace to Velero. - create: true - # Whether to create the cluster role binding to give administrator permissions to Velero - clusterAdministrator: true - # Name of the ClusterRole. - clusterAdministratorName: cluster-admin - -# Information about the Kubernetes service account Velero uses. -serviceAccount: - server: - create: true - name: - annotations: - labels: - imagePullSecrets: [] - # - registrySecretName - -# Info about the secret to be used by the Velero deployment, which -# should contain credentials for the cloud provider IAM account you've -# set up for Velero. -credentials: - # Whether a secret should be used. Set to false if, for examples: - # - using kube2iam or kiam to provide AWS IAM credentials instead of providing the key file. (AWS only) - # - using workload identity instead of providing the key file. (Azure/GCP only) - useSecret: true - # Name of the secret to create if `useSecret` is true and `existingSecret` is empty - name: - # Name of a pre-existing secret (if any) in the Velero namespace - # that should be used to get IAM account credentials. Optional. - existingSecret: - # Data to be stored in the Velero secret, if `useSecret` is true and `existingSecret` is empty. - # As of the current Velero release, Velero only uses one secret key/value at a time. - # The key must be named `cloud`, and the value corresponds to the entire content of your IAM credentials file. - # Note that the format will be different for different providers, please check their documentation. - # Here is a list of documentation for plugins maintained by the Velero team: - # [AWS] https://github.com/vmware-tanzu/velero-plugin-for-aws/blob/main/README.md - # [GCP] https://github.com/vmware-tanzu/velero-plugin-for-gcp/blob/main/README.md - # [Azure] https://github.com/vmware-tanzu/velero-plugin-for-microsoft-azure/blob/main/README.md - secretContents: {} - # cloud: | - # [default] - # aws_access_key_id= - # aws_secret_access_key= - # additional key/value pairs to be used as environment variables such as "DIGITALOCEAN_TOKEN: ". Values will be stored in the secret. - extraEnvVars: {} - # Name of a pre-existing secret (if any) in the Velero namespace - # that will be used to load environment variables into velero and node-agent. - # Secret should be in format - https://kubernetes.io/docs/concepts/configuration/secret/#use-case-as-container-environment-variables - extraSecretRef: "" - -# Whether to create backupstoragelocation crd, if false => do not create a default backup location -backupsEnabled: true -# Whether to create volumesnapshotlocation crd, if false => disable snapshot feature -snapshotsEnabled: true - -# Whether to deploy the node-agent daemonset. -deployNodeAgent: false - -nodeAgent: - podVolumePath: /var/lib/kubelet/pods - # Pod priority class name to use for the node-agent daemonset. Optional. - priorityClassName: "" - # Resource requests/limits to specify for the node-agent daemonset deployment. Optional. - # https://velero.io/docs/v1.6/customize-installation/#customize-resource-requests-and-limits - resources: {} - # requests: - # cpu: 500m - # memory: 512Mi - # limits: - # cpu: 1000m - # memory: 1024Mi - - # Tolerations to use for the node-agent daemonset. Optional. - tolerations: [] - - # Annotations to set for the node-agent daemonset. Optional. - annotations: {} - - # labels to set for the node-agent daemonset. Optional. - labels: {} - - # will map /scratch to emptyDir. Set to false and specify your own volume - # via extraVolumes and extraVolumeMounts that maps to /scratch - # if you don't want to use emptyDir. - useScratchEmptyDir: true - - # Extra volumes for the node-agent daemonset. Optional. - extraVolumes: [] - - # Extra volumeMounts for the node-agent daemonset. Optional. - extraVolumeMounts: [] - - # Key/value pairs to be used as environment variables for the node-agent daemonset. Optional. - extraEnvVars: {} - - # Additional command-line arguments that will be passed to the node-agent. Optional. - # e.g.: extraArgs: ["--foo=bar"] - extraArgs: [] - - # Configure the dnsPolicy of the node-agent daemonset - # See: https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/#pod-s-dns-policy - dnsPolicy: ClusterFirst - - # SecurityContext to use for the Velero deployment. Optional. - # Set fsGroup for `AWS IAM Roles for Service Accounts` - # see more informations at: https://docs.aws.amazon.com/eks/latest/userguide/iam-roles-for-service-accounts.html - podSecurityContext: - runAsUser: 0 - # fsGroup: 1337 - - # Container Level Security Context for the 'node-agent' container of the node-agent daemonset. Optional. - # See: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container - containerSecurityContext: {} - - # Container Lifecycle Hooks to use for the node-agent daemonset. Optional. - lifecycle: {} - - # Node selector to use for the node-agent daemonset. Optional. - nodeSelector: {} - - # Affinity to use with node-agent daemonset. Optional. - affinity: {} - - # DNS configuration to use for the node-agent daemonset. Optional. - dnsConfig: {} - -# Backup schedules to create. -# Eg: -# schedules: -# mybackup: -# disabled: false -# labels: -# myenv: foo -# annotations: -# myenv: foo -# schedule: "0 0 * * *" -# useOwnerReferencesInBackup: false -# paused: false -# template: -# ttl: "240h" -# storageLocation: default -# includedNamespaces: -# - foo -# # See: https://velero.io/docs/v1.14/resource-filtering/#excludes -# excludedNamespaceScopedResources: -# - persistentVolumeClaims -# excludedClusterScopedResources: -# - persistentVolumes -schedules: {} - -# Velero ConfigMaps. -# Eg: -# configMaps: - # See: https://velero.io/docs/v1.11/file-system-backup/ -# fs-restore-action-config: -# labels: -# velero.io/plugin-config: "" -# velero.io/pod-volume-restore: RestoreItemAction -# data: -# image: velero/velero-restore-helper:v1.10.2 -# cpuRequest: 200m -# memRequest: 128Mi -# cpuLimit: 200m -# memLimit: 128Mi -# secCtx: | -# capabilities: -# drop: -# - ALL -# add: [] -# allowPrivilegeEscalation: false -# readOnlyRootFilesystem: true -# runAsUser: 1001 -# runAsGroup: 999 -configMaps: {} - -## -## End of additional Velero resource settings. -## diff --git a/config/deploy.yml b/config/deploy.yml index a9affbf..983522b 100644 --- a/config/deploy.yml +++ b/config/deploy.yml @@ -9,7 +9,8 @@ image: futureporn/bright # Deploy to these servers. servers: web: - - 194.163.140.228 + - 45.76.57.101 + # - 127.0.0.1 # job: # hosts: # - 192.168.0.1 @@ -39,7 +40,6 @@ registry: # Configure builder setup. builder: arch: amd64 - context: ../ dockerfile: ./dockerfiles/bright.dockerfile # Pass in additional build args needed for your Dockerfile. # args: @@ -83,24 +83,23 @@ ssh: # Use accessory services (secrets come from .kamal/secrets). # -# accessories: -# db: -# image: mysql:8.0 -# host: 192.168.0.2 -# port: 3306 -# env: -# clear: -# MYSQL_ROOT_HOST: '%' -# secret: -# - MYSQL_ROOT_PASSWORD -# files: -# - config/mysql/production.cnf:/etc/mysql/my.cnf -# - db/production.sql:/docker-entrypoint-initdb.d/setup.sql -# directories: -# - data:/var/lib/mysql -# redis: -# image: valkey/valkey:8 -# host: 192.168.0.2 -# port: 6379 -# directories: -# - data:/data +accessories: + db: + image: postgres:15 + host: 45.76.57.101 + port: 5432 + env: + clear: + POSTGRES_USER: postgres + POSTGRES_DB: phoenix_dev + secret: + - POSTGRES_PASSWORD + directories: + - pg_data:/var/lib/postgresql/data + redis: + image: valkey/valkey:8 + host: 45.76.57.101 + port: 6379 + directories: + - data:/data + diff --git a/contrib/superstreamer b/contrib/superstreamer deleted file mode 160000 index 9e868ac..0000000 --- a/contrib/superstreamer +++ /dev/null @@ -1 +0,0 @@ -Subproject commit 9e868acede851f396b3db98fb9799ab4bf712b02 diff --git a/devbox.json b/devbox.json index 400c5a8..962b648 100644 --- a/devbox.json +++ b/devbox.json @@ -2,20 +2,13 @@ "$schema": "https://raw.githubusercontent.com/jetify-com/devbox/0.12.0/.schema/devbox.schema.json", "packages": [ "nodejs@20", - "tilt@latest", - "ctlptl@latest", - "kubectl@latest", - "cmctl@latest", - "kubernetes-helm@latest", - "k9s@latest", "ffmpeg@latest", "yt-dlp@latest", "python310@latest", "python310Packages.pip@latest", - "vips@latest", "hcloud@latest", - "ruby@latest", - "doppler@latest" + "lazydocker@latest", + "ruby@latest" ], "env": { "DEVBOX_COREPACK_ENABLED": "true", @@ -33,7 +26,7 @@ "test": [ "echo \"Error: no test specified\" && exit 1" ], - "tilt": "tilt up -f ./Tiltfile" + "backup": "docker exec -t postgres_db pg_dumpall -c -U postgres > ./backups/dev_`date +%Y-%m-%d_%H_%M_%S`.sql" } } } diff --git a/devbox.lock b/devbox.lock index cc30d9e..2ba8f4c 100644 --- a/devbox.lock +++ b/devbox.lock @@ -1,150 +1,6 @@ { "lockfile_version": "1", "packages": { - "cmctl@latest": { - "last_modified": "2024-07-07T07:43:47Z", - "resolved": "github:NixOS/nixpkgs/b60793b86201040d9dee019a05089a9150d08b5b#cmctl", - "source": "devbox-search", - "version": "1.14.7", - "systems": { - "aarch64-darwin": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/69lc5y36840ccy6d1pzph994psadk4bm-cmctl-1.14.7", - "default": true - } - ], - "store_path": "/nix/store/69lc5y36840ccy6d1pzph994psadk4bm-cmctl-1.14.7" - }, - "aarch64-linux": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/d5l61iil1gaax5sipnzg80mb0p1hqk9f-cmctl-1.14.7", - "default": true - } - ], - "store_path": "/nix/store/d5l61iil1gaax5sipnzg80mb0p1hqk9f-cmctl-1.14.7" - }, - "x86_64-darwin": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/j0r3wavi836mp31l0s7r1c3rjryw2z62-cmctl-1.14.7", - "default": true - } - ], - "store_path": "/nix/store/j0r3wavi836mp31l0s7r1c3rjryw2z62-cmctl-1.14.7" - }, - "x86_64-linux": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/lnmy96wjzmjna7z9f0dbqd16nf2x5qbv-cmctl-1.14.7", - "default": true - } - ], - "store_path": "/nix/store/lnmy96wjzmjna7z9f0dbqd16nf2x5qbv-cmctl-1.14.7" - } - } - }, - "ctlptl@latest": { - "last_modified": "2024-07-07T07:43:47Z", - "resolved": "github:NixOS/nixpkgs/b60793b86201040d9dee019a05089a9150d08b5b#ctlptl", - "source": "devbox-search", - "version": "0.8.29", - "systems": { - "aarch64-darwin": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/gvnmvb315zngbg5a0idynlwxcc45gmyd-ctlptl-0.8.29", - "default": true - } - ], - "store_path": "/nix/store/gvnmvb315zngbg5a0idynlwxcc45gmyd-ctlptl-0.8.29" - }, - "aarch64-linux": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/vgwbxwaf773mmgcbc1j5h6n5a7g587lf-ctlptl-0.8.29", - "default": true - } - ], - "store_path": "/nix/store/vgwbxwaf773mmgcbc1j5h6n5a7g587lf-ctlptl-0.8.29" - }, - "x86_64-darwin": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/ga4rp2188c6k24162s2n23nfi4846790-ctlptl-0.8.29", - "default": true - } - ], - "store_path": "/nix/store/ga4rp2188c6k24162s2n23nfi4846790-ctlptl-0.8.29" - }, - "x86_64-linux": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/wbfjbk575ny949rfyqpm4vai4ap5rbpp-ctlptl-0.8.29", - "default": true - } - ], - "store_path": "/nix/store/wbfjbk575ny949rfyqpm4vai4ap5rbpp-ctlptl-0.8.29" - } - } - }, - "doppler@latest": { - "last_modified": "2024-12-23T21:10:33Z", - "resolved": "github:NixOS/nixpkgs/de1864217bfa9b5845f465e771e0ecb48b30e02d#doppler", - "source": "devbox-search", - "version": "3.71.0", - "systems": { - "aarch64-darwin": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/6820wwrx8r525zq48bpv295wssiad4s1-doppler-3.71.0", - "default": true - } - ], - "store_path": "/nix/store/6820wwrx8r525zq48bpv295wssiad4s1-doppler-3.71.0" - }, - "aarch64-linux": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/lz24rlrrazcwfhz2kn5dszjwdcasvivw-doppler-3.71.0", - "default": true - } - ], - "store_path": "/nix/store/lz24rlrrazcwfhz2kn5dszjwdcasvivw-doppler-3.71.0" - }, - "x86_64-darwin": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/k908q0cadca9p70jrfr2lcrfclpr343n-doppler-3.71.0", - "default": true - } - ], - "store_path": "/nix/store/k908q0cadca9p70jrfr2lcrfclpr343n-doppler-3.71.0" - }, - "x86_64-linux": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/0vsjlajvmzg816sfphg5mswv4ply20ys-doppler-3.71.0", - "default": true - } - ], - "store_path": "/nix/store/0vsjlajvmzg816sfphg5mswv4ply20ys-doppler-3.71.0" - } - } - }, "ffmpeg@latest": { "last_modified": "2024-07-24T00:53:51Z", "resolved": "github:NixOS/nixpkgs/4f02464258baaf54992debfd010a7a3662a25536#ffmpeg", @@ -341,183 +197,51 @@ } } }, - "k9s@latest": { - "last_modified": "2024-07-20T09:11:00Z", - "resolved": "github:NixOS/nixpkgs/6e14bbce7bea6c4efd7adfa88a40dac750d80100#k9s", + "lazydocker@latest": { + "last_modified": "2024-12-23T21:10:33Z", + "resolved": "github:NixOS/nixpkgs/de1864217bfa9b5845f465e771e0ecb48b30e02d#lazydocker", "source": "devbox-search", - "version": "0.32.5", + "version": "0.24.1", "systems": { "aarch64-darwin": { "outputs": [ { "name": "out", - "path": "/nix/store/gcbiad83pqc4xyc3qr85gc7vdzn31yvl-k9s-0.32.5", + "path": "/nix/store/5hv519gz95pwhg4vr9ka8g98bdd91mvx-lazydocker-0.24.1", "default": true } ], - "store_path": "/nix/store/gcbiad83pqc4xyc3qr85gc7vdzn31yvl-k9s-0.32.5" + "store_path": "/nix/store/5hv519gz95pwhg4vr9ka8g98bdd91mvx-lazydocker-0.24.1" }, "aarch64-linux": { "outputs": [ { "name": "out", - "path": "/nix/store/p977rq0i7cqirirnv5gzj4kdvi2gz0av-k9s-0.32.5", + "path": "/nix/store/5wb57rjn6xwyf4nc0mifgjd934i545ph-lazydocker-0.24.1", "default": true } ], - "store_path": "/nix/store/p977rq0i7cqirirnv5gzj4kdvi2gz0av-k9s-0.32.5" + "store_path": "/nix/store/5wb57rjn6xwyf4nc0mifgjd934i545ph-lazydocker-0.24.1" }, "x86_64-darwin": { "outputs": [ { "name": "out", - "path": "/nix/store/xsv5smy3931nznpilp2vlva8slzk63ps-k9s-0.32.5", + "path": "/nix/store/bj056q5lmy3dx6xafwlmvhjqgxjr3awq-lazydocker-0.24.1", "default": true } ], - "store_path": "/nix/store/xsv5smy3931nznpilp2vlva8slzk63ps-k9s-0.32.5" + "store_path": "/nix/store/bj056q5lmy3dx6xafwlmvhjqgxjr3awq-lazydocker-0.24.1" }, "x86_64-linux": { "outputs": [ { "name": "out", - "path": "/nix/store/6zmdvw89ql0ani1zjh2im6wfhm3i0c94-k9s-0.32.5", + "path": "/nix/store/4gwbhlb13pvdbxcb31mslkgd6pr62gyk-lazydocker-0.24.1", "default": true } ], - "store_path": "/nix/store/6zmdvw89ql0ani1zjh2im6wfhm3i0c94-k9s-0.32.5" - } - } - }, - "kubectl@latest": { - "last_modified": "2024-07-07T07:43:47Z", - "resolved": "github:NixOS/nixpkgs/b60793b86201040d9dee019a05089a9150d08b5b#kubectl", - "source": "devbox-search", - "version": "1.30.2", - "systems": { - "aarch64-darwin": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/i1zidf41bkfzs2l1pq9fi1frymsfgywc-kubectl-1.30.2", - "default": true - }, - { - "name": "man", - "path": "/nix/store/dzxnn9mk9plcx3w9862jyd0nxys2yywz-kubectl-1.30.2-man", - "default": true - }, - { - "name": "convert", - "path": "/nix/store/v9ij5fnxxa02jkzpjvkbxw2jc4p9cbld-kubectl-1.30.2-convert" - } - ], - "store_path": "/nix/store/i1zidf41bkfzs2l1pq9fi1frymsfgywc-kubectl-1.30.2" - }, - "aarch64-linux": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/k7ql4247qs6ny27m3iz5c9xf5gb248a2-kubectl-1.30.2", - "default": true - }, - { - "name": "man", - "path": "/nix/store/wy64r4nn3isydw4nx257h95qy2x2z4mx-kubectl-1.30.2-man", - "default": true - }, - { - "name": "convert", - "path": "/nix/store/ic8za302hvb4kf4zrs55ivr4q2n2lznn-kubectl-1.30.2-convert" - } - ], - "store_path": "/nix/store/k7ql4247qs6ny27m3iz5c9xf5gb248a2-kubectl-1.30.2" - }, - "x86_64-darwin": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/v029n959l5b289br0cq591b04yc48516-kubectl-1.30.2", - "default": true - }, - { - "name": "man", - "path": "/nix/store/0dvcxn7gsi2ycy9blb7pcy506w4xp2vi-kubectl-1.30.2-man", - "default": true - }, - { - "name": "convert", - "path": "/nix/store/2nfq4ivwa4a7jwc0183f2wpl1jxbn754-kubectl-1.30.2-convert" - } - ], - "store_path": "/nix/store/v029n959l5b289br0cq591b04yc48516-kubectl-1.30.2" - }, - "x86_64-linux": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/3vkf0406s1i6l89hk5wrakh4bbn0p1p2-kubectl-1.30.2", - "default": true - }, - { - "name": "man", - "path": "/nix/store/3wbvgkkka1knkxvyr4c8qbpr448smw8i-kubectl-1.30.2-man", - "default": true - }, - { - "name": "convert", - "path": "/nix/store/h5zxz8db6wligwhw5gnwk4gbc8j1ixik-kubectl-1.30.2-convert" - } - ], - "store_path": "/nix/store/3vkf0406s1i6l89hk5wrakh4bbn0p1p2-kubectl-1.30.2" - } - } - }, - "kubernetes-helm@latest": { - "last_modified": "2024-07-20T09:11:00Z", - "resolved": "github:NixOS/nixpkgs/6e14bbce7bea6c4efd7adfa88a40dac750d80100#kubernetes-helm", - "source": "devbox-search", - "version": "3.15.3", - "systems": { - "aarch64-darwin": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/z4w7bnylg9h3f543yrf9bcwkxzfs82z2-kubernetes-helm-3.15.3", - "default": true - } - ], - "store_path": "/nix/store/z4w7bnylg9h3f543yrf9bcwkxzfs82z2-kubernetes-helm-3.15.3" - }, - "aarch64-linux": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/aa4jksq9ljgha8plw5cqyxf60n931dir-kubernetes-helm-3.15.3", - "default": true - } - ], - "store_path": "/nix/store/aa4jksq9ljgha8plw5cqyxf60n931dir-kubernetes-helm-3.15.3" - }, - "x86_64-darwin": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/5gjk6w3agm49ljiwi991ailvmw35zq1j-kubernetes-helm-3.15.3", - "default": true - } - ], - "store_path": "/nix/store/5gjk6w3agm49ljiwi991ailvmw35zq1j-kubernetes-helm-3.15.3" - }, - "x86_64-linux": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/n4p0zh1s8jz9mqf1r1pki23kviq4waa7-kubernetes-helm-3.15.3", - "default": true - } - ], - "store_path": "/nix/store/n4p0zh1s8jz9mqf1r1pki23kviq4waa7-kubernetes-helm-3.15.3" + "store_path": "/nix/store/4gwbhlb13pvdbxcb31mslkgd6pr62gyk-lazydocker-0.24.1" } } }, @@ -793,162 +517,6 @@ } } }, - "tilt@latest": { - "last_modified": "2024-07-15T21:47:20Z", - "resolved": "github:NixOS/nixpkgs/b2c1f10bfbb3f617ea8e8669ac13f3f56ceb2ea2#tilt", - "source": "devbox-search", - "version": "0.33.17", - "systems": { - "aarch64-darwin": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/l19qinywsx7y86xp8vgwr3bgnbi0rfcj-tilt-0.33.17", - "default": true - } - ], - "store_path": "/nix/store/l19qinywsx7y86xp8vgwr3bgnbi0rfcj-tilt-0.33.17" - }, - "aarch64-linux": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/f6swxifmvnxjxifxyw4k4aiyxh0dgfyq-tilt-0.33.17", - "default": true - } - ], - "store_path": "/nix/store/f6swxifmvnxjxifxyw4k4aiyxh0dgfyq-tilt-0.33.17" - }, - "x86_64-darwin": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/d59rlsabcqxax6bgw6d30zhmflw65ch0-tilt-0.33.17", - "default": true - } - ], - "store_path": "/nix/store/d59rlsabcqxax6bgw6d30zhmflw65ch0-tilt-0.33.17" - }, - "x86_64-linux": { - "outputs": [ - { - "name": "out", - "path": "/nix/store/qfv96sjcsslynqbilwj823x8nxvgj5cv-tilt-0.33.17", - "default": true - } - ], - "store_path": "/nix/store/qfv96sjcsslynqbilwj823x8nxvgj5cv-tilt-0.33.17" - } - } - }, - "vips@latest": { - "last_modified": "2024-09-01T03:39:50Z", - "resolved": "github:NixOS/nixpkgs/4a9443e2a4e06cbaff89056b5cdf6777c1fe5755#vips", - "source": "devbox-search", - "version": "8.15.2", - "systems": { - "aarch64-darwin": { - "outputs": [ - { - "name": "bin", - "path": "/nix/store/06dc4n06zs4jygbqa8c7fg9dcg39q9j9-vips-8.15.2-bin", - "default": true - }, - { - "name": "man", - "path": "/nix/store/442bpcz5cx0x2ddc2cp4041arqf87ph0-vips-8.15.2-man", - "default": true - }, - { - "name": "dev", - "path": "/nix/store/1l68927a29x2g1kyxcf7kxsh0lpnshvn-vips-8.15.2-dev" - }, - { - "name": "out", - "path": "/nix/store/6jnr4f8i012ki83bp0aml84acfigsdqr-vips-8.15.2" - } - ], - "store_path": "/nix/store/06dc4n06zs4jygbqa8c7fg9dcg39q9j9-vips-8.15.2-bin" - }, - "aarch64-linux": { - "outputs": [ - { - "name": "bin", - "path": "/nix/store/17asmchn0w4s7gic59r75n3drqdv12cn-vips-8.15.2-bin", - "default": true - }, - { - "name": "man", - "path": "/nix/store/1syxsf45h4ppyx1q0nynx1wnl4dxrnnv-vips-8.15.2-man", - "default": true - }, - { - "name": "dev", - "path": "/nix/store/afy2j1kdch6sb3ryqljlf163khs7hwkl-vips-8.15.2-dev" - }, - { - "name": "devdoc", - "path": "/nix/store/sifwrcrirz82gv0d9n0a683qz88msfzp-vips-8.15.2-devdoc" - }, - { - "name": "out", - "path": "/nix/store/rhlb3vk0sqy6cn521806qpw1rqzji0iv-vips-8.15.2" - } - ], - "store_path": "/nix/store/17asmchn0w4s7gic59r75n3drqdv12cn-vips-8.15.2-bin" - }, - "x86_64-darwin": { - "outputs": [ - { - "name": "bin", - "path": "/nix/store/gzqa0mv4czbw0c5243r97bhw8kch8qmi-vips-8.15.2-bin", - "default": true - }, - { - "name": "man", - "path": "/nix/store/38knm1dzqngi6w517g147gnpd2q720b1-vips-8.15.2-man", - "default": true - }, - { - "name": "dev", - "path": "/nix/store/f9vf62bnkrn1azgyvcjnmgsh1jknc33b-vips-8.15.2-dev" - }, - { - "name": "out", - "path": "/nix/store/487rhwp65bxzhanzf7brma1lcbis7w6c-vips-8.15.2" - } - ], - "store_path": "/nix/store/gzqa0mv4czbw0c5243r97bhw8kch8qmi-vips-8.15.2-bin" - }, - "x86_64-linux": { - "outputs": [ - { - "name": "bin", - "path": "/nix/store/bfy830xkb34pn4dym88rhzsqvlav25bq-vips-8.15.2-bin", - "default": true - }, - { - "name": "man", - "path": "/nix/store/ppi3phzcnr8bh4q4jnz5pp2grvh1dki9-vips-8.15.2-man", - "default": true - }, - { - "name": "dev", - "path": "/nix/store/6v5dd9lz15dls9k62dqnmdpi2af1y59n-vips-8.15.2-dev" - }, - { - "name": "devdoc", - "path": "/nix/store/9kf40jgp84bkxc6202x8z4nqwzn8627v-vips-8.15.2-devdoc" - }, - { - "name": "out", - "path": "/nix/store/k6aq70ncjh3d0vjcw6r8grw764hlagn5-vips-8.15.2" - } - ], - "store_path": "/nix/store/bfy830xkb34pn4dym88rhzsqvlav25bq-vips-8.15.2-bin" - } - } - }, "yt-dlp@latest": { "last_modified": "2024-07-18T22:08:26Z", "resolved": "github:NixOS/nixpkgs/cfa5366588c940ab6ee3bee399b337175545c664#yt-dlp", diff --git a/dockerfiles/bright.dockerfile b/dockerfiles/bright.dockerfile index 3b4b59a..70864d7 100644 --- a/dockerfiles/bright.dockerfile +++ b/dockerfiles/bright.dockerfile @@ -16,27 +16,26 @@ # - https://hub.docker.com/_/debian?tab=tags&page=1&name=bullseye-20241202-slim - for the release image # - https://pkgs.org/ - resource for finding needed packages # - Ex: hexpm/elixir:1.17.3-erlang-27.1.2-debian-bullseye-20241202-slim -# -ARG ELIXIR_VERSION=1.17.3 -ARG OTP_VERSION=27.1.2 -ARG DEBIAN_VERSION=bullseye-20241202-slim - -ARG BUILDER_IMAGE="hexpm/elixir:${ELIXIR_VERSION}-erlang-${OTP_VERSION}-debian-${DEBIAN_VERSION}" -ARG RUNNER_IMAGE="debian:${DEBIAN_VERSION}" +# - https://gitlab.com/ericlathrop/dockerize_elixir/-/blob/main/priv/templates/Dockerfile?ref_type=heads - - - - -FROM ${BUILDER_IMAGE} AS dev +FROM elixir:1.17.2-alpine AS dev # install build dependencies -RUN apt-get update -y && apt-get install -y build-essential git inotify-tools \ - && apt-get clean && rm -f /var/lib/apt/lists/*_* +RUN apt-get update -y \ + && apt-get install -y build-essential \ + && apt-get install -y git \ + && apt-get install -y inotify-tools \ + && apt-get clean \ + && rm -f /var/lib/apt/lists/*_* \ + && mkdir /home/user && \ + chown 1000.1000 /home/user + +ENV HOME=/home/user +USER 1000:1000 # prepare build dir WORKDIR /app @@ -47,6 +46,8 @@ RUN mix local.hex --force && \ # set build ENV ENV MIX_ENV="dev" +ENV DATABASE_URL="" +ENV PORT="" # install mix dependencies COPY ./services/bright/mix.exs ./services/bright/mix.lock ./ @@ -64,47 +65,4 @@ COPY ./services/bright/lib lib COPY ./services/bright/assets assets - -CMD ["mix", "phx.server"] - - - - - - - - - - -# # start a new build stage so that the final image will only contain -# # the compiled release and other runtime necessities -# FROM ${RUNNER_IMAGE} - -# RUN apt-get update -y && \ -# apt-get install -y libstdc++6 openssl libncurses5 locales ca-certificates \ -# && apt-get clean && rm -f /var/lib/apt/lists/*_* - -# # Set the locale -# RUN sed -i '/en_US.UTF-8/s/^# //g' /etc/locale.gen && locale-gen - -# ENV LANG en_US.UTF-8 -# ENV LANGUAGE en_US:en -# ENV LC_ALL en_US.UTF-8 - -# WORKDIR "/app" -# RUN chown nobody /app - -# # set runner ENV -# ENV MIX_ENV="prod" - -# # Only copy the final release from the build stage -# COPY --from=builder --chown=nobody:root /app/_build/${MIX_ENV}/rel/bright ./ - -# USER nobody - -# # If using an environment that doesn't automatically reap zombie processes, it is -# # advised to add an init process such as tini via `apt-get install` -# # above and adding an entrypoint. See https://github.com/krallin/tini for details -# # ENTRYPOINT ["/tini", "--"] - -# CMD ["/app/bin/server"] +CMD [ "mix", "ecto.setup", "&&", "mix", "phx.server" ] diff --git a/flux/README.md b/flux/README.md deleted file mode 100644 index 8d52bed..0000000 --- a/flux/README.md +++ /dev/null @@ -1,3 +0,0 @@ -# Futureporn flux - -Gitops https://fluxcd.io/flux/get-started/ \ No newline at end of file diff --git a/flux/apps/base/bitnami/kustomization.yaml b/flux/apps/base/bitnami/kustomization.yaml deleted file mode 100644 index 2bac203..0000000 --- a/flux/apps/base/bitnami/kustomization.yaml +++ /dev/null @@ -1,5 +0,0 @@ -apiVersion: kustomize.config.k8s.io/v1beta1 -kind: Kustomization -namespace: futureporn -resources: - - repository.yaml diff --git a/flux/apps/base/bitnami/repository.yaml b/flux/apps/base/bitnami/repository.yaml deleted file mode 100644 index 8d2af4c..0000000 --- a/flux/apps/base/bitnami/repository.yaml +++ /dev/null @@ -1,8 +0,0 @@ -apiVersion: source.toolkit.fluxcd.io/v1beta2 -kind: HelmRepository -metadata: - name: bitnami - namespace: futureporn -spec: - interval: 5m - url: https://charts.bitnami.com/bitnami diff --git a/flux/apps/base/drupal/kustomization.yaml b/flux/apps/base/drupal/kustomization.yaml deleted file mode 100644 index 7d4eed5..0000000 --- a/flux/apps/base/drupal/kustomization.yaml +++ /dev/null @@ -1,5 +0,0 @@ -apiVersion: kustomize.config.k8s.io/v1beta1 -kind: Kustomization -namespace: futureporn -resources: - - release.yaml diff --git a/flux/apps/base/drupal/release.yaml b/flux/apps/base/drupal/release.yaml deleted file mode 100644 index f0cb671..0000000 --- a/flux/apps/base/drupal/release.yaml +++ /dev/null @@ -1,25 +0,0 @@ -apiVersion: helm.toolkit.fluxcd.io/v2 -kind: HelmRelease -metadata: - namespace: futureporn - name: drupal -spec: - releaseName: drupal - chart: - spec: - version: "20.0.10" - chart: drupal - sourceRef: - kind: HelmRepository - name: bitnami - interval: 5m - install: - remediation: - retries: -1 - # Default values - # https://artifacthub.io/packages/helm/bitnami/drupal?modal=values - values: - fullnameOverride: drupal - mariadb: - enabled: true - allowEmptyPassword: true diff --git a/flux/apps/base/fp/kustomization.yaml b/flux/apps/base/fp/kustomization.yaml deleted file mode 100644 index c9b68af..0000000 --- a/flux/apps/base/fp/kustomization.yaml +++ /dev/null @@ -1,6 +0,0 @@ -apiVersion: kustomize.config.k8s.io/v1beta1 -kind: Kustomization -namespace: futureporn -resources: - - repository.yaml - - release.yaml diff --git a/flux/apps/base/fp/release.yaml b/flux/apps/base/fp/release.yaml deleted file mode 100644 index 77201fa..0000000 --- a/flux/apps/base/fp/release.yaml +++ /dev/null @@ -1,57 +0,0 @@ -apiVersion: helm.toolkit.fluxcd.io/v2 -kind: HelmRelease -metadata: - name: fp - namespace: futureporn - annotations: - meta.helm.sh/release-name: fp - meta.helm.sh/release-namespace: futureporn - labels: - app.kubernetes.io/managed-by: Helm -spec: - chart: - spec: - chart: ./charts/fp - ignoreMissingValuesFiles: false - sourceRef: - kind: GitRepository - name: fp - interval: 1m - install: - remediation: - retries: -1 - values: - storageClassName: vultr-block-storage-hdd - link2cid: - imageName: gitea.futureporn.net/futureporn/link2cid:latest - next: - imageName: gitea.futureporn.net/futureporn/next:latest - hostname: next.futureporn.svc.cluster.local - ingressClassName: nginx - capture: - imageName: gitea.futureporn.net/futureporn/capture:latest - worker: - imageName: gitea.futureporn.net/futureporn/worker:latest - replicas: 2 - scout: - manager: - imageName: gitea.futureporn.net/futureporn/scout-manager:latest - pubsubServerUrl: https://realtime.futureporn.svc.cluster.local/faye - hostname: next.futureporn.svc.cluster.local - cdnBucketUrl: https://fp-dev.b-cdn.net - s3BucketName: fp-dev - imageName: fp/scout - strapi: - imageName: fp/strapi - port: 1339 - url: https://strapi.fp.sbtp.xyz - hostname: strapi.futureporn.svc.cluster.local - ingressClassName: nginx - realtime: - imageName: gitea.futureporn.net/futureporn/realtime:latest - adminEmail: cj@futureporn.net - certManager: - issuer: letsencrypt-staging - echo: - hostname: echo.fp.sbtp.xyz - \ No newline at end of file diff --git a/flux/apps/base/fp/repository.yaml b/flux/apps/base/fp/repository.yaml deleted file mode 100644 index 88b5528..0000000 --- a/flux/apps/base/fp/repository.yaml +++ /dev/null @@ -1,15 +0,0 @@ -apiVersion: source.toolkit.fluxcd.io/v1 -kind: GitRepository -metadata: - name: fp - namespace: futureporn -spec: - interval: 1m - url: https://gitea.futureporn.net/futureporn/fp - ref: - branch: main - ignore: | - # exclude all - /* - # include charts directory - !/charts/ diff --git a/flux/apps/base/mariadb/kustomization.yaml b/flux/apps/base/mariadb/kustomization.yaml deleted file mode 100644 index 7d4eed5..0000000 --- a/flux/apps/base/mariadb/kustomization.yaml +++ /dev/null @@ -1,5 +0,0 @@ -apiVersion: kustomize.config.k8s.io/v1beta1 -kind: Kustomization -namespace: futureporn -resources: - - release.yaml diff --git a/flux/apps/base/mariadb/release.yaml b/flux/apps/base/mariadb/release.yaml deleted file mode 100644 index 7220dc6..0000000 --- a/flux/apps/base/mariadb/release.yaml +++ /dev/null @@ -1,23 +0,0 @@ -apiVersion: helm.toolkit.fluxcd.io/v2 -kind: HelmRelease -metadata: - namespace: futureporn - name: mariadb -spec: - releaseName: mariadb - chart: - spec: - version: "19.0.1" - chart: mariadb - sourceRef: - kind: HelmRepository - name: bitnami - interval: 5m - install: - remediation: - retries: -1 - # Default values - # https://artifacthub.io/packages/helm/bitnami/mariadb?modal=values - values: - fullnameOverride: mariadb-via-kustomize - diff --git a/flux/apps/base/podinfo/kustomization.yaml b/flux/apps/base/podinfo/kustomization.yaml deleted file mode 100644 index 026ca9d..0000000 --- a/flux/apps/base/podinfo/kustomization.yaml +++ /dev/null @@ -1,7 +0,0 @@ -apiVersion: kustomize.config.k8s.io/v1beta1 -kind: Kustomization -namespace: futureporn -resources: - - namespace.yaml - - repository.yaml - - release.yaml diff --git a/flux/apps/base/podinfo/namespace.yaml b/flux/apps/base/podinfo/namespace.yaml deleted file mode 100644 index 5128776..0000000 --- a/flux/apps/base/podinfo/namespace.yaml +++ /dev/null @@ -1,4 +0,0 @@ -apiVersion: v1 -kind: Namespace -metadata: - name: podinfo diff --git a/flux/apps/base/podinfo/release.yaml b/flux/apps/base/podinfo/release.yaml deleted file mode 100644 index 01871d5..0000000 --- a/flux/apps/base/podinfo/release.yaml +++ /dev/null @@ -1,39 +0,0 @@ -apiVersion: helm.toolkit.fluxcd.io/v2 -kind: HelmRelease -metadata: - name: podinfo - namespace: futureporn -spec: - releaseName: podinfo - chart: - spec: - version: '6.6.3' - chart: podinfo - sourceRef: - kind: HelmRepository - name: podinfo - interval: 10m - install: - remediation: - retries: 3 - # Default values - # https://github.com/stefanprodan/podinfo/blob/master/charts/podinfo/values.yaml - values: - redis: - enabled: true - ingress: - enabled: true - className: "nginx" - annotations: - external-dns.alpha.kubernetes.io/hostname: podinfo.fp.sbtp.xyz - # cert-manager.io/cluster-issuer: "letsencrypt-staging" - # ingress.kubernetes.io/ssl-redirect: "true" - hosts: - - host: podinfo.fp.sbtp.xyz - paths: - - path: / - pathType: Prefix - tls: - - secretName: podinfo-tls - hosts: - - podinfo.fp.sbtp.xyz diff --git a/flux/apps/base/podinfo/repository.yaml b/flux/apps/base/podinfo/repository.yaml deleted file mode 100644 index e678008..0000000 --- a/flux/apps/base/podinfo/repository.yaml +++ /dev/null @@ -1,8 +0,0 @@ -apiVersion: source.toolkit.fluxcd.io/v1beta2 -kind: HelmRepository -metadata: - name: podinfo - namespace: futureporn -spec: - interval: 5m - url: https://stefanprodan.github.io/podinfo diff --git a/flux/apps/base/postgresql/kustomization.yaml b/flux/apps/base/postgresql/kustomization.yaml deleted file mode 100644 index 7d4eed5..0000000 --- a/flux/apps/base/postgresql/kustomization.yaml +++ /dev/null @@ -1,5 +0,0 @@ -apiVersion: kustomize.config.k8s.io/v1beta1 -kind: Kustomization -namespace: futureporn -resources: - - release.yaml diff --git a/flux/apps/base/postgresql/release.yaml b/flux/apps/base/postgresql/release.yaml deleted file mode 100644 index 6bd646b..0000000 --- a/flux/apps/base/postgresql/release.yaml +++ /dev/null @@ -1,36 +0,0 @@ -apiVersion: helm.toolkit.fluxcd.io/v2 -kind: HelmRelease -metadata: - namespace: futureporn - name: postgresql -spec: - releaseName: postgresql - chart: - spec: - version: "15.5.17" - chart: postgresql - sourceRef: - kind: HelmRepository - name: bitnami - interval: 5m - install: - remediation: - retries: -1 - # Default values - # https://artifacthub.io/packages/helm/bitnami/postgresql?modal=values - values: - auth: - enablePostgresUser: true - existingSecret: postgres - architecture: replication - readReplicas: - replicaCount: 3 - persistence: - size: 40Gi ## Vultr HDD minimum size - replication: - syncronousCommit: "on" - numSyncronousReplicas: 3 - applicationName: futureporn - primary: - persistence: - size: 40Gi ## Vultr HDD minimum size diff --git a/flux/apps/base/redis/kustomization.yaml b/flux/apps/base/redis/kustomization.yaml deleted file mode 100644 index 7d4eed5..0000000 --- a/flux/apps/base/redis/kustomization.yaml +++ /dev/null @@ -1,5 +0,0 @@ -apiVersion: kustomize.config.k8s.io/v1beta1 -kind: Kustomization -namespace: futureporn -resources: - - release.yaml diff --git a/flux/apps/base/redis/release.yaml b/flux/apps/base/redis/release.yaml deleted file mode 100644 index ef730cc..0000000 --- a/flux/apps/base/redis/release.yaml +++ /dev/null @@ -1,30 +0,0 @@ -apiVersion: helm.toolkit.fluxcd.io/v2 -kind: HelmRelease -metadata: - namespace: futureporn - name: redis -spec: - releaseName: redis - chart: - spec: - version: "19.6.2" - chart: redis - sourceRef: - kind: HelmRepository - name: bitnami - interval: 5m - install: - remediation: - retries: -1 - ## see https://artifacthub.io/packages/helm/bitnami/redis?modal=values - values: - master: - persistence: - enabled: false - auth: - existingSecret: redis - existingSecretPasswordKey: password - replica: - persistence: - enabled: false - architecture: standalone \ No newline at end of file diff --git a/flux/apps/development/README.md b/flux/apps/development/README.md deleted file mode 100644 index 7275cc3..0000000 --- a/flux/apps/development/README.md +++ /dev/null @@ -1,10 +0,0 @@ -Right now, we use Tilt for dev, flux for prod. - -It would be very nice to have only one way of deploying things regardless of the environment (development|staging|production) - -however, this doesn't work because fp Helm chart is pulled from git rather than a folder relative to the flux dir. (kind: GitRepository) - -We are constrained to CrossNamespaceObjectReference kind list - -https://fluxcd.io/flux/components/helm/api/v2/#helm.toolkit.fluxcd.io/v2.CrossNamespaceObjectReference -https://github.com/fluxcd/helm-controller/blob/c8ae4b6ad225d37b19bacb634db784d6096908ac/api/v2beta2/reference_types.go#L53 \ No newline at end of file diff --git a/flux/apps/development/fp-values.yaml b/flux/apps/development/fp-values.yaml deleted file mode 100644 index 2379d25..0000000 --- a/flux/apps/development/fp-values.yaml +++ /dev/null @@ -1,20 +0,0 @@ -apiVersion: helm.toolkit.fluxcd.io/v2 -kind: HelmRelease -metadata: - name: fp - namespace: futureporn -spec: - values: - next: - hostname: next.fp.sbtp.xyz - scout: - cdnBucketUrl: https://fp-dev.b-cdn.net - s3BucketName: fp-dev - imageName: gitea.futureporn.net/futureporn/scout:latest - strapi: - url: https://strapi.sbtp.xyz - hostname: strapi.fp.sbtp.xyz - imageName: gitea.futureporn.net/futureporn/strapi:latest - managedBy: Helm - certManager: - issuer: letsencrypt-staging \ No newline at end of file diff --git a/flux/apps/development/kustomization.yaml b/flux/apps/development/kustomization.yaml deleted file mode 100644 index 7486b38..0000000 --- a/flux/apps/development/kustomization.yaml +++ /dev/null @@ -1,19 +0,0 @@ -apiVersion: kustomize.config.k8s.io/v1beta1 -kind: Kustomization -namespace: futureporn -resources: - - ../base/bitnami - - ../base/podinfo - - ../base/postgresql - - ../base/mariadb - - ../base/drupal - # - ../base/fp -patches: - # - path: podinfo-values.yaml - # target: - # kind: HelmRelease - # name: podinfo - - path: fp-values.yaml - target: - kind: HelmRelease - name: fp diff --git a/flux/apps/development/podinfo-values.yaml b/flux/apps/development/podinfo-values.yaml deleted file mode 100644 index 174cc82..0000000 --- a/flux/apps/development/podinfo-values.yaml +++ /dev/null @@ -1,19 +0,0 @@ -apiVersion: helm.toolkit.fluxcd.io/v2 -kind: HelmRelease -metadata: - name: podinfo - namespace: futureporn -spec: - chart: - spec: - version: ">=1.0.0-alpha" - test: - enable: false - values: - fullnameOverride: mariadb-taco - ingress: - hosts: - - host: podinfo.fp.sbtp.xyz - paths: - - path: / - pathType: ImplementationSpecific diff --git a/flux/apps/production/fp-values.yaml b/flux/apps/production/fp-values.yaml deleted file mode 100644 index e15715f..0000000 --- a/flux/apps/production/fp-values.yaml +++ /dev/null @@ -1,18 +0,0 @@ -apiVersion: helm.toolkit.fluxcd.io/v2 -kind: HelmRelease -metadata: - name: fp - namespace: futureporn -spec: - values: - next: - certIssuer: letsencrypt-prod - hostname: next.futureporn.net - scout: - cdnBucketUrl: https://futureporn-b2.b-cdn.net - s3BucketName: futureporn-b2 - strapi: - url: https://portal.futureporn.net - hostname: portal.futureporn.net - certManager: - issuer: letsencrypt-production \ No newline at end of file diff --git a/flux/apps/production/kustomization.yaml b/flux/apps/production/kustomization.yaml deleted file mode 100644 index 1e2099c..0000000 --- a/flux/apps/production/kustomization.yaml +++ /dev/null @@ -1,13 +0,0 @@ -apiVersion: kustomize.config.k8s.io/v1beta1 -kind: Kustomization -namespace: futureporn -resources: - - ../base/bitnami - - ../base/redis - - ../base/postgresql - - ../base/fp -patches: - - path: fp-values.yaml - target: - kind: HelmRelease - name: fp diff --git a/flux/apps/staging/fp-values.yaml b/flux/apps/staging/fp-values.yaml deleted file mode 100644 index 2379d25..0000000 --- a/flux/apps/staging/fp-values.yaml +++ /dev/null @@ -1,20 +0,0 @@ -apiVersion: helm.toolkit.fluxcd.io/v2 -kind: HelmRelease -metadata: - name: fp - namespace: futureporn -spec: - values: - next: - hostname: next.fp.sbtp.xyz - scout: - cdnBucketUrl: https://fp-dev.b-cdn.net - s3BucketName: fp-dev - imageName: gitea.futureporn.net/futureporn/scout:latest - strapi: - url: https://strapi.sbtp.xyz - hostname: strapi.fp.sbtp.xyz - imageName: gitea.futureporn.net/futureporn/strapi:latest - managedBy: Helm - certManager: - issuer: letsencrypt-staging \ No newline at end of file diff --git a/flux/apps/staging/kustomization.yaml b/flux/apps/staging/kustomization.yaml deleted file mode 100644 index 0a9ec56..0000000 --- a/flux/apps/staging/kustomization.yaml +++ /dev/null @@ -1,18 +0,0 @@ -apiVersion: kustomize.config.k8s.io/v1beta1 -kind: Kustomization -namespace: futureporn -resources: - - ../base/bitnami - - ../base/redis - - ../base/podinfo - - ../base/postgresql - - ../base/fp -patches: - # - path: podinfo-values.yaml - # target: - # kind: HelmRelease - # name: podinfo - - path: fp-values.yaml - target: - kind: HelmRelease - name: fp diff --git a/flux/apps/staging/podinfo-values.yaml b/flux/apps/staging/podinfo-values.yaml deleted file mode 100644 index b0ee71a..0000000 --- a/flux/apps/staging/podinfo-values.yaml +++ /dev/null @@ -1,18 +0,0 @@ -apiVersion: helm.toolkit.fluxcd.io/v2 -kind: HelmRelease -metadata: - name: podinfo - namespace: futureporn -spec: - chart: - spec: - version: ">=1.0.0-alpha" - test: - enable: false - values: - ingress: - hosts: - - host: podinfo.fp.sbtp.xyz - paths: - - path: / - pathType: ImplementationSpecific diff --git a/flux/clusters/production/apps.yaml b/flux/clusters/production/apps.yaml deleted file mode 100644 index 0495f92..0000000 --- a/flux/clusters/production/apps.yaml +++ /dev/null @@ -1,17 +0,0 @@ ---- -apiVersion: kustomize.toolkit.fluxcd.io/v1 -kind: Kustomization -metadata: - name: apps - namespace: flux-system -spec: - interval: 10m0s - dependsOn: - - name: infra-configs - sourceRef: - kind: GitRepository - name: flux-system - path: ./apps/production - prune: true - wait: true - timeout: 5m0s diff --git a/flux/clusters/production/flux-system/gotk-components.yaml b/flux/clusters/production/flux-system/gotk-components.yaml deleted file mode 100644 index 8fd99ab..0000000 --- a/flux/clusters/production/flux-system/gotk-components.yaml +++ /dev/null @@ -1,12385 +0,0 @@ ---- -# This manifest was generated by flux. DO NOT EDIT. -# Flux Version: v2.3.0 -# Components: source-controller,kustomize-controller,helm-controller,notification-controller -apiVersion: v1 -kind: Namespace -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - pod-security.kubernetes.io/warn: restricted - pod-security.kubernetes.io/warn-version: latest - name: flux-system ---- -apiVersion: networking.k8s.io/v1 -kind: NetworkPolicy -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: allow-egress - namespace: flux-system -spec: - egress: - - {} - ingress: - - from: - - podSelector: {} - podSelector: {} - policyTypes: - - Ingress - - Egress ---- -apiVersion: networking.k8s.io/v1 -kind: NetworkPolicy -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: allow-scraping - namespace: flux-system -spec: - ingress: - - from: - - namespaceSelector: {} - ports: - - port: 8080 - protocol: TCP - podSelector: {} - policyTypes: - - Ingress ---- -apiVersion: networking.k8s.io/v1 -kind: NetworkPolicy -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: allow-webhooks - namespace: flux-system -spec: - ingress: - - from: - - namespaceSelector: {} - podSelector: - matchLabels: - app: notification-controller - policyTypes: - - Ingress ---- -apiVersion: v1 -kind: ResourceQuota -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: critical-pods-flux-system - namespace: flux-system -spec: - hard: - pods: "1000" - scopeSelector: - matchExpressions: - - operator: In - scopeName: PriorityClass - values: - - system-node-critical - - system-cluster-critical ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: crd-controller-flux-system -rules: -- apiGroups: - - source.toolkit.fluxcd.io - resources: - - '*' - verbs: - - '*' -- apiGroups: - - kustomize.toolkit.fluxcd.io - resources: - - '*' - verbs: - - '*' -- apiGroups: - - helm.toolkit.fluxcd.io - resources: - - '*' - verbs: - - '*' -- apiGroups: - - notification.toolkit.fluxcd.io - resources: - - '*' - verbs: - - '*' -- apiGroups: - - image.toolkit.fluxcd.io - resources: - - '*' - verbs: - - '*' -- apiGroups: - - "" - resources: - - namespaces - - secrets - - configmaps - - serviceaccounts - verbs: - - get - - list - - watch -- apiGroups: - - "" - resources: - - events - verbs: - - create - - patch -- apiGroups: - - "" - resources: - - configmaps - verbs: - - get - - list - - watch - - create - - update - - patch - - delete -- apiGroups: - - "" - resources: - - configmaps/status - verbs: - - get - - update - - patch -- apiGroups: - - coordination.k8s.io - resources: - - leases - verbs: - - get - - list - - watch - - create - - update - - patch - - delete -- nonResourceURLs: - - /livez/ping - verbs: - - head ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - rbac.authorization.k8s.io/aggregate-to-admin: "true" - rbac.authorization.k8s.io/aggregate-to-edit: "true" - name: flux-edit-flux-system -rules: -- apiGroups: - - notification.toolkit.fluxcd.io - - source.toolkit.fluxcd.io - - helm.toolkit.fluxcd.io - - image.toolkit.fluxcd.io - - kustomize.toolkit.fluxcd.io - resources: - - '*' - verbs: - - create - - delete - - deletecollection - - patch - - update ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - rbac.authorization.k8s.io/aggregate-to-admin: "true" - rbac.authorization.k8s.io/aggregate-to-edit: "true" - rbac.authorization.k8s.io/aggregate-to-view: "true" - name: flux-view-flux-system -rules: -- apiGroups: - - notification.toolkit.fluxcd.io - - source.toolkit.fluxcd.io - - helm.toolkit.fluxcd.io - - image.toolkit.fluxcd.io - - kustomize.toolkit.fluxcd.io - resources: - - '*' - verbs: - - get - - list - - watch ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: cluster-reconciler-flux-system -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: cluster-admin -subjects: -- kind: ServiceAccount - name: kustomize-controller - namespace: flux-system -- kind: ServiceAccount - name: helm-controller - namespace: flux-system ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: crd-controller-flux-system -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: crd-controller-flux-system -subjects: -- kind: ServiceAccount - name: kustomize-controller - namespace: flux-system -- kind: ServiceAccount - name: helm-controller - namespace: flux-system -- kind: ServiceAccount - name: source-controller - namespace: flux-system -- kind: ServiceAccount - name: notification-controller - namespace: flux-system -- kind: ServiceAccount - name: image-reflector-controller - namespace: flux-system -- kind: ServiceAccount - name: image-automation-controller - namespace: flux-system ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: buckets.source.toolkit.fluxcd.io -spec: - group: source.toolkit.fluxcd.io - names: - kind: Bucket - listKind: BucketList - plural: buckets - singular: bucket - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.endpoint - name: Endpoint - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - deprecated: true - deprecationWarning: v1beta1 Bucket is deprecated, upgrade to v1beta2 - name: v1beta1 - schema: - openAPIV3Schema: - description: Bucket is the Schema for the buckets API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: BucketSpec defines the desired state of an S3 compatible - bucket - properties: - accessFrom: - description: AccessFrom defines an Access Control List for allowing - cross-namespace references to this object. - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - bucketName: - description: The bucket name. - type: string - endpoint: - description: The bucket endpoint address. - type: string - ignore: - description: |- - Ignore overrides the set of excluded patterns in the .sourceignore format - (which is the same as .gitignore). If not provided, a default will be used, - consult the documentation for your version to find out what those are. - type: string - insecure: - description: Insecure allows connecting to a non-TLS S3 HTTP endpoint. - type: boolean - interval: - description: The interval at which to check for bucket updates. - type: string - provider: - default: generic - description: The S3 compatible storage provider name, default ('generic'). - enum: - - generic - - aws - - gcp - type: string - region: - description: The bucket region. - type: string - secretRef: - description: |- - The name of the secret containing authentication credentials - for the Bucket. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: This flag tells the controller to suspend the reconciliation - of this source. - type: boolean - timeout: - default: 60s - description: The timeout for download operations, defaults to 60s. - type: string - required: - - bucketName - - endpoint - - interval - type: object - status: - default: - observedGeneration: -1 - description: BucketStatus defines the observed state of a bucket - properties: - artifact: - description: Artifact represents the output of the last successful - Bucket sync. - properties: - checksum: - description: Checksum is the SHA256 checksum of the artifact. - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of this - artifact. - format: date-time - type: string - path: - description: Path is the relative file path of this artifact. - type: string - revision: - description: |- - Revision is a human readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm index timestamp, a Helm - chart version, etc. - type: string - url: - description: URL is the HTTP address of this artifact. - type: string - required: - - path - - url - type: object - conditions: - description: Conditions holds the conditions for the Bucket. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - url: - description: URL is the download link for the artifact output of the - last Bucket sync. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.endpoint - name: Endpoint - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - name: v1beta2 - schema: - openAPIV3Schema: - description: Bucket is the Schema for the buckets API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - BucketSpec specifies the required configuration to produce an Artifact for - an object storage bucket. - properties: - accessFrom: - description: |- - AccessFrom specifies an Access Control List for allowing cross-namespace - references to this object. - NOTE: Not implemented, provisional as of https://github.com/fluxcd/flux2/pull/2092 - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - bucketName: - description: BucketName is the name of the object storage bucket. - type: string - endpoint: - description: Endpoint is the object storage address the BucketName - is located at. - type: string - ignore: - description: |- - Ignore overrides the set of excluded patterns in the .sourceignore format - (which is the same as .gitignore). If not provided, a default will be used, - consult the documentation for your version to find out what those are. - type: string - insecure: - description: Insecure allows connecting to a non-TLS HTTP Endpoint. - type: boolean - interval: - description: |- - Interval at which the Bucket Endpoint is checked for updates. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - prefix: - description: Prefix to use for server-side filtering of files in the - Bucket. - type: string - provider: - default: generic - description: |- - Provider of the object storage bucket. - Defaults to 'generic', which expects an S3 (API) compatible object - storage. - enum: - - generic - - aws - - gcp - - azure - type: string - region: - description: Region of the Endpoint where the BucketName is located - in. - type: string - secretRef: - description: |- - SecretRef specifies the Secret containing authentication credentials - for the Bucket. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend the reconciliation of this - Bucket. - type: boolean - timeout: - default: 60s - description: Timeout for fetch operations, defaults to 60s. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - required: - - bucketName - - endpoint - - interval - type: object - status: - default: - observedGeneration: -1 - description: BucketStatus records the observed state of a Bucket. - properties: - artifact: - description: Artifact represents the last successful Bucket reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the Bucket. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation of - the Bucket object. - format: int64 - type: integer - observedIgnore: - description: |- - ObservedIgnore is the observed exclusion patterns used for constructing - the source artifact. - type: string - url: - description: |- - URL is the dynamic fetch link for the latest Artifact. - It is provided on a "best effort" basis, and using the precise - BucketStatus.Artifact data is recommended. - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: gitrepositories.source.toolkit.fluxcd.io -spec: - group: source.toolkit.fluxcd.io - names: - kind: GitRepository - listKind: GitRepositoryList - plural: gitrepositories - shortNames: - - gitrepo - singular: gitrepository - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.url - name: URL - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - name: v1 - schema: - openAPIV3Schema: - description: GitRepository is the Schema for the gitrepositories API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - GitRepositorySpec specifies the required configuration to produce an - Artifact for a Git repository. - properties: - ignore: - description: |- - Ignore overrides the set of excluded patterns in the .sourceignore format - (which is the same as .gitignore). If not provided, a default will be used, - consult the documentation for your version to find out what those are. - type: string - include: - description: |- - Include specifies a list of GitRepository resources which Artifacts - should be included in the Artifact produced for this GitRepository. - items: - description: |- - GitRepositoryInclude specifies a local reference to a GitRepository which - Artifact (sub-)contents must be included, and where they should be placed. - properties: - fromPath: - description: |- - FromPath specifies the path to copy contents from, defaults to the root - of the Artifact. - type: string - repository: - description: |- - GitRepositoryRef specifies the GitRepository which Artifact contents - must be included. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - toPath: - description: |- - ToPath specifies the path to copy contents to, defaults to the name of - the GitRepositoryRef. - type: string - required: - - repository - type: object - type: array - interval: - description: |- - Interval at which the GitRepository URL is checked for updates. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - proxySecretRef: - description: |- - ProxySecretRef specifies the Secret containing the proxy configuration - to use while communicating with the Git server. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - recurseSubmodules: - description: |- - RecurseSubmodules enables the initialization of all submodules within - the GitRepository as cloned from the URL, using their default settings. - type: boolean - ref: - description: |- - Reference specifies the Git reference to resolve and monitor for - changes, defaults to the 'master' branch. - properties: - branch: - description: Branch to check out, defaults to 'master' if no other - field is defined. - type: string - commit: - description: |- - Commit SHA to check out, takes precedence over all reference fields. - - - This can be combined with Branch to shallow clone the branch, in which - the commit is expected to exist. - type: string - name: - description: |- - Name of the reference to check out; takes precedence over Branch, Tag and SemVer. - - - It must be a valid Git reference: https://git-scm.com/docs/git-check-ref-format#_description - Examples: "refs/heads/main", "refs/tags/v0.1.0", "refs/pull/420/head", "refs/merge-requests/1/head" - type: string - semver: - description: SemVer tag expression to check out, takes precedence - over Tag. - type: string - tag: - description: Tag to check out, takes precedence over Branch. - type: string - type: object - secretRef: - description: |- - SecretRef specifies the Secret containing authentication credentials for - the GitRepository. - For HTTPS repositories the Secret must contain 'username' and 'password' - fields for basic auth or 'bearerToken' field for token auth. - For SSH repositories the Secret must contain 'identity' - and 'known_hosts' fields. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend the reconciliation of this - GitRepository. - type: boolean - timeout: - default: 60s - description: Timeout for Git operations like cloning, defaults to - 60s. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - url: - description: URL specifies the Git repository URL, it can be an HTTP/S - or SSH address. - pattern: ^(http|https|ssh)://.*$ - type: string - verify: - description: |- - Verification specifies the configuration to verify the Git commit - signature(s). - properties: - mode: - default: HEAD - description: |- - Mode specifies which Git object(s) should be verified. - - - The variants "head" and "HEAD" both imply the same thing, i.e. verify - the commit that the HEAD of the Git repository points to. The variant - "head" solely exists to ensure backwards compatibility. - enum: - - head - - HEAD - - Tag - - TagAndHEAD - type: string - secretRef: - description: |- - SecretRef specifies the Secret containing the public keys of trusted Git - authors. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - secretRef - type: object - required: - - interval - - url - type: object - status: - default: - observedGeneration: -1 - description: GitRepositoryStatus records the observed state of a Git repository. - properties: - artifact: - description: Artifact represents the last successful GitRepository - reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the GitRepository. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - includedArtifacts: - description: |- - IncludedArtifacts contains a list of the last successfully included - Artifacts as instructed by GitRepositorySpec.Include. - items: - description: Artifact represents the output of a Source reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of - ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI - annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: |- - ObservedGeneration is the last observed generation of the GitRepository - object. - format: int64 - type: integer - observedIgnore: - description: |- - ObservedIgnore is the observed exclusion patterns used for constructing - the source artifact. - type: string - observedInclude: - description: |- - ObservedInclude is the observed list of GitRepository resources used to - produce the current Artifact. - items: - description: |- - GitRepositoryInclude specifies a local reference to a GitRepository which - Artifact (sub-)contents must be included, and where they should be placed. - properties: - fromPath: - description: |- - FromPath specifies the path to copy contents from, defaults to the root - of the Artifact. - type: string - repository: - description: |- - GitRepositoryRef specifies the GitRepository which Artifact contents - must be included. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - toPath: - description: |- - ToPath specifies the path to copy contents to, defaults to the name of - the GitRepositoryRef. - type: string - required: - - repository - type: object - type: array - observedRecurseSubmodules: - description: |- - ObservedRecurseSubmodules is the observed resource submodules - configuration used to produce the current Artifact. - type: boolean - sourceVerificationMode: - description: |- - SourceVerificationMode is the last used verification mode indicating - which Git object(s) have been verified. - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.url - name: URL - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - deprecated: true - deprecationWarning: v1beta1 GitRepository is deprecated, upgrade to v1 - name: v1beta1 - schema: - openAPIV3Schema: - description: GitRepository is the Schema for the gitrepositories API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: GitRepositorySpec defines the desired state of a Git repository. - properties: - accessFrom: - description: AccessFrom defines an Access Control List for allowing - cross-namespace references to this object. - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - gitImplementation: - default: go-git - description: |- - Determines which git client library to use. - Defaults to go-git, valid values are ('go-git', 'libgit2'). - enum: - - go-git - - libgit2 - type: string - ignore: - description: |- - Ignore overrides the set of excluded patterns in the .sourceignore format - (which is the same as .gitignore). If not provided, a default will be used, - consult the documentation for your version to find out what those are. - type: string - include: - description: Extra git repositories to map into the repository - items: - description: GitRepositoryInclude defines a source with a from and - to path. - properties: - fromPath: - description: The path to copy contents from, defaults to the - root directory. - type: string - repository: - description: Reference to a GitRepository to include. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - toPath: - description: The path to copy contents to, defaults to the name - of the source ref. - type: string - required: - - repository - type: object - type: array - interval: - description: The interval at which to check for repository updates. - type: string - recurseSubmodules: - description: |- - When enabled, after the clone is created, initializes all submodules within, - using their default settings. - This option is available only when using the 'go-git' GitImplementation. - type: boolean - ref: - description: |- - The Git reference to checkout and monitor for changes, defaults to - master branch. - properties: - branch: - description: The Git branch to checkout, defaults to master. - type: string - commit: - description: The Git commit SHA to checkout, if specified Tag - filters will be ignored. - type: string - semver: - description: The Git tag semver expression, takes precedence over - Tag. - type: string - tag: - description: The Git tag to checkout, takes precedence over Branch. - type: string - type: object - secretRef: - description: |- - The secret name containing the Git credentials. - For HTTPS repositories the secret must contain username and password - fields. - For SSH repositories the secret must contain identity and known_hosts - fields. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: This flag tells the controller to suspend the reconciliation - of this source. - type: boolean - timeout: - default: 60s - description: The timeout for remote Git operations like cloning, defaults - to 60s. - type: string - url: - description: The repository URL, can be a HTTP/S or SSH address. - pattern: ^(http|https|ssh)://.*$ - type: string - verify: - description: Verify OpenPGP signature for the Git commit HEAD points - to. - properties: - mode: - description: Mode describes what git object should be verified, - currently ('head'). - enum: - - head - type: string - secretRef: - description: The secret name containing the public keys of all - trusted Git authors. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - mode - type: object - required: - - interval - - url - type: object - status: - default: - observedGeneration: -1 - description: GitRepositoryStatus defines the observed state of a Git repository. - properties: - artifact: - description: Artifact represents the output of the last successful - repository sync. - properties: - checksum: - description: Checksum is the SHA256 checksum of the artifact. - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of this - artifact. - format: date-time - type: string - path: - description: Path is the relative file path of this artifact. - type: string - revision: - description: |- - Revision is a human readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm index timestamp, a Helm - chart version, etc. - type: string - url: - description: URL is the HTTP address of this artifact. - type: string - required: - - path - - url - type: object - conditions: - description: Conditions holds the conditions for the GitRepository. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - includedArtifacts: - description: IncludedArtifacts represents the included artifacts from - the last successful repository sync. - items: - description: Artifact represents the output of a source synchronisation. - properties: - checksum: - description: Checksum is the SHA256 checksum of the artifact. - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of this - artifact. - format: date-time - type: string - path: - description: Path is the relative file path of this artifact. - type: string - revision: - description: |- - Revision is a human readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm index timestamp, a Helm - chart version, etc. - type: string - url: - description: URL is the HTTP address of this artifact. - type: string - required: - - path - - url - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - url: - description: |- - URL is the download link for the artifact output of the last repository - sync. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.url - name: URL - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta2 GitRepository is deprecated, upgrade to v1 - name: v1beta2 - schema: - openAPIV3Schema: - description: GitRepository is the Schema for the gitrepositories API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - GitRepositorySpec specifies the required configuration to produce an - Artifact for a Git repository. - properties: - accessFrom: - description: |- - AccessFrom specifies an Access Control List for allowing cross-namespace - references to this object. - NOTE: Not implemented, provisional as of https://github.com/fluxcd/flux2/pull/2092 - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - gitImplementation: - default: go-git - description: |- - GitImplementation specifies which Git client library implementation to - use. Defaults to 'go-git', valid values are ('go-git', 'libgit2'). - Deprecated: gitImplementation is deprecated now that 'go-git' is the - only supported implementation. - enum: - - go-git - - libgit2 - type: string - ignore: - description: |- - Ignore overrides the set of excluded patterns in the .sourceignore format - (which is the same as .gitignore). If not provided, a default will be used, - consult the documentation for your version to find out what those are. - type: string - include: - description: |- - Include specifies a list of GitRepository resources which Artifacts - should be included in the Artifact produced for this GitRepository. - items: - description: |- - GitRepositoryInclude specifies a local reference to a GitRepository which - Artifact (sub-)contents must be included, and where they should be placed. - properties: - fromPath: - description: |- - FromPath specifies the path to copy contents from, defaults to the root - of the Artifact. - type: string - repository: - description: |- - GitRepositoryRef specifies the GitRepository which Artifact contents - must be included. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - toPath: - description: |- - ToPath specifies the path to copy contents to, defaults to the name of - the GitRepositoryRef. - type: string - required: - - repository - type: object - type: array - interval: - description: Interval at which to check the GitRepository for updates. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - recurseSubmodules: - description: |- - RecurseSubmodules enables the initialization of all submodules within - the GitRepository as cloned from the URL, using their default settings. - type: boolean - ref: - description: |- - Reference specifies the Git reference to resolve and monitor for - changes, defaults to the 'master' branch. - properties: - branch: - description: Branch to check out, defaults to 'master' if no other - field is defined. - type: string - commit: - description: |- - Commit SHA to check out, takes precedence over all reference fields. - - - This can be combined with Branch to shallow clone the branch, in which - the commit is expected to exist. - type: string - name: - description: |- - Name of the reference to check out; takes precedence over Branch, Tag and SemVer. - - - It must be a valid Git reference: https://git-scm.com/docs/git-check-ref-format#_description - Examples: "refs/heads/main", "refs/tags/v0.1.0", "refs/pull/420/head", "refs/merge-requests/1/head" - type: string - semver: - description: SemVer tag expression to check out, takes precedence - over Tag. - type: string - tag: - description: Tag to check out, takes precedence over Branch. - type: string - type: object - secretRef: - description: |- - SecretRef specifies the Secret containing authentication credentials for - the GitRepository. - For HTTPS repositories the Secret must contain 'username' and 'password' - fields for basic auth or 'bearerToken' field for token auth. - For SSH repositories the Secret must contain 'identity' - and 'known_hosts' fields. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend the reconciliation of this - GitRepository. - type: boolean - timeout: - default: 60s - description: Timeout for Git operations like cloning, defaults to - 60s. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - url: - description: URL specifies the Git repository URL, it can be an HTTP/S - or SSH address. - pattern: ^(http|https|ssh)://.*$ - type: string - verify: - description: |- - Verification specifies the configuration to verify the Git commit - signature(s). - properties: - mode: - description: Mode specifies what Git object should be verified, - currently ('head'). - enum: - - head - type: string - secretRef: - description: |- - SecretRef specifies the Secret containing the public keys of trusted Git - authors. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - mode - - secretRef - type: object - required: - - interval - - url - type: object - status: - default: - observedGeneration: -1 - description: GitRepositoryStatus records the observed state of a Git repository. - properties: - artifact: - description: Artifact represents the last successful GitRepository - reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the GitRepository. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - contentConfigChecksum: - description: |- - ContentConfigChecksum is a checksum of all the configurations related to - the content of the source artifact: - - .spec.ignore - - .spec.recurseSubmodules - - .spec.included and the checksum of the included artifacts - observed in .status.observedGeneration version of the object. This can - be used to determine if the content of the included repository has - changed. - It has the format of `:`, for example: `sha256:`. - - - Deprecated: Replaced with explicit fields for observed artifact content - config in the status. - type: string - includedArtifacts: - description: |- - IncludedArtifacts contains a list of the last successfully included - Artifacts as instructed by GitRepositorySpec.Include. - items: - description: Artifact represents the output of a Source reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of - ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI - annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: |- - ObservedGeneration is the last observed generation of the GitRepository - object. - format: int64 - type: integer - observedIgnore: - description: |- - ObservedIgnore is the observed exclusion patterns used for constructing - the source artifact. - type: string - observedInclude: - description: |- - ObservedInclude is the observed list of GitRepository resources used to - to produce the current Artifact. - items: - description: |- - GitRepositoryInclude specifies a local reference to a GitRepository which - Artifact (sub-)contents must be included, and where they should be placed. - properties: - fromPath: - description: |- - FromPath specifies the path to copy contents from, defaults to the root - of the Artifact. - type: string - repository: - description: |- - GitRepositoryRef specifies the GitRepository which Artifact contents - must be included. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - toPath: - description: |- - ToPath specifies the path to copy contents to, defaults to the name of - the GitRepositoryRef. - type: string - required: - - repository - type: object - type: array - observedRecurseSubmodules: - description: |- - ObservedRecurseSubmodules is the observed resource submodules - configuration used to produce the current Artifact. - type: boolean - url: - description: |- - URL is the dynamic fetch link for the latest Artifact. - It is provided on a "best effort" basis, and using the precise - GitRepositoryStatus.Artifact data is recommended. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: helmcharts.source.toolkit.fluxcd.io -spec: - group: source.toolkit.fluxcd.io - names: - kind: HelmChart - listKind: HelmChartList - plural: helmcharts - shortNames: - - hc - singular: helmchart - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.chart - name: Chart - type: string - - jsonPath: .spec.version - name: Version - type: string - - jsonPath: .spec.sourceRef.kind - name: Source Kind - type: string - - jsonPath: .spec.sourceRef.name - name: Source Name - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - name: v1 - schema: - openAPIV3Schema: - description: HelmChart is the Schema for the helmcharts API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HelmChartSpec specifies the desired state of a Helm chart. - properties: - chart: - description: |- - Chart is the name or path the Helm chart is available at in the - SourceRef. - type: string - ignoreMissingValuesFiles: - description: |- - IgnoreMissingValuesFiles controls whether to silently ignore missing values - files rather than failing. - type: boolean - interval: - description: |- - Interval at which the HelmChart SourceRef is checked for updates. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - reconcileStrategy: - default: ChartVersion - description: |- - ReconcileStrategy determines what enables the creation of a new artifact. - Valid values are ('ChartVersion', 'Revision'). - See the documentation of the values for an explanation on their behavior. - Defaults to ChartVersion when omitted. - enum: - - ChartVersion - - Revision - type: string - sourceRef: - description: SourceRef is the reference to the Source the chart is - available at. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: |- - Kind of the referent, valid values are ('HelmRepository', 'GitRepository', - 'Bucket'). - enum: - - HelmRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - type: string - required: - - kind - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend the reconciliation of this - source. - type: boolean - valuesFiles: - description: |- - ValuesFiles is an alternative list of values files to use as the chart - values (values.yaml is not included by default), expected to be a - relative path in the SourceRef. - Values files are merged in the order of this list with the last file - overriding the first. Ignored when omitted. - items: - type: string - type: array - verify: - description: |- - Verify contains the secret name containing the trusted public keys - used to verify the signature and specifies which provider to use to check - whether OCI image is authentic. - This field is only supported when using HelmRepository source with spec.type 'oci'. - Chart dependencies, which are not bundled in the umbrella chart artifact, are not verified. - properties: - matchOIDCIdentity: - description: |- - MatchOIDCIdentity specifies the identity matching criteria to use - while verifying an OCI artifact which was signed using Cosign keyless - signing. The artifact's identity is deemed to be verified if any of the - specified matchers match against the identity. - items: - description: |- - OIDCIdentityMatch specifies options for verifying the certificate identity, - i.e. the issuer and the subject of the certificate. - properties: - issuer: - description: |- - Issuer specifies the regex pattern to match against to verify - the OIDC issuer in the Fulcio certificate. The pattern must be a - valid Go regular expression. - type: string - subject: - description: |- - Subject specifies the regex pattern to match against to verify - the identity subject in the Fulcio certificate. The pattern must - be a valid Go regular expression. - type: string - required: - - issuer - - subject - type: object - type: array - provider: - default: cosign - description: Provider specifies the technology used to sign the - OCI Artifact. - enum: - - cosign - - notation - type: string - secretRef: - description: |- - SecretRef specifies the Kubernetes Secret containing the - trusted public keys. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - version: - default: '*' - description: |- - Version is the chart version semver expression, ignored for charts from - GitRepository and Bucket sources. Defaults to latest when omitted. - type: string - required: - - chart - - interval - - sourceRef - type: object - status: - default: - observedGeneration: -1 - description: HelmChartStatus records the observed state of the HelmChart. - properties: - artifact: - description: Artifact represents the output of the last successful - reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the HelmChart. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedChartName: - description: |- - ObservedChartName is the last observed chart name as specified by the - resolved chart reference. - type: string - observedGeneration: - description: |- - ObservedGeneration is the last observed generation of the HelmChart - object. - format: int64 - type: integer - observedSourceArtifactRevision: - description: |- - ObservedSourceArtifactRevision is the last observed Artifact.Revision - of the HelmChartSpec.SourceRef. - type: string - observedValuesFiles: - description: |- - ObservedValuesFiles are the observed value files of the last successful - reconciliation. - It matches the chart in the last successfully reconciled artifact. - items: - type: string - type: array - url: - description: |- - URL is the dynamic fetch link for the latest Artifact. - It is provided on a "best effort" basis, and using the precise - BucketStatus.Artifact data is recommended. - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.chart - name: Chart - type: string - - jsonPath: .spec.version - name: Version - type: string - - jsonPath: .spec.sourceRef.kind - name: Source Kind - type: string - - jsonPath: .spec.sourceRef.name - name: Source Name - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - deprecated: true - deprecationWarning: v1beta1 HelmChart is deprecated, upgrade to v1 - name: v1beta1 - schema: - openAPIV3Schema: - description: HelmChart is the Schema for the helmcharts API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HelmChartSpec defines the desired state of a Helm chart. - properties: - accessFrom: - description: AccessFrom defines an Access Control List for allowing - cross-namespace references to this object. - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - chart: - description: The name or path the Helm chart is available at in the - SourceRef. - type: string - interval: - description: The interval at which to check the Source for updates. - type: string - reconcileStrategy: - default: ChartVersion - description: |- - Determines what enables the creation of a new artifact. Valid values are - ('ChartVersion', 'Revision'). - See the documentation of the values for an explanation on their behavior. - Defaults to ChartVersion when omitted. - enum: - - ChartVersion - - Revision - type: string - sourceRef: - description: The reference to the Source the chart is available at. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: |- - Kind of the referent, valid values are ('HelmRepository', 'GitRepository', - 'Bucket'). - enum: - - HelmRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - type: string - required: - - kind - - name - type: object - suspend: - description: This flag tells the controller to suspend the reconciliation - of this source. - type: boolean - valuesFile: - description: |- - Alternative values file to use as the default chart values, expected to - be a relative path in the SourceRef. Deprecated in favor of ValuesFiles, - for backwards compatibility the file defined here is merged before the - ValuesFiles items. Ignored when omitted. - type: string - valuesFiles: - description: |- - Alternative list of values files to use as the chart values (values.yaml - is not included by default), expected to be a relative path in the SourceRef. - Values files are merged in the order of this list with the last file overriding - the first. Ignored when omitted. - items: - type: string - type: array - version: - default: '*' - description: |- - The chart version semver expression, ignored for charts from GitRepository - and Bucket sources. Defaults to latest when omitted. - type: string - required: - - chart - - interval - - sourceRef - type: object - status: - default: - observedGeneration: -1 - description: HelmChartStatus defines the observed state of the HelmChart. - properties: - artifact: - description: Artifact represents the output of the last successful - chart sync. - properties: - checksum: - description: Checksum is the SHA256 checksum of the artifact. - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of this - artifact. - format: date-time - type: string - path: - description: Path is the relative file path of this artifact. - type: string - revision: - description: |- - Revision is a human readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm index timestamp, a Helm - chart version, etc. - type: string - url: - description: URL is the HTTP address of this artifact. - type: string - required: - - path - - url - type: object - conditions: - description: Conditions holds the conditions for the HelmChart. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - url: - description: URL is the download link for the last chart pulled. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.chart - name: Chart - type: string - - jsonPath: .spec.version - name: Version - type: string - - jsonPath: .spec.sourceRef.kind - name: Source Kind - type: string - - jsonPath: .spec.sourceRef.name - name: Source Name - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta2 HelmChart is deprecated, upgrade to v1 - name: v1beta2 - schema: - openAPIV3Schema: - description: HelmChart is the Schema for the helmcharts API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HelmChartSpec specifies the desired state of a Helm chart. - properties: - accessFrom: - description: |- - AccessFrom specifies an Access Control List for allowing cross-namespace - references to this object. - NOTE: Not implemented, provisional as of https://github.com/fluxcd/flux2/pull/2092 - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - chart: - description: |- - Chart is the name or path the Helm chart is available at in the - SourceRef. - type: string - ignoreMissingValuesFiles: - description: |- - IgnoreMissingValuesFiles controls whether to silently ignore missing values - files rather than failing. - type: boolean - interval: - description: |- - Interval at which the HelmChart SourceRef is checked for updates. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - reconcileStrategy: - default: ChartVersion - description: |- - ReconcileStrategy determines what enables the creation of a new artifact. - Valid values are ('ChartVersion', 'Revision'). - See the documentation of the values for an explanation on their behavior. - Defaults to ChartVersion when omitted. - enum: - - ChartVersion - - Revision - type: string - sourceRef: - description: SourceRef is the reference to the Source the chart is - available at. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: |- - Kind of the referent, valid values are ('HelmRepository', 'GitRepository', - 'Bucket'). - enum: - - HelmRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - type: string - required: - - kind - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend the reconciliation of this - source. - type: boolean - valuesFile: - description: |- - ValuesFile is an alternative values file to use as the default chart - values, expected to be a relative path in the SourceRef. Deprecated in - favor of ValuesFiles, for backwards compatibility the file specified here - is merged before the ValuesFiles items. Ignored when omitted. - type: string - valuesFiles: - description: |- - ValuesFiles is an alternative list of values files to use as the chart - values (values.yaml is not included by default), expected to be a - relative path in the SourceRef. - Values files are merged in the order of this list with the last file - overriding the first. Ignored when omitted. - items: - type: string - type: array - verify: - description: |- - Verify contains the secret name containing the trusted public keys - used to verify the signature and specifies which provider to use to check - whether OCI image is authentic. - This field is only supported when using HelmRepository source with spec.type 'oci'. - Chart dependencies, which are not bundled in the umbrella chart artifact, are not verified. - properties: - matchOIDCIdentity: - description: |- - MatchOIDCIdentity specifies the identity matching criteria to use - while verifying an OCI artifact which was signed using Cosign keyless - signing. The artifact's identity is deemed to be verified if any of the - specified matchers match against the identity. - items: - description: |- - OIDCIdentityMatch specifies options for verifying the certificate identity, - i.e. the issuer and the subject of the certificate. - properties: - issuer: - description: |- - Issuer specifies the regex pattern to match against to verify - the OIDC issuer in the Fulcio certificate. The pattern must be a - valid Go regular expression. - type: string - subject: - description: |- - Subject specifies the regex pattern to match against to verify - the identity subject in the Fulcio certificate. The pattern must - be a valid Go regular expression. - type: string - required: - - issuer - - subject - type: object - type: array - provider: - default: cosign - description: Provider specifies the technology used to sign the - OCI Artifact. - enum: - - cosign - - notation - type: string - secretRef: - description: |- - SecretRef specifies the Kubernetes Secret containing the - trusted public keys. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - version: - default: '*' - description: |- - Version is the chart version semver expression, ignored for charts from - GitRepository and Bucket sources. Defaults to latest when omitted. - type: string - required: - - chart - - interval - - sourceRef - type: object - status: - default: - observedGeneration: -1 - description: HelmChartStatus records the observed state of the HelmChart. - properties: - artifact: - description: Artifact represents the output of the last successful - reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the HelmChart. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedChartName: - description: |- - ObservedChartName is the last observed chart name as specified by the - resolved chart reference. - type: string - observedGeneration: - description: |- - ObservedGeneration is the last observed generation of the HelmChart - object. - format: int64 - type: integer - observedSourceArtifactRevision: - description: |- - ObservedSourceArtifactRevision is the last observed Artifact.Revision - of the HelmChartSpec.SourceRef. - type: string - observedValuesFiles: - description: |- - ObservedValuesFiles are the observed value files of the last successful - reconciliation. - It matches the chart in the last successfully reconciled artifact. - items: - type: string - type: array - url: - description: |- - URL is the dynamic fetch link for the latest Artifact. - It is provided on a "best effort" basis, and using the precise - BucketStatus.Artifact data is recommended. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: helmrepositories.source.toolkit.fluxcd.io -spec: - group: source.toolkit.fluxcd.io - names: - kind: HelmRepository - listKind: HelmRepositoryList - plural: helmrepositories - shortNames: - - helmrepo - singular: helmrepository - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.url - name: URL - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - name: v1 - schema: - openAPIV3Schema: - description: HelmRepository is the Schema for the helmrepositories API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - HelmRepositorySpec specifies the required configuration to produce an - Artifact for a Helm repository index YAML. - properties: - accessFrom: - description: |- - AccessFrom specifies an Access Control List for allowing cross-namespace - references to this object. - NOTE: Not implemented, provisional as of https://github.com/fluxcd/flux2/pull/2092 - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - certSecretRef: - description: |- - CertSecretRef can be given the name of a Secret containing - either or both of - - - - a PEM-encoded client certificate (`tls.crt`) and private - key (`tls.key`); - - a PEM-encoded CA certificate (`ca.crt`) - - - and whichever are supplied, will be used for connecting to the - registry. The client cert and key are useful if you are - authenticating with a certificate; the CA cert is useful if - you are using a self-signed server certificate. The Secret must - be of type `Opaque` or `kubernetes.io/tls`. - - - It takes precedence over the values specified in the Secret referred - to by `.spec.secretRef`. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - insecure: - description: |- - Insecure allows connecting to a non-TLS HTTP container registry. - This field is only taken into account if the .spec.type field is set to 'oci'. - type: boolean - interval: - description: |- - Interval at which the HelmRepository URL is checked for updates. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - passCredentials: - description: |- - PassCredentials allows the credentials from the SecretRef to be passed - on to a host that does not match the host as defined in URL. - This may be required if the host of the advertised chart URLs in the - index differ from the defined URL. - Enabling this should be done with caution, as it can potentially result - in credentials getting stolen in a MITM-attack. - type: boolean - provider: - default: generic - description: |- - Provider used for authentication, can be 'aws', 'azure', 'gcp' or 'generic'. - This field is optional, and only taken into account if the .spec.type field is set to 'oci'. - When not specified, defaults to 'generic'. - enum: - - generic - - aws - - azure - - gcp - type: string - secretRef: - description: |- - SecretRef specifies the Secret containing authentication credentials - for the HelmRepository. - For HTTP/S basic auth the secret must contain 'username' and 'password' - fields. - Support for TLS auth using the 'certFile' and 'keyFile', and/or 'caFile' - keys is deprecated. Please use `.spec.certSecretRef` instead. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend the reconciliation of this - HelmRepository. - type: boolean - timeout: - description: |- - Timeout is used for the index fetch operation for an HTTPS helm repository, - and for remote OCI Repository operations like pulling for an OCI helm - chart by the associated HelmChart. - Its default value is 60s. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - type: - description: |- - Type of the HelmRepository. - When this field is set to "oci", the URL field value must be prefixed with "oci://". - enum: - - default - - oci - type: string - url: - description: |- - URL of the Helm repository, a valid URL contains at least a protocol and - host. - pattern: ^(http|https|oci)://.*$ - type: string - required: - - url - type: object - status: - default: - observedGeneration: -1 - description: HelmRepositoryStatus records the observed state of the HelmRepository. - properties: - artifact: - description: Artifact represents the last successful HelmRepository - reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the HelmRepository. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: |- - ObservedGeneration is the last observed generation of the HelmRepository - object. - format: int64 - type: integer - url: - description: |- - URL is the dynamic fetch link for the latest Artifact. - It is provided on a "best effort" basis, and using the precise - HelmRepositoryStatus.Artifact data is recommended. - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.url - name: URL - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - deprecated: true - deprecationWarning: v1beta1 HelmRepository is deprecated, upgrade to v1 - name: v1beta1 - schema: - openAPIV3Schema: - description: HelmRepository is the Schema for the helmrepositories API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HelmRepositorySpec defines the reference to a Helm repository. - properties: - accessFrom: - description: AccessFrom defines an Access Control List for allowing - cross-namespace references to this object. - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - interval: - description: The interval at which to check the upstream for updates. - type: string - passCredentials: - description: |- - PassCredentials allows the credentials from the SecretRef to be passed on to - a host that does not match the host as defined in URL. - This may be required if the host of the advertised chart URLs in the index - differ from the defined URL. - Enabling this should be done with caution, as it can potentially result in - credentials getting stolen in a MITM-attack. - type: boolean - secretRef: - description: |- - The name of the secret containing authentication credentials for the Helm - repository. - For HTTP/S basic auth the secret must contain username and - password fields. - For TLS the secret must contain a certFile and keyFile, and/or - caFile fields. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: This flag tells the controller to suspend the reconciliation - of this source. - type: boolean - timeout: - default: 60s - description: The timeout of index downloading, defaults to 60s. - type: string - url: - description: The Helm repository URL, a valid URL contains at least - a protocol and host. - type: string - required: - - interval - - url - type: object - status: - default: - observedGeneration: -1 - description: HelmRepositoryStatus defines the observed state of the HelmRepository. - properties: - artifact: - description: Artifact represents the output of the last successful - repository sync. - properties: - checksum: - description: Checksum is the SHA256 checksum of the artifact. - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of this - artifact. - format: date-time - type: string - path: - description: Path is the relative file path of this artifact. - type: string - revision: - description: |- - Revision is a human readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm index timestamp, a Helm - chart version, etc. - type: string - url: - description: URL is the HTTP address of this artifact. - type: string - required: - - path - - url - type: object - conditions: - description: Conditions holds the conditions for the HelmRepository. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - url: - description: URL is the download link for the last index fetched. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.url - name: URL - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta2 HelmRepository is deprecated, upgrade to v1 - name: v1beta2 - schema: - openAPIV3Schema: - description: HelmRepository is the Schema for the helmrepositories API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - HelmRepositorySpec specifies the required configuration to produce an - Artifact for a Helm repository index YAML. - properties: - accessFrom: - description: |- - AccessFrom specifies an Access Control List for allowing cross-namespace - references to this object. - NOTE: Not implemented, provisional as of https://github.com/fluxcd/flux2/pull/2092 - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - certSecretRef: - description: |- - CertSecretRef can be given the name of a Secret containing - either or both of - - - - a PEM-encoded client certificate (`tls.crt`) and private - key (`tls.key`); - - a PEM-encoded CA certificate (`ca.crt`) - - - and whichever are supplied, will be used for connecting to the - registry. The client cert and key are useful if you are - authenticating with a certificate; the CA cert is useful if - you are using a self-signed server certificate. The Secret must - be of type `Opaque` or `kubernetes.io/tls`. - - - It takes precedence over the values specified in the Secret referred - to by `.spec.secretRef`. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - insecure: - description: |- - Insecure allows connecting to a non-TLS HTTP container registry. - This field is only taken into account if the .spec.type field is set to 'oci'. - type: boolean - interval: - description: |- - Interval at which the HelmRepository URL is checked for updates. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - passCredentials: - description: |- - PassCredentials allows the credentials from the SecretRef to be passed - on to a host that does not match the host as defined in URL. - This may be required if the host of the advertised chart URLs in the - index differ from the defined URL. - Enabling this should be done with caution, as it can potentially result - in credentials getting stolen in a MITM-attack. - type: boolean - provider: - default: generic - description: |- - Provider used for authentication, can be 'aws', 'azure', 'gcp' or 'generic'. - This field is optional, and only taken into account if the .spec.type field is set to 'oci'. - When not specified, defaults to 'generic'. - enum: - - generic - - aws - - azure - - gcp - type: string - secretRef: - description: |- - SecretRef specifies the Secret containing authentication credentials - for the HelmRepository. - For HTTP/S basic auth the secret must contain 'username' and 'password' - fields. - Support for TLS auth using the 'certFile' and 'keyFile', and/or 'caFile' - keys is deprecated. Please use `.spec.certSecretRef` instead. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend the reconciliation of this - HelmRepository. - type: boolean - timeout: - description: |- - Timeout is used for the index fetch operation for an HTTPS helm repository, - and for remote OCI Repository operations like pulling for an OCI helm - chart by the associated HelmChart. - Its default value is 60s. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - type: - description: |- - Type of the HelmRepository. - When this field is set to "oci", the URL field value must be prefixed with "oci://". - enum: - - default - - oci - type: string - url: - description: |- - URL of the Helm repository, a valid URL contains at least a protocol and - host. - pattern: ^(http|https|oci)://.*$ - type: string - required: - - url - type: object - status: - default: - observedGeneration: -1 - description: HelmRepositoryStatus records the observed state of the HelmRepository. - properties: - artifact: - description: Artifact represents the last successful HelmRepository - reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the HelmRepository. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: |- - ObservedGeneration is the last observed generation of the HelmRepository - object. - format: int64 - type: integer - url: - description: |- - URL is the dynamic fetch link for the latest Artifact. - It is provided on a "best effort" basis, and using the precise - HelmRepositoryStatus.Artifact data is recommended. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: ocirepositories.source.toolkit.fluxcd.io -spec: - group: source.toolkit.fluxcd.io - names: - kind: OCIRepository - listKind: OCIRepositoryList - plural: ocirepositories - shortNames: - - ocirepo - singular: ocirepository - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.url - name: URL - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1beta2 - schema: - openAPIV3Schema: - description: OCIRepository is the Schema for the ocirepositories API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: OCIRepositorySpec defines the desired state of OCIRepository - properties: - certSecretRef: - description: |- - CertSecretRef can be given the name of a Secret containing - either or both of - - - - a PEM-encoded client certificate (`tls.crt`) and private - key (`tls.key`); - - a PEM-encoded CA certificate (`ca.crt`) - - - and whichever are supplied, will be used for connecting to the - registry. The client cert and key are useful if you are - authenticating with a certificate; the CA cert is useful if - you are using a self-signed server certificate. The Secret must - be of type `Opaque` or `kubernetes.io/tls`. - - - Note: Support for the `caFile`, `certFile` and `keyFile` keys have - been deprecated. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - ignore: - description: |- - Ignore overrides the set of excluded patterns in the .sourceignore format - (which is the same as .gitignore). If not provided, a default will be used, - consult the documentation for your version to find out what those are. - type: string - insecure: - description: Insecure allows connecting to a non-TLS HTTP container - registry. - type: boolean - interval: - description: |- - Interval at which the OCIRepository URL is checked for updates. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - layerSelector: - description: |- - LayerSelector specifies which layer should be extracted from the OCI artifact. - When not specified, the first layer found in the artifact is selected. - properties: - mediaType: - description: |- - MediaType specifies the OCI media type of the layer - which should be extracted from the OCI Artifact. The - first layer matching this type is selected. - type: string - operation: - description: |- - Operation specifies how the selected layer should be processed. - By default, the layer compressed content is extracted to storage. - When the operation is set to 'copy', the layer compressed content - is persisted to storage as it is. - enum: - - extract - - copy - type: string - type: object - provider: - default: generic - description: |- - The provider used for authentication, can be 'aws', 'azure', 'gcp' or 'generic'. - When not specified, defaults to 'generic'. - enum: - - generic - - aws - - azure - - gcp - type: string - ref: - description: |- - The OCI reference to pull and monitor for changes, - defaults to the latest tag. - properties: - digest: - description: |- - Digest is the image digest to pull, takes precedence over SemVer. - The value should be in the format 'sha256:'. - type: string - semver: - description: |- - SemVer is the range of tags to pull selecting the latest within - the range, takes precedence over Tag. - type: string - semverFilter: - description: SemverFilter is a regex pattern to filter the tags - within the SemVer range. - type: string - tag: - description: Tag is the image tag to pull, defaults to latest. - type: string - type: object - secretRef: - description: |- - SecretRef contains the secret name containing the registry login - credentials to resolve image metadata. - The secret must be of type kubernetes.io/dockerconfigjson. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - serviceAccountName: - description: |- - ServiceAccountName is the name of the Kubernetes ServiceAccount used to authenticate - the image pull if the service account has attached pull secrets. For more information: - https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/#add-imagepullsecrets-to-a-service-account - type: string - suspend: - description: This flag tells the controller to suspend the reconciliation - of this source. - type: boolean - timeout: - default: 60s - description: The timeout for remote OCI Repository operations like - pulling, defaults to 60s. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - url: - description: |- - URL is a reference to an OCI artifact repository hosted - on a remote container registry. - pattern: ^oci://.*$ - type: string - verify: - description: |- - Verify contains the secret name containing the trusted public keys - used to verify the signature and specifies which provider to use to check - whether OCI image is authentic. - properties: - matchOIDCIdentity: - description: |- - MatchOIDCIdentity specifies the identity matching criteria to use - while verifying an OCI artifact which was signed using Cosign keyless - signing. The artifact's identity is deemed to be verified if any of the - specified matchers match against the identity. - items: - description: |- - OIDCIdentityMatch specifies options for verifying the certificate identity, - i.e. the issuer and the subject of the certificate. - properties: - issuer: - description: |- - Issuer specifies the regex pattern to match against to verify - the OIDC issuer in the Fulcio certificate. The pattern must be a - valid Go regular expression. - type: string - subject: - description: |- - Subject specifies the regex pattern to match against to verify - the identity subject in the Fulcio certificate. The pattern must - be a valid Go regular expression. - type: string - required: - - issuer - - subject - type: object - type: array - provider: - default: cosign - description: Provider specifies the technology used to sign the - OCI Artifact. - enum: - - cosign - - notation - type: string - secretRef: - description: |- - SecretRef specifies the Kubernetes Secret containing the - trusted public keys. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - required: - - interval - - url - type: object - status: - default: - observedGeneration: -1 - description: OCIRepositoryStatus defines the observed state of OCIRepository - properties: - artifact: - description: Artifact represents the output of the last successful - OCI Repository sync. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the OCIRepository. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - contentConfigChecksum: - description: |- - ContentConfigChecksum is a checksum of all the configurations related to - the content of the source artifact: - - .spec.ignore - - .spec.layerSelector - observed in .status.observedGeneration version of the object. This can - be used to determine if the content configuration has changed and the - artifact needs to be rebuilt. - It has the format of `:`, for example: `sha256:`. - - - Deprecated: Replaced with explicit fields for observed artifact content - config in the status. - type: string - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - observedIgnore: - description: |- - ObservedIgnore is the observed exclusion patterns used for constructing - the source artifact. - type: string - observedLayerSelector: - description: |- - ObservedLayerSelector is the observed layer selector used for constructing - the source artifact. - properties: - mediaType: - description: |- - MediaType specifies the OCI media type of the layer - which should be extracted from the OCI Artifact. The - first layer matching this type is selected. - type: string - operation: - description: |- - Operation specifies how the selected layer should be processed. - By default, the layer compressed content is extracted to storage. - When the operation is set to 'copy', the layer compressed content - is persisted to storage as it is. - enum: - - extract - - copy - type: string - type: object - url: - description: URL is the download link for the artifact output of the - last OCI Repository sync. - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} ---- -apiVersion: v1 -kind: ServiceAccount -metadata: - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: source-controller - namespace: flux-system ---- -apiVersion: v1 -kind: Service -metadata: - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - control-plane: controller - name: source-controller - namespace: flux-system -spec: - ports: - - name: http - port: 80 - protocol: TCP - targetPort: http - selector: - app: source-controller - type: ClusterIP ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - control-plane: controller - name: source-controller - namespace: flux-system -spec: - replicas: 1 - selector: - matchLabels: - app: source-controller - strategy: - type: Recreate - template: - metadata: - annotations: - prometheus.io/port: "8080" - prometheus.io/scrape: "true" - labels: - app: source-controller - spec: - containers: - - args: - - --events-addr=http://notification-controller.flux-system.svc.cluster.local./ - - --watch-all-namespaces=true - - --log-level=info - - --log-encoding=json - - --enable-leader-election - - --storage-path=/data - - --storage-adv-addr=source-controller.$(RUNTIME_NAMESPACE).svc.cluster.local. - env: - - name: RUNTIME_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - - name: TUF_ROOT - value: /tmp/.sigstore - - name: GOMAXPROCS - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.cpu - - name: GOMEMLIMIT - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.memory - image: ghcr.io/fluxcd/source-controller:v1.3.0 - imagePullPolicy: IfNotPresent - livenessProbe: - httpGet: - path: /healthz - port: healthz - name: manager - ports: - - containerPort: 9090 - name: http - protocol: TCP - - containerPort: 8080 - name: http-prom - protocol: TCP - - containerPort: 9440 - name: healthz - protocol: TCP - readinessProbe: - httpGet: - path: / - port: http - resources: - limits: - cpu: 1000m - memory: 1Gi - requests: - cpu: 50m - memory: 64Mi - securityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - readOnlyRootFilesystem: true - runAsNonRoot: true - seccompProfile: - type: RuntimeDefault - volumeMounts: - - mountPath: /data - name: data - - mountPath: /tmp - name: tmp - nodeSelector: - kubernetes.io/os: linux - priorityClassName: system-cluster-critical - securityContext: - fsGroup: 1337 - serviceAccountName: source-controller - terminationGracePeriodSeconds: 10 - volumes: - - emptyDir: {} - name: data - - emptyDir: {} - name: tmp ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: kustomize-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: kustomizations.kustomize.toolkit.fluxcd.io -spec: - group: kustomize.toolkit.fluxcd.io - names: - kind: Kustomization - listKind: KustomizationList - plural: kustomizations - shortNames: - - ks - singular: kustomization - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - name: v1 - schema: - openAPIV3Schema: - description: Kustomization is the Schema for the kustomizations API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - KustomizationSpec defines the configuration to calculate the desired state - from a Source using Kustomize. - properties: - commonMetadata: - description: |- - CommonMetadata specifies the common labels and annotations that are - applied to all resources. Any existing label or annotation will be - overridden if its key matches a common one. - properties: - annotations: - additionalProperties: - type: string - description: Annotations to be added to the object's metadata. - type: object - labels: - additionalProperties: - type: string - description: Labels to be added to the object's metadata. - type: object - type: object - components: - description: Components specifies relative paths to specifications - of other Components. - items: - type: string - type: array - decryption: - description: Decrypt Kubernetes secrets before applying them on the - cluster. - properties: - provider: - description: Provider is the name of the decryption engine. - enum: - - sops - type: string - secretRef: - description: The secret name containing the private OpenPGP keys - used for decryption. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - dependsOn: - description: |- - DependsOn may contain a meta.NamespacedObjectReference slice - with references to Kustomization resources that must be ready before this - Kustomization can be reconciled. - items: - description: |- - NamespacedObjectReference contains enough information to locate the referenced Kubernetes resource object in any - namespace. - properties: - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - name - type: object - type: array - force: - default: false - description: |- - Force instructs the controller to recreate resources - when patching fails due to an immutable field change. - type: boolean - healthChecks: - description: A list of resources to be included in the health assessment. - items: - description: |- - NamespacedObjectKindReference contains enough information to locate the typed referenced Kubernetes resource object - in any namespace. - properties: - apiVersion: - description: API version of the referent, if not specified the - Kubernetes preferred version will be used. - type: string - kind: - description: Kind of the referent. - type: string - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - kind - - name - type: object - type: array - images: - description: |- - Images is a list of (image name, new name, new tag or digest) - for changing image names, tags or digests. This can also be achieved with a - patch, but this operator is simpler to specify. - items: - description: Image contains an image name, a new name, a new tag - or digest, which will replace the original name and tag. - properties: - digest: - description: |- - Digest is the value used to replace the original image tag. - If digest is present NewTag value is ignored. - type: string - name: - description: Name is a tag-less image name. - type: string - newName: - description: NewName is the value used to replace the original - name. - type: string - newTag: - description: NewTag is the value used to replace the original - tag. - type: string - required: - - name - type: object - type: array - interval: - description: |- - The interval at which to reconcile the Kustomization. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - kubeConfig: - description: |- - The KubeConfig for reconciling the Kustomization on a remote cluster. - When used in combination with KustomizationSpec.ServiceAccountName, - forces the controller to act on behalf of that Service Account at the - target cluster. - If the --default-service-account flag is set, its value will be used as - a controller level fallback for when KustomizationSpec.ServiceAccountName - is empty. - properties: - secretRef: - description: |- - SecretRef holds the name of a secret that contains a key with - the kubeconfig file as the value. If no key is set, the key will default - to 'value'. - It is recommended that the kubeconfig is self-contained, and the secret - is regularly updated if credentials such as a cloud-access-token expire. - Cloud specific `cmd-path` auth helpers will not function without adding - binaries and credentials to the Pod that is responsible for reconciling - Kubernetes resources. - properties: - key: - description: Key in the Secret, when not specified an implementation-specific - default key is used. - type: string - name: - description: Name of the Secret. - type: string - required: - - name - type: object - required: - - secretRef - type: object - namePrefix: - description: NamePrefix will prefix the names of all managed resources. - maxLength: 200 - minLength: 1 - type: string - nameSuffix: - description: NameSuffix will suffix the names of all managed resources. - maxLength: 200 - minLength: 1 - type: string - patches: - description: |- - Strategic merge and JSON patches, defined as inline YAML objects, - capable of targeting objects based on kind, label and annotation selectors. - items: - description: |- - Patch contains an inline StrategicMerge or JSON6902 patch, and the target the patch should - be applied to. - properties: - patch: - description: |- - Patch contains an inline StrategicMerge patch or an inline JSON6902 patch with - an array of operation objects. - type: string - target: - description: Target points to the resources that the patch document - should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - type: object - type: array - path: - description: |- - Path to the directory containing the kustomization.yaml file, or the - set of plain YAMLs a kustomization.yaml should be generated for. - Defaults to 'None', which translates to the root path of the SourceRef. - type: string - postBuild: - description: |- - PostBuild describes which actions to perform on the YAML manifest - generated by building the kustomize overlay. - properties: - substitute: - additionalProperties: - type: string - description: |- - Substitute holds a map of key/value pairs. - The variables defined in your YAML manifests that match any of the keys - defined in the map will be substituted with the set value. - Includes support for bash string replacement functions - e.g. ${var:=default}, ${var:position} and ${var/substring/replacement}. - type: object - substituteFrom: - description: |- - SubstituteFrom holds references to ConfigMaps and Secrets containing - the variables and their values to be substituted in the YAML manifests. - The ConfigMap and the Secret data keys represent the var names, and they - must match the vars declared in the manifests for the substitution to - happen. - items: - description: |- - SubstituteReference contains a reference to a resource containing - the variables name and value. - properties: - kind: - description: Kind of the values referent, valid values are - ('Secret', 'ConfigMap'). - enum: - - Secret - - ConfigMap - type: string - name: - description: |- - Name of the values referent. Should reside in the same namespace as the - referring resource. - maxLength: 253 - minLength: 1 - type: string - optional: - default: false - description: |- - Optional indicates whether the referenced resource must exist, or whether to - tolerate its absence. If true and the referenced resource is absent, proceed - as if the resource was present but empty, without any variables defined. - type: boolean - required: - - kind - - name - type: object - type: array - type: object - prune: - description: Prune enables garbage collection. - type: boolean - retryInterval: - description: |- - The interval at which to retry a previously failed reconciliation. - When not specified, the controller uses the KustomizationSpec.Interval - value to retry failures. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - serviceAccountName: - description: |- - The name of the Kubernetes service account to impersonate - when reconciling this Kustomization. - type: string - sourceRef: - description: Reference of the source where the kustomization file - is. - properties: - apiVersion: - description: API version of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - OCIRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - type: string - namespace: - description: |- - Namespace of the referent, defaults to the namespace of the Kubernetes - resource object that contains the reference. - type: string - required: - - kind - - name - type: object - suspend: - description: |- - This flag tells the controller to suspend subsequent kustomize executions, - it does not apply to already started executions. Defaults to false. - type: boolean - targetNamespace: - description: |- - TargetNamespace sets or overrides the namespace in the - kustomization.yaml file. - maxLength: 63 - minLength: 1 - type: string - timeout: - description: |- - Timeout for validation, apply and health checking operations. - Defaults to 'Interval' duration. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - wait: - description: |- - Wait instructs the controller to check the health of all the reconciled - resources. When enabled, the HealthChecks are ignored. Defaults to false. - type: boolean - required: - - interval - - prune - - sourceRef - type: object - status: - default: - observedGeneration: -1 - description: KustomizationStatus defines the observed state of a kustomization. - properties: - conditions: - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - inventory: - description: |- - Inventory contains the list of Kubernetes resource object references that - have been successfully applied. - properties: - entries: - description: Entries of Kubernetes resource object references. - items: - description: ResourceRef contains the information necessary - to locate a resource within a cluster. - properties: - id: - description: |- - ID is the string representation of the Kubernetes resource object's metadata, - in the format '___'. - type: string - v: - description: Version is the API version of the Kubernetes - resource object's kind. - type: string - required: - - id - - v - type: object - type: array - required: - - entries - type: object - lastAppliedRevision: - description: |- - The last successfully applied revision. - Equals the Revision of the applied Artifact from the referenced Source. - type: string - lastAttemptedRevision: - description: LastAttemptedRevision is the revision of the last reconciliation - attempt. - type: string - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last reconciled generation. - format: int64 - type: integer - type: object - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - deprecated: true - deprecationWarning: v1beta1 Kustomization is deprecated, upgrade to v1 - name: v1beta1 - schema: - openAPIV3Schema: - description: Kustomization is the Schema for the kustomizations API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: KustomizationSpec defines the desired state of a kustomization. - properties: - decryption: - description: Decrypt Kubernetes secrets before applying them on the - cluster. - properties: - provider: - description: Provider is the name of the decryption engine. - enum: - - sops - type: string - secretRef: - description: The secret name containing the private OpenPGP keys - used for decryption. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - dependsOn: - description: |- - DependsOn may contain a meta.NamespacedObjectReference slice - with references to Kustomization resources that must be ready before this - Kustomization can be reconciled. - items: - description: |- - NamespacedObjectReference contains enough information to locate the referenced Kubernetes resource object in any - namespace. - properties: - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - name - type: object - type: array - force: - default: false - description: |- - Force instructs the controller to recreate resources - when patching fails due to an immutable field change. - type: boolean - healthChecks: - description: A list of resources to be included in the health assessment. - items: - description: |- - NamespacedObjectKindReference contains enough information to locate the typed referenced Kubernetes resource object - in any namespace. - properties: - apiVersion: - description: API version of the referent, if not specified the - Kubernetes preferred version will be used. - type: string - kind: - description: Kind of the referent. - type: string - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - kind - - name - type: object - type: array - images: - description: |- - Images is a list of (image name, new name, new tag or digest) - for changing image names, tags or digests. This can also be achieved with a - patch, but this operator is simpler to specify. - items: - description: Image contains an image name, a new name, a new tag - or digest, which will replace the original name and tag. - properties: - digest: - description: |- - Digest is the value used to replace the original image tag. - If digest is present NewTag value is ignored. - type: string - name: - description: Name is a tag-less image name. - type: string - newName: - description: NewName is the value used to replace the original - name. - type: string - newTag: - description: NewTag is the value used to replace the original - tag. - type: string - required: - - name - type: object - type: array - interval: - description: The interval at which to reconcile the Kustomization. - type: string - kubeConfig: - description: |- - The KubeConfig for reconciling the Kustomization on a remote cluster. - When specified, KubeConfig takes precedence over ServiceAccountName. - properties: - secretRef: - description: |- - SecretRef holds the name to a secret that contains a 'value' key with - the kubeconfig file as the value. It must be in the same namespace as - the Kustomization. - It is recommended that the kubeconfig is self-contained, and the secret - is regularly updated if credentials such as a cloud-access-token expire. - Cloud specific `cmd-path` auth helpers will not function without adding - binaries and credentials to the Pod that is responsible for reconciling - the Kustomization. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - type: object - patches: - description: |- - Strategic merge and JSON patches, defined as inline YAML objects, - capable of targeting objects based on kind, label and annotation selectors. - items: - description: |- - Patch contains an inline StrategicMerge or JSON6902 patch, and the target the patch should - be applied to. - properties: - patch: - description: |- - Patch contains an inline StrategicMerge patch or an inline JSON6902 patch with - an array of operation objects. - type: string - target: - description: Target points to the resources that the patch document - should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - type: object - type: array - patchesJson6902: - description: JSON 6902 patches, defined as inline YAML objects. - items: - description: JSON6902Patch contains a JSON6902 patch and the target - the patch should be applied to. - properties: - patch: - description: Patch contains the JSON6902 patch document with - an array of operation objects. - items: - description: |- - JSON6902 is a JSON6902 operation object. - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - properties: - from: - description: |- - From contains a JSON-pointer value that references a location within the target document where the operation is - performed. The meaning of the value depends on the value of Op, and is NOT taken into account by all operations. - type: string - op: - description: |- - Op indicates the operation to perform. Its value MUST be one of "add", "remove", "replace", "move", "copy", or - "test". - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - enum: - - test - - remove - - add - - replace - - move - - copy - type: string - path: - description: |- - Path contains the JSON-pointer value that references a location within the target document where the operation - is performed. The meaning of the value depends on the value of Op. - type: string - value: - description: |- - Value contains a valid JSON structure. The meaning of the value depends on the value of Op, and is NOT taken into - account by all operations. - x-kubernetes-preserve-unknown-fields: true - required: - - op - - path - type: object - type: array - target: - description: Target points to the resources that the patch document - should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - - target - type: object - type: array - patchesStrategicMerge: - description: Strategic merge patches, defined as inline YAML objects. - items: - x-kubernetes-preserve-unknown-fields: true - type: array - path: - description: |- - Path to the directory containing the kustomization.yaml file, or the - set of plain YAMLs a kustomization.yaml should be generated for. - Defaults to 'None', which translates to the root path of the SourceRef. - type: string - postBuild: - description: |- - PostBuild describes which actions to perform on the YAML manifest - generated by building the kustomize overlay. - properties: - substitute: - additionalProperties: - type: string - description: |- - Substitute holds a map of key/value pairs. - The variables defined in your YAML manifests - that match any of the keys defined in the map - will be substituted with the set value. - Includes support for bash string replacement functions - e.g. ${var:=default}, ${var:position} and ${var/substring/replacement}. - type: object - substituteFrom: - description: |- - SubstituteFrom holds references to ConfigMaps and Secrets containing - the variables and their values to be substituted in the YAML manifests. - The ConfigMap and the Secret data keys represent the var names and they - must match the vars declared in the manifests for the substitution to happen. - items: - description: |- - SubstituteReference contains a reference to a resource containing - the variables name and value. - properties: - kind: - description: Kind of the values referent, valid values are - ('Secret', 'ConfigMap'). - enum: - - Secret - - ConfigMap - type: string - name: - description: |- - Name of the values referent. Should reside in the same namespace as the - referring resource. - maxLength: 253 - minLength: 1 - type: string - required: - - kind - - name - type: object - type: array - type: object - prune: - description: Prune enables garbage collection. - type: boolean - retryInterval: - description: |- - The interval at which to retry a previously failed reconciliation. - When not specified, the controller uses the KustomizationSpec.Interval - value to retry failures. - type: string - serviceAccountName: - description: |- - The name of the Kubernetes service account to impersonate - when reconciling this Kustomization. - type: string - sourceRef: - description: Reference of the source where the kustomization file - is. - properties: - apiVersion: - description: API version of the referent - type: string - kind: - description: Kind of the referent - enum: - - GitRepository - - Bucket - type: string - name: - description: Name of the referent - type: string - namespace: - description: Namespace of the referent, defaults to the Kustomization - namespace - type: string - required: - - kind - - name - type: object - suspend: - description: |- - This flag tells the controller to suspend subsequent kustomize executions, - it does not apply to already started executions. Defaults to false. - type: boolean - targetNamespace: - description: |- - TargetNamespace sets or overrides the namespace in the - kustomization.yaml file. - maxLength: 63 - minLength: 1 - type: string - timeout: - description: |- - Timeout for validation, apply and health checking operations. - Defaults to 'Interval' duration. - type: string - validation: - description: |- - Validate the Kubernetes objects before applying them on the cluster. - The validation strategy can be 'client' (local dry-run), 'server' - (APIServer dry-run) or 'none'. - When 'Force' is 'true', validation will fallback to 'client' if set to - 'server' because server-side validation is not supported in this scenario. - enum: - - none - - client - - server - type: string - required: - - interval - - prune - - sourceRef - type: object - status: - default: - observedGeneration: -1 - description: KustomizationStatus defines the observed state of a kustomization. - properties: - conditions: - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastAppliedRevision: - description: |- - The last successfully applied revision. - The revision format for Git sources is /. - type: string - lastAttemptedRevision: - description: LastAttemptedRevision is the revision of the last reconciliation - attempt. - type: string - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last reconciled generation. - format: int64 - type: integer - snapshot: - description: The last successfully applied revision metadata. - properties: - checksum: - description: The manifests sha1 checksum. - type: string - entries: - description: A list of Kubernetes kinds grouped by namespace. - items: - description: |- - Snapshot holds the metadata of namespaced - Kubernetes objects - properties: - kinds: - additionalProperties: - type: string - description: The list of Kubernetes kinds. - type: object - namespace: - description: The namespace of this entry. - type: string - required: - - kinds - type: object - type: array - required: - - checksum - - entries - type: object - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta2 Kustomization is deprecated, upgrade to v1 - name: v1beta2 - schema: - openAPIV3Schema: - description: Kustomization is the Schema for the kustomizations API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: KustomizationSpec defines the configuration to calculate - the desired state from a Source using Kustomize. - properties: - commonMetadata: - description: |- - CommonMetadata specifies the common labels and annotations that are applied to all resources. - Any existing label or annotation will be overridden if its key matches a common one. - properties: - annotations: - additionalProperties: - type: string - description: Annotations to be added to the object's metadata. - type: object - labels: - additionalProperties: - type: string - description: Labels to be added to the object's metadata. - type: object - type: object - components: - description: Components specifies relative paths to specifications - of other Components. - items: - type: string - type: array - decryption: - description: Decrypt Kubernetes secrets before applying them on the - cluster. - properties: - provider: - description: Provider is the name of the decryption engine. - enum: - - sops - type: string - secretRef: - description: The secret name containing the private OpenPGP keys - used for decryption. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - dependsOn: - description: |- - DependsOn may contain a meta.NamespacedObjectReference slice - with references to Kustomization resources that must be ready before this - Kustomization can be reconciled. - items: - description: |- - NamespacedObjectReference contains enough information to locate the referenced Kubernetes resource object in any - namespace. - properties: - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - name - type: object - type: array - force: - default: false - description: |- - Force instructs the controller to recreate resources - when patching fails due to an immutable field change. - type: boolean - healthChecks: - description: A list of resources to be included in the health assessment. - items: - description: |- - NamespacedObjectKindReference contains enough information to locate the typed referenced Kubernetes resource object - in any namespace. - properties: - apiVersion: - description: API version of the referent, if not specified the - Kubernetes preferred version will be used. - type: string - kind: - description: Kind of the referent. - type: string - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - kind - - name - type: object - type: array - images: - description: |- - Images is a list of (image name, new name, new tag or digest) - for changing image names, tags or digests. This can also be achieved with a - patch, but this operator is simpler to specify. - items: - description: Image contains an image name, a new name, a new tag - or digest, which will replace the original name and tag. - properties: - digest: - description: |- - Digest is the value used to replace the original image tag. - If digest is present NewTag value is ignored. - type: string - name: - description: Name is a tag-less image name. - type: string - newName: - description: NewName is the value used to replace the original - name. - type: string - newTag: - description: NewTag is the value used to replace the original - tag. - type: string - required: - - name - type: object - type: array - interval: - description: The interval at which to reconcile the Kustomization. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - kubeConfig: - description: |- - The KubeConfig for reconciling the Kustomization on a remote cluster. - When used in combination with KustomizationSpec.ServiceAccountName, - forces the controller to act on behalf of that Service Account at the - target cluster. - If the --default-service-account flag is set, its value will be used as - a controller level fallback for when KustomizationSpec.ServiceAccountName - is empty. - properties: - secretRef: - description: |- - SecretRef holds the name of a secret that contains a key with - the kubeconfig file as the value. If no key is set, the key will default - to 'value'. - It is recommended that the kubeconfig is self-contained, and the secret - is regularly updated if credentials such as a cloud-access-token expire. - Cloud specific `cmd-path` auth helpers will not function without adding - binaries and credentials to the Pod that is responsible for reconciling - Kubernetes resources. - properties: - key: - description: Key in the Secret, when not specified an implementation-specific - default key is used. - type: string - name: - description: Name of the Secret. - type: string - required: - - name - type: object - required: - - secretRef - type: object - patches: - description: |- - Strategic merge and JSON patches, defined as inline YAML objects, - capable of targeting objects based on kind, label and annotation selectors. - items: - description: |- - Patch contains an inline StrategicMerge or JSON6902 patch, and the target the patch should - be applied to. - properties: - patch: - description: |- - Patch contains an inline StrategicMerge patch or an inline JSON6902 patch with - an array of operation objects. - type: string - target: - description: Target points to the resources that the patch document - should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - type: object - type: array - patchesJson6902: - description: |- - JSON 6902 patches, defined as inline YAML objects. - Deprecated: Use Patches instead. - items: - description: JSON6902Patch contains a JSON6902 patch and the target - the patch should be applied to. - properties: - patch: - description: Patch contains the JSON6902 patch document with - an array of operation objects. - items: - description: |- - JSON6902 is a JSON6902 operation object. - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - properties: - from: - description: |- - From contains a JSON-pointer value that references a location within the target document where the operation is - performed. The meaning of the value depends on the value of Op, and is NOT taken into account by all operations. - type: string - op: - description: |- - Op indicates the operation to perform. Its value MUST be one of "add", "remove", "replace", "move", "copy", or - "test". - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - enum: - - test - - remove - - add - - replace - - move - - copy - type: string - path: - description: |- - Path contains the JSON-pointer value that references a location within the target document where the operation - is performed. The meaning of the value depends on the value of Op. - type: string - value: - description: |- - Value contains a valid JSON structure. The meaning of the value depends on the value of Op, and is NOT taken into - account by all operations. - x-kubernetes-preserve-unknown-fields: true - required: - - op - - path - type: object - type: array - target: - description: Target points to the resources that the patch document - should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - - target - type: object - type: array - patchesStrategicMerge: - description: |- - Strategic merge patches, defined as inline YAML objects. - Deprecated: Use Patches instead. - items: - x-kubernetes-preserve-unknown-fields: true - type: array - path: - description: |- - Path to the directory containing the kustomization.yaml file, or the - set of plain YAMLs a kustomization.yaml should be generated for. - Defaults to 'None', which translates to the root path of the SourceRef. - type: string - postBuild: - description: |- - PostBuild describes which actions to perform on the YAML manifest - generated by building the kustomize overlay. - properties: - substitute: - additionalProperties: - type: string - description: |- - Substitute holds a map of key/value pairs. - The variables defined in your YAML manifests - that match any of the keys defined in the map - will be substituted with the set value. - Includes support for bash string replacement functions - e.g. ${var:=default}, ${var:position} and ${var/substring/replacement}. - type: object - substituteFrom: - description: |- - SubstituteFrom holds references to ConfigMaps and Secrets containing - the variables and their values to be substituted in the YAML manifests. - The ConfigMap and the Secret data keys represent the var names and they - must match the vars declared in the manifests for the substitution to happen. - items: - description: |- - SubstituteReference contains a reference to a resource containing - the variables name and value. - properties: - kind: - description: Kind of the values referent, valid values are - ('Secret', 'ConfigMap'). - enum: - - Secret - - ConfigMap - type: string - name: - description: |- - Name of the values referent. Should reside in the same namespace as the - referring resource. - maxLength: 253 - minLength: 1 - type: string - optional: - default: false - description: |- - Optional indicates whether the referenced resource must exist, or whether to - tolerate its absence. If true and the referenced resource is absent, proceed - as if the resource was present but empty, without any variables defined. - type: boolean - required: - - kind - - name - type: object - type: array - type: object - prune: - description: Prune enables garbage collection. - type: boolean - retryInterval: - description: |- - The interval at which to retry a previously failed reconciliation. - When not specified, the controller uses the KustomizationSpec.Interval - value to retry failures. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - serviceAccountName: - description: |- - The name of the Kubernetes service account to impersonate - when reconciling this Kustomization. - type: string - sourceRef: - description: Reference of the source where the kustomization file - is. - properties: - apiVersion: - description: API version of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - OCIRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, defaults to the namespace - of the Kubernetes resource object that contains the reference. - type: string - required: - - kind - - name - type: object - suspend: - description: |- - This flag tells the controller to suspend subsequent kustomize executions, - it does not apply to already started executions. Defaults to false. - type: boolean - targetNamespace: - description: |- - TargetNamespace sets or overrides the namespace in the - kustomization.yaml file. - maxLength: 63 - minLength: 1 - type: string - timeout: - description: |- - Timeout for validation, apply and health checking operations. - Defaults to 'Interval' duration. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - validation: - description: 'Deprecated: Not used in v1beta2.' - enum: - - none - - client - - server - type: string - wait: - description: |- - Wait instructs the controller to check the health of all the reconciled resources. - When enabled, the HealthChecks are ignored. Defaults to false. - type: boolean - required: - - interval - - prune - - sourceRef - type: object - status: - default: - observedGeneration: -1 - description: KustomizationStatus defines the observed state of a kustomization. - properties: - conditions: - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - inventory: - description: Inventory contains the list of Kubernetes resource object - references that have been successfully applied. - properties: - entries: - description: Entries of Kubernetes resource object references. - items: - description: ResourceRef contains the information necessary - to locate a resource within a cluster. - properties: - id: - description: |- - ID is the string representation of the Kubernetes resource object's metadata, - in the format '___'. - type: string - v: - description: Version is the API version of the Kubernetes - resource object's kind. - type: string - required: - - id - - v - type: object - type: array - required: - - entries - type: object - lastAppliedRevision: - description: |- - The last successfully applied revision. - Equals the Revision of the applied Artifact from the referenced Source. - type: string - lastAttemptedRevision: - description: LastAttemptedRevision is the revision of the last reconciliation - attempt. - type: string - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last reconciled generation. - format: int64 - type: integer - type: object - type: object - served: true - storage: false - subresources: - status: {} ---- -apiVersion: v1 -kind: ServiceAccount -metadata: - labels: - app.kubernetes.io/component: kustomize-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: kustomize-controller - namespace: flux-system ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - labels: - app.kubernetes.io/component: kustomize-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - control-plane: controller - name: kustomize-controller - namespace: flux-system -spec: - replicas: 1 - selector: - matchLabels: - app: kustomize-controller - template: - metadata: - annotations: - prometheus.io/port: "8080" - prometheus.io/scrape: "true" - labels: - app: kustomize-controller - spec: - containers: - - args: - - --events-addr=http://notification-controller.flux-system.svc.cluster.local./ - - --watch-all-namespaces=true - - --log-level=info - - --log-encoding=json - - --enable-leader-election - env: - - name: RUNTIME_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - - name: GOMAXPROCS - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.cpu - - name: GOMEMLIMIT - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.memory - image: ghcr.io/fluxcd/kustomize-controller:v1.3.0 - imagePullPolicy: IfNotPresent - livenessProbe: - httpGet: - path: /healthz - port: healthz - name: manager - ports: - - containerPort: 8080 - name: http-prom - protocol: TCP - - containerPort: 9440 - name: healthz - protocol: TCP - readinessProbe: - httpGet: - path: /readyz - port: healthz - resources: - limits: - cpu: 1000m - memory: 1Gi - requests: - cpu: 100m - memory: 64Mi - securityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - readOnlyRootFilesystem: true - runAsNonRoot: true - seccompProfile: - type: RuntimeDefault - volumeMounts: - - mountPath: /tmp - name: temp - nodeSelector: - kubernetes.io/os: linux - priorityClassName: system-cluster-critical - securityContext: - fsGroup: 1337 - serviceAccountName: kustomize-controller - terminationGracePeriodSeconds: 60 - volumes: - - emptyDir: {} - name: temp ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: helm-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: helmreleases.helm.toolkit.fluxcd.io -spec: - group: helm.toolkit.fluxcd.io - names: - kind: HelmRelease - listKind: HelmReleaseList - plural: helmreleases - shortNames: - - hr - singular: helmrelease - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - name: v2 - schema: - openAPIV3Schema: - description: HelmRelease is the Schema for the helmreleases API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HelmReleaseSpec defines the desired state of a Helm release. - properties: - chart: - description: |- - Chart defines the template of the v1.HelmChart that should be created - for this HelmRelease. - properties: - metadata: - description: ObjectMeta holds the template for metadata like labels - and annotations. - properties: - annotations: - additionalProperties: - type: string - description: |- - Annotations is an unstructured key value map stored with a resource that may be - set by external tools to store and retrieve arbitrary metadata. They are not - queryable and should be preserved when modifying objects. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ - type: object - labels: - additionalProperties: - type: string - description: |- - Map of string keys and values that can be used to organize and categorize - (scope and select) objects. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/ - type: object - type: object - spec: - description: Spec holds the template for the v1.HelmChartSpec - for this HelmRelease. - properties: - chart: - description: The name or path the Helm chart is available - at in the SourceRef. - maxLength: 2048 - minLength: 1 - type: string - ignoreMissingValuesFiles: - description: IgnoreMissingValuesFiles controls whether to - silently ignore missing values files rather than failing. - type: boolean - interval: - description: |- - Interval at which to check the v1.Source for updates. Defaults to - 'HelmReleaseSpec.Interval'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - reconcileStrategy: - default: ChartVersion - description: |- - Determines what enables the creation of a new artifact. Valid values are - ('ChartVersion', 'Revision'). - See the documentation of the values for an explanation on their behavior. - Defaults to ChartVersion when omitted. - enum: - - ChartVersion - - Revision - type: string - sourceRef: - description: The name and namespace of the v1.Source the chart - is available at. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - HelmRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: Namespace of the referent. - maxLength: 63 - minLength: 1 - type: string - required: - - name - type: object - valuesFiles: - description: |- - Alternative list of values files to use as the chart values (values.yaml - is not included by default), expected to be a relative path in the SourceRef. - Values files are merged in the order of this list with the last file overriding - the first. Ignored when omitted. - items: - type: string - type: array - verify: - description: |- - Verify contains the secret name containing the trusted public keys - used to verify the signature and specifies which provider to use to check - whether OCI image is authentic. - This field is only supported for OCI sources. - Chart dependencies, which are not bundled in the umbrella chart artifact, - are not verified. - properties: - provider: - default: cosign - description: Provider specifies the technology used to - sign the OCI Helm chart. - enum: - - cosign - - notation - type: string - secretRef: - description: |- - SecretRef specifies the Kubernetes Secret containing the - trusted public keys. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - version: - default: '*' - description: |- - Version semver expression, ignored for charts from v1.GitRepository and - v1beta2.Bucket sources. Defaults to latest when omitted. - type: string - required: - - chart - - sourceRef - type: object - required: - - spec - type: object - chartRef: - description: |- - ChartRef holds a reference to a source controller resource containing the - Helm chart artifact. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - OCIRepository - - HelmChart - type: string - name: - description: Name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace of the referent, defaults to the namespace of the Kubernetes - resource object that contains the reference. - maxLength: 63 - minLength: 1 - type: string - required: - - kind - - name - type: object - dependsOn: - description: |- - DependsOn may contain a meta.NamespacedObjectReference slice with - references to HelmRelease resources that must be ready before this HelmRelease - can be reconciled. - items: - description: |- - NamespacedObjectReference contains enough information to locate the referenced Kubernetes resource object in any - namespace. - properties: - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - name - type: object - type: array - driftDetection: - description: |- - DriftDetection holds the configuration for detecting and handling - differences between the manifest in the Helm storage and the resources - currently existing in the cluster. - properties: - ignore: - description: |- - Ignore contains a list of rules for specifying which changes to ignore - during diffing. - items: - description: |- - IgnoreRule defines a rule to selectively disregard specific changes during - the drift detection process. - properties: - paths: - description: |- - Paths is a list of JSON Pointer (RFC 6901) paths to be excluded from - consideration in a Kubernetes object. - items: - type: string - type: array - target: - description: |- - Target is a selector for specifying Kubernetes objects to which this - rule applies. - If Target is not set, the Paths will be ignored for all Kubernetes - objects within the manifest of the Helm release. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - paths - type: object - type: array - mode: - description: |- - Mode defines how differences should be handled between the Helm manifest - and the manifest currently applied to the cluster. - If not explicitly set, it defaults to DiffModeDisabled. - enum: - - enabled - - warn - - disabled - type: string - type: object - install: - description: Install holds the configuration for Helm install actions - for this HelmRelease. - properties: - crds: - description: |- - CRDs upgrade CRDs from the Helm Chart's crds directory according - to the CRD upgrade policy provided here. Valid values are `Skip`, - `Create` or `CreateReplace`. Default is `Create` and if omitted - CRDs are installed but not updated. - - - Skip: do neither install nor replace (update) any CRDs. - - - Create: new CRDs are created, existing CRDs are neither updated nor deleted. - - - CreateReplace: new CRDs are created, existing CRDs are updated (replaced) - but not deleted. - - - By default, CRDs are applied (installed) during Helm install action. - With this option users can opt in to CRD replace existing CRDs on Helm - install actions, which is not (yet) natively supported by Helm. - https://helm.sh/docs/chart_best_practices/custom_resource_definitions. - enum: - - Skip - - Create - - CreateReplace - type: string - createNamespace: - description: |- - CreateNamespace tells the Helm install action to create the - HelmReleaseSpec.TargetNamespace if it does not exist yet. - On uninstall, the namespace will not be garbage collected. - type: boolean - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm install action. - type: boolean - disableOpenAPIValidation: - description: |- - DisableOpenAPIValidation prevents the Helm install action from validating - rendered templates against the Kubernetes OpenAPI Schema. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - install has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - install has been performed. - type: boolean - remediation: - description: |- - Remediation holds the remediation configuration for when the Helm install - action for the HelmRelease fails. The default is to not perform any action. - properties: - ignoreTestFailures: - description: |- - IgnoreTestFailures tells the controller to skip remediation when the Helm - tests are run after an install action but fail. Defaults to - 'Test.IgnoreFailures'. - type: boolean - remediateLastFailure: - description: |- - RemediateLastFailure tells the controller to remediate the last failure, when - no retries remain. Defaults to 'false'. - type: boolean - retries: - description: |- - Retries is the number of retries that should be attempted on failures before - bailing. Remediation, using an uninstall, is performed between each attempt. - Defaults to '0', a negative integer equals to unlimited retries. - type: integer - type: object - replace: - description: |- - Replace tells the Helm install action to re-use the 'ReleaseName', but only - if that name is a deleted release which remains in the history. - type: boolean - skipCRDs: - description: |- - SkipCRDs tells the Helm install action to not install any CRDs. By default, - CRDs are installed if not already present. - - - Deprecated use CRD policy (`crds`) attribute with value `Skip` instead. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm install action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - interval: - description: Interval at which to reconcile the Helm release. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - kubeConfig: - description: |- - KubeConfig for reconciling the HelmRelease on a remote cluster. - When used in combination with HelmReleaseSpec.ServiceAccountName, - forces the controller to act on behalf of that Service Account at the - target cluster. - If the --default-service-account flag is set, its value will be used as - a controller level fallback for when HelmReleaseSpec.ServiceAccountName - is empty. - properties: - secretRef: - description: |- - SecretRef holds the name of a secret that contains a key with - the kubeconfig file as the value. If no key is set, the key will default - to 'value'. - It is recommended that the kubeconfig is self-contained, and the secret - is regularly updated if credentials such as a cloud-access-token expire. - Cloud specific `cmd-path` auth helpers will not function without adding - binaries and credentials to the Pod that is responsible for reconciling - Kubernetes resources. - properties: - key: - description: Key in the Secret, when not specified an implementation-specific - default key is used. - type: string - name: - description: Name of the Secret. - type: string - required: - - name - type: object - required: - - secretRef - type: object - maxHistory: - description: |- - MaxHistory is the number of revisions saved by Helm for this HelmRelease. - Use '0' for an unlimited number of revisions; defaults to '5'. - type: integer - persistentClient: - description: |- - PersistentClient tells the controller to use a persistent Kubernetes - client for this release. When enabled, the client will be reused for the - duration of the reconciliation, instead of being created and destroyed - for each (step of a) Helm action. - - - This can improve performance, but may cause issues with some Helm charts - that for example do create Custom Resource Definitions during installation - outside Helm's CRD lifecycle hooks, which are then not observed to be - available by e.g. post-install hooks. - - - If not set, it defaults to true. - type: boolean - postRenderers: - description: |- - PostRenderers holds an array of Helm PostRenderers, which will be applied in order - of their definition. - items: - description: PostRenderer contains a Helm PostRenderer specification. - properties: - kustomize: - description: Kustomization to apply as PostRenderer. - properties: - images: - description: |- - Images is a list of (image name, new name, new tag or digest) - for changing image names, tags or digests. This can also be achieved with a - patch, but this operator is simpler to specify. - items: - description: Image contains an image name, a new name, - a new tag or digest, which will replace the original - name and tag. - properties: - digest: - description: |- - Digest is the value used to replace the original image tag. - If digest is present NewTag value is ignored. - type: string - name: - description: Name is a tag-less image name. - type: string - newName: - description: NewName is the value used to replace - the original name. - type: string - newTag: - description: NewTag is the value used to replace the - original tag. - type: string - required: - - name - type: object - type: array - patches: - description: |- - Strategic merge and JSON patches, defined as inline YAML objects, - capable of targeting objects based on kind, label and annotation selectors. - items: - description: |- - Patch contains an inline StrategicMerge or JSON6902 patch, and the target the patch should - be applied to. - properties: - patch: - description: |- - Patch contains an inline StrategicMerge patch or an inline JSON6902 patch with - an array of operation objects. - type: string - target: - description: Target points to the resources that the - patch document should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - type: object - type: array - type: object - type: object - type: array - releaseName: - description: |- - ReleaseName used for the Helm release. Defaults to a composition of - '[TargetNamespace-]Name'. - maxLength: 53 - minLength: 1 - type: string - rollback: - description: Rollback holds the configuration for Helm rollback actions - for this HelmRelease. - properties: - cleanupOnFail: - description: |- - CleanupOnFail allows deletion of new resources created during the Helm - rollback action when it fails. - type: boolean - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm rollback action. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - rollback has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - rollback has been performed. - type: boolean - force: - description: Force forces resource updates through a replacement - strategy. - type: boolean - recreate: - description: Recreate performs pod restarts for the resource if - applicable. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm rollback action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - serviceAccountName: - description: |- - The name of the Kubernetes service account to impersonate - when reconciling this HelmRelease. - maxLength: 253 - minLength: 1 - type: string - storageNamespace: - description: |- - StorageNamespace used for the Helm storage. - Defaults to the namespace of the HelmRelease. - maxLength: 63 - minLength: 1 - type: string - suspend: - description: |- - Suspend tells the controller to suspend reconciliation for this HelmRelease, - it does not apply to already started reconciliations. Defaults to false. - type: boolean - targetNamespace: - description: |- - TargetNamespace to target when performing operations for the HelmRelease. - Defaults to the namespace of the HelmRelease. - maxLength: 63 - minLength: 1 - type: string - test: - description: Test holds the configuration for Helm test actions for - this HelmRelease. - properties: - enable: - description: |- - Enable enables Helm test actions for this HelmRelease after an Helm install - or upgrade action has been performed. - type: boolean - filters: - description: Filters is a list of tests to run or exclude from - running. - items: - description: Filter holds the configuration for individual Helm - test filters. - properties: - exclude: - description: Exclude specifies whether the named test should - be excluded. - type: boolean - name: - description: Name is the name of the test. - maxLength: 253 - minLength: 1 - type: string - required: - - name - type: object - type: array - ignoreFailures: - description: |- - IgnoreFailures tells the controller to skip remediation when the Helm tests - are run but fail. Can be overwritten for tests run after install or upgrade - actions in 'Install.IgnoreTestFailures' and 'Upgrade.IgnoreTestFailures'. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation during - the performance of a Helm test action. Defaults to 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like Jobs - for hooks) during the performance of a Helm action. Defaults to '5m0s'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - uninstall: - description: Uninstall holds the configuration for Helm uninstall - actions for this HelmRelease. - properties: - deletionPropagation: - default: background - description: |- - DeletionPropagation specifies the deletion propagation policy when - a Helm uninstall is performed. - enum: - - background - - foreground - - orphan - type: string - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm rollback action. - type: boolean - disableWait: - description: |- - DisableWait disables waiting for all the resources to be deleted after - a Helm uninstall is performed. - type: boolean - keepHistory: - description: |- - KeepHistory tells Helm to remove all associated resources and mark the - release as deleted, but retain the release history. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm uninstall action. Defaults - to 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - upgrade: - description: Upgrade holds the configuration for Helm upgrade actions - for this HelmRelease. - properties: - cleanupOnFail: - description: |- - CleanupOnFail allows deletion of new resources created during the Helm - upgrade action when it fails. - type: boolean - crds: - description: |- - CRDs upgrade CRDs from the Helm Chart's crds directory according - to the CRD upgrade policy provided here. Valid values are `Skip`, - `Create` or `CreateReplace`. Default is `Skip` and if omitted - CRDs are neither installed nor upgraded. - - - Skip: do neither install nor replace (update) any CRDs. - - - Create: new CRDs are created, existing CRDs are neither updated nor deleted. - - - CreateReplace: new CRDs are created, existing CRDs are updated (replaced) - but not deleted. - - - By default, CRDs are not applied during Helm upgrade action. With this - option users can opt-in to CRD upgrade, which is not (yet) natively supported by Helm. - https://helm.sh/docs/chart_best_practices/custom_resource_definitions. - enum: - - Skip - - Create - - CreateReplace - type: string - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm upgrade action. - type: boolean - disableOpenAPIValidation: - description: |- - DisableOpenAPIValidation prevents the Helm upgrade action from validating - rendered templates against the Kubernetes OpenAPI Schema. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - upgrade has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - upgrade has been performed. - type: boolean - force: - description: Force forces resource updates through a replacement - strategy. - type: boolean - preserveValues: - description: |- - PreserveValues will make Helm reuse the last release's values and merge in - overrides from 'Values'. Setting this flag makes the HelmRelease - non-declarative. - type: boolean - remediation: - description: |- - Remediation holds the remediation configuration for when the Helm upgrade - action for the HelmRelease fails. The default is to not perform any action. - properties: - ignoreTestFailures: - description: |- - IgnoreTestFailures tells the controller to skip remediation when the Helm - tests are run after an upgrade action but fail. - Defaults to 'Test.IgnoreFailures'. - type: boolean - remediateLastFailure: - description: |- - RemediateLastFailure tells the controller to remediate the last failure, when - no retries remain. Defaults to 'false' unless 'Retries' is greater than 0. - type: boolean - retries: - description: |- - Retries is the number of retries that should be attempted on failures before - bailing. Remediation, using 'Strategy', is performed between each attempt. - Defaults to '0', a negative integer equals to unlimited retries. - type: integer - strategy: - description: Strategy to use for failure remediation. Defaults - to 'rollback'. - enum: - - rollback - - uninstall - type: string - type: object - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm upgrade action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - values: - description: Values holds the values for this Helm release. - x-kubernetes-preserve-unknown-fields: true - valuesFrom: - description: |- - ValuesFrom holds references to resources containing Helm values for this HelmRelease, - and information about how they should be merged. - items: - description: |- - ValuesReference contains a reference to a resource containing Helm values, - and optionally the key they can be found at. - properties: - kind: - description: Kind of the values referent, valid values are ('Secret', - 'ConfigMap'). - enum: - - Secret - - ConfigMap - type: string - name: - description: |- - Name of the values referent. Should reside in the same namespace as the - referring resource. - maxLength: 253 - minLength: 1 - type: string - optional: - description: |- - Optional marks this ValuesReference as optional. When set, a not found error - for the values reference is ignored, but any ValuesKey, TargetPath or - transient error will still result in a reconciliation failure. - type: boolean - targetPath: - description: |- - TargetPath is the YAML dot notation path the value should be merged at. When - set, the ValuesKey is expected to be a single flat value. Defaults to 'None', - which results in the values getting merged at the root. - maxLength: 250 - pattern: ^([a-zA-Z0-9_\-.\\\/]|\[[0-9]{1,5}\])+$ - type: string - valuesKey: - description: |- - ValuesKey is the data key where the values.yaml or a specific value can be - found at. Defaults to 'values.yaml'. - maxLength: 253 - pattern: ^[\-._a-zA-Z0-9]+$ - type: string - required: - - kind - - name - type: object - type: array - required: - - interval - type: object - x-kubernetes-validations: - - message: either chart or chartRef must be set - rule: (has(self.chart) && !has(self.chartRef)) || (!has(self.chart) - && has(self.chartRef)) - status: - default: - observedGeneration: -1 - description: HelmReleaseStatus defines the observed state of a HelmRelease. - properties: - conditions: - description: Conditions holds the conditions for the HelmRelease. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - failures: - description: |- - Failures is the reconciliation failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - helmChart: - description: |- - HelmChart is the namespaced name of the HelmChart resource created by - the controller for the HelmRelease. - type: string - history: - description: |- - History holds the history of Helm releases performed for this HelmRelease - up to the last successfully completed release. - items: - description: |- - Snapshot captures a point-in-time copy of the status information for a Helm release, - as managed by the controller. - properties: - apiVersion: - description: |- - APIVersion is the API version of the Snapshot. - Provisional: when the calculation method of the Digest field is changed, - this field will be used to distinguish between the old and new methods. - type: string - appVersion: - description: AppVersion is the chart app version of the release - object in storage. - type: string - chartName: - description: ChartName is the chart name of the release object - in storage. - type: string - chartVersion: - description: |- - ChartVersion is the chart version of the release object in - storage. - type: string - configDigest: - description: |- - ConfigDigest is the checksum of the config (better known as - "values") of the release object in storage. - It has the format of `:`. - type: string - deleted: - description: Deleted is when the release was deleted. - format: date-time - type: string - digest: - description: |- - Digest is the checksum of the release object in storage. - It has the format of `:`. - type: string - firstDeployed: - description: FirstDeployed is when the release was first deployed. - format: date-time - type: string - lastDeployed: - description: LastDeployed is when the release was last deployed. - format: date-time - type: string - name: - description: Name is the name of the release. - type: string - namespace: - description: Namespace is the namespace the release is deployed - to. - type: string - ociDigest: - description: OCIDigest is the digest of the OCI artifact associated - with the release. - type: string - status: - description: Status is the current state of the release. - type: string - testHooks: - additionalProperties: - description: |- - TestHookStatus holds the status information for a test hook as observed - to be run by the controller. - properties: - lastCompleted: - description: LastCompleted is the time the test hook last - completed. - format: date-time - type: string - lastStarted: - description: LastStarted is the time the test hook was - last started. - format: date-time - type: string - phase: - description: Phase the test hook was observed to be in. - type: string - type: object - description: |- - TestHooks is the list of test hooks for the release as observed to be - run by the controller. - type: object - version: - description: Version is the version of the release object in - storage. - type: integer - required: - - chartName - - chartVersion - - configDigest - - digest - - firstDeployed - - lastDeployed - - name - - namespace - - status - - version - type: object - type: array - installFailures: - description: |- - InstallFailures is the install failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - lastAttemptedConfigDigest: - description: |- - LastAttemptedConfigDigest is the digest for the config (better known as - "values") of the last reconciliation attempt. - type: string - lastAttemptedGeneration: - description: |- - LastAttemptedGeneration is the last generation the controller attempted - to reconcile. - format: int64 - type: integer - lastAttemptedReleaseAction: - description: |- - LastAttemptedReleaseAction is the last release action performed for this - HelmRelease. It is used to determine the active remediation strategy. - enum: - - install - - upgrade - type: string - lastAttemptedRevision: - description: |- - LastAttemptedRevision is the Source revision of the last reconciliation - attempt. For OCIRepository sources, the 12 first characters of the digest are - appended to the chart version e.g. "1.2.3+1234567890ab". - type: string - lastAttemptedRevisionDigest: - description: |- - LastAttemptedRevisionDigest is the digest of the last reconciliation attempt. - This is only set for OCIRepository sources. - type: string - lastAttemptedValuesChecksum: - description: |- - LastAttemptedValuesChecksum is the SHA1 checksum for the values of the last - reconciliation attempt. - Deprecated: Use LastAttemptedConfigDigest instead. - type: string - lastHandledForceAt: - description: |- - LastHandledForceAt holds the value of the most recent force request - value, so a change of the annotation value can be detected. - type: string - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - lastHandledResetAt: - description: |- - LastHandledResetAt holds the value of the most recent reset request - value, so a change of the annotation value can be detected. - type: string - lastReleaseRevision: - description: |- - LastReleaseRevision is the revision of the last successful Helm release. - Deprecated: Use History instead. - type: integer - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - observedPostRenderersDigest: - description: |- - ObservedPostRenderersDigest is the digest for the post-renderers of - the last successful reconciliation attempt. - type: string - storageNamespace: - description: |- - StorageNamespace is the namespace of the Helm release storage for the - current release. - maxLength: 63 - minLength: 1 - type: string - upgradeFailures: - description: |- - UpgradeFailures is the upgrade failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - type: object - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v2beta1 HelmRelease is deprecated, upgrade to v2 - name: v2beta1 - schema: - openAPIV3Schema: - description: HelmRelease is the Schema for the helmreleases API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HelmReleaseSpec defines the desired state of a Helm release. - properties: - chart: - description: |- - Chart defines the template of the v1beta2.HelmChart that should be created - for this HelmRelease. - properties: - metadata: - description: ObjectMeta holds the template for metadata like labels - and annotations. - properties: - annotations: - additionalProperties: - type: string - description: |- - Annotations is an unstructured key value map stored with a resource that may be - set by external tools to store and retrieve arbitrary metadata. They are not - queryable and should be preserved when modifying objects. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ - type: object - labels: - additionalProperties: - type: string - description: |- - Map of string keys and values that can be used to organize and categorize - (scope and select) objects. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/ - type: object - type: object - spec: - description: Spec holds the template for the v1beta2.HelmChartSpec - for this HelmRelease. - properties: - chart: - description: The name or path the Helm chart is available - at in the SourceRef. - type: string - interval: - description: |- - Interval at which to check the v1beta2.Source for updates. Defaults to - 'HelmReleaseSpec.Interval'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - reconcileStrategy: - default: ChartVersion - description: |- - Determines what enables the creation of a new artifact. Valid values are - ('ChartVersion', 'Revision'). - See the documentation of the values for an explanation on their behavior. - Defaults to ChartVersion when omitted. - enum: - - ChartVersion - - Revision - type: string - sourceRef: - description: The name and namespace of the v1beta2.Source - the chart is available at. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - HelmRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: Namespace of the referent. - maxLength: 63 - minLength: 1 - type: string - required: - - name - type: object - valuesFile: - description: |- - Alternative values file to use as the default chart values, expected to - be a relative path in the SourceRef. Deprecated in favor of ValuesFiles, - for backwards compatibility the file defined here is merged before the - ValuesFiles items. Ignored when omitted. - type: string - valuesFiles: - description: |- - Alternative list of values files to use as the chart values (values.yaml - is not included by default), expected to be a relative path in the SourceRef. - Values files are merged in the order of this list with the last file overriding - the first. Ignored when omitted. - items: - type: string - type: array - verify: - description: |- - Verify contains the secret name containing the trusted public keys - used to verify the signature and specifies which provider to use to check - whether OCI image is authentic. - This field is only supported for OCI sources. - Chart dependencies, which are not bundled in the umbrella chart artifact, are not verified. - properties: - provider: - default: cosign - description: Provider specifies the technology used to - sign the OCI Helm chart. - enum: - - cosign - type: string - secretRef: - description: |- - SecretRef specifies the Kubernetes Secret containing the - trusted public keys. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - version: - default: '*' - description: |- - Version semver expression, ignored for charts from v1beta2.GitRepository and - v1beta2.Bucket sources. Defaults to latest when omitted. - type: string - required: - - chart - - sourceRef - type: object - required: - - spec - type: object - chartRef: - description: |- - ChartRef holds a reference to a source controller resource containing the - Helm chart artifact. - - - Note: this field is provisional to the v2 API, and not actively used - by v2beta1 HelmReleases. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - OCIRepository - - HelmChart - type: string - name: - description: Name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace of the referent, defaults to the namespace of the Kubernetes - resource object that contains the reference. - maxLength: 63 - minLength: 1 - type: string - required: - - kind - - name - type: object - dependsOn: - description: |- - DependsOn may contain a meta.NamespacedObjectReference slice with - references to HelmRelease resources that must be ready before this HelmRelease - can be reconciled. - items: - description: |- - NamespacedObjectReference contains enough information to locate the referenced Kubernetes resource object in any - namespace. - properties: - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - name - type: object - type: array - driftDetection: - description: |- - DriftDetection holds the configuration for detecting and handling - differences between the manifest in the Helm storage and the resources - currently existing in the cluster. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - properties: - ignore: - description: |- - Ignore contains a list of rules for specifying which changes to ignore - during diffing. - items: - description: |- - IgnoreRule defines a rule to selectively disregard specific changes during - the drift detection process. - properties: - paths: - description: |- - Paths is a list of JSON Pointer (RFC 6901) paths to be excluded from - consideration in a Kubernetes object. - items: - type: string - type: array - target: - description: |- - Target is a selector for specifying Kubernetes objects to which this - rule applies. - If Target is not set, the Paths will be ignored for all Kubernetes - objects within the manifest of the Helm release. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - paths - type: object - type: array - mode: - description: |- - Mode defines how differences should be handled between the Helm manifest - and the manifest currently applied to the cluster. - If not explicitly set, it defaults to DiffModeDisabled. - enum: - - enabled - - warn - - disabled - type: string - type: object - install: - description: Install holds the configuration for Helm install actions - for this HelmRelease. - properties: - crds: - description: |- - CRDs upgrade CRDs from the Helm Chart's crds directory according - to the CRD upgrade policy provided here. Valid values are `Skip`, - `Create` or `CreateReplace`. Default is `Create` and if omitted - CRDs are installed but not updated. - - - Skip: do neither install nor replace (update) any CRDs. - - - Create: new CRDs are created, existing CRDs are neither updated nor deleted. - - - CreateReplace: new CRDs are created, existing CRDs are updated (replaced) - but not deleted. - - - By default, CRDs are applied (installed) during Helm install action. - With this option users can opt-in to CRD replace existing CRDs on Helm - install actions, which is not (yet) natively supported by Helm. - https://helm.sh/docs/chart_best_practices/custom_resource_definitions. - enum: - - Skip - - Create - - CreateReplace - type: string - createNamespace: - description: |- - CreateNamespace tells the Helm install action to create the - HelmReleaseSpec.TargetNamespace if it does not exist yet. - On uninstall, the namespace will not be garbage collected. - type: boolean - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm install action. - type: boolean - disableOpenAPIValidation: - description: |- - DisableOpenAPIValidation prevents the Helm install action from validating - rendered templates against the Kubernetes OpenAPI Schema. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - install has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - install has been performed. - type: boolean - remediation: - description: |- - Remediation holds the remediation configuration for when the Helm install - action for the HelmRelease fails. The default is to not perform any action. - properties: - ignoreTestFailures: - description: |- - IgnoreTestFailures tells the controller to skip remediation when the Helm - tests are run after an install action but fail. Defaults to - 'Test.IgnoreFailures'. - type: boolean - remediateLastFailure: - description: |- - RemediateLastFailure tells the controller to remediate the last failure, when - no retries remain. Defaults to 'false'. - type: boolean - retries: - description: |- - Retries is the number of retries that should be attempted on failures before - bailing. Remediation, using an uninstall, is performed between each attempt. - Defaults to '0', a negative integer equals to unlimited retries. - type: integer - type: object - replace: - description: |- - Replace tells the Helm install action to re-use the 'ReleaseName', but only - if that name is a deleted release which remains in the history. - type: boolean - skipCRDs: - description: |- - SkipCRDs tells the Helm install action to not install any CRDs. By default, - CRDs are installed if not already present. - - - Deprecated use CRD policy (`crds`) attribute with value `Skip` instead. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm install action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - interval: - description: |- - Interval at which to reconcile the Helm release. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - kubeConfig: - description: |- - KubeConfig for reconciling the HelmRelease on a remote cluster. - When used in combination with HelmReleaseSpec.ServiceAccountName, - forces the controller to act on behalf of that Service Account at the - target cluster. - If the --default-service-account flag is set, its value will be used as - a controller level fallback for when HelmReleaseSpec.ServiceAccountName - is empty. - properties: - secretRef: - description: |- - SecretRef holds the name of a secret that contains a key with - the kubeconfig file as the value. If no key is set, the key will default - to 'value'. - It is recommended that the kubeconfig is self-contained, and the secret - is regularly updated if credentials such as a cloud-access-token expire. - Cloud specific `cmd-path` auth helpers will not function without adding - binaries and credentials to the Pod that is responsible for reconciling - Kubernetes resources. - properties: - key: - description: Key in the Secret, when not specified an implementation-specific - default key is used. - type: string - name: - description: Name of the Secret. - type: string - required: - - name - type: object - required: - - secretRef - type: object - maxHistory: - description: |- - MaxHistory is the number of revisions saved by Helm for this HelmRelease. - Use '0' for an unlimited number of revisions; defaults to '10'. - type: integer - persistentClient: - description: |- - PersistentClient tells the controller to use a persistent Kubernetes - client for this release. When enabled, the client will be reused for the - duration of the reconciliation, instead of being created and destroyed - for each (step of a) Helm action. - - - This can improve performance, but may cause issues with some Helm charts - that for example do create Custom Resource Definitions during installation - outside Helm's CRD lifecycle hooks, which are then not observed to be - available by e.g. post-install hooks. - - - If not set, it defaults to true. - type: boolean - postRenderers: - description: |- - PostRenderers holds an array of Helm PostRenderers, which will be applied in order - of their definition. - items: - description: PostRenderer contains a Helm PostRenderer specification. - properties: - kustomize: - description: Kustomization to apply as PostRenderer. - properties: - images: - description: |- - Images is a list of (image name, new name, new tag or digest) - for changing image names, tags or digests. This can also be achieved with a - patch, but this operator is simpler to specify. - items: - description: Image contains an image name, a new name, - a new tag or digest, which will replace the original - name and tag. - properties: - digest: - description: |- - Digest is the value used to replace the original image tag. - If digest is present NewTag value is ignored. - type: string - name: - description: Name is a tag-less image name. - type: string - newName: - description: NewName is the value used to replace - the original name. - type: string - newTag: - description: NewTag is the value used to replace the - original tag. - type: string - required: - - name - type: object - type: array - patches: - description: |- - Strategic merge and JSON patches, defined as inline YAML objects, - capable of targeting objects based on kind, label and annotation selectors. - items: - description: |- - Patch contains an inline StrategicMerge or JSON6902 patch, and the target the patch should - be applied to. - properties: - patch: - description: |- - Patch contains an inline StrategicMerge patch or an inline JSON6902 patch with - an array of operation objects. - type: string - target: - description: Target points to the resources that the - patch document should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - type: object - type: array - patchesJson6902: - description: JSON 6902 patches, defined as inline YAML objects. - items: - description: JSON6902Patch contains a JSON6902 patch and - the target the patch should be applied to. - properties: - patch: - description: Patch contains the JSON6902 patch document - with an array of operation objects. - items: - description: |- - JSON6902 is a JSON6902 operation object. - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - properties: - from: - description: |- - From contains a JSON-pointer value that references a location within the target document where the operation is - performed. The meaning of the value depends on the value of Op, and is NOT taken into account by all operations. - type: string - op: - description: |- - Op indicates the operation to perform. Its value MUST be one of "add", "remove", "replace", "move", "copy", or - "test". - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - enum: - - test - - remove - - add - - replace - - move - - copy - type: string - path: - description: |- - Path contains the JSON-pointer value that references a location within the target document where the operation - is performed. The meaning of the value depends on the value of Op. - type: string - value: - description: |- - Value contains a valid JSON structure. The meaning of the value depends on the value of Op, and is NOT taken into - account by all operations. - x-kubernetes-preserve-unknown-fields: true - required: - - op - - path - type: object - type: array - target: - description: Target points to the resources that the - patch document should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - - target - type: object - type: array - patchesStrategicMerge: - description: Strategic merge patches, defined as inline - YAML objects. - items: - x-kubernetes-preserve-unknown-fields: true - type: array - type: object - type: object - type: array - releaseName: - description: |- - ReleaseName used for the Helm release. Defaults to a composition of - '[TargetNamespace-]Name'. - maxLength: 53 - minLength: 1 - type: string - rollback: - description: Rollback holds the configuration for Helm rollback actions - for this HelmRelease. - properties: - cleanupOnFail: - description: |- - CleanupOnFail allows deletion of new resources created during the Helm - rollback action when it fails. - type: boolean - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm rollback action. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - rollback has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - rollback has been performed. - type: boolean - force: - description: Force forces resource updates through a replacement - strategy. - type: boolean - recreate: - description: Recreate performs pod restarts for the resource if - applicable. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm rollback action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - serviceAccountName: - description: |- - The name of the Kubernetes service account to impersonate - when reconciling this HelmRelease. - type: string - storageNamespace: - description: |- - StorageNamespace used for the Helm storage. - Defaults to the namespace of the HelmRelease. - maxLength: 63 - minLength: 1 - type: string - suspend: - description: |- - Suspend tells the controller to suspend reconciliation for this HelmRelease, - it does not apply to already started reconciliations. Defaults to false. - type: boolean - targetNamespace: - description: |- - TargetNamespace to target when performing operations for the HelmRelease. - Defaults to the namespace of the HelmRelease. - maxLength: 63 - minLength: 1 - type: string - test: - description: Test holds the configuration for Helm test actions for - this HelmRelease. - properties: - enable: - description: |- - Enable enables Helm test actions for this HelmRelease after an Helm install - or upgrade action has been performed. - type: boolean - ignoreFailures: - description: |- - IgnoreFailures tells the controller to skip remediation when the Helm tests - are run but fail. Can be overwritten for tests run after install or upgrade - actions in 'Install.IgnoreTestFailures' and 'Upgrade.IgnoreTestFailures'. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation during - the performance of a Helm test action. Defaults to 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like Jobs - for hooks) during the performance of a Helm action. Defaults to '5m0s'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - uninstall: - description: Uninstall holds the configuration for Helm uninstall - actions for this HelmRelease. - properties: - deletionPropagation: - default: background - description: |- - DeletionPropagation specifies the deletion propagation policy when - a Helm uninstall is performed. - enum: - - background - - foreground - - orphan - type: string - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm rollback action. - type: boolean - disableWait: - description: |- - DisableWait disables waiting for all the resources to be deleted after - a Helm uninstall is performed. - type: boolean - keepHistory: - description: |- - KeepHistory tells Helm to remove all associated resources and mark the - release as deleted, but retain the release history. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm uninstall action. Defaults - to 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - upgrade: - description: Upgrade holds the configuration for Helm upgrade actions - for this HelmRelease. - properties: - cleanupOnFail: - description: |- - CleanupOnFail allows deletion of new resources created during the Helm - upgrade action when it fails. - type: boolean - crds: - description: |- - CRDs upgrade CRDs from the Helm Chart's crds directory according - to the CRD upgrade policy provided here. Valid values are `Skip`, - `Create` or `CreateReplace`. Default is `Skip` and if omitted - CRDs are neither installed nor upgraded. - - - Skip: do neither install nor replace (update) any CRDs. - - - Create: new CRDs are created, existing CRDs are neither updated nor deleted. - - - CreateReplace: new CRDs are created, existing CRDs are updated (replaced) - but not deleted. - - - By default, CRDs are not applied during Helm upgrade action. With this - option users can opt-in to CRD upgrade, which is not (yet) natively supported by Helm. - https://helm.sh/docs/chart_best_practices/custom_resource_definitions. - enum: - - Skip - - Create - - CreateReplace - type: string - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm upgrade action. - type: boolean - disableOpenAPIValidation: - description: |- - DisableOpenAPIValidation prevents the Helm upgrade action from validating - rendered templates against the Kubernetes OpenAPI Schema. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - upgrade has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - upgrade has been performed. - type: boolean - force: - description: Force forces resource updates through a replacement - strategy. - type: boolean - preserveValues: - description: |- - PreserveValues will make Helm reuse the last release's values and merge in - overrides from 'Values'. Setting this flag makes the HelmRelease - non-declarative. - type: boolean - remediation: - description: |- - Remediation holds the remediation configuration for when the Helm upgrade - action for the HelmRelease fails. The default is to not perform any action. - properties: - ignoreTestFailures: - description: |- - IgnoreTestFailures tells the controller to skip remediation when the Helm - tests are run after an upgrade action but fail. - Defaults to 'Test.IgnoreFailures'. - type: boolean - remediateLastFailure: - description: |- - RemediateLastFailure tells the controller to remediate the last failure, when - no retries remain. Defaults to 'false' unless 'Retries' is greater than 0. - type: boolean - retries: - description: |- - Retries is the number of retries that should be attempted on failures before - bailing. Remediation, using 'Strategy', is performed between each attempt. - Defaults to '0', a negative integer equals to unlimited retries. - type: integer - strategy: - description: Strategy to use for failure remediation. Defaults - to 'rollback'. - enum: - - rollback - - uninstall - type: string - type: object - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm upgrade action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - values: - description: Values holds the values for this Helm release. - x-kubernetes-preserve-unknown-fields: true - valuesFrom: - description: |- - ValuesFrom holds references to resources containing Helm values for this HelmRelease, - and information about how they should be merged. - items: - description: |- - ValuesReference contains a reference to a resource containing Helm values, - and optionally the key they can be found at. - properties: - kind: - description: Kind of the values referent, valid values are ('Secret', - 'ConfigMap'). - enum: - - Secret - - ConfigMap - type: string - name: - description: |- - Name of the values referent. Should reside in the same namespace as the - referring resource. - maxLength: 253 - minLength: 1 - type: string - optional: - description: |- - Optional marks this ValuesReference as optional. When set, a not found error - for the values reference is ignored, but any ValuesKey, TargetPath or - transient error will still result in a reconciliation failure. - type: boolean - targetPath: - description: |- - TargetPath is the YAML dot notation path the value should be merged at. When - set, the ValuesKey is expected to be a single flat value. Defaults to 'None', - which results in the values getting merged at the root. - maxLength: 250 - pattern: ^([a-zA-Z0-9_\-.\\\/]|\[[0-9]{1,5}\])+$ - type: string - valuesKey: - description: |- - ValuesKey is the data key where the values.yaml or a specific value can be - found at. Defaults to 'values.yaml'. - When set, must be a valid Data Key, consisting of alphanumeric characters, - '-', '_' or '.'. - maxLength: 253 - pattern: ^[\-._a-zA-Z0-9]+$ - type: string - required: - - kind - - name - type: object - type: array - required: - - interval - type: object - status: - default: - observedGeneration: -1 - description: HelmReleaseStatus defines the observed state of a HelmRelease. - properties: - conditions: - description: Conditions holds the conditions for the HelmRelease. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - failures: - description: |- - Failures is the reconciliation failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - helmChart: - description: |- - HelmChart is the namespaced name of the HelmChart resource created by - the controller for the HelmRelease. - type: string - history: - description: |- - History holds the history of Helm releases performed for this HelmRelease - up to the last successfully completed release. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - items: - description: |- - Snapshot captures a point-in-time copy of the status information for a Helm release, - as managed by the controller. - properties: - apiVersion: - description: |- - APIVersion is the API version of the Snapshot. - Provisional: when the calculation method of the Digest field is changed, - this field will be used to distinguish between the old and new methods. - type: string - appVersion: - description: AppVersion is the chart app version of the release - object in storage. - type: string - chartName: - description: ChartName is the chart name of the release object - in storage. - type: string - chartVersion: - description: |- - ChartVersion is the chart version of the release object in - storage. - type: string - configDigest: - description: |- - ConfigDigest is the checksum of the config (better known as - "values") of the release object in storage. - It has the format of `:`. - type: string - deleted: - description: Deleted is when the release was deleted. - format: date-time - type: string - digest: - description: |- - Digest is the checksum of the release object in storage. - It has the format of `:`. - type: string - firstDeployed: - description: FirstDeployed is when the release was first deployed. - format: date-time - type: string - lastDeployed: - description: LastDeployed is when the release was last deployed. - format: date-time - type: string - name: - description: Name is the name of the release. - type: string - namespace: - description: Namespace is the namespace the release is deployed - to. - type: string - ociDigest: - description: OCIDigest is the digest of the OCI artifact associated - with the release. - type: string - status: - description: Status is the current state of the release. - type: string - testHooks: - additionalProperties: - description: |- - TestHookStatus holds the status information for a test hook as observed - to be run by the controller. - properties: - lastCompleted: - description: LastCompleted is the time the test hook last - completed. - format: date-time - type: string - lastStarted: - description: LastStarted is the time the test hook was - last started. - format: date-time - type: string - phase: - description: Phase the test hook was observed to be in. - type: string - type: object - description: |- - TestHooks is the list of test hooks for the release as observed to be - run by the controller. - type: object - version: - description: Version is the version of the release object in - storage. - type: integer - required: - - chartName - - chartVersion - - configDigest - - digest - - firstDeployed - - lastDeployed - - name - - namespace - - status - - version - type: object - type: array - installFailures: - description: |- - InstallFailures is the install failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - lastAppliedRevision: - description: LastAppliedRevision is the revision of the last successfully - applied source. - type: string - lastAttemptedConfigDigest: - description: |- - LastAttemptedConfigDigest is the digest for the config (better known as - "values") of the last reconciliation attempt. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - type: string - lastAttemptedGeneration: - description: |- - LastAttemptedGeneration is the last generation the controller attempted - to reconcile. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - format: int64 - type: integer - lastAttemptedReleaseAction: - description: |- - LastAttemptedReleaseAction is the last release action performed for this - HelmRelease. It is used to determine the active remediation strategy. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - type: string - lastAttemptedRevision: - description: LastAttemptedRevision is the revision of the last reconciliation - attempt. - type: string - lastAttemptedValuesChecksum: - description: |- - LastAttemptedValuesChecksum is the SHA1 checksum of the values of the last - reconciliation attempt. - type: string - lastHandledForceAt: - description: |- - LastHandledForceAt holds the value of the most recent force request - value, so a change of the annotation value can be detected. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - type: string - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - lastHandledResetAt: - description: |- - LastHandledResetAt holds the value of the most recent reset request - value, so a change of the annotation value can be detected. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - type: string - lastReleaseRevision: - description: LastReleaseRevision is the revision of the last successful - Helm release. - type: integer - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - observedPostRenderersDigest: - description: |- - ObservedPostRenderersDigest is the digest for the post-renderers of - the last successful reconciliation attempt. - type: string - storageNamespace: - description: |- - StorageNamespace is the namespace of the Helm release storage for the - current release. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - type: string - upgradeFailures: - description: |- - UpgradeFailures is the upgrade failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v2beta2 HelmRelease is deprecated, upgrade to v2 - name: v2beta2 - schema: - openAPIV3Schema: - description: HelmRelease is the Schema for the helmreleases API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HelmReleaseSpec defines the desired state of a Helm release. - properties: - chart: - description: |- - Chart defines the template of the v1beta2.HelmChart that should be created - for this HelmRelease. - properties: - metadata: - description: ObjectMeta holds the template for metadata like labels - and annotations. - properties: - annotations: - additionalProperties: - type: string - description: |- - Annotations is an unstructured key value map stored with a resource that may be - set by external tools to store and retrieve arbitrary metadata. They are not - queryable and should be preserved when modifying objects. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ - type: object - labels: - additionalProperties: - type: string - description: |- - Map of string keys and values that can be used to organize and categorize - (scope and select) objects. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/ - type: object - type: object - spec: - description: Spec holds the template for the v1beta2.HelmChartSpec - for this HelmRelease. - properties: - chart: - description: The name or path the Helm chart is available - at in the SourceRef. - maxLength: 2048 - minLength: 1 - type: string - ignoreMissingValuesFiles: - description: IgnoreMissingValuesFiles controls whether to - silently ignore missing values files rather than failing. - type: boolean - interval: - description: |- - Interval at which to check the v1.Source for updates. Defaults to - 'HelmReleaseSpec.Interval'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - reconcileStrategy: - default: ChartVersion - description: |- - Determines what enables the creation of a new artifact. Valid values are - ('ChartVersion', 'Revision'). - See the documentation of the values for an explanation on their behavior. - Defaults to ChartVersion when omitted. - enum: - - ChartVersion - - Revision - type: string - sourceRef: - description: The name and namespace of the v1.Source the chart - is available at. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - HelmRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: Namespace of the referent. - maxLength: 63 - minLength: 1 - type: string - required: - - name - type: object - valuesFile: - description: |- - Alternative values file to use as the default chart values, expected to - be a relative path in the SourceRef. Deprecated in favor of ValuesFiles, - for backwards compatibility the file defined here is merged before the - ValuesFiles items. Ignored when omitted. - type: string - valuesFiles: - description: |- - Alternative list of values files to use as the chart values (values.yaml - is not included by default), expected to be a relative path in the SourceRef. - Values files are merged in the order of this list with the last file overriding - the first. Ignored when omitted. - items: - type: string - type: array - verify: - description: |- - Verify contains the secret name containing the trusted public keys - used to verify the signature and specifies which provider to use to check - whether OCI image is authentic. - This field is only supported for OCI sources. - Chart dependencies, which are not bundled in the umbrella chart artifact, - are not verified. - properties: - provider: - default: cosign - description: Provider specifies the technology used to - sign the OCI Helm chart. - enum: - - cosign - - notation - type: string - secretRef: - description: |- - SecretRef specifies the Kubernetes Secret containing the - trusted public keys. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - version: - default: '*' - description: |- - Version semver expression, ignored for charts from v1beta2.GitRepository and - v1beta2.Bucket sources. Defaults to latest when omitted. - type: string - required: - - chart - - sourceRef - type: object - required: - - spec - type: object - chartRef: - description: |- - ChartRef holds a reference to a source controller resource containing the - Helm chart artifact. - - - Note: this field is provisional to the v2 API, and not actively used - by v2beta2 HelmReleases. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - OCIRepository - - HelmChart - type: string - name: - description: Name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace of the referent, defaults to the namespace of the Kubernetes - resource object that contains the reference. - maxLength: 63 - minLength: 1 - type: string - required: - - kind - - name - type: object - dependsOn: - description: |- - DependsOn may contain a meta.NamespacedObjectReference slice with - references to HelmRelease resources that must be ready before this HelmRelease - can be reconciled. - items: - description: |- - NamespacedObjectReference contains enough information to locate the referenced Kubernetes resource object in any - namespace. - properties: - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - name - type: object - type: array - driftDetection: - description: |- - DriftDetection holds the configuration for detecting and handling - differences between the manifest in the Helm storage and the resources - currently existing in the cluster. - properties: - ignore: - description: |- - Ignore contains a list of rules for specifying which changes to ignore - during diffing. - items: - description: |- - IgnoreRule defines a rule to selectively disregard specific changes during - the drift detection process. - properties: - paths: - description: |- - Paths is a list of JSON Pointer (RFC 6901) paths to be excluded from - consideration in a Kubernetes object. - items: - type: string - type: array - target: - description: |- - Target is a selector for specifying Kubernetes objects to which this - rule applies. - If Target is not set, the Paths will be ignored for all Kubernetes - objects within the manifest of the Helm release. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - paths - type: object - type: array - mode: - description: |- - Mode defines how differences should be handled between the Helm manifest - and the manifest currently applied to the cluster. - If not explicitly set, it defaults to DiffModeDisabled. - enum: - - enabled - - warn - - disabled - type: string - type: object - install: - description: Install holds the configuration for Helm install actions - for this HelmRelease. - properties: - crds: - description: |- - CRDs upgrade CRDs from the Helm Chart's crds directory according - to the CRD upgrade policy provided here. Valid values are `Skip`, - `Create` or `CreateReplace`. Default is `Create` and if omitted - CRDs are installed but not updated. - - - Skip: do neither install nor replace (update) any CRDs. - - - Create: new CRDs are created, existing CRDs are neither updated nor deleted. - - - CreateReplace: new CRDs are created, existing CRDs are updated (replaced) - but not deleted. - - - By default, CRDs are applied (installed) during Helm install action. - With this option users can opt in to CRD replace existing CRDs on Helm - install actions, which is not (yet) natively supported by Helm. - https://helm.sh/docs/chart_best_practices/custom_resource_definitions. - enum: - - Skip - - Create - - CreateReplace - type: string - createNamespace: - description: |- - CreateNamespace tells the Helm install action to create the - HelmReleaseSpec.TargetNamespace if it does not exist yet. - On uninstall, the namespace will not be garbage collected. - type: boolean - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm install action. - type: boolean - disableOpenAPIValidation: - description: |- - DisableOpenAPIValidation prevents the Helm install action from validating - rendered templates against the Kubernetes OpenAPI Schema. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - install has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - install has been performed. - type: boolean - remediation: - description: |- - Remediation holds the remediation configuration for when the Helm install - action for the HelmRelease fails. The default is to not perform any action. - properties: - ignoreTestFailures: - description: |- - IgnoreTestFailures tells the controller to skip remediation when the Helm - tests are run after an install action but fail. Defaults to - 'Test.IgnoreFailures'. - type: boolean - remediateLastFailure: - description: |- - RemediateLastFailure tells the controller to remediate the last failure, when - no retries remain. Defaults to 'false'. - type: boolean - retries: - description: |- - Retries is the number of retries that should be attempted on failures before - bailing. Remediation, using an uninstall, is performed between each attempt. - Defaults to '0', a negative integer equals to unlimited retries. - type: integer - type: object - replace: - description: |- - Replace tells the Helm install action to re-use the 'ReleaseName', but only - if that name is a deleted release which remains in the history. - type: boolean - skipCRDs: - description: |- - SkipCRDs tells the Helm install action to not install any CRDs. By default, - CRDs are installed if not already present. - - - Deprecated use CRD policy (`crds`) attribute with value `Skip` instead. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm install action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - interval: - description: Interval at which to reconcile the Helm release. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - kubeConfig: - description: |- - KubeConfig for reconciling the HelmRelease on a remote cluster. - When used in combination with HelmReleaseSpec.ServiceAccountName, - forces the controller to act on behalf of that Service Account at the - target cluster. - If the --default-service-account flag is set, its value will be used as - a controller level fallback for when HelmReleaseSpec.ServiceAccountName - is empty. - properties: - secretRef: - description: |- - SecretRef holds the name of a secret that contains a key with - the kubeconfig file as the value. If no key is set, the key will default - to 'value'. - It is recommended that the kubeconfig is self-contained, and the secret - is regularly updated if credentials such as a cloud-access-token expire. - Cloud specific `cmd-path` auth helpers will not function without adding - binaries and credentials to the Pod that is responsible for reconciling - Kubernetes resources. - properties: - key: - description: Key in the Secret, when not specified an implementation-specific - default key is used. - type: string - name: - description: Name of the Secret. - type: string - required: - - name - type: object - required: - - secretRef - type: object - maxHistory: - description: |- - MaxHistory is the number of revisions saved by Helm for this HelmRelease. - Use '0' for an unlimited number of revisions; defaults to '5'. - type: integer - persistentClient: - description: |- - PersistentClient tells the controller to use a persistent Kubernetes - client for this release. When enabled, the client will be reused for the - duration of the reconciliation, instead of being created and destroyed - for each (step of a) Helm action. - - - This can improve performance, but may cause issues with some Helm charts - that for example do create Custom Resource Definitions during installation - outside Helm's CRD lifecycle hooks, which are then not observed to be - available by e.g. post-install hooks. - - - If not set, it defaults to true. - type: boolean - postRenderers: - description: |- - PostRenderers holds an array of Helm PostRenderers, which will be applied in order - of their definition. - items: - description: PostRenderer contains a Helm PostRenderer specification. - properties: - kustomize: - description: Kustomization to apply as PostRenderer. - properties: - images: - description: |- - Images is a list of (image name, new name, new tag or digest) - for changing image names, tags or digests. This can also be achieved with a - patch, but this operator is simpler to specify. - items: - description: Image contains an image name, a new name, - a new tag or digest, which will replace the original - name and tag. - properties: - digest: - description: |- - Digest is the value used to replace the original image tag. - If digest is present NewTag value is ignored. - type: string - name: - description: Name is a tag-less image name. - type: string - newName: - description: NewName is the value used to replace - the original name. - type: string - newTag: - description: NewTag is the value used to replace the - original tag. - type: string - required: - - name - type: object - type: array - patches: - description: |- - Strategic merge and JSON patches, defined as inline YAML objects, - capable of targeting objects based on kind, label and annotation selectors. - items: - description: |- - Patch contains an inline StrategicMerge or JSON6902 patch, and the target the patch should - be applied to. - properties: - patch: - description: |- - Patch contains an inline StrategicMerge patch or an inline JSON6902 patch with - an array of operation objects. - type: string - target: - description: Target points to the resources that the - patch document should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - type: object - type: array - patchesJson6902: - description: |- - JSON 6902 patches, defined as inline YAML objects. - Deprecated: use Patches instead. - items: - description: JSON6902Patch contains a JSON6902 patch and - the target the patch should be applied to. - properties: - patch: - description: Patch contains the JSON6902 patch document - with an array of operation objects. - items: - description: |- - JSON6902 is a JSON6902 operation object. - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - properties: - from: - description: |- - From contains a JSON-pointer value that references a location within the target document where the operation is - performed. The meaning of the value depends on the value of Op, and is NOT taken into account by all operations. - type: string - op: - description: |- - Op indicates the operation to perform. Its value MUST be one of "add", "remove", "replace", "move", "copy", or - "test". - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - enum: - - test - - remove - - add - - replace - - move - - copy - type: string - path: - description: |- - Path contains the JSON-pointer value that references a location within the target document where the operation - is performed. The meaning of the value depends on the value of Op. - type: string - value: - description: |- - Value contains a valid JSON structure. The meaning of the value depends on the value of Op, and is NOT taken into - account by all operations. - x-kubernetes-preserve-unknown-fields: true - required: - - op - - path - type: object - type: array - target: - description: Target points to the resources that the - patch document should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - - target - type: object - type: array - patchesStrategicMerge: - description: |- - Strategic merge patches, defined as inline YAML objects. - Deprecated: use Patches instead. - items: - x-kubernetes-preserve-unknown-fields: true - type: array - type: object - type: object - type: array - releaseName: - description: |- - ReleaseName used for the Helm release. Defaults to a composition of - '[TargetNamespace-]Name'. - maxLength: 53 - minLength: 1 - type: string - rollback: - description: Rollback holds the configuration for Helm rollback actions - for this HelmRelease. - properties: - cleanupOnFail: - description: |- - CleanupOnFail allows deletion of new resources created during the Helm - rollback action when it fails. - type: boolean - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm rollback action. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - rollback has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - rollback has been performed. - type: boolean - force: - description: Force forces resource updates through a replacement - strategy. - type: boolean - recreate: - description: Recreate performs pod restarts for the resource if - applicable. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm rollback action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - serviceAccountName: - description: |- - The name of the Kubernetes service account to impersonate - when reconciling this HelmRelease. - maxLength: 253 - minLength: 1 - type: string - storageNamespace: - description: |- - StorageNamespace used for the Helm storage. - Defaults to the namespace of the HelmRelease. - maxLength: 63 - minLength: 1 - type: string - suspend: - description: |- - Suspend tells the controller to suspend reconciliation for this HelmRelease, - it does not apply to already started reconciliations. Defaults to false. - type: boolean - targetNamespace: - description: |- - TargetNamespace to target when performing operations for the HelmRelease. - Defaults to the namespace of the HelmRelease. - maxLength: 63 - minLength: 1 - type: string - test: - description: Test holds the configuration for Helm test actions for - this HelmRelease. - properties: - enable: - description: |- - Enable enables Helm test actions for this HelmRelease after an Helm install - or upgrade action has been performed. - type: boolean - filters: - description: Filters is a list of tests to run or exclude from - running. - items: - description: Filter holds the configuration for individual Helm - test filters. - properties: - exclude: - description: Exclude specifies whether the named test should - be excluded. - type: boolean - name: - description: Name is the name of the test. - maxLength: 253 - minLength: 1 - type: string - required: - - name - type: object - type: array - ignoreFailures: - description: |- - IgnoreFailures tells the controller to skip remediation when the Helm tests - are run but fail. Can be overwritten for tests run after install or upgrade - actions in 'Install.IgnoreTestFailures' and 'Upgrade.IgnoreTestFailures'. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation during - the performance of a Helm test action. Defaults to 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like Jobs - for hooks) during the performance of a Helm action. Defaults to '5m0s'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - uninstall: - description: Uninstall holds the configuration for Helm uninstall - actions for this HelmRelease. - properties: - deletionPropagation: - default: background - description: |- - DeletionPropagation specifies the deletion propagation policy when - a Helm uninstall is performed. - enum: - - background - - foreground - - orphan - type: string - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm rollback action. - type: boolean - disableWait: - description: |- - DisableWait disables waiting for all the resources to be deleted after - a Helm uninstall is performed. - type: boolean - keepHistory: - description: |- - KeepHistory tells Helm to remove all associated resources and mark the - release as deleted, but retain the release history. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm uninstall action. Defaults - to 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - upgrade: - description: Upgrade holds the configuration for Helm upgrade actions - for this HelmRelease. - properties: - cleanupOnFail: - description: |- - CleanupOnFail allows deletion of new resources created during the Helm - upgrade action when it fails. - type: boolean - crds: - description: |- - CRDs upgrade CRDs from the Helm Chart's crds directory according - to the CRD upgrade policy provided here. Valid values are `Skip`, - `Create` or `CreateReplace`. Default is `Skip` and if omitted - CRDs are neither installed nor upgraded. - - - Skip: do neither install nor replace (update) any CRDs. - - - Create: new CRDs are created, existing CRDs are neither updated nor deleted. - - - CreateReplace: new CRDs are created, existing CRDs are updated (replaced) - but not deleted. - - - By default, CRDs are not applied during Helm upgrade action. With this - option users can opt-in to CRD upgrade, which is not (yet) natively supported by Helm. - https://helm.sh/docs/chart_best_practices/custom_resource_definitions. - enum: - - Skip - - Create - - CreateReplace - type: string - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm upgrade action. - type: boolean - disableOpenAPIValidation: - description: |- - DisableOpenAPIValidation prevents the Helm upgrade action from validating - rendered templates against the Kubernetes OpenAPI Schema. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - upgrade has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - upgrade has been performed. - type: boolean - force: - description: Force forces resource updates through a replacement - strategy. - type: boolean - preserveValues: - description: |- - PreserveValues will make Helm reuse the last release's values and merge in - overrides from 'Values'. Setting this flag makes the HelmRelease - non-declarative. - type: boolean - remediation: - description: |- - Remediation holds the remediation configuration for when the Helm upgrade - action for the HelmRelease fails. The default is to not perform any action. - properties: - ignoreTestFailures: - description: |- - IgnoreTestFailures tells the controller to skip remediation when the Helm - tests are run after an upgrade action but fail. - Defaults to 'Test.IgnoreFailures'. - type: boolean - remediateLastFailure: - description: |- - RemediateLastFailure tells the controller to remediate the last failure, when - no retries remain. Defaults to 'false' unless 'Retries' is greater than 0. - type: boolean - retries: - description: |- - Retries is the number of retries that should be attempted on failures before - bailing. Remediation, using 'Strategy', is performed between each attempt. - Defaults to '0', a negative integer equals to unlimited retries. - type: integer - strategy: - description: Strategy to use for failure remediation. Defaults - to 'rollback'. - enum: - - rollback - - uninstall - type: string - type: object - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm upgrade action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - values: - description: Values holds the values for this Helm release. - x-kubernetes-preserve-unknown-fields: true - valuesFrom: - description: |- - ValuesFrom holds references to resources containing Helm values for this HelmRelease, - and information about how they should be merged. - items: - description: |- - ValuesReference contains a reference to a resource containing Helm values, - and optionally the key they can be found at. - properties: - kind: - description: Kind of the values referent, valid values are ('Secret', - 'ConfigMap'). - enum: - - Secret - - ConfigMap - type: string - name: - description: |- - Name of the values referent. Should reside in the same namespace as the - referring resource. - maxLength: 253 - minLength: 1 - type: string - optional: - description: |- - Optional marks this ValuesReference as optional. When set, a not found error - for the values reference is ignored, but any ValuesKey, TargetPath or - transient error will still result in a reconciliation failure. - type: boolean - targetPath: - description: |- - TargetPath is the YAML dot notation path the value should be merged at. When - set, the ValuesKey is expected to be a single flat value. Defaults to 'None', - which results in the values getting merged at the root. - maxLength: 250 - pattern: ^([a-zA-Z0-9_\-.\\\/]|\[[0-9]{1,5}\])+$ - type: string - valuesKey: - description: |- - ValuesKey is the data key where the values.yaml or a specific value can be - found at. Defaults to 'values.yaml'. - maxLength: 253 - pattern: ^[\-._a-zA-Z0-9]+$ - type: string - required: - - kind - - name - type: object - type: array - required: - - interval - type: object - x-kubernetes-validations: - - message: either chart or chartRef must be set - rule: (has(self.chart) && !has(self.chartRef)) || (!has(self.chart) - && has(self.chartRef)) - status: - default: - observedGeneration: -1 - description: HelmReleaseStatus defines the observed state of a HelmRelease. - properties: - conditions: - description: Conditions holds the conditions for the HelmRelease. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - failures: - description: |- - Failures is the reconciliation failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - helmChart: - description: |- - HelmChart is the namespaced name of the HelmChart resource created by - the controller for the HelmRelease. - type: string - history: - description: |- - History holds the history of Helm releases performed for this HelmRelease - up to the last successfully completed release. - items: - description: |- - Snapshot captures a point-in-time copy of the status information for a Helm release, - as managed by the controller. - properties: - apiVersion: - description: |- - APIVersion is the API version of the Snapshot. - Provisional: when the calculation method of the Digest field is changed, - this field will be used to distinguish between the old and new methods. - type: string - appVersion: - description: AppVersion is the chart app version of the release - object in storage. - type: string - chartName: - description: ChartName is the chart name of the release object - in storage. - type: string - chartVersion: - description: |- - ChartVersion is the chart version of the release object in - storage. - type: string - configDigest: - description: |- - ConfigDigest is the checksum of the config (better known as - "values") of the release object in storage. - It has the format of `:`. - type: string - deleted: - description: Deleted is when the release was deleted. - format: date-time - type: string - digest: - description: |- - Digest is the checksum of the release object in storage. - It has the format of `:`. - type: string - firstDeployed: - description: FirstDeployed is when the release was first deployed. - format: date-time - type: string - lastDeployed: - description: LastDeployed is when the release was last deployed. - format: date-time - type: string - name: - description: Name is the name of the release. - type: string - namespace: - description: Namespace is the namespace the release is deployed - to. - type: string - ociDigest: - description: OCIDigest is the digest of the OCI artifact associated - with the release. - type: string - status: - description: Status is the current state of the release. - type: string - testHooks: - additionalProperties: - description: |- - TestHookStatus holds the status information for a test hook as observed - to be run by the controller. - properties: - lastCompleted: - description: LastCompleted is the time the test hook last - completed. - format: date-time - type: string - lastStarted: - description: LastStarted is the time the test hook was - last started. - format: date-time - type: string - phase: - description: Phase the test hook was observed to be in. - type: string - type: object - description: |- - TestHooks is the list of test hooks for the release as observed to be - run by the controller. - type: object - version: - description: Version is the version of the release object in - storage. - type: integer - required: - - chartName - - chartVersion - - configDigest - - digest - - firstDeployed - - lastDeployed - - name - - namespace - - status - - version - type: object - type: array - installFailures: - description: |- - InstallFailures is the install failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - lastAppliedRevision: - description: |- - LastAppliedRevision is the revision of the last successfully applied - source. - Deprecated: the revision can now be found in the History. - type: string - lastAttemptedConfigDigest: - description: |- - LastAttemptedConfigDigest is the digest for the config (better known as - "values") of the last reconciliation attempt. - type: string - lastAttemptedGeneration: - description: |- - LastAttemptedGeneration is the last generation the controller attempted - to reconcile. - format: int64 - type: integer - lastAttemptedReleaseAction: - description: |- - LastAttemptedReleaseAction is the last release action performed for this - HelmRelease. It is used to determine the active remediation strategy. - enum: - - install - - upgrade - type: string - lastAttemptedRevision: - description: |- - LastAttemptedRevision is the Source revision of the last reconciliation - attempt. For OCIRepository sources, the 12 first characters of the digest are - appended to the chart version e.g. "1.2.3+1234567890ab". - type: string - lastAttemptedRevisionDigest: - description: |- - LastAttemptedRevisionDigest is the digest of the last reconciliation attempt. - This is only set for OCIRepository sources. - type: string - lastAttemptedValuesChecksum: - description: |- - LastAttemptedValuesChecksum is the SHA1 checksum for the values of the last - reconciliation attempt. - Deprecated: Use LastAttemptedConfigDigest instead. - type: string - lastHandledForceAt: - description: |- - LastHandledForceAt holds the value of the most recent force request - value, so a change of the annotation value can be detected. - type: string - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - lastHandledResetAt: - description: |- - LastHandledResetAt holds the value of the most recent reset request - value, so a change of the annotation value can be detected. - type: string - lastReleaseRevision: - description: |- - LastReleaseRevision is the revision of the last successful Helm release. - Deprecated: Use History instead. - type: integer - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - observedPostRenderersDigest: - description: |- - ObservedPostRenderersDigest is the digest for the post-renderers of - the last successful reconciliation attempt. - type: string - storageNamespace: - description: |- - StorageNamespace is the namespace of the Helm release storage for the - current release. - maxLength: 63 - minLength: 1 - type: string - upgradeFailures: - description: |- - UpgradeFailures is the upgrade failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - type: object - type: object - served: true - storage: false - subresources: - status: {} ---- -apiVersion: v1 -kind: ServiceAccount -metadata: - labels: - app.kubernetes.io/component: helm-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: helm-controller - namespace: flux-system ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - labels: - app.kubernetes.io/component: helm-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - control-plane: controller - name: helm-controller - namespace: flux-system -spec: - replicas: 1 - selector: - matchLabels: - app: helm-controller - template: - metadata: - annotations: - prometheus.io/port: "8080" - prometheus.io/scrape: "true" - labels: - app: helm-controller - spec: - containers: - - args: - - --events-addr=http://notification-controller.flux-system.svc.cluster.local./ - - --watch-all-namespaces=true - - --log-level=info - - --log-encoding=json - - --enable-leader-election - env: - - name: RUNTIME_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - - name: GOMAXPROCS - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.cpu - - name: GOMEMLIMIT - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.memory - image: ghcr.io/fluxcd/helm-controller:v1.0.1 - imagePullPolicy: IfNotPresent - livenessProbe: - httpGet: - path: /healthz - port: healthz - name: manager - ports: - - containerPort: 8080 - name: http-prom - protocol: TCP - - containerPort: 9440 - name: healthz - protocol: TCP - readinessProbe: - httpGet: - path: /readyz - port: healthz - resources: - limits: - cpu: 1000m - memory: 1Gi - requests: - cpu: 100m - memory: 64Mi - securityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - readOnlyRootFilesystem: true - runAsNonRoot: true - seccompProfile: - type: RuntimeDefault - volumeMounts: - - mountPath: /tmp - name: temp - nodeSelector: - kubernetes.io/os: linux - priorityClassName: system-cluster-critical - securityContext: - fsGroup: 1337 - serviceAccountName: helm-controller - terminationGracePeriodSeconds: 600 - volumes: - - emptyDir: {} - name: temp ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: notification-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: alerts.notification.toolkit.fluxcd.io -spec: - group: notification.toolkit.fluxcd.io - names: - kind: Alert - listKind: AlertList - plural: alerts - singular: alert - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta1 Alert is deprecated, upgrade to v1beta3 - name: v1beta1 - schema: - openAPIV3Schema: - description: Alert is the Schema for the alerts API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: AlertSpec defines an alerting rule for events involving a - list of objects - properties: - eventSeverity: - default: info - description: |- - Filter events based on severity, defaults to ('info'). - If set to 'info' no events will be filtered. - enum: - - info - - error - type: string - eventSources: - description: Filter events based on the involved objects. - items: - description: |- - CrossNamespaceObjectReference contains enough information to let you locate the - typed referenced object at cluster level - properties: - apiVersion: - description: API version of the referent - type: string - kind: - description: Kind of the referent - enum: - - Bucket - - GitRepository - - Kustomization - - HelmRelease - - HelmChart - - HelmRepository - - ImageRepository - - ImagePolicy - - ImageUpdateAutomation - - OCIRepository - type: string - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - name: - description: Name of the referent - maxLength: 53 - minLength: 1 - type: string - namespace: - description: Namespace of the referent - maxLength: 53 - minLength: 1 - type: string - required: - - name - type: object - type: array - exclusionList: - description: A list of Golang regular expressions to be used for excluding - messages. - items: - type: string - type: array - providerRef: - description: Send events using this provider. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - summary: - description: Short description of the impact and affected cluster. - type: string - suspend: - description: |- - This flag tells the controller to suspend subsequent events dispatching. - Defaults to false. - type: boolean - required: - - eventSources - - providerRef - type: object - status: - default: - observedGeneration: -1 - description: AlertStatus defines the observed state of Alert - properties: - conditions: - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta2 Alert is deprecated, upgrade to v1beta3 - name: v1beta2 - schema: - openAPIV3Schema: - description: Alert is the Schema for the alerts API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: AlertSpec defines an alerting rule for events involving a - list of objects. - properties: - eventMetadata: - additionalProperties: - type: string - description: |- - EventMetadata is an optional field for adding metadata to events dispatched by the - controller. This can be used for enhancing the context of the event. If a field - would override one already present on the original event as generated by the emitter, - then the override doesn't happen, i.e. the original value is preserved, and an info - log is printed. - type: object - eventSeverity: - default: info - description: |- - EventSeverity specifies how to filter events based on severity. - If set to 'info' no events will be filtered. - enum: - - info - - error - type: string - eventSources: - description: |- - EventSources specifies how to filter events based - on the involved object kind, name and namespace. - items: - description: |- - CrossNamespaceObjectReference contains enough information to let you locate the - typed referenced object at cluster level - properties: - apiVersion: - description: API version of the referent - type: string - kind: - description: Kind of the referent - enum: - - Bucket - - GitRepository - - Kustomization - - HelmRelease - - HelmChart - - HelmRepository - - ImageRepository - - ImagePolicy - - ImageUpdateAutomation - - OCIRepository - type: string - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - MatchLabels requires the name to be set to `*`. - type: object - name: - description: |- - Name of the referent - If multiple resources are targeted `*` may be set. - maxLength: 53 - minLength: 1 - type: string - namespace: - description: Namespace of the referent - maxLength: 53 - minLength: 1 - type: string - required: - - kind - - name - type: object - type: array - exclusionList: - description: |- - ExclusionList specifies a list of Golang regular expressions - to be used for excluding messages. - items: - type: string - type: array - inclusionList: - description: |- - InclusionList specifies a list of Golang regular expressions - to be used for including messages. - items: - type: string - type: array - providerRef: - description: ProviderRef specifies which Provider this Alert should - use. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - summary: - description: Summary holds a short description of the impact and affected - cluster. - maxLength: 255 - type: string - suspend: - description: |- - Suspend tells the controller to suspend subsequent - events handling for this Alert. - type: boolean - required: - - eventSources - - providerRef - type: object - status: - default: - observedGeneration: -1 - description: AlertStatus defines the observed state of the Alert. - properties: - conditions: - description: Conditions holds the conditions for the Alert. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1beta3 - schema: - openAPIV3Schema: - description: Alert is the Schema for the alerts API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: AlertSpec defines an alerting rule for events involving a - list of objects. - properties: - eventMetadata: - additionalProperties: - type: string - description: |- - EventMetadata is an optional field for adding metadata to events dispatched by the - controller. This can be used for enhancing the context of the event. If a field - would override one already present on the original event as generated by the emitter, - then the override doesn't happen, i.e. the original value is preserved, and an info - log is printed. - type: object - eventSeverity: - default: info - description: |- - EventSeverity specifies how to filter events based on severity. - If set to 'info' no events will be filtered. - enum: - - info - - error - type: string - eventSources: - description: |- - EventSources specifies how to filter events based - on the involved object kind, name and namespace. - items: - description: |- - CrossNamespaceObjectReference contains enough information to let you locate the - typed referenced object at cluster level - properties: - apiVersion: - description: API version of the referent - type: string - kind: - description: Kind of the referent - enum: - - Bucket - - GitRepository - - Kustomization - - HelmRelease - - HelmChart - - HelmRepository - - ImageRepository - - ImagePolicy - - ImageUpdateAutomation - - OCIRepository - type: string - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - MatchLabels requires the name to be set to `*`. - type: object - name: - description: |- - Name of the referent - If multiple resources are targeted `*` may be set. - maxLength: 53 - minLength: 1 - type: string - namespace: - description: Namespace of the referent - maxLength: 53 - minLength: 1 - type: string - required: - - kind - - name - type: object - type: array - exclusionList: - description: |- - ExclusionList specifies a list of Golang regular expressions - to be used for excluding messages. - items: - type: string - type: array - inclusionList: - description: |- - InclusionList specifies a list of Golang regular expressions - to be used for including messages. - items: - type: string - type: array - providerRef: - description: ProviderRef specifies which Provider this Alert should - use. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - summary: - description: Summary holds a short description of the impact and affected - cluster. - maxLength: 255 - type: string - suspend: - description: |- - Suspend tells the controller to suspend subsequent - events handling for this Alert. - type: boolean - required: - - eventSources - - providerRef - type: object - type: object - served: true - storage: true - subresources: {} ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: notification-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: providers.notification.toolkit.fluxcd.io -spec: - group: notification.toolkit.fluxcd.io - names: - kind: Provider - listKind: ProviderList - plural: providers - singular: provider - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta1 Provider is deprecated, upgrade to v1beta3 - name: v1beta1 - schema: - openAPIV3Schema: - description: Provider is the Schema for the providers API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ProviderSpec defines the desired state of Provider - properties: - address: - description: HTTP/S webhook address of this provider - pattern: ^(http|https):// - type: string - certSecretRef: - description: |- - CertSecretRef can be given the name of a secret containing - a PEM-encoded CA certificate (`caFile`) - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - channel: - description: Alert channel for this provider - type: string - proxy: - description: HTTP/S address of the proxy - pattern: ^(http|https):// - type: string - secretRef: - description: |- - Secret reference containing the provider webhook URL - using "address" as data key - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - This flag tells the controller to suspend subsequent events handling. - Defaults to false. - type: boolean - timeout: - description: Timeout for sending alerts to the provider. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - type: - description: Type of provider - enum: - - slack - - discord - - msteams - - rocket - - generic - - generic-hmac - - github - - gitlab - - bitbucket - - azuredevops - - googlechat - - webex - - sentry - - azureeventhub - - telegram - - lark - - matrix - - opsgenie - - alertmanager - - grafana - - githubdispatch - type: string - username: - description: Bot username for this provider - type: string - required: - - type - type: object - status: - default: - observedGeneration: -1 - description: ProviderStatus defines the observed state of Provider - properties: - conditions: - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - observedGeneration: - description: ObservedGeneration is the last reconciled generation. - format: int64 - type: integer - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta2 Provider is deprecated, upgrade to v1beta3 - name: v1beta2 - schema: - openAPIV3Schema: - description: Provider is the Schema for the providers API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ProviderSpec defines the desired state of the Provider. - properties: - address: - description: |- - Address specifies the endpoint, in a generic sense, to where alerts are sent. - What kind of endpoint depends on the specific Provider type being used. - For the generic Provider, for example, this is an HTTP/S address. - For other Provider types this could be a project ID or a namespace. - maxLength: 2048 - type: string - certSecretRef: - description: |- - CertSecretRef specifies the Secret containing - a PEM-encoded CA certificate (in the `ca.crt` key). - - - Note: Support for the `caFile` key has - been deprecated. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - channel: - description: Channel specifies the destination channel where events - should be posted. - maxLength: 2048 - type: string - interval: - description: Interval at which to reconcile the Provider with its - Secret references. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - proxy: - description: Proxy the HTTP/S address of the proxy server. - maxLength: 2048 - pattern: ^(http|https)://.*$ - type: string - secretRef: - description: |- - SecretRef specifies the Secret containing the authentication - credentials for this Provider. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend subsequent - events handling for this Provider. - type: boolean - timeout: - description: Timeout for sending alerts to the Provider. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - type: - description: Type specifies which Provider implementation to use. - enum: - - slack - - discord - - msteams - - rocket - - generic - - generic-hmac - - github - - gitlab - - gitea - - bitbucketserver - - bitbucket - - azuredevops - - googlechat - - googlepubsub - - webex - - sentry - - azureeventhub - - telegram - - lark - - matrix - - opsgenie - - alertmanager - - grafana - - githubdispatch - - pagerduty - - datadog - type: string - username: - description: Username specifies the name under which events are posted. - maxLength: 2048 - type: string - required: - - type - type: object - status: - default: - observedGeneration: -1 - description: ProviderStatus defines the observed state of the Provider. - properties: - conditions: - description: Conditions holds the conditions for the Provider. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last reconciled generation. - format: int64 - type: integer - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1beta3 - schema: - openAPIV3Schema: - description: Provider is the Schema for the providers API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ProviderSpec defines the desired state of the Provider. - properties: - address: - description: |- - Address specifies the endpoint, in a generic sense, to where alerts are sent. - What kind of endpoint depends on the specific Provider type being used. - For the generic Provider, for example, this is an HTTP/S address. - For other Provider types this could be a project ID or a namespace. - maxLength: 2048 - type: string - certSecretRef: - description: |- - CertSecretRef specifies the Secret containing - a PEM-encoded CA certificate (in the `ca.crt` key). - - - Note: Support for the `caFile` key has - been deprecated. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - channel: - description: Channel specifies the destination channel where events - should be posted. - maxLength: 2048 - type: string - interval: - description: |- - Interval at which to reconcile the Provider with its Secret references. - Deprecated and not used in v1beta3. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - proxy: - description: Proxy the HTTP/S address of the proxy server. - maxLength: 2048 - pattern: ^(http|https)://.*$ - type: string - secretRef: - description: |- - SecretRef specifies the Secret containing the authentication - credentials for this Provider. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend subsequent - events handling for this Provider. - type: boolean - timeout: - description: Timeout for sending alerts to the Provider. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - type: - description: Type specifies which Provider implementation to use. - enum: - - slack - - discord - - msteams - - rocket - - generic - - generic-hmac - - github - - gitlab - - gitea - - bitbucketserver - - bitbucket - - azuredevops - - googlechat - - googlepubsub - - webex - - sentry - - azureeventhub - - telegram - - lark - - matrix - - opsgenie - - alertmanager - - grafana - - githubdispatch - - pagerduty - - datadog - - nats - type: string - username: - description: Username specifies the name under which events are posted. - maxLength: 2048 - type: string - required: - - type - type: object - type: object - served: true - storage: true - subresources: {} ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: notification-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: receivers.notification.toolkit.fluxcd.io -spec: - group: notification.toolkit.fluxcd.io - names: - kind: Receiver - listKind: ReceiverList - plural: receivers - singular: receiver - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - name: v1 - schema: - openAPIV3Schema: - description: Receiver is the Schema for the receivers API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ReceiverSpec defines the desired state of the Receiver. - properties: - events: - description: |- - Events specifies the list of event types to handle, - e.g. 'push' for GitHub or 'Push Hook' for GitLab. - items: - type: string - type: array - interval: - default: 10m - description: Interval at which to reconcile the Receiver with its - Secret references. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - resources: - description: A list of resources to be notified about changes. - items: - description: |- - CrossNamespaceObjectReference contains enough information to let you locate the - typed referenced object at cluster level - properties: - apiVersion: - description: API version of the referent - type: string - kind: - description: Kind of the referent - enum: - - Bucket - - GitRepository - - Kustomization - - HelmRelease - - HelmChart - - HelmRepository - - ImageRepository - - ImagePolicy - - ImageUpdateAutomation - - OCIRepository - type: string - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - MatchLabels requires the name to be set to `*`. - type: object - name: - description: |- - Name of the referent - If multiple resources are targeted `*` may be set. - maxLength: 53 - minLength: 1 - type: string - namespace: - description: Namespace of the referent - maxLength: 53 - minLength: 1 - type: string - required: - - kind - - name - type: object - type: array - secretRef: - description: |- - SecretRef specifies the Secret containing the token used - to validate the payload authenticity. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend subsequent - events handling for this receiver. - type: boolean - type: - description: |- - Type of webhook sender, used to determine - the validation procedure and payload deserialization. - enum: - - generic - - generic-hmac - - github - - gitlab - - bitbucket - - harbor - - dockerhub - - quay - - gcr - - nexus - - acr - - cdevents - type: string - required: - - resources - - secretRef - - type - type: object - status: - default: - observedGeneration: -1 - description: ReceiverStatus defines the observed state of the Receiver. - properties: - conditions: - description: Conditions holds the conditions for the Receiver. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation of - the Receiver object. - format: int64 - type: integer - webhookPath: - description: |- - WebhookPath is the generated incoming webhook address in the format - of '/hook/sha256sum(token+name+namespace)'. - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta1 Receiver is deprecated, upgrade to v1 - name: v1beta1 - schema: - openAPIV3Schema: - description: Receiver is the Schema for the receivers API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ReceiverSpec defines the desired state of Receiver - properties: - events: - description: |- - A list of events to handle, - e.g. 'push' for GitHub or 'Push Hook' for GitLab. - items: - type: string - type: array - resources: - description: A list of resources to be notified about changes. - items: - description: |- - CrossNamespaceObjectReference contains enough information to let you locate the - typed referenced object at cluster level - properties: - apiVersion: - description: API version of the referent - type: string - kind: - description: Kind of the referent - enum: - - Bucket - - GitRepository - - Kustomization - - HelmRelease - - HelmChart - - HelmRepository - - ImageRepository - - ImagePolicy - - ImageUpdateAutomation - - OCIRepository - type: string - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - name: - description: Name of the referent - maxLength: 53 - minLength: 1 - type: string - namespace: - description: Namespace of the referent - maxLength: 53 - minLength: 1 - type: string - required: - - name - type: object - type: array - secretRef: - description: |- - Secret reference containing the token used - to validate the payload authenticity - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - This flag tells the controller to suspend subsequent events handling. - Defaults to false. - type: boolean - type: - description: |- - Type of webhook sender, used to determine - the validation procedure and payload deserialization. - enum: - - generic - - generic-hmac - - github - - gitlab - - bitbucket - - harbor - - dockerhub - - quay - - gcr - - nexus - - acr - type: string - required: - - resources - - type - type: object - status: - default: - observedGeneration: -1 - description: ReceiverStatus defines the observed state of Receiver - properties: - conditions: - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - url: - description: |- - Generated webhook URL in the format - of '/hook/sha256sum(token+name+namespace)'. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta2 Receiver is deprecated, upgrade to v1 - name: v1beta2 - schema: - openAPIV3Schema: - description: Receiver is the Schema for the receivers API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ReceiverSpec defines the desired state of the Receiver. - properties: - events: - description: |- - Events specifies the list of event types to handle, - e.g. 'push' for GitHub or 'Push Hook' for GitLab. - items: - type: string - type: array - interval: - description: Interval at which to reconcile the Receiver with its - Secret references. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - resources: - description: A list of resources to be notified about changes. - items: - description: |- - CrossNamespaceObjectReference contains enough information to let you locate the - typed referenced object at cluster level - properties: - apiVersion: - description: API version of the referent - type: string - kind: - description: Kind of the referent - enum: - - Bucket - - GitRepository - - Kustomization - - HelmRelease - - HelmChart - - HelmRepository - - ImageRepository - - ImagePolicy - - ImageUpdateAutomation - - OCIRepository - type: string - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - MatchLabels requires the name to be set to `*`. - type: object - name: - description: |- - Name of the referent - If multiple resources are targeted `*` may be set. - maxLength: 53 - minLength: 1 - type: string - namespace: - description: Namespace of the referent - maxLength: 53 - minLength: 1 - type: string - required: - - kind - - name - type: object - type: array - secretRef: - description: |- - SecretRef specifies the Secret containing the token used - to validate the payload authenticity. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend subsequent - events handling for this receiver. - type: boolean - type: - description: |- - Type of webhook sender, used to determine - the validation procedure and payload deserialization. - enum: - - generic - - generic-hmac - - github - - gitlab - - bitbucket - - harbor - - dockerhub - - quay - - gcr - - nexus - - acr - type: string - required: - - resources - - type - type: object - status: - default: - observedGeneration: -1 - description: ReceiverStatus defines the observed state of the Receiver. - properties: - conditions: - description: Conditions holds the conditions for the Receiver. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation of - the Receiver object. - format: int64 - type: integer - url: - description: |- - URL is the generated incoming webhook address in the format - of '/hook/sha256sum(token+name+namespace)'. - Deprecated: Replaced by WebhookPath. - type: string - webhookPath: - description: |- - WebhookPath is the generated incoming webhook address in the format - of '/hook/sha256sum(token+name+namespace)'. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} ---- -apiVersion: v1 -kind: ServiceAccount -metadata: - labels: - app.kubernetes.io/component: notification-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: notification-controller - namespace: flux-system ---- -apiVersion: v1 -kind: Service -metadata: - labels: - app.kubernetes.io/component: notification-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - control-plane: controller - name: notification-controller - namespace: flux-system -spec: - ports: - - name: http - port: 80 - protocol: TCP - targetPort: http - selector: - app: notification-controller - type: ClusterIP ---- -apiVersion: v1 -kind: Service -metadata: - labels: - app.kubernetes.io/component: notification-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - control-plane: controller - name: webhook-receiver - namespace: flux-system -spec: - ports: - - name: http - port: 80 - protocol: TCP - targetPort: http-webhook - selector: - app: notification-controller - type: ClusterIP ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - labels: - app.kubernetes.io/component: notification-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - control-plane: controller - name: notification-controller - namespace: flux-system -spec: - replicas: 1 - selector: - matchLabels: - app: notification-controller - template: - metadata: - annotations: - prometheus.io/port: "8080" - prometheus.io/scrape: "true" - labels: - app: notification-controller - spec: - containers: - - args: - - --watch-all-namespaces=true - - --log-level=info - - --log-encoding=json - - --enable-leader-election - env: - - name: RUNTIME_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - - name: GOMAXPROCS - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.cpu - - name: GOMEMLIMIT - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.memory - image: ghcr.io/fluxcd/notification-controller:v1.3.0 - imagePullPolicy: IfNotPresent - livenessProbe: - httpGet: - path: /healthz - port: healthz - name: manager - ports: - - containerPort: 9090 - name: http - protocol: TCP - - containerPort: 9292 - name: http-webhook - protocol: TCP - - containerPort: 8080 - name: http-prom - protocol: TCP - - containerPort: 9440 - name: healthz - protocol: TCP - readinessProbe: - httpGet: - path: /readyz - port: healthz - resources: - limits: - cpu: 1000m - memory: 1Gi - requests: - cpu: 100m - memory: 64Mi - securityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - readOnlyRootFilesystem: true - runAsNonRoot: true - seccompProfile: - type: RuntimeDefault - volumeMounts: - - mountPath: /tmp - name: temp - nodeSelector: - kubernetes.io/os: linux - securityContext: - fsGroup: 1337 - serviceAccountName: notification-controller - terminationGracePeriodSeconds: 10 - volumes: - - emptyDir: {} - name: temp diff --git a/flux/clusters/production/flux-system/gotk-sync.yaml b/flux/clusters/production/flux-system/gotk-sync.yaml deleted file mode 100644 index 00a6fbd..0000000 --- a/flux/clusters/production/flux-system/gotk-sync.yaml +++ /dev/null @@ -1,27 +0,0 @@ -# This manifest was generated by flux. DO NOT EDIT. ---- -apiVersion: source.toolkit.fluxcd.io/v1 -kind: GitRepository -metadata: - name: flux-system - namespace: flux-system -spec: - interval: 1m0s - ref: - branch: main - secretRef: - name: flux-system - url: ssh://git@gitea.futureporn.net:2222/futureporn/fp ---- -apiVersion: kustomize.toolkit.fluxcd.io/v1 -kind: Kustomization -metadata: - name: flux-system - namespace: flux-system -spec: - interval: 10m0s - path: ./clusters/production - prune: true - sourceRef: - kind: GitRepository - name: flux-system diff --git a/flux/clusters/production/flux-system/kustomization.yaml b/flux/clusters/production/flux-system/kustomization.yaml deleted file mode 100644 index a3a32be..0000000 --- a/flux/clusters/production/flux-system/kustomization.yaml +++ /dev/null @@ -1,19 +0,0 @@ -apiVersion: kustomize.config.k8s.io/v1beta1 -kind: Kustomization -resources: - - gotk-components.yaml - - gotk-sync.yaml -labels: - - pairs: - toolkit.fluxcd.io/tenant: sre-team -patches: - - patch: | - - op: add - path: /spec/template/spec/containers/0/args/- - value: --concurrent=20 - - op: add - path: /spec/template/spec/containers/0/args/- - value: --requeue-dependency=5s - target: - kind: Deployment - name: "(kustomize-controller|helm-controller|source-controller)" diff --git a/flux/clusters/production/infrastructure.yaml b/flux/clusters/production/infrastructure.yaml deleted file mode 100644 index 72bc16f..0000000 --- a/flux/clusters/production/infrastructure.yaml +++ /dev/null @@ -1,56 +0,0 @@ ---- -apiVersion: kustomize.toolkit.fluxcd.io/v1 -kind: Kustomization -metadata: - name: infra-controllers - namespace: flux-system -spec: - interval: 1h - retryInterval: 1m - timeout: 5m - sourceRef: - kind: GitRepository - name: flux-system - path: ./flux/infrastructure/controllers - prune: true - wait: true - ---- -apiVersion: kustomize.toolkit.fluxcd.io/v1 -kind: Kustomization -metadata: - name: infra-configs - namespace: flux-system -spec: - dependsOn: - - name: infra-controllers - interval: 1h - retryInterval: 1m - timeout: 5m - sourceRef: - kind: GitRepository - name: flux-system - path: ./flux/infrastructure/configs - prune: true - patches: - - patch: | - - op: replace - path: /spec/acme/privateKeySecretRef/name - value: letsencrypt-production - target: - kind: ClusterIssuer - name: letsencrypt - - patch: | - - op: replace - path: /spec/acme/server - value: https://acme-staging-v02.api.letsencrypt.org/directory - target: - kind: ClusterIssuer - name: letsencrypt - - patch: | - - op: replace - path: /metadata/name - value: letsencrypt-production - target: - kind: ClusterIssuer - name: letsencrypt \ No newline at end of file diff --git a/flux/clusters/staging/apps.yaml b/flux/clusters/staging/apps.yaml deleted file mode 100644 index 1213410..0000000 --- a/flux/clusters/staging/apps.yaml +++ /dev/null @@ -1,16 +0,0 @@ -apiVersion: kustomize.toolkit.fluxcd.io/v1 -kind: Kustomization -metadata: - name: apps - namespace: flux-system -spec: - interval: 1m0s - dependsOn: - - name: infra-configs - sourceRef: - kind: GitRepository - name: flux-system - path: ./flux/apps/staging - prune: true - wait: true - timeout: 5m0s diff --git a/flux/clusters/staging/flux-system/gotk-components.yaml b/flux/clusters/staging/flux-system/gotk-components.yaml deleted file mode 100644 index 8fd99ab..0000000 --- a/flux/clusters/staging/flux-system/gotk-components.yaml +++ /dev/null @@ -1,12385 +0,0 @@ ---- -# This manifest was generated by flux. DO NOT EDIT. -# Flux Version: v2.3.0 -# Components: source-controller,kustomize-controller,helm-controller,notification-controller -apiVersion: v1 -kind: Namespace -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - pod-security.kubernetes.io/warn: restricted - pod-security.kubernetes.io/warn-version: latest - name: flux-system ---- -apiVersion: networking.k8s.io/v1 -kind: NetworkPolicy -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: allow-egress - namespace: flux-system -spec: - egress: - - {} - ingress: - - from: - - podSelector: {} - podSelector: {} - policyTypes: - - Ingress - - Egress ---- -apiVersion: networking.k8s.io/v1 -kind: NetworkPolicy -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: allow-scraping - namespace: flux-system -spec: - ingress: - - from: - - namespaceSelector: {} - ports: - - port: 8080 - protocol: TCP - podSelector: {} - policyTypes: - - Ingress ---- -apiVersion: networking.k8s.io/v1 -kind: NetworkPolicy -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: allow-webhooks - namespace: flux-system -spec: - ingress: - - from: - - namespaceSelector: {} - podSelector: - matchLabels: - app: notification-controller - policyTypes: - - Ingress ---- -apiVersion: v1 -kind: ResourceQuota -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: critical-pods-flux-system - namespace: flux-system -spec: - hard: - pods: "1000" - scopeSelector: - matchExpressions: - - operator: In - scopeName: PriorityClass - values: - - system-node-critical - - system-cluster-critical ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: crd-controller-flux-system -rules: -- apiGroups: - - source.toolkit.fluxcd.io - resources: - - '*' - verbs: - - '*' -- apiGroups: - - kustomize.toolkit.fluxcd.io - resources: - - '*' - verbs: - - '*' -- apiGroups: - - helm.toolkit.fluxcd.io - resources: - - '*' - verbs: - - '*' -- apiGroups: - - notification.toolkit.fluxcd.io - resources: - - '*' - verbs: - - '*' -- apiGroups: - - image.toolkit.fluxcd.io - resources: - - '*' - verbs: - - '*' -- apiGroups: - - "" - resources: - - namespaces - - secrets - - configmaps - - serviceaccounts - verbs: - - get - - list - - watch -- apiGroups: - - "" - resources: - - events - verbs: - - create - - patch -- apiGroups: - - "" - resources: - - configmaps - verbs: - - get - - list - - watch - - create - - update - - patch - - delete -- apiGroups: - - "" - resources: - - configmaps/status - verbs: - - get - - update - - patch -- apiGroups: - - coordination.k8s.io - resources: - - leases - verbs: - - get - - list - - watch - - create - - update - - patch - - delete -- nonResourceURLs: - - /livez/ping - verbs: - - head ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - rbac.authorization.k8s.io/aggregate-to-admin: "true" - rbac.authorization.k8s.io/aggregate-to-edit: "true" - name: flux-edit-flux-system -rules: -- apiGroups: - - notification.toolkit.fluxcd.io - - source.toolkit.fluxcd.io - - helm.toolkit.fluxcd.io - - image.toolkit.fluxcd.io - - kustomize.toolkit.fluxcd.io - resources: - - '*' - verbs: - - create - - delete - - deletecollection - - patch - - update ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - rbac.authorization.k8s.io/aggregate-to-admin: "true" - rbac.authorization.k8s.io/aggregate-to-edit: "true" - rbac.authorization.k8s.io/aggregate-to-view: "true" - name: flux-view-flux-system -rules: -- apiGroups: - - notification.toolkit.fluxcd.io - - source.toolkit.fluxcd.io - - helm.toolkit.fluxcd.io - - image.toolkit.fluxcd.io - - kustomize.toolkit.fluxcd.io - resources: - - '*' - verbs: - - get - - list - - watch ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: cluster-reconciler-flux-system -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: cluster-admin -subjects: -- kind: ServiceAccount - name: kustomize-controller - namespace: flux-system -- kind: ServiceAccount - name: helm-controller - namespace: flux-system ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - labels: - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: crd-controller-flux-system -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: crd-controller-flux-system -subjects: -- kind: ServiceAccount - name: kustomize-controller - namespace: flux-system -- kind: ServiceAccount - name: helm-controller - namespace: flux-system -- kind: ServiceAccount - name: source-controller - namespace: flux-system -- kind: ServiceAccount - name: notification-controller - namespace: flux-system -- kind: ServiceAccount - name: image-reflector-controller - namespace: flux-system -- kind: ServiceAccount - name: image-automation-controller - namespace: flux-system ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: buckets.source.toolkit.fluxcd.io -spec: - group: source.toolkit.fluxcd.io - names: - kind: Bucket - listKind: BucketList - plural: buckets - singular: bucket - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.endpoint - name: Endpoint - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - deprecated: true - deprecationWarning: v1beta1 Bucket is deprecated, upgrade to v1beta2 - name: v1beta1 - schema: - openAPIV3Schema: - description: Bucket is the Schema for the buckets API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: BucketSpec defines the desired state of an S3 compatible - bucket - properties: - accessFrom: - description: AccessFrom defines an Access Control List for allowing - cross-namespace references to this object. - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - bucketName: - description: The bucket name. - type: string - endpoint: - description: The bucket endpoint address. - type: string - ignore: - description: |- - Ignore overrides the set of excluded patterns in the .sourceignore format - (which is the same as .gitignore). If not provided, a default will be used, - consult the documentation for your version to find out what those are. - type: string - insecure: - description: Insecure allows connecting to a non-TLS S3 HTTP endpoint. - type: boolean - interval: - description: The interval at which to check for bucket updates. - type: string - provider: - default: generic - description: The S3 compatible storage provider name, default ('generic'). - enum: - - generic - - aws - - gcp - type: string - region: - description: The bucket region. - type: string - secretRef: - description: |- - The name of the secret containing authentication credentials - for the Bucket. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: This flag tells the controller to suspend the reconciliation - of this source. - type: boolean - timeout: - default: 60s - description: The timeout for download operations, defaults to 60s. - type: string - required: - - bucketName - - endpoint - - interval - type: object - status: - default: - observedGeneration: -1 - description: BucketStatus defines the observed state of a bucket - properties: - artifact: - description: Artifact represents the output of the last successful - Bucket sync. - properties: - checksum: - description: Checksum is the SHA256 checksum of the artifact. - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of this - artifact. - format: date-time - type: string - path: - description: Path is the relative file path of this artifact. - type: string - revision: - description: |- - Revision is a human readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm index timestamp, a Helm - chart version, etc. - type: string - url: - description: URL is the HTTP address of this artifact. - type: string - required: - - path - - url - type: object - conditions: - description: Conditions holds the conditions for the Bucket. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - url: - description: URL is the download link for the artifact output of the - last Bucket sync. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.endpoint - name: Endpoint - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - name: v1beta2 - schema: - openAPIV3Schema: - description: Bucket is the Schema for the buckets API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - BucketSpec specifies the required configuration to produce an Artifact for - an object storage bucket. - properties: - accessFrom: - description: |- - AccessFrom specifies an Access Control List for allowing cross-namespace - references to this object. - NOTE: Not implemented, provisional as of https://github.com/fluxcd/flux2/pull/2092 - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - bucketName: - description: BucketName is the name of the object storage bucket. - type: string - endpoint: - description: Endpoint is the object storage address the BucketName - is located at. - type: string - ignore: - description: |- - Ignore overrides the set of excluded patterns in the .sourceignore format - (which is the same as .gitignore). If not provided, a default will be used, - consult the documentation for your version to find out what those are. - type: string - insecure: - description: Insecure allows connecting to a non-TLS HTTP Endpoint. - type: boolean - interval: - description: |- - Interval at which the Bucket Endpoint is checked for updates. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - prefix: - description: Prefix to use for server-side filtering of files in the - Bucket. - type: string - provider: - default: generic - description: |- - Provider of the object storage bucket. - Defaults to 'generic', which expects an S3 (API) compatible object - storage. - enum: - - generic - - aws - - gcp - - azure - type: string - region: - description: Region of the Endpoint where the BucketName is located - in. - type: string - secretRef: - description: |- - SecretRef specifies the Secret containing authentication credentials - for the Bucket. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend the reconciliation of this - Bucket. - type: boolean - timeout: - default: 60s - description: Timeout for fetch operations, defaults to 60s. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - required: - - bucketName - - endpoint - - interval - type: object - status: - default: - observedGeneration: -1 - description: BucketStatus records the observed state of a Bucket. - properties: - artifact: - description: Artifact represents the last successful Bucket reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the Bucket. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation of - the Bucket object. - format: int64 - type: integer - observedIgnore: - description: |- - ObservedIgnore is the observed exclusion patterns used for constructing - the source artifact. - type: string - url: - description: |- - URL is the dynamic fetch link for the latest Artifact. - It is provided on a "best effort" basis, and using the precise - BucketStatus.Artifact data is recommended. - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: gitrepositories.source.toolkit.fluxcd.io -spec: - group: source.toolkit.fluxcd.io - names: - kind: GitRepository - listKind: GitRepositoryList - plural: gitrepositories - shortNames: - - gitrepo - singular: gitrepository - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.url - name: URL - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - name: v1 - schema: - openAPIV3Schema: - description: GitRepository is the Schema for the gitrepositories API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - GitRepositorySpec specifies the required configuration to produce an - Artifact for a Git repository. - properties: - ignore: - description: |- - Ignore overrides the set of excluded patterns in the .sourceignore format - (which is the same as .gitignore). If not provided, a default will be used, - consult the documentation for your version to find out what those are. - type: string - include: - description: |- - Include specifies a list of GitRepository resources which Artifacts - should be included in the Artifact produced for this GitRepository. - items: - description: |- - GitRepositoryInclude specifies a local reference to a GitRepository which - Artifact (sub-)contents must be included, and where they should be placed. - properties: - fromPath: - description: |- - FromPath specifies the path to copy contents from, defaults to the root - of the Artifact. - type: string - repository: - description: |- - GitRepositoryRef specifies the GitRepository which Artifact contents - must be included. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - toPath: - description: |- - ToPath specifies the path to copy contents to, defaults to the name of - the GitRepositoryRef. - type: string - required: - - repository - type: object - type: array - interval: - description: |- - Interval at which the GitRepository URL is checked for updates. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - proxySecretRef: - description: |- - ProxySecretRef specifies the Secret containing the proxy configuration - to use while communicating with the Git server. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - recurseSubmodules: - description: |- - RecurseSubmodules enables the initialization of all submodules within - the GitRepository as cloned from the URL, using their default settings. - type: boolean - ref: - description: |- - Reference specifies the Git reference to resolve and monitor for - changes, defaults to the 'master' branch. - properties: - branch: - description: Branch to check out, defaults to 'master' if no other - field is defined. - type: string - commit: - description: |- - Commit SHA to check out, takes precedence over all reference fields. - - - This can be combined with Branch to shallow clone the branch, in which - the commit is expected to exist. - type: string - name: - description: |- - Name of the reference to check out; takes precedence over Branch, Tag and SemVer. - - - It must be a valid Git reference: https://git-scm.com/docs/git-check-ref-format#_description - Examples: "refs/heads/main", "refs/tags/v0.1.0", "refs/pull/420/head", "refs/merge-requests/1/head" - type: string - semver: - description: SemVer tag expression to check out, takes precedence - over Tag. - type: string - tag: - description: Tag to check out, takes precedence over Branch. - type: string - type: object - secretRef: - description: |- - SecretRef specifies the Secret containing authentication credentials for - the GitRepository. - For HTTPS repositories the Secret must contain 'username' and 'password' - fields for basic auth or 'bearerToken' field for token auth. - For SSH repositories the Secret must contain 'identity' - and 'known_hosts' fields. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend the reconciliation of this - GitRepository. - type: boolean - timeout: - default: 60s - description: Timeout for Git operations like cloning, defaults to - 60s. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - url: - description: URL specifies the Git repository URL, it can be an HTTP/S - or SSH address. - pattern: ^(http|https|ssh)://.*$ - type: string - verify: - description: |- - Verification specifies the configuration to verify the Git commit - signature(s). - properties: - mode: - default: HEAD - description: |- - Mode specifies which Git object(s) should be verified. - - - The variants "head" and "HEAD" both imply the same thing, i.e. verify - the commit that the HEAD of the Git repository points to. The variant - "head" solely exists to ensure backwards compatibility. - enum: - - head - - HEAD - - Tag - - TagAndHEAD - type: string - secretRef: - description: |- - SecretRef specifies the Secret containing the public keys of trusted Git - authors. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - secretRef - type: object - required: - - interval - - url - type: object - status: - default: - observedGeneration: -1 - description: GitRepositoryStatus records the observed state of a Git repository. - properties: - artifact: - description: Artifact represents the last successful GitRepository - reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the GitRepository. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - includedArtifacts: - description: |- - IncludedArtifacts contains a list of the last successfully included - Artifacts as instructed by GitRepositorySpec.Include. - items: - description: Artifact represents the output of a Source reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of - ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI - annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: |- - ObservedGeneration is the last observed generation of the GitRepository - object. - format: int64 - type: integer - observedIgnore: - description: |- - ObservedIgnore is the observed exclusion patterns used for constructing - the source artifact. - type: string - observedInclude: - description: |- - ObservedInclude is the observed list of GitRepository resources used to - produce the current Artifact. - items: - description: |- - GitRepositoryInclude specifies a local reference to a GitRepository which - Artifact (sub-)contents must be included, and where they should be placed. - properties: - fromPath: - description: |- - FromPath specifies the path to copy contents from, defaults to the root - of the Artifact. - type: string - repository: - description: |- - GitRepositoryRef specifies the GitRepository which Artifact contents - must be included. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - toPath: - description: |- - ToPath specifies the path to copy contents to, defaults to the name of - the GitRepositoryRef. - type: string - required: - - repository - type: object - type: array - observedRecurseSubmodules: - description: |- - ObservedRecurseSubmodules is the observed resource submodules - configuration used to produce the current Artifact. - type: boolean - sourceVerificationMode: - description: |- - SourceVerificationMode is the last used verification mode indicating - which Git object(s) have been verified. - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.url - name: URL - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - deprecated: true - deprecationWarning: v1beta1 GitRepository is deprecated, upgrade to v1 - name: v1beta1 - schema: - openAPIV3Schema: - description: GitRepository is the Schema for the gitrepositories API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: GitRepositorySpec defines the desired state of a Git repository. - properties: - accessFrom: - description: AccessFrom defines an Access Control List for allowing - cross-namespace references to this object. - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - gitImplementation: - default: go-git - description: |- - Determines which git client library to use. - Defaults to go-git, valid values are ('go-git', 'libgit2'). - enum: - - go-git - - libgit2 - type: string - ignore: - description: |- - Ignore overrides the set of excluded patterns in the .sourceignore format - (which is the same as .gitignore). If not provided, a default will be used, - consult the documentation for your version to find out what those are. - type: string - include: - description: Extra git repositories to map into the repository - items: - description: GitRepositoryInclude defines a source with a from and - to path. - properties: - fromPath: - description: The path to copy contents from, defaults to the - root directory. - type: string - repository: - description: Reference to a GitRepository to include. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - toPath: - description: The path to copy contents to, defaults to the name - of the source ref. - type: string - required: - - repository - type: object - type: array - interval: - description: The interval at which to check for repository updates. - type: string - recurseSubmodules: - description: |- - When enabled, after the clone is created, initializes all submodules within, - using their default settings. - This option is available only when using the 'go-git' GitImplementation. - type: boolean - ref: - description: |- - The Git reference to checkout and monitor for changes, defaults to - master branch. - properties: - branch: - description: The Git branch to checkout, defaults to master. - type: string - commit: - description: The Git commit SHA to checkout, if specified Tag - filters will be ignored. - type: string - semver: - description: The Git tag semver expression, takes precedence over - Tag. - type: string - tag: - description: The Git tag to checkout, takes precedence over Branch. - type: string - type: object - secretRef: - description: |- - The secret name containing the Git credentials. - For HTTPS repositories the secret must contain username and password - fields. - For SSH repositories the secret must contain identity and known_hosts - fields. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: This flag tells the controller to suspend the reconciliation - of this source. - type: boolean - timeout: - default: 60s - description: The timeout for remote Git operations like cloning, defaults - to 60s. - type: string - url: - description: The repository URL, can be a HTTP/S or SSH address. - pattern: ^(http|https|ssh)://.*$ - type: string - verify: - description: Verify OpenPGP signature for the Git commit HEAD points - to. - properties: - mode: - description: Mode describes what git object should be verified, - currently ('head'). - enum: - - head - type: string - secretRef: - description: The secret name containing the public keys of all - trusted Git authors. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - mode - type: object - required: - - interval - - url - type: object - status: - default: - observedGeneration: -1 - description: GitRepositoryStatus defines the observed state of a Git repository. - properties: - artifact: - description: Artifact represents the output of the last successful - repository sync. - properties: - checksum: - description: Checksum is the SHA256 checksum of the artifact. - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of this - artifact. - format: date-time - type: string - path: - description: Path is the relative file path of this artifact. - type: string - revision: - description: |- - Revision is a human readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm index timestamp, a Helm - chart version, etc. - type: string - url: - description: URL is the HTTP address of this artifact. - type: string - required: - - path - - url - type: object - conditions: - description: Conditions holds the conditions for the GitRepository. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - includedArtifacts: - description: IncludedArtifacts represents the included artifacts from - the last successful repository sync. - items: - description: Artifact represents the output of a source synchronisation. - properties: - checksum: - description: Checksum is the SHA256 checksum of the artifact. - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of this - artifact. - format: date-time - type: string - path: - description: Path is the relative file path of this artifact. - type: string - revision: - description: |- - Revision is a human readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm index timestamp, a Helm - chart version, etc. - type: string - url: - description: URL is the HTTP address of this artifact. - type: string - required: - - path - - url - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - url: - description: |- - URL is the download link for the artifact output of the last repository - sync. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.url - name: URL - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta2 GitRepository is deprecated, upgrade to v1 - name: v1beta2 - schema: - openAPIV3Schema: - description: GitRepository is the Schema for the gitrepositories API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - GitRepositorySpec specifies the required configuration to produce an - Artifact for a Git repository. - properties: - accessFrom: - description: |- - AccessFrom specifies an Access Control List for allowing cross-namespace - references to this object. - NOTE: Not implemented, provisional as of https://github.com/fluxcd/flux2/pull/2092 - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - gitImplementation: - default: go-git - description: |- - GitImplementation specifies which Git client library implementation to - use. Defaults to 'go-git', valid values are ('go-git', 'libgit2'). - Deprecated: gitImplementation is deprecated now that 'go-git' is the - only supported implementation. - enum: - - go-git - - libgit2 - type: string - ignore: - description: |- - Ignore overrides the set of excluded patterns in the .sourceignore format - (which is the same as .gitignore). If not provided, a default will be used, - consult the documentation for your version to find out what those are. - type: string - include: - description: |- - Include specifies a list of GitRepository resources which Artifacts - should be included in the Artifact produced for this GitRepository. - items: - description: |- - GitRepositoryInclude specifies a local reference to a GitRepository which - Artifact (sub-)contents must be included, and where they should be placed. - properties: - fromPath: - description: |- - FromPath specifies the path to copy contents from, defaults to the root - of the Artifact. - type: string - repository: - description: |- - GitRepositoryRef specifies the GitRepository which Artifact contents - must be included. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - toPath: - description: |- - ToPath specifies the path to copy contents to, defaults to the name of - the GitRepositoryRef. - type: string - required: - - repository - type: object - type: array - interval: - description: Interval at which to check the GitRepository for updates. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - recurseSubmodules: - description: |- - RecurseSubmodules enables the initialization of all submodules within - the GitRepository as cloned from the URL, using their default settings. - type: boolean - ref: - description: |- - Reference specifies the Git reference to resolve and monitor for - changes, defaults to the 'master' branch. - properties: - branch: - description: Branch to check out, defaults to 'master' if no other - field is defined. - type: string - commit: - description: |- - Commit SHA to check out, takes precedence over all reference fields. - - - This can be combined with Branch to shallow clone the branch, in which - the commit is expected to exist. - type: string - name: - description: |- - Name of the reference to check out; takes precedence over Branch, Tag and SemVer. - - - It must be a valid Git reference: https://git-scm.com/docs/git-check-ref-format#_description - Examples: "refs/heads/main", "refs/tags/v0.1.0", "refs/pull/420/head", "refs/merge-requests/1/head" - type: string - semver: - description: SemVer tag expression to check out, takes precedence - over Tag. - type: string - tag: - description: Tag to check out, takes precedence over Branch. - type: string - type: object - secretRef: - description: |- - SecretRef specifies the Secret containing authentication credentials for - the GitRepository. - For HTTPS repositories the Secret must contain 'username' and 'password' - fields for basic auth or 'bearerToken' field for token auth. - For SSH repositories the Secret must contain 'identity' - and 'known_hosts' fields. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend the reconciliation of this - GitRepository. - type: boolean - timeout: - default: 60s - description: Timeout for Git operations like cloning, defaults to - 60s. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - url: - description: URL specifies the Git repository URL, it can be an HTTP/S - or SSH address. - pattern: ^(http|https|ssh)://.*$ - type: string - verify: - description: |- - Verification specifies the configuration to verify the Git commit - signature(s). - properties: - mode: - description: Mode specifies what Git object should be verified, - currently ('head'). - enum: - - head - type: string - secretRef: - description: |- - SecretRef specifies the Secret containing the public keys of trusted Git - authors. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - mode - - secretRef - type: object - required: - - interval - - url - type: object - status: - default: - observedGeneration: -1 - description: GitRepositoryStatus records the observed state of a Git repository. - properties: - artifact: - description: Artifact represents the last successful GitRepository - reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the GitRepository. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - contentConfigChecksum: - description: |- - ContentConfigChecksum is a checksum of all the configurations related to - the content of the source artifact: - - .spec.ignore - - .spec.recurseSubmodules - - .spec.included and the checksum of the included artifacts - observed in .status.observedGeneration version of the object. This can - be used to determine if the content of the included repository has - changed. - It has the format of `:`, for example: `sha256:`. - - - Deprecated: Replaced with explicit fields for observed artifact content - config in the status. - type: string - includedArtifacts: - description: |- - IncludedArtifacts contains a list of the last successfully included - Artifacts as instructed by GitRepositorySpec.Include. - items: - description: Artifact represents the output of a Source reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of - ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI - annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: |- - ObservedGeneration is the last observed generation of the GitRepository - object. - format: int64 - type: integer - observedIgnore: - description: |- - ObservedIgnore is the observed exclusion patterns used for constructing - the source artifact. - type: string - observedInclude: - description: |- - ObservedInclude is the observed list of GitRepository resources used to - to produce the current Artifact. - items: - description: |- - GitRepositoryInclude specifies a local reference to a GitRepository which - Artifact (sub-)contents must be included, and where they should be placed. - properties: - fromPath: - description: |- - FromPath specifies the path to copy contents from, defaults to the root - of the Artifact. - type: string - repository: - description: |- - GitRepositoryRef specifies the GitRepository which Artifact contents - must be included. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - toPath: - description: |- - ToPath specifies the path to copy contents to, defaults to the name of - the GitRepositoryRef. - type: string - required: - - repository - type: object - type: array - observedRecurseSubmodules: - description: |- - ObservedRecurseSubmodules is the observed resource submodules - configuration used to produce the current Artifact. - type: boolean - url: - description: |- - URL is the dynamic fetch link for the latest Artifact. - It is provided on a "best effort" basis, and using the precise - GitRepositoryStatus.Artifact data is recommended. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: helmcharts.source.toolkit.fluxcd.io -spec: - group: source.toolkit.fluxcd.io - names: - kind: HelmChart - listKind: HelmChartList - plural: helmcharts - shortNames: - - hc - singular: helmchart - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.chart - name: Chart - type: string - - jsonPath: .spec.version - name: Version - type: string - - jsonPath: .spec.sourceRef.kind - name: Source Kind - type: string - - jsonPath: .spec.sourceRef.name - name: Source Name - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - name: v1 - schema: - openAPIV3Schema: - description: HelmChart is the Schema for the helmcharts API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HelmChartSpec specifies the desired state of a Helm chart. - properties: - chart: - description: |- - Chart is the name or path the Helm chart is available at in the - SourceRef. - type: string - ignoreMissingValuesFiles: - description: |- - IgnoreMissingValuesFiles controls whether to silently ignore missing values - files rather than failing. - type: boolean - interval: - description: |- - Interval at which the HelmChart SourceRef is checked for updates. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - reconcileStrategy: - default: ChartVersion - description: |- - ReconcileStrategy determines what enables the creation of a new artifact. - Valid values are ('ChartVersion', 'Revision'). - See the documentation of the values for an explanation on their behavior. - Defaults to ChartVersion when omitted. - enum: - - ChartVersion - - Revision - type: string - sourceRef: - description: SourceRef is the reference to the Source the chart is - available at. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: |- - Kind of the referent, valid values are ('HelmRepository', 'GitRepository', - 'Bucket'). - enum: - - HelmRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - type: string - required: - - kind - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend the reconciliation of this - source. - type: boolean - valuesFiles: - description: |- - ValuesFiles is an alternative list of values files to use as the chart - values (values.yaml is not included by default), expected to be a - relative path in the SourceRef. - Values files are merged in the order of this list with the last file - overriding the first. Ignored when omitted. - items: - type: string - type: array - verify: - description: |- - Verify contains the secret name containing the trusted public keys - used to verify the signature and specifies which provider to use to check - whether OCI image is authentic. - This field is only supported when using HelmRepository source with spec.type 'oci'. - Chart dependencies, which are not bundled in the umbrella chart artifact, are not verified. - properties: - matchOIDCIdentity: - description: |- - MatchOIDCIdentity specifies the identity matching criteria to use - while verifying an OCI artifact which was signed using Cosign keyless - signing. The artifact's identity is deemed to be verified if any of the - specified matchers match against the identity. - items: - description: |- - OIDCIdentityMatch specifies options for verifying the certificate identity, - i.e. the issuer and the subject of the certificate. - properties: - issuer: - description: |- - Issuer specifies the regex pattern to match against to verify - the OIDC issuer in the Fulcio certificate. The pattern must be a - valid Go regular expression. - type: string - subject: - description: |- - Subject specifies the regex pattern to match against to verify - the identity subject in the Fulcio certificate. The pattern must - be a valid Go regular expression. - type: string - required: - - issuer - - subject - type: object - type: array - provider: - default: cosign - description: Provider specifies the technology used to sign the - OCI Artifact. - enum: - - cosign - - notation - type: string - secretRef: - description: |- - SecretRef specifies the Kubernetes Secret containing the - trusted public keys. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - version: - default: '*' - description: |- - Version is the chart version semver expression, ignored for charts from - GitRepository and Bucket sources. Defaults to latest when omitted. - type: string - required: - - chart - - interval - - sourceRef - type: object - status: - default: - observedGeneration: -1 - description: HelmChartStatus records the observed state of the HelmChart. - properties: - artifact: - description: Artifact represents the output of the last successful - reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the HelmChart. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedChartName: - description: |- - ObservedChartName is the last observed chart name as specified by the - resolved chart reference. - type: string - observedGeneration: - description: |- - ObservedGeneration is the last observed generation of the HelmChart - object. - format: int64 - type: integer - observedSourceArtifactRevision: - description: |- - ObservedSourceArtifactRevision is the last observed Artifact.Revision - of the HelmChartSpec.SourceRef. - type: string - observedValuesFiles: - description: |- - ObservedValuesFiles are the observed value files of the last successful - reconciliation. - It matches the chart in the last successfully reconciled artifact. - items: - type: string - type: array - url: - description: |- - URL is the dynamic fetch link for the latest Artifact. - It is provided on a "best effort" basis, and using the precise - BucketStatus.Artifact data is recommended. - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.chart - name: Chart - type: string - - jsonPath: .spec.version - name: Version - type: string - - jsonPath: .spec.sourceRef.kind - name: Source Kind - type: string - - jsonPath: .spec.sourceRef.name - name: Source Name - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - deprecated: true - deprecationWarning: v1beta1 HelmChart is deprecated, upgrade to v1 - name: v1beta1 - schema: - openAPIV3Schema: - description: HelmChart is the Schema for the helmcharts API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HelmChartSpec defines the desired state of a Helm chart. - properties: - accessFrom: - description: AccessFrom defines an Access Control List for allowing - cross-namespace references to this object. - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - chart: - description: The name or path the Helm chart is available at in the - SourceRef. - type: string - interval: - description: The interval at which to check the Source for updates. - type: string - reconcileStrategy: - default: ChartVersion - description: |- - Determines what enables the creation of a new artifact. Valid values are - ('ChartVersion', 'Revision'). - See the documentation of the values for an explanation on their behavior. - Defaults to ChartVersion when omitted. - enum: - - ChartVersion - - Revision - type: string - sourceRef: - description: The reference to the Source the chart is available at. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: |- - Kind of the referent, valid values are ('HelmRepository', 'GitRepository', - 'Bucket'). - enum: - - HelmRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - type: string - required: - - kind - - name - type: object - suspend: - description: This flag tells the controller to suspend the reconciliation - of this source. - type: boolean - valuesFile: - description: |- - Alternative values file to use as the default chart values, expected to - be a relative path in the SourceRef. Deprecated in favor of ValuesFiles, - for backwards compatibility the file defined here is merged before the - ValuesFiles items. Ignored when omitted. - type: string - valuesFiles: - description: |- - Alternative list of values files to use as the chart values (values.yaml - is not included by default), expected to be a relative path in the SourceRef. - Values files are merged in the order of this list with the last file overriding - the first. Ignored when omitted. - items: - type: string - type: array - version: - default: '*' - description: |- - The chart version semver expression, ignored for charts from GitRepository - and Bucket sources. Defaults to latest when omitted. - type: string - required: - - chart - - interval - - sourceRef - type: object - status: - default: - observedGeneration: -1 - description: HelmChartStatus defines the observed state of the HelmChart. - properties: - artifact: - description: Artifact represents the output of the last successful - chart sync. - properties: - checksum: - description: Checksum is the SHA256 checksum of the artifact. - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of this - artifact. - format: date-time - type: string - path: - description: Path is the relative file path of this artifact. - type: string - revision: - description: |- - Revision is a human readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm index timestamp, a Helm - chart version, etc. - type: string - url: - description: URL is the HTTP address of this artifact. - type: string - required: - - path - - url - type: object - conditions: - description: Conditions holds the conditions for the HelmChart. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - url: - description: URL is the download link for the last chart pulled. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.chart - name: Chart - type: string - - jsonPath: .spec.version - name: Version - type: string - - jsonPath: .spec.sourceRef.kind - name: Source Kind - type: string - - jsonPath: .spec.sourceRef.name - name: Source Name - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta2 HelmChart is deprecated, upgrade to v1 - name: v1beta2 - schema: - openAPIV3Schema: - description: HelmChart is the Schema for the helmcharts API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HelmChartSpec specifies the desired state of a Helm chart. - properties: - accessFrom: - description: |- - AccessFrom specifies an Access Control List for allowing cross-namespace - references to this object. - NOTE: Not implemented, provisional as of https://github.com/fluxcd/flux2/pull/2092 - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - chart: - description: |- - Chart is the name or path the Helm chart is available at in the - SourceRef. - type: string - ignoreMissingValuesFiles: - description: |- - IgnoreMissingValuesFiles controls whether to silently ignore missing values - files rather than failing. - type: boolean - interval: - description: |- - Interval at which the HelmChart SourceRef is checked for updates. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - reconcileStrategy: - default: ChartVersion - description: |- - ReconcileStrategy determines what enables the creation of a new artifact. - Valid values are ('ChartVersion', 'Revision'). - See the documentation of the values for an explanation on their behavior. - Defaults to ChartVersion when omitted. - enum: - - ChartVersion - - Revision - type: string - sourceRef: - description: SourceRef is the reference to the Source the chart is - available at. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: |- - Kind of the referent, valid values are ('HelmRepository', 'GitRepository', - 'Bucket'). - enum: - - HelmRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - type: string - required: - - kind - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend the reconciliation of this - source. - type: boolean - valuesFile: - description: |- - ValuesFile is an alternative values file to use as the default chart - values, expected to be a relative path in the SourceRef. Deprecated in - favor of ValuesFiles, for backwards compatibility the file specified here - is merged before the ValuesFiles items. Ignored when omitted. - type: string - valuesFiles: - description: |- - ValuesFiles is an alternative list of values files to use as the chart - values (values.yaml is not included by default), expected to be a - relative path in the SourceRef. - Values files are merged in the order of this list with the last file - overriding the first. Ignored when omitted. - items: - type: string - type: array - verify: - description: |- - Verify contains the secret name containing the trusted public keys - used to verify the signature and specifies which provider to use to check - whether OCI image is authentic. - This field is only supported when using HelmRepository source with spec.type 'oci'. - Chart dependencies, which are not bundled in the umbrella chart artifact, are not verified. - properties: - matchOIDCIdentity: - description: |- - MatchOIDCIdentity specifies the identity matching criteria to use - while verifying an OCI artifact which was signed using Cosign keyless - signing. The artifact's identity is deemed to be verified if any of the - specified matchers match against the identity. - items: - description: |- - OIDCIdentityMatch specifies options for verifying the certificate identity, - i.e. the issuer and the subject of the certificate. - properties: - issuer: - description: |- - Issuer specifies the regex pattern to match against to verify - the OIDC issuer in the Fulcio certificate. The pattern must be a - valid Go regular expression. - type: string - subject: - description: |- - Subject specifies the regex pattern to match against to verify - the identity subject in the Fulcio certificate. The pattern must - be a valid Go regular expression. - type: string - required: - - issuer - - subject - type: object - type: array - provider: - default: cosign - description: Provider specifies the technology used to sign the - OCI Artifact. - enum: - - cosign - - notation - type: string - secretRef: - description: |- - SecretRef specifies the Kubernetes Secret containing the - trusted public keys. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - version: - default: '*' - description: |- - Version is the chart version semver expression, ignored for charts from - GitRepository and Bucket sources. Defaults to latest when omitted. - type: string - required: - - chart - - interval - - sourceRef - type: object - status: - default: - observedGeneration: -1 - description: HelmChartStatus records the observed state of the HelmChart. - properties: - artifact: - description: Artifact represents the output of the last successful - reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the HelmChart. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedChartName: - description: |- - ObservedChartName is the last observed chart name as specified by the - resolved chart reference. - type: string - observedGeneration: - description: |- - ObservedGeneration is the last observed generation of the HelmChart - object. - format: int64 - type: integer - observedSourceArtifactRevision: - description: |- - ObservedSourceArtifactRevision is the last observed Artifact.Revision - of the HelmChartSpec.SourceRef. - type: string - observedValuesFiles: - description: |- - ObservedValuesFiles are the observed value files of the last successful - reconciliation. - It matches the chart in the last successfully reconciled artifact. - items: - type: string - type: array - url: - description: |- - URL is the dynamic fetch link for the latest Artifact. - It is provided on a "best effort" basis, and using the precise - BucketStatus.Artifact data is recommended. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: helmrepositories.source.toolkit.fluxcd.io -spec: - group: source.toolkit.fluxcd.io - names: - kind: HelmRepository - listKind: HelmRepositoryList - plural: helmrepositories - shortNames: - - helmrepo - singular: helmrepository - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.url - name: URL - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - name: v1 - schema: - openAPIV3Schema: - description: HelmRepository is the Schema for the helmrepositories API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - HelmRepositorySpec specifies the required configuration to produce an - Artifact for a Helm repository index YAML. - properties: - accessFrom: - description: |- - AccessFrom specifies an Access Control List for allowing cross-namespace - references to this object. - NOTE: Not implemented, provisional as of https://github.com/fluxcd/flux2/pull/2092 - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - certSecretRef: - description: |- - CertSecretRef can be given the name of a Secret containing - either or both of - - - - a PEM-encoded client certificate (`tls.crt`) and private - key (`tls.key`); - - a PEM-encoded CA certificate (`ca.crt`) - - - and whichever are supplied, will be used for connecting to the - registry. The client cert and key are useful if you are - authenticating with a certificate; the CA cert is useful if - you are using a self-signed server certificate. The Secret must - be of type `Opaque` or `kubernetes.io/tls`. - - - It takes precedence over the values specified in the Secret referred - to by `.spec.secretRef`. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - insecure: - description: |- - Insecure allows connecting to a non-TLS HTTP container registry. - This field is only taken into account if the .spec.type field is set to 'oci'. - type: boolean - interval: - description: |- - Interval at which the HelmRepository URL is checked for updates. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - passCredentials: - description: |- - PassCredentials allows the credentials from the SecretRef to be passed - on to a host that does not match the host as defined in URL. - This may be required if the host of the advertised chart URLs in the - index differ from the defined URL. - Enabling this should be done with caution, as it can potentially result - in credentials getting stolen in a MITM-attack. - type: boolean - provider: - default: generic - description: |- - Provider used for authentication, can be 'aws', 'azure', 'gcp' or 'generic'. - This field is optional, and only taken into account if the .spec.type field is set to 'oci'. - When not specified, defaults to 'generic'. - enum: - - generic - - aws - - azure - - gcp - type: string - secretRef: - description: |- - SecretRef specifies the Secret containing authentication credentials - for the HelmRepository. - For HTTP/S basic auth the secret must contain 'username' and 'password' - fields. - Support for TLS auth using the 'certFile' and 'keyFile', and/or 'caFile' - keys is deprecated. Please use `.spec.certSecretRef` instead. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend the reconciliation of this - HelmRepository. - type: boolean - timeout: - description: |- - Timeout is used for the index fetch operation for an HTTPS helm repository, - and for remote OCI Repository operations like pulling for an OCI helm - chart by the associated HelmChart. - Its default value is 60s. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - type: - description: |- - Type of the HelmRepository. - When this field is set to "oci", the URL field value must be prefixed with "oci://". - enum: - - default - - oci - type: string - url: - description: |- - URL of the Helm repository, a valid URL contains at least a protocol and - host. - pattern: ^(http|https|oci)://.*$ - type: string - required: - - url - type: object - status: - default: - observedGeneration: -1 - description: HelmRepositoryStatus records the observed state of the HelmRepository. - properties: - artifact: - description: Artifact represents the last successful HelmRepository - reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the HelmRepository. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: |- - ObservedGeneration is the last observed generation of the HelmRepository - object. - format: int64 - type: integer - url: - description: |- - URL is the dynamic fetch link for the latest Artifact. - It is provided on a "best effort" basis, and using the precise - HelmRepositoryStatus.Artifact data is recommended. - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.url - name: URL - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - deprecated: true - deprecationWarning: v1beta1 HelmRepository is deprecated, upgrade to v1 - name: v1beta1 - schema: - openAPIV3Schema: - description: HelmRepository is the Schema for the helmrepositories API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HelmRepositorySpec defines the reference to a Helm repository. - properties: - accessFrom: - description: AccessFrom defines an Access Control List for allowing - cross-namespace references to this object. - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - interval: - description: The interval at which to check the upstream for updates. - type: string - passCredentials: - description: |- - PassCredentials allows the credentials from the SecretRef to be passed on to - a host that does not match the host as defined in URL. - This may be required if the host of the advertised chart URLs in the index - differ from the defined URL. - Enabling this should be done with caution, as it can potentially result in - credentials getting stolen in a MITM-attack. - type: boolean - secretRef: - description: |- - The name of the secret containing authentication credentials for the Helm - repository. - For HTTP/S basic auth the secret must contain username and - password fields. - For TLS the secret must contain a certFile and keyFile, and/or - caFile fields. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: This flag tells the controller to suspend the reconciliation - of this source. - type: boolean - timeout: - default: 60s - description: The timeout of index downloading, defaults to 60s. - type: string - url: - description: The Helm repository URL, a valid URL contains at least - a protocol and host. - type: string - required: - - interval - - url - type: object - status: - default: - observedGeneration: -1 - description: HelmRepositoryStatus defines the observed state of the HelmRepository. - properties: - artifact: - description: Artifact represents the output of the last successful - repository sync. - properties: - checksum: - description: Checksum is the SHA256 checksum of the artifact. - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of this - artifact. - format: date-time - type: string - path: - description: Path is the relative file path of this artifact. - type: string - revision: - description: |- - Revision is a human readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm index timestamp, a Helm - chart version, etc. - type: string - url: - description: URL is the HTTP address of this artifact. - type: string - required: - - path - - url - type: object - conditions: - description: Conditions holds the conditions for the HelmRepository. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - url: - description: URL is the download link for the last index fetched. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .spec.url - name: URL - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta2 HelmRepository is deprecated, upgrade to v1 - name: v1beta2 - schema: - openAPIV3Schema: - description: HelmRepository is the Schema for the helmrepositories API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - HelmRepositorySpec specifies the required configuration to produce an - Artifact for a Helm repository index YAML. - properties: - accessFrom: - description: |- - AccessFrom specifies an Access Control List for allowing cross-namespace - references to this object. - NOTE: Not implemented, provisional as of https://github.com/fluxcd/flux2/pull/2092 - properties: - namespaceSelectors: - description: |- - NamespaceSelectors is the list of namespace selectors to which this ACL applies. - Items in this list are evaluated using a logical OR operation. - items: - description: |- - NamespaceSelector selects the namespaces to which this ACL applies. - An empty map of MatchLabels matches all namespaces in a cluster. - properties: - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - type: object - type: array - required: - - namespaceSelectors - type: object - certSecretRef: - description: |- - CertSecretRef can be given the name of a Secret containing - either or both of - - - - a PEM-encoded client certificate (`tls.crt`) and private - key (`tls.key`); - - a PEM-encoded CA certificate (`ca.crt`) - - - and whichever are supplied, will be used for connecting to the - registry. The client cert and key are useful if you are - authenticating with a certificate; the CA cert is useful if - you are using a self-signed server certificate. The Secret must - be of type `Opaque` or `kubernetes.io/tls`. - - - It takes precedence over the values specified in the Secret referred - to by `.spec.secretRef`. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - insecure: - description: |- - Insecure allows connecting to a non-TLS HTTP container registry. - This field is only taken into account if the .spec.type field is set to 'oci'. - type: boolean - interval: - description: |- - Interval at which the HelmRepository URL is checked for updates. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - passCredentials: - description: |- - PassCredentials allows the credentials from the SecretRef to be passed - on to a host that does not match the host as defined in URL. - This may be required if the host of the advertised chart URLs in the - index differ from the defined URL. - Enabling this should be done with caution, as it can potentially result - in credentials getting stolen in a MITM-attack. - type: boolean - provider: - default: generic - description: |- - Provider used for authentication, can be 'aws', 'azure', 'gcp' or 'generic'. - This field is optional, and only taken into account if the .spec.type field is set to 'oci'. - When not specified, defaults to 'generic'. - enum: - - generic - - aws - - azure - - gcp - type: string - secretRef: - description: |- - SecretRef specifies the Secret containing authentication credentials - for the HelmRepository. - For HTTP/S basic auth the secret must contain 'username' and 'password' - fields. - Support for TLS auth using the 'certFile' and 'keyFile', and/or 'caFile' - keys is deprecated. Please use `.spec.certSecretRef` instead. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend the reconciliation of this - HelmRepository. - type: boolean - timeout: - description: |- - Timeout is used for the index fetch operation for an HTTPS helm repository, - and for remote OCI Repository operations like pulling for an OCI helm - chart by the associated HelmChart. - Its default value is 60s. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - type: - description: |- - Type of the HelmRepository. - When this field is set to "oci", the URL field value must be prefixed with "oci://". - enum: - - default - - oci - type: string - url: - description: |- - URL of the Helm repository, a valid URL contains at least a protocol and - host. - pattern: ^(http|https|oci)://.*$ - type: string - required: - - url - type: object - status: - default: - observedGeneration: -1 - description: HelmRepositoryStatus records the observed state of the HelmRepository. - properties: - artifact: - description: Artifact represents the last successful HelmRepository - reconciliation. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the HelmRepository. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: |- - ObservedGeneration is the last observed generation of the HelmRepository - object. - format: int64 - type: integer - url: - description: |- - URL is the dynamic fetch link for the latest Artifact. - It is provided on a "best effort" basis, and using the precise - HelmRepositoryStatus.Artifact data is recommended. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: ocirepositories.source.toolkit.fluxcd.io -spec: - group: source.toolkit.fluxcd.io - names: - kind: OCIRepository - listKind: OCIRepositoryList - plural: ocirepositories - shortNames: - - ocirepo - singular: ocirepository - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .spec.url - name: URL - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1beta2 - schema: - openAPIV3Schema: - description: OCIRepository is the Schema for the ocirepositories API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: OCIRepositorySpec defines the desired state of OCIRepository - properties: - certSecretRef: - description: |- - CertSecretRef can be given the name of a Secret containing - either or both of - - - - a PEM-encoded client certificate (`tls.crt`) and private - key (`tls.key`); - - a PEM-encoded CA certificate (`ca.crt`) - - - and whichever are supplied, will be used for connecting to the - registry. The client cert and key are useful if you are - authenticating with a certificate; the CA cert is useful if - you are using a self-signed server certificate. The Secret must - be of type `Opaque` or `kubernetes.io/tls`. - - - Note: Support for the `caFile`, `certFile` and `keyFile` keys have - been deprecated. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - ignore: - description: |- - Ignore overrides the set of excluded patterns in the .sourceignore format - (which is the same as .gitignore). If not provided, a default will be used, - consult the documentation for your version to find out what those are. - type: string - insecure: - description: Insecure allows connecting to a non-TLS HTTP container - registry. - type: boolean - interval: - description: |- - Interval at which the OCIRepository URL is checked for updates. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - layerSelector: - description: |- - LayerSelector specifies which layer should be extracted from the OCI artifact. - When not specified, the first layer found in the artifact is selected. - properties: - mediaType: - description: |- - MediaType specifies the OCI media type of the layer - which should be extracted from the OCI Artifact. The - first layer matching this type is selected. - type: string - operation: - description: |- - Operation specifies how the selected layer should be processed. - By default, the layer compressed content is extracted to storage. - When the operation is set to 'copy', the layer compressed content - is persisted to storage as it is. - enum: - - extract - - copy - type: string - type: object - provider: - default: generic - description: |- - The provider used for authentication, can be 'aws', 'azure', 'gcp' or 'generic'. - When not specified, defaults to 'generic'. - enum: - - generic - - aws - - azure - - gcp - type: string - ref: - description: |- - The OCI reference to pull and monitor for changes, - defaults to the latest tag. - properties: - digest: - description: |- - Digest is the image digest to pull, takes precedence over SemVer. - The value should be in the format 'sha256:'. - type: string - semver: - description: |- - SemVer is the range of tags to pull selecting the latest within - the range, takes precedence over Tag. - type: string - semverFilter: - description: SemverFilter is a regex pattern to filter the tags - within the SemVer range. - type: string - tag: - description: Tag is the image tag to pull, defaults to latest. - type: string - type: object - secretRef: - description: |- - SecretRef contains the secret name containing the registry login - credentials to resolve image metadata. - The secret must be of type kubernetes.io/dockerconfigjson. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - serviceAccountName: - description: |- - ServiceAccountName is the name of the Kubernetes ServiceAccount used to authenticate - the image pull if the service account has attached pull secrets. For more information: - https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/#add-imagepullsecrets-to-a-service-account - type: string - suspend: - description: This flag tells the controller to suspend the reconciliation - of this source. - type: boolean - timeout: - default: 60s - description: The timeout for remote OCI Repository operations like - pulling, defaults to 60s. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - url: - description: |- - URL is a reference to an OCI artifact repository hosted - on a remote container registry. - pattern: ^oci://.*$ - type: string - verify: - description: |- - Verify contains the secret name containing the trusted public keys - used to verify the signature and specifies which provider to use to check - whether OCI image is authentic. - properties: - matchOIDCIdentity: - description: |- - MatchOIDCIdentity specifies the identity matching criteria to use - while verifying an OCI artifact which was signed using Cosign keyless - signing. The artifact's identity is deemed to be verified if any of the - specified matchers match against the identity. - items: - description: |- - OIDCIdentityMatch specifies options for verifying the certificate identity, - i.e. the issuer and the subject of the certificate. - properties: - issuer: - description: |- - Issuer specifies the regex pattern to match against to verify - the OIDC issuer in the Fulcio certificate. The pattern must be a - valid Go regular expression. - type: string - subject: - description: |- - Subject specifies the regex pattern to match against to verify - the identity subject in the Fulcio certificate. The pattern must - be a valid Go regular expression. - type: string - required: - - issuer - - subject - type: object - type: array - provider: - default: cosign - description: Provider specifies the technology used to sign the - OCI Artifact. - enum: - - cosign - - notation - type: string - secretRef: - description: |- - SecretRef specifies the Kubernetes Secret containing the - trusted public keys. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - required: - - interval - - url - type: object - status: - default: - observedGeneration: -1 - description: OCIRepositoryStatus defines the observed state of OCIRepository - properties: - artifact: - description: Artifact represents the output of the last successful - OCI Repository sync. - properties: - digest: - description: Digest is the digest of the file in the form of ':'. - pattern: ^[a-z0-9]+(?:[.+_-][a-z0-9]+)*:[a-zA-Z0-9=_-]+$ - type: string - lastUpdateTime: - description: |- - LastUpdateTime is the timestamp corresponding to the last update of the - Artifact. - format: date-time - type: string - metadata: - additionalProperties: - type: string - description: Metadata holds upstream information such as OCI annotations. - type: object - path: - description: |- - Path is the relative file path of the Artifact. It can be used to locate - the file in the root of the Artifact storage on the local file system of - the controller managing the Source. - type: string - revision: - description: |- - Revision is a human-readable identifier traceable in the origin source - system. It can be a Git commit SHA, Git tag, a Helm chart version, etc. - type: string - size: - description: Size is the number of bytes in the file. - format: int64 - type: integer - url: - description: |- - URL is the HTTP address of the Artifact as exposed by the controller - managing the Source. It can be used to retrieve the Artifact for - consumption, e.g. by another controller applying the Artifact contents. - type: string - required: - - lastUpdateTime - - path - - revision - - url - type: object - conditions: - description: Conditions holds the conditions for the OCIRepository. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - contentConfigChecksum: - description: |- - ContentConfigChecksum is a checksum of all the configurations related to - the content of the source artifact: - - .spec.ignore - - .spec.layerSelector - observed in .status.observedGeneration version of the object. This can - be used to determine if the content configuration has changed and the - artifact needs to be rebuilt. - It has the format of `:`, for example: `sha256:`. - - - Deprecated: Replaced with explicit fields for observed artifact content - config in the status. - type: string - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - observedIgnore: - description: |- - ObservedIgnore is the observed exclusion patterns used for constructing - the source artifact. - type: string - observedLayerSelector: - description: |- - ObservedLayerSelector is the observed layer selector used for constructing - the source artifact. - properties: - mediaType: - description: |- - MediaType specifies the OCI media type of the layer - which should be extracted from the OCI Artifact. The - first layer matching this type is selected. - type: string - operation: - description: |- - Operation specifies how the selected layer should be processed. - By default, the layer compressed content is extracted to storage. - When the operation is set to 'copy', the layer compressed content - is persisted to storage as it is. - enum: - - extract - - copy - type: string - type: object - url: - description: URL is the download link for the artifact output of the - last OCI Repository sync. - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} ---- -apiVersion: v1 -kind: ServiceAccount -metadata: - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: source-controller - namespace: flux-system ---- -apiVersion: v1 -kind: Service -metadata: - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - control-plane: controller - name: source-controller - namespace: flux-system -spec: - ports: - - name: http - port: 80 - protocol: TCP - targetPort: http - selector: - app: source-controller - type: ClusterIP ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - labels: - app.kubernetes.io/component: source-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - control-plane: controller - name: source-controller - namespace: flux-system -spec: - replicas: 1 - selector: - matchLabels: - app: source-controller - strategy: - type: Recreate - template: - metadata: - annotations: - prometheus.io/port: "8080" - prometheus.io/scrape: "true" - labels: - app: source-controller - spec: - containers: - - args: - - --events-addr=http://notification-controller.flux-system.svc.cluster.local./ - - --watch-all-namespaces=true - - --log-level=info - - --log-encoding=json - - --enable-leader-election - - --storage-path=/data - - --storage-adv-addr=source-controller.$(RUNTIME_NAMESPACE).svc.cluster.local. - env: - - name: RUNTIME_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - - name: TUF_ROOT - value: /tmp/.sigstore - - name: GOMAXPROCS - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.cpu - - name: GOMEMLIMIT - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.memory - image: ghcr.io/fluxcd/source-controller:v1.3.0 - imagePullPolicy: IfNotPresent - livenessProbe: - httpGet: - path: /healthz - port: healthz - name: manager - ports: - - containerPort: 9090 - name: http - protocol: TCP - - containerPort: 8080 - name: http-prom - protocol: TCP - - containerPort: 9440 - name: healthz - protocol: TCP - readinessProbe: - httpGet: - path: / - port: http - resources: - limits: - cpu: 1000m - memory: 1Gi - requests: - cpu: 50m - memory: 64Mi - securityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - readOnlyRootFilesystem: true - runAsNonRoot: true - seccompProfile: - type: RuntimeDefault - volumeMounts: - - mountPath: /data - name: data - - mountPath: /tmp - name: tmp - nodeSelector: - kubernetes.io/os: linux - priorityClassName: system-cluster-critical - securityContext: - fsGroup: 1337 - serviceAccountName: source-controller - terminationGracePeriodSeconds: 10 - volumes: - - emptyDir: {} - name: data - - emptyDir: {} - name: tmp ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: kustomize-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: kustomizations.kustomize.toolkit.fluxcd.io -spec: - group: kustomize.toolkit.fluxcd.io - names: - kind: Kustomization - listKind: KustomizationList - plural: kustomizations - shortNames: - - ks - singular: kustomization - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - name: v1 - schema: - openAPIV3Schema: - description: Kustomization is the Schema for the kustomizations API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: |- - KustomizationSpec defines the configuration to calculate the desired state - from a Source using Kustomize. - properties: - commonMetadata: - description: |- - CommonMetadata specifies the common labels and annotations that are - applied to all resources. Any existing label or annotation will be - overridden if its key matches a common one. - properties: - annotations: - additionalProperties: - type: string - description: Annotations to be added to the object's metadata. - type: object - labels: - additionalProperties: - type: string - description: Labels to be added to the object's metadata. - type: object - type: object - components: - description: Components specifies relative paths to specifications - of other Components. - items: - type: string - type: array - decryption: - description: Decrypt Kubernetes secrets before applying them on the - cluster. - properties: - provider: - description: Provider is the name of the decryption engine. - enum: - - sops - type: string - secretRef: - description: The secret name containing the private OpenPGP keys - used for decryption. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - dependsOn: - description: |- - DependsOn may contain a meta.NamespacedObjectReference slice - with references to Kustomization resources that must be ready before this - Kustomization can be reconciled. - items: - description: |- - NamespacedObjectReference contains enough information to locate the referenced Kubernetes resource object in any - namespace. - properties: - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - name - type: object - type: array - force: - default: false - description: |- - Force instructs the controller to recreate resources - when patching fails due to an immutable field change. - type: boolean - healthChecks: - description: A list of resources to be included in the health assessment. - items: - description: |- - NamespacedObjectKindReference contains enough information to locate the typed referenced Kubernetes resource object - in any namespace. - properties: - apiVersion: - description: API version of the referent, if not specified the - Kubernetes preferred version will be used. - type: string - kind: - description: Kind of the referent. - type: string - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - kind - - name - type: object - type: array - images: - description: |- - Images is a list of (image name, new name, new tag or digest) - for changing image names, tags or digests. This can also be achieved with a - patch, but this operator is simpler to specify. - items: - description: Image contains an image name, a new name, a new tag - or digest, which will replace the original name and tag. - properties: - digest: - description: |- - Digest is the value used to replace the original image tag. - If digest is present NewTag value is ignored. - type: string - name: - description: Name is a tag-less image name. - type: string - newName: - description: NewName is the value used to replace the original - name. - type: string - newTag: - description: NewTag is the value used to replace the original - tag. - type: string - required: - - name - type: object - type: array - interval: - description: |- - The interval at which to reconcile the Kustomization. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - kubeConfig: - description: |- - The KubeConfig for reconciling the Kustomization on a remote cluster. - When used in combination with KustomizationSpec.ServiceAccountName, - forces the controller to act on behalf of that Service Account at the - target cluster. - If the --default-service-account flag is set, its value will be used as - a controller level fallback for when KustomizationSpec.ServiceAccountName - is empty. - properties: - secretRef: - description: |- - SecretRef holds the name of a secret that contains a key with - the kubeconfig file as the value. If no key is set, the key will default - to 'value'. - It is recommended that the kubeconfig is self-contained, and the secret - is regularly updated if credentials such as a cloud-access-token expire. - Cloud specific `cmd-path` auth helpers will not function without adding - binaries and credentials to the Pod that is responsible for reconciling - Kubernetes resources. - properties: - key: - description: Key in the Secret, when not specified an implementation-specific - default key is used. - type: string - name: - description: Name of the Secret. - type: string - required: - - name - type: object - required: - - secretRef - type: object - namePrefix: - description: NamePrefix will prefix the names of all managed resources. - maxLength: 200 - minLength: 1 - type: string - nameSuffix: - description: NameSuffix will suffix the names of all managed resources. - maxLength: 200 - minLength: 1 - type: string - patches: - description: |- - Strategic merge and JSON patches, defined as inline YAML objects, - capable of targeting objects based on kind, label and annotation selectors. - items: - description: |- - Patch contains an inline StrategicMerge or JSON6902 patch, and the target the patch should - be applied to. - properties: - patch: - description: |- - Patch contains an inline StrategicMerge patch or an inline JSON6902 patch with - an array of operation objects. - type: string - target: - description: Target points to the resources that the patch document - should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - type: object - type: array - path: - description: |- - Path to the directory containing the kustomization.yaml file, or the - set of plain YAMLs a kustomization.yaml should be generated for. - Defaults to 'None', which translates to the root path of the SourceRef. - type: string - postBuild: - description: |- - PostBuild describes which actions to perform on the YAML manifest - generated by building the kustomize overlay. - properties: - substitute: - additionalProperties: - type: string - description: |- - Substitute holds a map of key/value pairs. - The variables defined in your YAML manifests that match any of the keys - defined in the map will be substituted with the set value. - Includes support for bash string replacement functions - e.g. ${var:=default}, ${var:position} and ${var/substring/replacement}. - type: object - substituteFrom: - description: |- - SubstituteFrom holds references to ConfigMaps and Secrets containing - the variables and their values to be substituted in the YAML manifests. - The ConfigMap and the Secret data keys represent the var names, and they - must match the vars declared in the manifests for the substitution to - happen. - items: - description: |- - SubstituteReference contains a reference to a resource containing - the variables name and value. - properties: - kind: - description: Kind of the values referent, valid values are - ('Secret', 'ConfigMap'). - enum: - - Secret - - ConfigMap - type: string - name: - description: |- - Name of the values referent. Should reside in the same namespace as the - referring resource. - maxLength: 253 - minLength: 1 - type: string - optional: - default: false - description: |- - Optional indicates whether the referenced resource must exist, or whether to - tolerate its absence. If true and the referenced resource is absent, proceed - as if the resource was present but empty, without any variables defined. - type: boolean - required: - - kind - - name - type: object - type: array - type: object - prune: - description: Prune enables garbage collection. - type: boolean - retryInterval: - description: |- - The interval at which to retry a previously failed reconciliation. - When not specified, the controller uses the KustomizationSpec.Interval - value to retry failures. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - serviceAccountName: - description: |- - The name of the Kubernetes service account to impersonate - when reconciling this Kustomization. - type: string - sourceRef: - description: Reference of the source where the kustomization file - is. - properties: - apiVersion: - description: API version of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - OCIRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - type: string - namespace: - description: |- - Namespace of the referent, defaults to the namespace of the Kubernetes - resource object that contains the reference. - type: string - required: - - kind - - name - type: object - suspend: - description: |- - This flag tells the controller to suspend subsequent kustomize executions, - it does not apply to already started executions. Defaults to false. - type: boolean - targetNamespace: - description: |- - TargetNamespace sets or overrides the namespace in the - kustomization.yaml file. - maxLength: 63 - minLength: 1 - type: string - timeout: - description: |- - Timeout for validation, apply and health checking operations. - Defaults to 'Interval' duration. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - wait: - description: |- - Wait instructs the controller to check the health of all the reconciled - resources. When enabled, the HealthChecks are ignored. Defaults to false. - type: boolean - required: - - interval - - prune - - sourceRef - type: object - status: - default: - observedGeneration: -1 - description: KustomizationStatus defines the observed state of a kustomization. - properties: - conditions: - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - inventory: - description: |- - Inventory contains the list of Kubernetes resource object references that - have been successfully applied. - properties: - entries: - description: Entries of Kubernetes resource object references. - items: - description: ResourceRef contains the information necessary - to locate a resource within a cluster. - properties: - id: - description: |- - ID is the string representation of the Kubernetes resource object's metadata, - in the format '___'. - type: string - v: - description: Version is the API version of the Kubernetes - resource object's kind. - type: string - required: - - id - - v - type: object - type: array - required: - - entries - type: object - lastAppliedRevision: - description: |- - The last successfully applied revision. - Equals the Revision of the applied Artifact from the referenced Source. - type: string - lastAttemptedRevision: - description: LastAttemptedRevision is the revision of the last reconciliation - attempt. - type: string - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last reconciled generation. - format: int64 - type: integer - type: object - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - deprecated: true - deprecationWarning: v1beta1 Kustomization is deprecated, upgrade to v1 - name: v1beta1 - schema: - openAPIV3Schema: - description: Kustomization is the Schema for the kustomizations API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: KustomizationSpec defines the desired state of a kustomization. - properties: - decryption: - description: Decrypt Kubernetes secrets before applying them on the - cluster. - properties: - provider: - description: Provider is the name of the decryption engine. - enum: - - sops - type: string - secretRef: - description: The secret name containing the private OpenPGP keys - used for decryption. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - dependsOn: - description: |- - DependsOn may contain a meta.NamespacedObjectReference slice - with references to Kustomization resources that must be ready before this - Kustomization can be reconciled. - items: - description: |- - NamespacedObjectReference contains enough information to locate the referenced Kubernetes resource object in any - namespace. - properties: - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - name - type: object - type: array - force: - default: false - description: |- - Force instructs the controller to recreate resources - when patching fails due to an immutable field change. - type: boolean - healthChecks: - description: A list of resources to be included in the health assessment. - items: - description: |- - NamespacedObjectKindReference contains enough information to locate the typed referenced Kubernetes resource object - in any namespace. - properties: - apiVersion: - description: API version of the referent, if not specified the - Kubernetes preferred version will be used. - type: string - kind: - description: Kind of the referent. - type: string - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - kind - - name - type: object - type: array - images: - description: |- - Images is a list of (image name, new name, new tag or digest) - for changing image names, tags or digests. This can also be achieved with a - patch, but this operator is simpler to specify. - items: - description: Image contains an image name, a new name, a new tag - or digest, which will replace the original name and tag. - properties: - digest: - description: |- - Digest is the value used to replace the original image tag. - If digest is present NewTag value is ignored. - type: string - name: - description: Name is a tag-less image name. - type: string - newName: - description: NewName is the value used to replace the original - name. - type: string - newTag: - description: NewTag is the value used to replace the original - tag. - type: string - required: - - name - type: object - type: array - interval: - description: The interval at which to reconcile the Kustomization. - type: string - kubeConfig: - description: |- - The KubeConfig for reconciling the Kustomization on a remote cluster. - When specified, KubeConfig takes precedence over ServiceAccountName. - properties: - secretRef: - description: |- - SecretRef holds the name to a secret that contains a 'value' key with - the kubeconfig file as the value. It must be in the same namespace as - the Kustomization. - It is recommended that the kubeconfig is self-contained, and the secret - is regularly updated if credentials such as a cloud-access-token expire. - Cloud specific `cmd-path` auth helpers will not function without adding - binaries and credentials to the Pod that is responsible for reconciling - the Kustomization. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - type: object - patches: - description: |- - Strategic merge and JSON patches, defined as inline YAML objects, - capable of targeting objects based on kind, label and annotation selectors. - items: - description: |- - Patch contains an inline StrategicMerge or JSON6902 patch, and the target the patch should - be applied to. - properties: - patch: - description: |- - Patch contains an inline StrategicMerge patch or an inline JSON6902 patch with - an array of operation objects. - type: string - target: - description: Target points to the resources that the patch document - should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - type: object - type: array - patchesJson6902: - description: JSON 6902 patches, defined as inline YAML objects. - items: - description: JSON6902Patch contains a JSON6902 patch and the target - the patch should be applied to. - properties: - patch: - description: Patch contains the JSON6902 patch document with - an array of operation objects. - items: - description: |- - JSON6902 is a JSON6902 operation object. - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - properties: - from: - description: |- - From contains a JSON-pointer value that references a location within the target document where the operation is - performed. The meaning of the value depends on the value of Op, and is NOT taken into account by all operations. - type: string - op: - description: |- - Op indicates the operation to perform. Its value MUST be one of "add", "remove", "replace", "move", "copy", or - "test". - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - enum: - - test - - remove - - add - - replace - - move - - copy - type: string - path: - description: |- - Path contains the JSON-pointer value that references a location within the target document where the operation - is performed. The meaning of the value depends on the value of Op. - type: string - value: - description: |- - Value contains a valid JSON structure. The meaning of the value depends on the value of Op, and is NOT taken into - account by all operations. - x-kubernetes-preserve-unknown-fields: true - required: - - op - - path - type: object - type: array - target: - description: Target points to the resources that the patch document - should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - - target - type: object - type: array - patchesStrategicMerge: - description: Strategic merge patches, defined as inline YAML objects. - items: - x-kubernetes-preserve-unknown-fields: true - type: array - path: - description: |- - Path to the directory containing the kustomization.yaml file, or the - set of plain YAMLs a kustomization.yaml should be generated for. - Defaults to 'None', which translates to the root path of the SourceRef. - type: string - postBuild: - description: |- - PostBuild describes which actions to perform on the YAML manifest - generated by building the kustomize overlay. - properties: - substitute: - additionalProperties: - type: string - description: |- - Substitute holds a map of key/value pairs. - The variables defined in your YAML manifests - that match any of the keys defined in the map - will be substituted with the set value. - Includes support for bash string replacement functions - e.g. ${var:=default}, ${var:position} and ${var/substring/replacement}. - type: object - substituteFrom: - description: |- - SubstituteFrom holds references to ConfigMaps and Secrets containing - the variables and their values to be substituted in the YAML manifests. - The ConfigMap and the Secret data keys represent the var names and they - must match the vars declared in the manifests for the substitution to happen. - items: - description: |- - SubstituteReference contains a reference to a resource containing - the variables name and value. - properties: - kind: - description: Kind of the values referent, valid values are - ('Secret', 'ConfigMap'). - enum: - - Secret - - ConfigMap - type: string - name: - description: |- - Name of the values referent. Should reside in the same namespace as the - referring resource. - maxLength: 253 - minLength: 1 - type: string - required: - - kind - - name - type: object - type: array - type: object - prune: - description: Prune enables garbage collection. - type: boolean - retryInterval: - description: |- - The interval at which to retry a previously failed reconciliation. - When not specified, the controller uses the KustomizationSpec.Interval - value to retry failures. - type: string - serviceAccountName: - description: |- - The name of the Kubernetes service account to impersonate - when reconciling this Kustomization. - type: string - sourceRef: - description: Reference of the source where the kustomization file - is. - properties: - apiVersion: - description: API version of the referent - type: string - kind: - description: Kind of the referent - enum: - - GitRepository - - Bucket - type: string - name: - description: Name of the referent - type: string - namespace: - description: Namespace of the referent, defaults to the Kustomization - namespace - type: string - required: - - kind - - name - type: object - suspend: - description: |- - This flag tells the controller to suspend subsequent kustomize executions, - it does not apply to already started executions. Defaults to false. - type: boolean - targetNamespace: - description: |- - TargetNamespace sets or overrides the namespace in the - kustomization.yaml file. - maxLength: 63 - minLength: 1 - type: string - timeout: - description: |- - Timeout for validation, apply and health checking operations. - Defaults to 'Interval' duration. - type: string - validation: - description: |- - Validate the Kubernetes objects before applying them on the cluster. - The validation strategy can be 'client' (local dry-run), 'server' - (APIServer dry-run) or 'none'. - When 'Force' is 'true', validation will fallback to 'client' if set to - 'server' because server-side validation is not supported in this scenario. - enum: - - none - - client - - server - type: string - required: - - interval - - prune - - sourceRef - type: object - status: - default: - observedGeneration: -1 - description: KustomizationStatus defines the observed state of a kustomization. - properties: - conditions: - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastAppliedRevision: - description: |- - The last successfully applied revision. - The revision format for Git sources is /. - type: string - lastAttemptedRevision: - description: LastAttemptedRevision is the revision of the last reconciliation - attempt. - type: string - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last reconciled generation. - format: int64 - type: integer - snapshot: - description: The last successfully applied revision metadata. - properties: - checksum: - description: The manifests sha1 checksum. - type: string - entries: - description: A list of Kubernetes kinds grouped by namespace. - items: - description: |- - Snapshot holds the metadata of namespaced - Kubernetes objects - properties: - kinds: - additionalProperties: - type: string - description: The list of Kubernetes kinds. - type: object - namespace: - description: The namespace of this entry. - type: string - required: - - kinds - type: object - type: array - required: - - checksum - - entries - type: object - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta2 Kustomization is deprecated, upgrade to v1 - name: v1beta2 - schema: - openAPIV3Schema: - description: Kustomization is the Schema for the kustomizations API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: KustomizationSpec defines the configuration to calculate - the desired state from a Source using Kustomize. - properties: - commonMetadata: - description: |- - CommonMetadata specifies the common labels and annotations that are applied to all resources. - Any existing label or annotation will be overridden if its key matches a common one. - properties: - annotations: - additionalProperties: - type: string - description: Annotations to be added to the object's metadata. - type: object - labels: - additionalProperties: - type: string - description: Labels to be added to the object's metadata. - type: object - type: object - components: - description: Components specifies relative paths to specifications - of other Components. - items: - type: string - type: array - decryption: - description: Decrypt Kubernetes secrets before applying them on the - cluster. - properties: - provider: - description: Provider is the name of the decryption engine. - enum: - - sops - type: string - secretRef: - description: The secret name containing the private OpenPGP keys - used for decryption. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - dependsOn: - description: |- - DependsOn may contain a meta.NamespacedObjectReference slice - with references to Kustomization resources that must be ready before this - Kustomization can be reconciled. - items: - description: |- - NamespacedObjectReference contains enough information to locate the referenced Kubernetes resource object in any - namespace. - properties: - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - name - type: object - type: array - force: - default: false - description: |- - Force instructs the controller to recreate resources - when patching fails due to an immutable field change. - type: boolean - healthChecks: - description: A list of resources to be included in the health assessment. - items: - description: |- - NamespacedObjectKindReference contains enough information to locate the typed referenced Kubernetes resource object - in any namespace. - properties: - apiVersion: - description: API version of the referent, if not specified the - Kubernetes preferred version will be used. - type: string - kind: - description: Kind of the referent. - type: string - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - kind - - name - type: object - type: array - images: - description: |- - Images is a list of (image name, new name, new tag or digest) - for changing image names, tags or digests. This can also be achieved with a - patch, but this operator is simpler to specify. - items: - description: Image contains an image name, a new name, a new tag - or digest, which will replace the original name and tag. - properties: - digest: - description: |- - Digest is the value used to replace the original image tag. - If digest is present NewTag value is ignored. - type: string - name: - description: Name is a tag-less image name. - type: string - newName: - description: NewName is the value used to replace the original - name. - type: string - newTag: - description: NewTag is the value used to replace the original - tag. - type: string - required: - - name - type: object - type: array - interval: - description: The interval at which to reconcile the Kustomization. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - kubeConfig: - description: |- - The KubeConfig for reconciling the Kustomization on a remote cluster. - When used in combination with KustomizationSpec.ServiceAccountName, - forces the controller to act on behalf of that Service Account at the - target cluster. - If the --default-service-account flag is set, its value will be used as - a controller level fallback for when KustomizationSpec.ServiceAccountName - is empty. - properties: - secretRef: - description: |- - SecretRef holds the name of a secret that contains a key with - the kubeconfig file as the value. If no key is set, the key will default - to 'value'. - It is recommended that the kubeconfig is self-contained, and the secret - is regularly updated if credentials such as a cloud-access-token expire. - Cloud specific `cmd-path` auth helpers will not function without adding - binaries and credentials to the Pod that is responsible for reconciling - Kubernetes resources. - properties: - key: - description: Key in the Secret, when not specified an implementation-specific - default key is used. - type: string - name: - description: Name of the Secret. - type: string - required: - - name - type: object - required: - - secretRef - type: object - patches: - description: |- - Strategic merge and JSON patches, defined as inline YAML objects, - capable of targeting objects based on kind, label and annotation selectors. - items: - description: |- - Patch contains an inline StrategicMerge or JSON6902 patch, and the target the patch should - be applied to. - properties: - patch: - description: |- - Patch contains an inline StrategicMerge patch or an inline JSON6902 patch with - an array of operation objects. - type: string - target: - description: Target points to the resources that the patch document - should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - type: object - type: array - patchesJson6902: - description: |- - JSON 6902 patches, defined as inline YAML objects. - Deprecated: Use Patches instead. - items: - description: JSON6902Patch contains a JSON6902 patch and the target - the patch should be applied to. - properties: - patch: - description: Patch contains the JSON6902 patch document with - an array of operation objects. - items: - description: |- - JSON6902 is a JSON6902 operation object. - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - properties: - from: - description: |- - From contains a JSON-pointer value that references a location within the target document where the operation is - performed. The meaning of the value depends on the value of Op, and is NOT taken into account by all operations. - type: string - op: - description: |- - Op indicates the operation to perform. Its value MUST be one of "add", "remove", "replace", "move", "copy", or - "test". - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - enum: - - test - - remove - - add - - replace - - move - - copy - type: string - path: - description: |- - Path contains the JSON-pointer value that references a location within the target document where the operation - is performed. The meaning of the value depends on the value of Op. - type: string - value: - description: |- - Value contains a valid JSON structure. The meaning of the value depends on the value of Op, and is NOT taken into - account by all operations. - x-kubernetes-preserve-unknown-fields: true - required: - - op - - path - type: object - type: array - target: - description: Target points to the resources that the patch document - should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - - target - type: object - type: array - patchesStrategicMerge: - description: |- - Strategic merge patches, defined as inline YAML objects. - Deprecated: Use Patches instead. - items: - x-kubernetes-preserve-unknown-fields: true - type: array - path: - description: |- - Path to the directory containing the kustomization.yaml file, or the - set of plain YAMLs a kustomization.yaml should be generated for. - Defaults to 'None', which translates to the root path of the SourceRef. - type: string - postBuild: - description: |- - PostBuild describes which actions to perform on the YAML manifest - generated by building the kustomize overlay. - properties: - substitute: - additionalProperties: - type: string - description: |- - Substitute holds a map of key/value pairs. - The variables defined in your YAML manifests - that match any of the keys defined in the map - will be substituted with the set value. - Includes support for bash string replacement functions - e.g. ${var:=default}, ${var:position} and ${var/substring/replacement}. - type: object - substituteFrom: - description: |- - SubstituteFrom holds references to ConfigMaps and Secrets containing - the variables and their values to be substituted in the YAML manifests. - The ConfigMap and the Secret data keys represent the var names and they - must match the vars declared in the manifests for the substitution to happen. - items: - description: |- - SubstituteReference contains a reference to a resource containing - the variables name and value. - properties: - kind: - description: Kind of the values referent, valid values are - ('Secret', 'ConfigMap'). - enum: - - Secret - - ConfigMap - type: string - name: - description: |- - Name of the values referent. Should reside in the same namespace as the - referring resource. - maxLength: 253 - minLength: 1 - type: string - optional: - default: false - description: |- - Optional indicates whether the referenced resource must exist, or whether to - tolerate its absence. If true and the referenced resource is absent, proceed - as if the resource was present but empty, without any variables defined. - type: boolean - required: - - kind - - name - type: object - type: array - type: object - prune: - description: Prune enables garbage collection. - type: boolean - retryInterval: - description: |- - The interval at which to retry a previously failed reconciliation. - When not specified, the controller uses the KustomizationSpec.Interval - value to retry failures. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - serviceAccountName: - description: |- - The name of the Kubernetes service account to impersonate - when reconciling this Kustomization. - type: string - sourceRef: - description: Reference of the source where the kustomization file - is. - properties: - apiVersion: - description: API version of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - OCIRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, defaults to the namespace - of the Kubernetes resource object that contains the reference. - type: string - required: - - kind - - name - type: object - suspend: - description: |- - This flag tells the controller to suspend subsequent kustomize executions, - it does not apply to already started executions. Defaults to false. - type: boolean - targetNamespace: - description: |- - TargetNamespace sets or overrides the namespace in the - kustomization.yaml file. - maxLength: 63 - minLength: 1 - type: string - timeout: - description: |- - Timeout for validation, apply and health checking operations. - Defaults to 'Interval' duration. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - validation: - description: 'Deprecated: Not used in v1beta2.' - enum: - - none - - client - - server - type: string - wait: - description: |- - Wait instructs the controller to check the health of all the reconciled resources. - When enabled, the HealthChecks are ignored. Defaults to false. - type: boolean - required: - - interval - - prune - - sourceRef - type: object - status: - default: - observedGeneration: -1 - description: KustomizationStatus defines the observed state of a kustomization. - properties: - conditions: - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - inventory: - description: Inventory contains the list of Kubernetes resource object - references that have been successfully applied. - properties: - entries: - description: Entries of Kubernetes resource object references. - items: - description: ResourceRef contains the information necessary - to locate a resource within a cluster. - properties: - id: - description: |- - ID is the string representation of the Kubernetes resource object's metadata, - in the format '___'. - type: string - v: - description: Version is the API version of the Kubernetes - resource object's kind. - type: string - required: - - id - - v - type: object - type: array - required: - - entries - type: object - lastAppliedRevision: - description: |- - The last successfully applied revision. - Equals the Revision of the applied Artifact from the referenced Source. - type: string - lastAttemptedRevision: - description: LastAttemptedRevision is the revision of the last reconciliation - attempt. - type: string - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last reconciled generation. - format: int64 - type: integer - type: object - type: object - served: true - storage: false - subresources: - status: {} ---- -apiVersion: v1 -kind: ServiceAccount -metadata: - labels: - app.kubernetes.io/component: kustomize-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: kustomize-controller - namespace: flux-system ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - labels: - app.kubernetes.io/component: kustomize-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - control-plane: controller - name: kustomize-controller - namespace: flux-system -spec: - replicas: 1 - selector: - matchLabels: - app: kustomize-controller - template: - metadata: - annotations: - prometheus.io/port: "8080" - prometheus.io/scrape: "true" - labels: - app: kustomize-controller - spec: - containers: - - args: - - --events-addr=http://notification-controller.flux-system.svc.cluster.local./ - - --watch-all-namespaces=true - - --log-level=info - - --log-encoding=json - - --enable-leader-election - env: - - name: RUNTIME_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - - name: GOMAXPROCS - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.cpu - - name: GOMEMLIMIT - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.memory - image: ghcr.io/fluxcd/kustomize-controller:v1.3.0 - imagePullPolicy: IfNotPresent - livenessProbe: - httpGet: - path: /healthz - port: healthz - name: manager - ports: - - containerPort: 8080 - name: http-prom - protocol: TCP - - containerPort: 9440 - name: healthz - protocol: TCP - readinessProbe: - httpGet: - path: /readyz - port: healthz - resources: - limits: - cpu: 1000m - memory: 1Gi - requests: - cpu: 100m - memory: 64Mi - securityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - readOnlyRootFilesystem: true - runAsNonRoot: true - seccompProfile: - type: RuntimeDefault - volumeMounts: - - mountPath: /tmp - name: temp - nodeSelector: - kubernetes.io/os: linux - priorityClassName: system-cluster-critical - securityContext: - fsGroup: 1337 - serviceAccountName: kustomize-controller - terminationGracePeriodSeconds: 60 - volumes: - - emptyDir: {} - name: temp ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: helm-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: helmreleases.helm.toolkit.fluxcd.io -spec: - group: helm.toolkit.fluxcd.io - names: - kind: HelmRelease - listKind: HelmReleaseList - plural: helmreleases - shortNames: - - hr - singular: helmrelease - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - name: v2 - schema: - openAPIV3Schema: - description: HelmRelease is the Schema for the helmreleases API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HelmReleaseSpec defines the desired state of a Helm release. - properties: - chart: - description: |- - Chart defines the template of the v1.HelmChart that should be created - for this HelmRelease. - properties: - metadata: - description: ObjectMeta holds the template for metadata like labels - and annotations. - properties: - annotations: - additionalProperties: - type: string - description: |- - Annotations is an unstructured key value map stored with a resource that may be - set by external tools to store and retrieve arbitrary metadata. They are not - queryable and should be preserved when modifying objects. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ - type: object - labels: - additionalProperties: - type: string - description: |- - Map of string keys and values that can be used to organize and categorize - (scope and select) objects. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/ - type: object - type: object - spec: - description: Spec holds the template for the v1.HelmChartSpec - for this HelmRelease. - properties: - chart: - description: The name or path the Helm chart is available - at in the SourceRef. - maxLength: 2048 - minLength: 1 - type: string - ignoreMissingValuesFiles: - description: IgnoreMissingValuesFiles controls whether to - silently ignore missing values files rather than failing. - type: boolean - interval: - description: |- - Interval at which to check the v1.Source for updates. Defaults to - 'HelmReleaseSpec.Interval'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - reconcileStrategy: - default: ChartVersion - description: |- - Determines what enables the creation of a new artifact. Valid values are - ('ChartVersion', 'Revision'). - See the documentation of the values for an explanation on their behavior. - Defaults to ChartVersion when omitted. - enum: - - ChartVersion - - Revision - type: string - sourceRef: - description: The name and namespace of the v1.Source the chart - is available at. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - HelmRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: Namespace of the referent. - maxLength: 63 - minLength: 1 - type: string - required: - - name - type: object - valuesFiles: - description: |- - Alternative list of values files to use as the chart values (values.yaml - is not included by default), expected to be a relative path in the SourceRef. - Values files are merged in the order of this list with the last file overriding - the first. Ignored when omitted. - items: - type: string - type: array - verify: - description: |- - Verify contains the secret name containing the trusted public keys - used to verify the signature and specifies which provider to use to check - whether OCI image is authentic. - This field is only supported for OCI sources. - Chart dependencies, which are not bundled in the umbrella chart artifact, - are not verified. - properties: - provider: - default: cosign - description: Provider specifies the technology used to - sign the OCI Helm chart. - enum: - - cosign - - notation - type: string - secretRef: - description: |- - SecretRef specifies the Kubernetes Secret containing the - trusted public keys. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - version: - default: '*' - description: |- - Version semver expression, ignored for charts from v1.GitRepository and - v1beta2.Bucket sources. Defaults to latest when omitted. - type: string - required: - - chart - - sourceRef - type: object - required: - - spec - type: object - chartRef: - description: |- - ChartRef holds a reference to a source controller resource containing the - Helm chart artifact. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - OCIRepository - - HelmChart - type: string - name: - description: Name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace of the referent, defaults to the namespace of the Kubernetes - resource object that contains the reference. - maxLength: 63 - minLength: 1 - type: string - required: - - kind - - name - type: object - dependsOn: - description: |- - DependsOn may contain a meta.NamespacedObjectReference slice with - references to HelmRelease resources that must be ready before this HelmRelease - can be reconciled. - items: - description: |- - NamespacedObjectReference contains enough information to locate the referenced Kubernetes resource object in any - namespace. - properties: - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - name - type: object - type: array - driftDetection: - description: |- - DriftDetection holds the configuration for detecting and handling - differences between the manifest in the Helm storage and the resources - currently existing in the cluster. - properties: - ignore: - description: |- - Ignore contains a list of rules for specifying which changes to ignore - during diffing. - items: - description: |- - IgnoreRule defines a rule to selectively disregard specific changes during - the drift detection process. - properties: - paths: - description: |- - Paths is a list of JSON Pointer (RFC 6901) paths to be excluded from - consideration in a Kubernetes object. - items: - type: string - type: array - target: - description: |- - Target is a selector for specifying Kubernetes objects to which this - rule applies. - If Target is not set, the Paths will be ignored for all Kubernetes - objects within the manifest of the Helm release. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - paths - type: object - type: array - mode: - description: |- - Mode defines how differences should be handled between the Helm manifest - and the manifest currently applied to the cluster. - If not explicitly set, it defaults to DiffModeDisabled. - enum: - - enabled - - warn - - disabled - type: string - type: object - install: - description: Install holds the configuration for Helm install actions - for this HelmRelease. - properties: - crds: - description: |- - CRDs upgrade CRDs from the Helm Chart's crds directory according - to the CRD upgrade policy provided here. Valid values are `Skip`, - `Create` or `CreateReplace`. Default is `Create` and if omitted - CRDs are installed but not updated. - - - Skip: do neither install nor replace (update) any CRDs. - - - Create: new CRDs are created, existing CRDs are neither updated nor deleted. - - - CreateReplace: new CRDs are created, existing CRDs are updated (replaced) - but not deleted. - - - By default, CRDs are applied (installed) during Helm install action. - With this option users can opt in to CRD replace existing CRDs on Helm - install actions, which is not (yet) natively supported by Helm. - https://helm.sh/docs/chart_best_practices/custom_resource_definitions. - enum: - - Skip - - Create - - CreateReplace - type: string - createNamespace: - description: |- - CreateNamespace tells the Helm install action to create the - HelmReleaseSpec.TargetNamespace if it does not exist yet. - On uninstall, the namespace will not be garbage collected. - type: boolean - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm install action. - type: boolean - disableOpenAPIValidation: - description: |- - DisableOpenAPIValidation prevents the Helm install action from validating - rendered templates against the Kubernetes OpenAPI Schema. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - install has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - install has been performed. - type: boolean - remediation: - description: |- - Remediation holds the remediation configuration for when the Helm install - action for the HelmRelease fails. The default is to not perform any action. - properties: - ignoreTestFailures: - description: |- - IgnoreTestFailures tells the controller to skip remediation when the Helm - tests are run after an install action but fail. Defaults to - 'Test.IgnoreFailures'. - type: boolean - remediateLastFailure: - description: |- - RemediateLastFailure tells the controller to remediate the last failure, when - no retries remain. Defaults to 'false'. - type: boolean - retries: - description: |- - Retries is the number of retries that should be attempted on failures before - bailing. Remediation, using an uninstall, is performed between each attempt. - Defaults to '0', a negative integer equals to unlimited retries. - type: integer - type: object - replace: - description: |- - Replace tells the Helm install action to re-use the 'ReleaseName', but only - if that name is a deleted release which remains in the history. - type: boolean - skipCRDs: - description: |- - SkipCRDs tells the Helm install action to not install any CRDs. By default, - CRDs are installed if not already present. - - - Deprecated use CRD policy (`crds`) attribute with value `Skip` instead. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm install action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - interval: - description: Interval at which to reconcile the Helm release. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - kubeConfig: - description: |- - KubeConfig for reconciling the HelmRelease on a remote cluster. - When used in combination with HelmReleaseSpec.ServiceAccountName, - forces the controller to act on behalf of that Service Account at the - target cluster. - If the --default-service-account flag is set, its value will be used as - a controller level fallback for when HelmReleaseSpec.ServiceAccountName - is empty. - properties: - secretRef: - description: |- - SecretRef holds the name of a secret that contains a key with - the kubeconfig file as the value. If no key is set, the key will default - to 'value'. - It is recommended that the kubeconfig is self-contained, and the secret - is regularly updated if credentials such as a cloud-access-token expire. - Cloud specific `cmd-path` auth helpers will not function without adding - binaries and credentials to the Pod that is responsible for reconciling - Kubernetes resources. - properties: - key: - description: Key in the Secret, when not specified an implementation-specific - default key is used. - type: string - name: - description: Name of the Secret. - type: string - required: - - name - type: object - required: - - secretRef - type: object - maxHistory: - description: |- - MaxHistory is the number of revisions saved by Helm for this HelmRelease. - Use '0' for an unlimited number of revisions; defaults to '5'. - type: integer - persistentClient: - description: |- - PersistentClient tells the controller to use a persistent Kubernetes - client for this release. When enabled, the client will be reused for the - duration of the reconciliation, instead of being created and destroyed - for each (step of a) Helm action. - - - This can improve performance, but may cause issues with some Helm charts - that for example do create Custom Resource Definitions during installation - outside Helm's CRD lifecycle hooks, which are then not observed to be - available by e.g. post-install hooks. - - - If not set, it defaults to true. - type: boolean - postRenderers: - description: |- - PostRenderers holds an array of Helm PostRenderers, which will be applied in order - of their definition. - items: - description: PostRenderer contains a Helm PostRenderer specification. - properties: - kustomize: - description: Kustomization to apply as PostRenderer. - properties: - images: - description: |- - Images is a list of (image name, new name, new tag or digest) - for changing image names, tags or digests. This can also be achieved with a - patch, but this operator is simpler to specify. - items: - description: Image contains an image name, a new name, - a new tag or digest, which will replace the original - name and tag. - properties: - digest: - description: |- - Digest is the value used to replace the original image tag. - If digest is present NewTag value is ignored. - type: string - name: - description: Name is a tag-less image name. - type: string - newName: - description: NewName is the value used to replace - the original name. - type: string - newTag: - description: NewTag is the value used to replace the - original tag. - type: string - required: - - name - type: object - type: array - patches: - description: |- - Strategic merge and JSON patches, defined as inline YAML objects, - capable of targeting objects based on kind, label and annotation selectors. - items: - description: |- - Patch contains an inline StrategicMerge or JSON6902 patch, and the target the patch should - be applied to. - properties: - patch: - description: |- - Patch contains an inline StrategicMerge patch or an inline JSON6902 patch with - an array of operation objects. - type: string - target: - description: Target points to the resources that the - patch document should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - type: object - type: array - type: object - type: object - type: array - releaseName: - description: |- - ReleaseName used for the Helm release. Defaults to a composition of - '[TargetNamespace-]Name'. - maxLength: 53 - minLength: 1 - type: string - rollback: - description: Rollback holds the configuration for Helm rollback actions - for this HelmRelease. - properties: - cleanupOnFail: - description: |- - CleanupOnFail allows deletion of new resources created during the Helm - rollback action when it fails. - type: boolean - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm rollback action. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - rollback has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - rollback has been performed. - type: boolean - force: - description: Force forces resource updates through a replacement - strategy. - type: boolean - recreate: - description: Recreate performs pod restarts for the resource if - applicable. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm rollback action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - serviceAccountName: - description: |- - The name of the Kubernetes service account to impersonate - when reconciling this HelmRelease. - maxLength: 253 - minLength: 1 - type: string - storageNamespace: - description: |- - StorageNamespace used for the Helm storage. - Defaults to the namespace of the HelmRelease. - maxLength: 63 - minLength: 1 - type: string - suspend: - description: |- - Suspend tells the controller to suspend reconciliation for this HelmRelease, - it does not apply to already started reconciliations. Defaults to false. - type: boolean - targetNamespace: - description: |- - TargetNamespace to target when performing operations for the HelmRelease. - Defaults to the namespace of the HelmRelease. - maxLength: 63 - minLength: 1 - type: string - test: - description: Test holds the configuration for Helm test actions for - this HelmRelease. - properties: - enable: - description: |- - Enable enables Helm test actions for this HelmRelease after an Helm install - or upgrade action has been performed. - type: boolean - filters: - description: Filters is a list of tests to run or exclude from - running. - items: - description: Filter holds the configuration for individual Helm - test filters. - properties: - exclude: - description: Exclude specifies whether the named test should - be excluded. - type: boolean - name: - description: Name is the name of the test. - maxLength: 253 - minLength: 1 - type: string - required: - - name - type: object - type: array - ignoreFailures: - description: |- - IgnoreFailures tells the controller to skip remediation when the Helm tests - are run but fail. Can be overwritten for tests run after install or upgrade - actions in 'Install.IgnoreTestFailures' and 'Upgrade.IgnoreTestFailures'. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation during - the performance of a Helm test action. Defaults to 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like Jobs - for hooks) during the performance of a Helm action. Defaults to '5m0s'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - uninstall: - description: Uninstall holds the configuration for Helm uninstall - actions for this HelmRelease. - properties: - deletionPropagation: - default: background - description: |- - DeletionPropagation specifies the deletion propagation policy when - a Helm uninstall is performed. - enum: - - background - - foreground - - orphan - type: string - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm rollback action. - type: boolean - disableWait: - description: |- - DisableWait disables waiting for all the resources to be deleted after - a Helm uninstall is performed. - type: boolean - keepHistory: - description: |- - KeepHistory tells Helm to remove all associated resources and mark the - release as deleted, but retain the release history. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm uninstall action. Defaults - to 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - upgrade: - description: Upgrade holds the configuration for Helm upgrade actions - for this HelmRelease. - properties: - cleanupOnFail: - description: |- - CleanupOnFail allows deletion of new resources created during the Helm - upgrade action when it fails. - type: boolean - crds: - description: |- - CRDs upgrade CRDs from the Helm Chart's crds directory according - to the CRD upgrade policy provided here. Valid values are `Skip`, - `Create` or `CreateReplace`. Default is `Skip` and if omitted - CRDs are neither installed nor upgraded. - - - Skip: do neither install nor replace (update) any CRDs. - - - Create: new CRDs are created, existing CRDs are neither updated nor deleted. - - - CreateReplace: new CRDs are created, existing CRDs are updated (replaced) - but not deleted. - - - By default, CRDs are not applied during Helm upgrade action. With this - option users can opt-in to CRD upgrade, which is not (yet) natively supported by Helm. - https://helm.sh/docs/chart_best_practices/custom_resource_definitions. - enum: - - Skip - - Create - - CreateReplace - type: string - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm upgrade action. - type: boolean - disableOpenAPIValidation: - description: |- - DisableOpenAPIValidation prevents the Helm upgrade action from validating - rendered templates against the Kubernetes OpenAPI Schema. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - upgrade has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - upgrade has been performed. - type: boolean - force: - description: Force forces resource updates through a replacement - strategy. - type: boolean - preserveValues: - description: |- - PreserveValues will make Helm reuse the last release's values and merge in - overrides from 'Values'. Setting this flag makes the HelmRelease - non-declarative. - type: boolean - remediation: - description: |- - Remediation holds the remediation configuration for when the Helm upgrade - action for the HelmRelease fails. The default is to not perform any action. - properties: - ignoreTestFailures: - description: |- - IgnoreTestFailures tells the controller to skip remediation when the Helm - tests are run after an upgrade action but fail. - Defaults to 'Test.IgnoreFailures'. - type: boolean - remediateLastFailure: - description: |- - RemediateLastFailure tells the controller to remediate the last failure, when - no retries remain. Defaults to 'false' unless 'Retries' is greater than 0. - type: boolean - retries: - description: |- - Retries is the number of retries that should be attempted on failures before - bailing. Remediation, using 'Strategy', is performed between each attempt. - Defaults to '0', a negative integer equals to unlimited retries. - type: integer - strategy: - description: Strategy to use for failure remediation. Defaults - to 'rollback'. - enum: - - rollback - - uninstall - type: string - type: object - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm upgrade action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - values: - description: Values holds the values for this Helm release. - x-kubernetes-preserve-unknown-fields: true - valuesFrom: - description: |- - ValuesFrom holds references to resources containing Helm values for this HelmRelease, - and information about how they should be merged. - items: - description: |- - ValuesReference contains a reference to a resource containing Helm values, - and optionally the key they can be found at. - properties: - kind: - description: Kind of the values referent, valid values are ('Secret', - 'ConfigMap'). - enum: - - Secret - - ConfigMap - type: string - name: - description: |- - Name of the values referent. Should reside in the same namespace as the - referring resource. - maxLength: 253 - minLength: 1 - type: string - optional: - description: |- - Optional marks this ValuesReference as optional. When set, a not found error - for the values reference is ignored, but any ValuesKey, TargetPath or - transient error will still result in a reconciliation failure. - type: boolean - targetPath: - description: |- - TargetPath is the YAML dot notation path the value should be merged at. When - set, the ValuesKey is expected to be a single flat value. Defaults to 'None', - which results in the values getting merged at the root. - maxLength: 250 - pattern: ^([a-zA-Z0-9_\-.\\\/]|\[[0-9]{1,5}\])+$ - type: string - valuesKey: - description: |- - ValuesKey is the data key where the values.yaml or a specific value can be - found at. Defaults to 'values.yaml'. - maxLength: 253 - pattern: ^[\-._a-zA-Z0-9]+$ - type: string - required: - - kind - - name - type: object - type: array - required: - - interval - type: object - x-kubernetes-validations: - - message: either chart or chartRef must be set - rule: (has(self.chart) && !has(self.chartRef)) || (!has(self.chart) - && has(self.chartRef)) - status: - default: - observedGeneration: -1 - description: HelmReleaseStatus defines the observed state of a HelmRelease. - properties: - conditions: - description: Conditions holds the conditions for the HelmRelease. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - failures: - description: |- - Failures is the reconciliation failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - helmChart: - description: |- - HelmChart is the namespaced name of the HelmChart resource created by - the controller for the HelmRelease. - type: string - history: - description: |- - History holds the history of Helm releases performed for this HelmRelease - up to the last successfully completed release. - items: - description: |- - Snapshot captures a point-in-time copy of the status information for a Helm release, - as managed by the controller. - properties: - apiVersion: - description: |- - APIVersion is the API version of the Snapshot. - Provisional: when the calculation method of the Digest field is changed, - this field will be used to distinguish between the old and new methods. - type: string - appVersion: - description: AppVersion is the chart app version of the release - object in storage. - type: string - chartName: - description: ChartName is the chart name of the release object - in storage. - type: string - chartVersion: - description: |- - ChartVersion is the chart version of the release object in - storage. - type: string - configDigest: - description: |- - ConfigDigest is the checksum of the config (better known as - "values") of the release object in storage. - It has the format of `:`. - type: string - deleted: - description: Deleted is when the release was deleted. - format: date-time - type: string - digest: - description: |- - Digest is the checksum of the release object in storage. - It has the format of `:`. - type: string - firstDeployed: - description: FirstDeployed is when the release was first deployed. - format: date-time - type: string - lastDeployed: - description: LastDeployed is when the release was last deployed. - format: date-time - type: string - name: - description: Name is the name of the release. - type: string - namespace: - description: Namespace is the namespace the release is deployed - to. - type: string - ociDigest: - description: OCIDigest is the digest of the OCI artifact associated - with the release. - type: string - status: - description: Status is the current state of the release. - type: string - testHooks: - additionalProperties: - description: |- - TestHookStatus holds the status information for a test hook as observed - to be run by the controller. - properties: - lastCompleted: - description: LastCompleted is the time the test hook last - completed. - format: date-time - type: string - lastStarted: - description: LastStarted is the time the test hook was - last started. - format: date-time - type: string - phase: - description: Phase the test hook was observed to be in. - type: string - type: object - description: |- - TestHooks is the list of test hooks for the release as observed to be - run by the controller. - type: object - version: - description: Version is the version of the release object in - storage. - type: integer - required: - - chartName - - chartVersion - - configDigest - - digest - - firstDeployed - - lastDeployed - - name - - namespace - - status - - version - type: object - type: array - installFailures: - description: |- - InstallFailures is the install failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - lastAttemptedConfigDigest: - description: |- - LastAttemptedConfigDigest is the digest for the config (better known as - "values") of the last reconciliation attempt. - type: string - lastAttemptedGeneration: - description: |- - LastAttemptedGeneration is the last generation the controller attempted - to reconcile. - format: int64 - type: integer - lastAttemptedReleaseAction: - description: |- - LastAttemptedReleaseAction is the last release action performed for this - HelmRelease. It is used to determine the active remediation strategy. - enum: - - install - - upgrade - type: string - lastAttemptedRevision: - description: |- - LastAttemptedRevision is the Source revision of the last reconciliation - attempt. For OCIRepository sources, the 12 first characters of the digest are - appended to the chart version e.g. "1.2.3+1234567890ab". - type: string - lastAttemptedRevisionDigest: - description: |- - LastAttemptedRevisionDigest is the digest of the last reconciliation attempt. - This is only set for OCIRepository sources. - type: string - lastAttemptedValuesChecksum: - description: |- - LastAttemptedValuesChecksum is the SHA1 checksum for the values of the last - reconciliation attempt. - Deprecated: Use LastAttemptedConfigDigest instead. - type: string - lastHandledForceAt: - description: |- - LastHandledForceAt holds the value of the most recent force request - value, so a change of the annotation value can be detected. - type: string - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - lastHandledResetAt: - description: |- - LastHandledResetAt holds the value of the most recent reset request - value, so a change of the annotation value can be detected. - type: string - lastReleaseRevision: - description: |- - LastReleaseRevision is the revision of the last successful Helm release. - Deprecated: Use History instead. - type: integer - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - observedPostRenderersDigest: - description: |- - ObservedPostRenderersDigest is the digest for the post-renderers of - the last successful reconciliation attempt. - type: string - storageNamespace: - description: |- - StorageNamespace is the namespace of the Helm release storage for the - current release. - maxLength: 63 - minLength: 1 - type: string - upgradeFailures: - description: |- - UpgradeFailures is the upgrade failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - type: object - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v2beta1 HelmRelease is deprecated, upgrade to v2 - name: v2beta1 - schema: - openAPIV3Schema: - description: HelmRelease is the Schema for the helmreleases API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HelmReleaseSpec defines the desired state of a Helm release. - properties: - chart: - description: |- - Chart defines the template of the v1beta2.HelmChart that should be created - for this HelmRelease. - properties: - metadata: - description: ObjectMeta holds the template for metadata like labels - and annotations. - properties: - annotations: - additionalProperties: - type: string - description: |- - Annotations is an unstructured key value map stored with a resource that may be - set by external tools to store and retrieve arbitrary metadata. They are not - queryable and should be preserved when modifying objects. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ - type: object - labels: - additionalProperties: - type: string - description: |- - Map of string keys and values that can be used to organize and categorize - (scope and select) objects. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/ - type: object - type: object - spec: - description: Spec holds the template for the v1beta2.HelmChartSpec - for this HelmRelease. - properties: - chart: - description: The name or path the Helm chart is available - at in the SourceRef. - type: string - interval: - description: |- - Interval at which to check the v1beta2.Source for updates. Defaults to - 'HelmReleaseSpec.Interval'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - reconcileStrategy: - default: ChartVersion - description: |- - Determines what enables the creation of a new artifact. Valid values are - ('ChartVersion', 'Revision'). - See the documentation of the values for an explanation on their behavior. - Defaults to ChartVersion when omitted. - enum: - - ChartVersion - - Revision - type: string - sourceRef: - description: The name and namespace of the v1beta2.Source - the chart is available at. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - HelmRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: Namespace of the referent. - maxLength: 63 - minLength: 1 - type: string - required: - - name - type: object - valuesFile: - description: |- - Alternative values file to use as the default chart values, expected to - be a relative path in the SourceRef. Deprecated in favor of ValuesFiles, - for backwards compatibility the file defined here is merged before the - ValuesFiles items. Ignored when omitted. - type: string - valuesFiles: - description: |- - Alternative list of values files to use as the chart values (values.yaml - is not included by default), expected to be a relative path in the SourceRef. - Values files are merged in the order of this list with the last file overriding - the first. Ignored when omitted. - items: - type: string - type: array - verify: - description: |- - Verify contains the secret name containing the trusted public keys - used to verify the signature and specifies which provider to use to check - whether OCI image is authentic. - This field is only supported for OCI sources. - Chart dependencies, which are not bundled in the umbrella chart artifact, are not verified. - properties: - provider: - default: cosign - description: Provider specifies the technology used to - sign the OCI Helm chart. - enum: - - cosign - type: string - secretRef: - description: |- - SecretRef specifies the Kubernetes Secret containing the - trusted public keys. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - version: - default: '*' - description: |- - Version semver expression, ignored for charts from v1beta2.GitRepository and - v1beta2.Bucket sources. Defaults to latest when omitted. - type: string - required: - - chart - - sourceRef - type: object - required: - - spec - type: object - chartRef: - description: |- - ChartRef holds a reference to a source controller resource containing the - Helm chart artifact. - - - Note: this field is provisional to the v2 API, and not actively used - by v2beta1 HelmReleases. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - OCIRepository - - HelmChart - type: string - name: - description: Name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace of the referent, defaults to the namespace of the Kubernetes - resource object that contains the reference. - maxLength: 63 - minLength: 1 - type: string - required: - - kind - - name - type: object - dependsOn: - description: |- - DependsOn may contain a meta.NamespacedObjectReference slice with - references to HelmRelease resources that must be ready before this HelmRelease - can be reconciled. - items: - description: |- - NamespacedObjectReference contains enough information to locate the referenced Kubernetes resource object in any - namespace. - properties: - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - name - type: object - type: array - driftDetection: - description: |- - DriftDetection holds the configuration for detecting and handling - differences between the manifest in the Helm storage and the resources - currently existing in the cluster. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - properties: - ignore: - description: |- - Ignore contains a list of rules for specifying which changes to ignore - during diffing. - items: - description: |- - IgnoreRule defines a rule to selectively disregard specific changes during - the drift detection process. - properties: - paths: - description: |- - Paths is a list of JSON Pointer (RFC 6901) paths to be excluded from - consideration in a Kubernetes object. - items: - type: string - type: array - target: - description: |- - Target is a selector for specifying Kubernetes objects to which this - rule applies. - If Target is not set, the Paths will be ignored for all Kubernetes - objects within the manifest of the Helm release. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - paths - type: object - type: array - mode: - description: |- - Mode defines how differences should be handled between the Helm manifest - and the manifest currently applied to the cluster. - If not explicitly set, it defaults to DiffModeDisabled. - enum: - - enabled - - warn - - disabled - type: string - type: object - install: - description: Install holds the configuration for Helm install actions - for this HelmRelease. - properties: - crds: - description: |- - CRDs upgrade CRDs from the Helm Chart's crds directory according - to the CRD upgrade policy provided here. Valid values are `Skip`, - `Create` or `CreateReplace`. Default is `Create` and if omitted - CRDs are installed but not updated. - - - Skip: do neither install nor replace (update) any CRDs. - - - Create: new CRDs are created, existing CRDs are neither updated nor deleted. - - - CreateReplace: new CRDs are created, existing CRDs are updated (replaced) - but not deleted. - - - By default, CRDs are applied (installed) during Helm install action. - With this option users can opt-in to CRD replace existing CRDs on Helm - install actions, which is not (yet) natively supported by Helm. - https://helm.sh/docs/chart_best_practices/custom_resource_definitions. - enum: - - Skip - - Create - - CreateReplace - type: string - createNamespace: - description: |- - CreateNamespace tells the Helm install action to create the - HelmReleaseSpec.TargetNamespace if it does not exist yet. - On uninstall, the namespace will not be garbage collected. - type: boolean - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm install action. - type: boolean - disableOpenAPIValidation: - description: |- - DisableOpenAPIValidation prevents the Helm install action from validating - rendered templates against the Kubernetes OpenAPI Schema. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - install has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - install has been performed. - type: boolean - remediation: - description: |- - Remediation holds the remediation configuration for when the Helm install - action for the HelmRelease fails. The default is to not perform any action. - properties: - ignoreTestFailures: - description: |- - IgnoreTestFailures tells the controller to skip remediation when the Helm - tests are run after an install action but fail. Defaults to - 'Test.IgnoreFailures'. - type: boolean - remediateLastFailure: - description: |- - RemediateLastFailure tells the controller to remediate the last failure, when - no retries remain. Defaults to 'false'. - type: boolean - retries: - description: |- - Retries is the number of retries that should be attempted on failures before - bailing. Remediation, using an uninstall, is performed between each attempt. - Defaults to '0', a negative integer equals to unlimited retries. - type: integer - type: object - replace: - description: |- - Replace tells the Helm install action to re-use the 'ReleaseName', but only - if that name is a deleted release which remains in the history. - type: boolean - skipCRDs: - description: |- - SkipCRDs tells the Helm install action to not install any CRDs. By default, - CRDs are installed if not already present. - - - Deprecated use CRD policy (`crds`) attribute with value `Skip` instead. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm install action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - interval: - description: |- - Interval at which to reconcile the Helm release. - This interval is approximate and may be subject to jitter to ensure - efficient use of resources. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - kubeConfig: - description: |- - KubeConfig for reconciling the HelmRelease on a remote cluster. - When used in combination with HelmReleaseSpec.ServiceAccountName, - forces the controller to act on behalf of that Service Account at the - target cluster. - If the --default-service-account flag is set, its value will be used as - a controller level fallback for when HelmReleaseSpec.ServiceAccountName - is empty. - properties: - secretRef: - description: |- - SecretRef holds the name of a secret that contains a key with - the kubeconfig file as the value. If no key is set, the key will default - to 'value'. - It is recommended that the kubeconfig is self-contained, and the secret - is regularly updated if credentials such as a cloud-access-token expire. - Cloud specific `cmd-path` auth helpers will not function without adding - binaries and credentials to the Pod that is responsible for reconciling - Kubernetes resources. - properties: - key: - description: Key in the Secret, when not specified an implementation-specific - default key is used. - type: string - name: - description: Name of the Secret. - type: string - required: - - name - type: object - required: - - secretRef - type: object - maxHistory: - description: |- - MaxHistory is the number of revisions saved by Helm for this HelmRelease. - Use '0' for an unlimited number of revisions; defaults to '10'. - type: integer - persistentClient: - description: |- - PersistentClient tells the controller to use a persistent Kubernetes - client for this release. When enabled, the client will be reused for the - duration of the reconciliation, instead of being created and destroyed - for each (step of a) Helm action. - - - This can improve performance, but may cause issues with some Helm charts - that for example do create Custom Resource Definitions during installation - outside Helm's CRD lifecycle hooks, which are then not observed to be - available by e.g. post-install hooks. - - - If not set, it defaults to true. - type: boolean - postRenderers: - description: |- - PostRenderers holds an array of Helm PostRenderers, which will be applied in order - of their definition. - items: - description: PostRenderer contains a Helm PostRenderer specification. - properties: - kustomize: - description: Kustomization to apply as PostRenderer. - properties: - images: - description: |- - Images is a list of (image name, new name, new tag or digest) - for changing image names, tags or digests. This can also be achieved with a - patch, but this operator is simpler to specify. - items: - description: Image contains an image name, a new name, - a new tag or digest, which will replace the original - name and tag. - properties: - digest: - description: |- - Digest is the value used to replace the original image tag. - If digest is present NewTag value is ignored. - type: string - name: - description: Name is a tag-less image name. - type: string - newName: - description: NewName is the value used to replace - the original name. - type: string - newTag: - description: NewTag is the value used to replace the - original tag. - type: string - required: - - name - type: object - type: array - patches: - description: |- - Strategic merge and JSON patches, defined as inline YAML objects, - capable of targeting objects based on kind, label and annotation selectors. - items: - description: |- - Patch contains an inline StrategicMerge or JSON6902 patch, and the target the patch should - be applied to. - properties: - patch: - description: |- - Patch contains an inline StrategicMerge patch or an inline JSON6902 patch with - an array of operation objects. - type: string - target: - description: Target points to the resources that the - patch document should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - type: object - type: array - patchesJson6902: - description: JSON 6902 patches, defined as inline YAML objects. - items: - description: JSON6902Patch contains a JSON6902 patch and - the target the patch should be applied to. - properties: - patch: - description: Patch contains the JSON6902 patch document - with an array of operation objects. - items: - description: |- - JSON6902 is a JSON6902 operation object. - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - properties: - from: - description: |- - From contains a JSON-pointer value that references a location within the target document where the operation is - performed. The meaning of the value depends on the value of Op, and is NOT taken into account by all operations. - type: string - op: - description: |- - Op indicates the operation to perform. Its value MUST be one of "add", "remove", "replace", "move", "copy", or - "test". - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - enum: - - test - - remove - - add - - replace - - move - - copy - type: string - path: - description: |- - Path contains the JSON-pointer value that references a location within the target document where the operation - is performed. The meaning of the value depends on the value of Op. - type: string - value: - description: |- - Value contains a valid JSON structure. The meaning of the value depends on the value of Op, and is NOT taken into - account by all operations. - x-kubernetes-preserve-unknown-fields: true - required: - - op - - path - type: object - type: array - target: - description: Target points to the resources that the - patch document should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - - target - type: object - type: array - patchesStrategicMerge: - description: Strategic merge patches, defined as inline - YAML objects. - items: - x-kubernetes-preserve-unknown-fields: true - type: array - type: object - type: object - type: array - releaseName: - description: |- - ReleaseName used for the Helm release. Defaults to a composition of - '[TargetNamespace-]Name'. - maxLength: 53 - minLength: 1 - type: string - rollback: - description: Rollback holds the configuration for Helm rollback actions - for this HelmRelease. - properties: - cleanupOnFail: - description: |- - CleanupOnFail allows deletion of new resources created during the Helm - rollback action when it fails. - type: boolean - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm rollback action. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - rollback has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - rollback has been performed. - type: boolean - force: - description: Force forces resource updates through a replacement - strategy. - type: boolean - recreate: - description: Recreate performs pod restarts for the resource if - applicable. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm rollback action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - serviceAccountName: - description: |- - The name of the Kubernetes service account to impersonate - when reconciling this HelmRelease. - type: string - storageNamespace: - description: |- - StorageNamespace used for the Helm storage. - Defaults to the namespace of the HelmRelease. - maxLength: 63 - minLength: 1 - type: string - suspend: - description: |- - Suspend tells the controller to suspend reconciliation for this HelmRelease, - it does not apply to already started reconciliations. Defaults to false. - type: boolean - targetNamespace: - description: |- - TargetNamespace to target when performing operations for the HelmRelease. - Defaults to the namespace of the HelmRelease. - maxLength: 63 - minLength: 1 - type: string - test: - description: Test holds the configuration for Helm test actions for - this HelmRelease. - properties: - enable: - description: |- - Enable enables Helm test actions for this HelmRelease after an Helm install - or upgrade action has been performed. - type: boolean - ignoreFailures: - description: |- - IgnoreFailures tells the controller to skip remediation when the Helm tests - are run but fail. Can be overwritten for tests run after install or upgrade - actions in 'Install.IgnoreTestFailures' and 'Upgrade.IgnoreTestFailures'. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation during - the performance of a Helm test action. Defaults to 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like Jobs - for hooks) during the performance of a Helm action. Defaults to '5m0s'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - uninstall: - description: Uninstall holds the configuration for Helm uninstall - actions for this HelmRelease. - properties: - deletionPropagation: - default: background - description: |- - DeletionPropagation specifies the deletion propagation policy when - a Helm uninstall is performed. - enum: - - background - - foreground - - orphan - type: string - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm rollback action. - type: boolean - disableWait: - description: |- - DisableWait disables waiting for all the resources to be deleted after - a Helm uninstall is performed. - type: boolean - keepHistory: - description: |- - KeepHistory tells Helm to remove all associated resources and mark the - release as deleted, but retain the release history. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm uninstall action. Defaults - to 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - upgrade: - description: Upgrade holds the configuration for Helm upgrade actions - for this HelmRelease. - properties: - cleanupOnFail: - description: |- - CleanupOnFail allows deletion of new resources created during the Helm - upgrade action when it fails. - type: boolean - crds: - description: |- - CRDs upgrade CRDs from the Helm Chart's crds directory according - to the CRD upgrade policy provided here. Valid values are `Skip`, - `Create` or `CreateReplace`. Default is `Skip` and if omitted - CRDs are neither installed nor upgraded. - - - Skip: do neither install nor replace (update) any CRDs. - - - Create: new CRDs are created, existing CRDs are neither updated nor deleted. - - - CreateReplace: new CRDs are created, existing CRDs are updated (replaced) - but not deleted. - - - By default, CRDs are not applied during Helm upgrade action. With this - option users can opt-in to CRD upgrade, which is not (yet) natively supported by Helm. - https://helm.sh/docs/chart_best_practices/custom_resource_definitions. - enum: - - Skip - - Create - - CreateReplace - type: string - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm upgrade action. - type: boolean - disableOpenAPIValidation: - description: |- - DisableOpenAPIValidation prevents the Helm upgrade action from validating - rendered templates against the Kubernetes OpenAPI Schema. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - upgrade has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - upgrade has been performed. - type: boolean - force: - description: Force forces resource updates through a replacement - strategy. - type: boolean - preserveValues: - description: |- - PreserveValues will make Helm reuse the last release's values and merge in - overrides from 'Values'. Setting this flag makes the HelmRelease - non-declarative. - type: boolean - remediation: - description: |- - Remediation holds the remediation configuration for when the Helm upgrade - action for the HelmRelease fails. The default is to not perform any action. - properties: - ignoreTestFailures: - description: |- - IgnoreTestFailures tells the controller to skip remediation when the Helm - tests are run after an upgrade action but fail. - Defaults to 'Test.IgnoreFailures'. - type: boolean - remediateLastFailure: - description: |- - RemediateLastFailure tells the controller to remediate the last failure, when - no retries remain. Defaults to 'false' unless 'Retries' is greater than 0. - type: boolean - retries: - description: |- - Retries is the number of retries that should be attempted on failures before - bailing. Remediation, using 'Strategy', is performed between each attempt. - Defaults to '0', a negative integer equals to unlimited retries. - type: integer - strategy: - description: Strategy to use for failure remediation. Defaults - to 'rollback'. - enum: - - rollback - - uninstall - type: string - type: object - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm upgrade action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - values: - description: Values holds the values for this Helm release. - x-kubernetes-preserve-unknown-fields: true - valuesFrom: - description: |- - ValuesFrom holds references to resources containing Helm values for this HelmRelease, - and information about how they should be merged. - items: - description: |- - ValuesReference contains a reference to a resource containing Helm values, - and optionally the key they can be found at. - properties: - kind: - description: Kind of the values referent, valid values are ('Secret', - 'ConfigMap'). - enum: - - Secret - - ConfigMap - type: string - name: - description: |- - Name of the values referent. Should reside in the same namespace as the - referring resource. - maxLength: 253 - minLength: 1 - type: string - optional: - description: |- - Optional marks this ValuesReference as optional. When set, a not found error - for the values reference is ignored, but any ValuesKey, TargetPath or - transient error will still result in a reconciliation failure. - type: boolean - targetPath: - description: |- - TargetPath is the YAML dot notation path the value should be merged at. When - set, the ValuesKey is expected to be a single flat value. Defaults to 'None', - which results in the values getting merged at the root. - maxLength: 250 - pattern: ^([a-zA-Z0-9_\-.\\\/]|\[[0-9]{1,5}\])+$ - type: string - valuesKey: - description: |- - ValuesKey is the data key where the values.yaml or a specific value can be - found at. Defaults to 'values.yaml'. - When set, must be a valid Data Key, consisting of alphanumeric characters, - '-', '_' or '.'. - maxLength: 253 - pattern: ^[\-._a-zA-Z0-9]+$ - type: string - required: - - kind - - name - type: object - type: array - required: - - interval - type: object - status: - default: - observedGeneration: -1 - description: HelmReleaseStatus defines the observed state of a HelmRelease. - properties: - conditions: - description: Conditions holds the conditions for the HelmRelease. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - failures: - description: |- - Failures is the reconciliation failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - helmChart: - description: |- - HelmChart is the namespaced name of the HelmChart resource created by - the controller for the HelmRelease. - type: string - history: - description: |- - History holds the history of Helm releases performed for this HelmRelease - up to the last successfully completed release. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - items: - description: |- - Snapshot captures a point-in-time copy of the status information for a Helm release, - as managed by the controller. - properties: - apiVersion: - description: |- - APIVersion is the API version of the Snapshot. - Provisional: when the calculation method of the Digest field is changed, - this field will be used to distinguish between the old and new methods. - type: string - appVersion: - description: AppVersion is the chart app version of the release - object in storage. - type: string - chartName: - description: ChartName is the chart name of the release object - in storage. - type: string - chartVersion: - description: |- - ChartVersion is the chart version of the release object in - storage. - type: string - configDigest: - description: |- - ConfigDigest is the checksum of the config (better known as - "values") of the release object in storage. - It has the format of `:`. - type: string - deleted: - description: Deleted is when the release was deleted. - format: date-time - type: string - digest: - description: |- - Digest is the checksum of the release object in storage. - It has the format of `:`. - type: string - firstDeployed: - description: FirstDeployed is when the release was first deployed. - format: date-time - type: string - lastDeployed: - description: LastDeployed is when the release was last deployed. - format: date-time - type: string - name: - description: Name is the name of the release. - type: string - namespace: - description: Namespace is the namespace the release is deployed - to. - type: string - ociDigest: - description: OCIDigest is the digest of the OCI artifact associated - with the release. - type: string - status: - description: Status is the current state of the release. - type: string - testHooks: - additionalProperties: - description: |- - TestHookStatus holds the status information for a test hook as observed - to be run by the controller. - properties: - lastCompleted: - description: LastCompleted is the time the test hook last - completed. - format: date-time - type: string - lastStarted: - description: LastStarted is the time the test hook was - last started. - format: date-time - type: string - phase: - description: Phase the test hook was observed to be in. - type: string - type: object - description: |- - TestHooks is the list of test hooks for the release as observed to be - run by the controller. - type: object - version: - description: Version is the version of the release object in - storage. - type: integer - required: - - chartName - - chartVersion - - configDigest - - digest - - firstDeployed - - lastDeployed - - name - - namespace - - status - - version - type: object - type: array - installFailures: - description: |- - InstallFailures is the install failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - lastAppliedRevision: - description: LastAppliedRevision is the revision of the last successfully - applied source. - type: string - lastAttemptedConfigDigest: - description: |- - LastAttemptedConfigDigest is the digest for the config (better known as - "values") of the last reconciliation attempt. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - type: string - lastAttemptedGeneration: - description: |- - LastAttemptedGeneration is the last generation the controller attempted - to reconcile. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - format: int64 - type: integer - lastAttemptedReleaseAction: - description: |- - LastAttemptedReleaseAction is the last release action performed for this - HelmRelease. It is used to determine the active remediation strategy. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - type: string - lastAttemptedRevision: - description: LastAttemptedRevision is the revision of the last reconciliation - attempt. - type: string - lastAttemptedValuesChecksum: - description: |- - LastAttemptedValuesChecksum is the SHA1 checksum of the values of the last - reconciliation attempt. - type: string - lastHandledForceAt: - description: |- - LastHandledForceAt holds the value of the most recent force request - value, so a change of the annotation value can be detected. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - type: string - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - lastHandledResetAt: - description: |- - LastHandledResetAt holds the value of the most recent reset request - value, so a change of the annotation value can be detected. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - type: string - lastReleaseRevision: - description: LastReleaseRevision is the revision of the last successful - Helm release. - type: integer - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - observedPostRenderersDigest: - description: |- - ObservedPostRenderersDigest is the digest for the post-renderers of - the last successful reconciliation attempt. - type: string - storageNamespace: - description: |- - StorageNamespace is the namespace of the Helm release storage for the - current release. - - - Note: this field is provisional to the v2beta2 API, and not actively used - by v2beta1 HelmReleases. - type: string - upgradeFailures: - description: |- - UpgradeFailures is the upgrade failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v2beta2 HelmRelease is deprecated, upgrade to v2 - name: v2beta2 - schema: - openAPIV3Schema: - description: HelmRelease is the Schema for the helmreleases API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: HelmReleaseSpec defines the desired state of a Helm release. - properties: - chart: - description: |- - Chart defines the template of the v1beta2.HelmChart that should be created - for this HelmRelease. - properties: - metadata: - description: ObjectMeta holds the template for metadata like labels - and annotations. - properties: - annotations: - additionalProperties: - type: string - description: |- - Annotations is an unstructured key value map stored with a resource that may be - set by external tools to store and retrieve arbitrary metadata. They are not - queryable and should be preserved when modifying objects. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ - type: object - labels: - additionalProperties: - type: string - description: |- - Map of string keys and values that can be used to organize and categorize - (scope and select) objects. - More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/ - type: object - type: object - spec: - description: Spec holds the template for the v1beta2.HelmChartSpec - for this HelmRelease. - properties: - chart: - description: The name or path the Helm chart is available - at in the SourceRef. - maxLength: 2048 - minLength: 1 - type: string - ignoreMissingValuesFiles: - description: IgnoreMissingValuesFiles controls whether to - silently ignore missing values files rather than failing. - type: boolean - interval: - description: |- - Interval at which to check the v1.Source for updates. Defaults to - 'HelmReleaseSpec.Interval'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - reconcileStrategy: - default: ChartVersion - description: |- - Determines what enables the creation of a new artifact. Valid values are - ('ChartVersion', 'Revision'). - See the documentation of the values for an explanation on their behavior. - Defaults to ChartVersion when omitted. - enum: - - ChartVersion - - Revision - type: string - sourceRef: - description: The name and namespace of the v1.Source the chart - is available at. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - HelmRepository - - GitRepository - - Bucket - type: string - name: - description: Name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: Namespace of the referent. - maxLength: 63 - minLength: 1 - type: string - required: - - name - type: object - valuesFile: - description: |- - Alternative values file to use as the default chart values, expected to - be a relative path in the SourceRef. Deprecated in favor of ValuesFiles, - for backwards compatibility the file defined here is merged before the - ValuesFiles items. Ignored when omitted. - type: string - valuesFiles: - description: |- - Alternative list of values files to use as the chart values (values.yaml - is not included by default), expected to be a relative path in the SourceRef. - Values files are merged in the order of this list with the last file overriding - the first. Ignored when omitted. - items: - type: string - type: array - verify: - description: |- - Verify contains the secret name containing the trusted public keys - used to verify the signature and specifies which provider to use to check - whether OCI image is authentic. - This field is only supported for OCI sources. - Chart dependencies, which are not bundled in the umbrella chart artifact, - are not verified. - properties: - provider: - default: cosign - description: Provider specifies the technology used to - sign the OCI Helm chart. - enum: - - cosign - - notation - type: string - secretRef: - description: |- - SecretRef specifies the Kubernetes Secret containing the - trusted public keys. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - required: - - provider - type: object - version: - default: '*' - description: |- - Version semver expression, ignored for charts from v1beta2.GitRepository and - v1beta2.Bucket sources. Defaults to latest when omitted. - type: string - required: - - chart - - sourceRef - type: object - required: - - spec - type: object - chartRef: - description: |- - ChartRef holds a reference to a source controller resource containing the - Helm chart artifact. - - - Note: this field is provisional to the v2 API, and not actively used - by v2beta2 HelmReleases. - properties: - apiVersion: - description: APIVersion of the referent. - type: string - kind: - description: Kind of the referent. - enum: - - OCIRepository - - HelmChart - type: string - name: - description: Name of the referent. - maxLength: 253 - minLength: 1 - type: string - namespace: - description: |- - Namespace of the referent, defaults to the namespace of the Kubernetes - resource object that contains the reference. - maxLength: 63 - minLength: 1 - type: string - required: - - kind - - name - type: object - dependsOn: - description: |- - DependsOn may contain a meta.NamespacedObjectReference slice with - references to HelmRelease resources that must be ready before this HelmRelease - can be reconciled. - items: - description: |- - NamespacedObjectReference contains enough information to locate the referenced Kubernetes resource object in any - namespace. - properties: - name: - description: Name of the referent. - type: string - namespace: - description: Namespace of the referent, when not specified it - acts as LocalObjectReference. - type: string - required: - - name - type: object - type: array - driftDetection: - description: |- - DriftDetection holds the configuration for detecting and handling - differences between the manifest in the Helm storage and the resources - currently existing in the cluster. - properties: - ignore: - description: |- - Ignore contains a list of rules for specifying which changes to ignore - during diffing. - items: - description: |- - IgnoreRule defines a rule to selectively disregard specific changes during - the drift detection process. - properties: - paths: - description: |- - Paths is a list of JSON Pointer (RFC 6901) paths to be excluded from - consideration in a Kubernetes object. - items: - type: string - type: array - target: - description: |- - Target is a selector for specifying Kubernetes objects to which this - rule applies. - If Target is not set, the Paths will be ignored for all Kubernetes - objects within the manifest of the Helm release. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - paths - type: object - type: array - mode: - description: |- - Mode defines how differences should be handled between the Helm manifest - and the manifest currently applied to the cluster. - If not explicitly set, it defaults to DiffModeDisabled. - enum: - - enabled - - warn - - disabled - type: string - type: object - install: - description: Install holds the configuration for Helm install actions - for this HelmRelease. - properties: - crds: - description: |- - CRDs upgrade CRDs from the Helm Chart's crds directory according - to the CRD upgrade policy provided here. Valid values are `Skip`, - `Create` or `CreateReplace`. Default is `Create` and if omitted - CRDs are installed but not updated. - - - Skip: do neither install nor replace (update) any CRDs. - - - Create: new CRDs are created, existing CRDs are neither updated nor deleted. - - - CreateReplace: new CRDs are created, existing CRDs are updated (replaced) - but not deleted. - - - By default, CRDs are applied (installed) during Helm install action. - With this option users can opt in to CRD replace existing CRDs on Helm - install actions, which is not (yet) natively supported by Helm. - https://helm.sh/docs/chart_best_practices/custom_resource_definitions. - enum: - - Skip - - Create - - CreateReplace - type: string - createNamespace: - description: |- - CreateNamespace tells the Helm install action to create the - HelmReleaseSpec.TargetNamespace if it does not exist yet. - On uninstall, the namespace will not be garbage collected. - type: boolean - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm install action. - type: boolean - disableOpenAPIValidation: - description: |- - DisableOpenAPIValidation prevents the Helm install action from validating - rendered templates against the Kubernetes OpenAPI Schema. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - install has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - install has been performed. - type: boolean - remediation: - description: |- - Remediation holds the remediation configuration for when the Helm install - action for the HelmRelease fails. The default is to not perform any action. - properties: - ignoreTestFailures: - description: |- - IgnoreTestFailures tells the controller to skip remediation when the Helm - tests are run after an install action but fail. Defaults to - 'Test.IgnoreFailures'. - type: boolean - remediateLastFailure: - description: |- - RemediateLastFailure tells the controller to remediate the last failure, when - no retries remain. Defaults to 'false'. - type: boolean - retries: - description: |- - Retries is the number of retries that should be attempted on failures before - bailing. Remediation, using an uninstall, is performed between each attempt. - Defaults to '0', a negative integer equals to unlimited retries. - type: integer - type: object - replace: - description: |- - Replace tells the Helm install action to re-use the 'ReleaseName', but only - if that name is a deleted release which remains in the history. - type: boolean - skipCRDs: - description: |- - SkipCRDs tells the Helm install action to not install any CRDs. By default, - CRDs are installed if not already present. - - - Deprecated use CRD policy (`crds`) attribute with value `Skip` instead. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm install action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - interval: - description: Interval at which to reconcile the Helm release. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - kubeConfig: - description: |- - KubeConfig for reconciling the HelmRelease on a remote cluster. - When used in combination with HelmReleaseSpec.ServiceAccountName, - forces the controller to act on behalf of that Service Account at the - target cluster. - If the --default-service-account flag is set, its value will be used as - a controller level fallback for when HelmReleaseSpec.ServiceAccountName - is empty. - properties: - secretRef: - description: |- - SecretRef holds the name of a secret that contains a key with - the kubeconfig file as the value. If no key is set, the key will default - to 'value'. - It is recommended that the kubeconfig is self-contained, and the secret - is regularly updated if credentials such as a cloud-access-token expire. - Cloud specific `cmd-path` auth helpers will not function without adding - binaries and credentials to the Pod that is responsible for reconciling - Kubernetes resources. - properties: - key: - description: Key in the Secret, when not specified an implementation-specific - default key is used. - type: string - name: - description: Name of the Secret. - type: string - required: - - name - type: object - required: - - secretRef - type: object - maxHistory: - description: |- - MaxHistory is the number of revisions saved by Helm for this HelmRelease. - Use '0' for an unlimited number of revisions; defaults to '5'. - type: integer - persistentClient: - description: |- - PersistentClient tells the controller to use a persistent Kubernetes - client for this release. When enabled, the client will be reused for the - duration of the reconciliation, instead of being created and destroyed - for each (step of a) Helm action. - - - This can improve performance, but may cause issues with some Helm charts - that for example do create Custom Resource Definitions during installation - outside Helm's CRD lifecycle hooks, which are then not observed to be - available by e.g. post-install hooks. - - - If not set, it defaults to true. - type: boolean - postRenderers: - description: |- - PostRenderers holds an array of Helm PostRenderers, which will be applied in order - of their definition. - items: - description: PostRenderer contains a Helm PostRenderer specification. - properties: - kustomize: - description: Kustomization to apply as PostRenderer. - properties: - images: - description: |- - Images is a list of (image name, new name, new tag or digest) - for changing image names, tags or digests. This can also be achieved with a - patch, but this operator is simpler to specify. - items: - description: Image contains an image name, a new name, - a new tag or digest, which will replace the original - name and tag. - properties: - digest: - description: |- - Digest is the value used to replace the original image tag. - If digest is present NewTag value is ignored. - type: string - name: - description: Name is a tag-less image name. - type: string - newName: - description: NewName is the value used to replace - the original name. - type: string - newTag: - description: NewTag is the value used to replace the - original tag. - type: string - required: - - name - type: object - type: array - patches: - description: |- - Strategic merge and JSON patches, defined as inline YAML objects, - capable of targeting objects based on kind, label and annotation selectors. - items: - description: |- - Patch contains an inline StrategicMerge or JSON6902 patch, and the target the patch should - be applied to. - properties: - patch: - description: |- - Patch contains an inline StrategicMerge patch or an inline JSON6902 patch with - an array of operation objects. - type: string - target: - description: Target points to the resources that the - patch document should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - type: object - type: array - patchesJson6902: - description: |- - JSON 6902 patches, defined as inline YAML objects. - Deprecated: use Patches instead. - items: - description: JSON6902Patch contains a JSON6902 patch and - the target the patch should be applied to. - properties: - patch: - description: Patch contains the JSON6902 patch document - with an array of operation objects. - items: - description: |- - JSON6902 is a JSON6902 operation object. - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - properties: - from: - description: |- - From contains a JSON-pointer value that references a location within the target document where the operation is - performed. The meaning of the value depends on the value of Op, and is NOT taken into account by all operations. - type: string - op: - description: |- - Op indicates the operation to perform. Its value MUST be one of "add", "remove", "replace", "move", "copy", or - "test". - https://datatracker.ietf.org/doc/html/rfc6902#section-4 - enum: - - test - - remove - - add - - replace - - move - - copy - type: string - path: - description: |- - Path contains the JSON-pointer value that references a location within the target document where the operation - is performed. The meaning of the value depends on the value of Op. - type: string - value: - description: |- - Value contains a valid JSON structure. The meaning of the value depends on the value of Op, and is NOT taken into - account by all operations. - x-kubernetes-preserve-unknown-fields: true - required: - - op - - path - type: object - type: array - target: - description: Target points to the resources that the - patch document should be applied to. - properties: - annotationSelector: - description: |- - AnnotationSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource annotations. - type: string - group: - description: |- - Group is the API group to select resources from. - Together with Version and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - kind: - description: |- - Kind of the API Group to select resources from. - Together with Group and Version it is capable of unambiguously - identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - labelSelector: - description: |- - LabelSelector is a string that follows the label selection expression - https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#api - It matches with the resource labels. - type: string - name: - description: Name to match resources with. - type: string - namespace: - description: Namespace to select resources from. - type: string - version: - description: |- - Version of the API Group to select resources from. - Together with Group and Kind it is capable of unambiguously identifying and/or selecting resources. - https://github.com/kubernetes/community/blob/master/contributors/design-proposals/api-machinery/api-group.md - type: string - type: object - required: - - patch - - target - type: object - type: array - patchesStrategicMerge: - description: |- - Strategic merge patches, defined as inline YAML objects. - Deprecated: use Patches instead. - items: - x-kubernetes-preserve-unknown-fields: true - type: array - type: object - type: object - type: array - releaseName: - description: |- - ReleaseName used for the Helm release. Defaults to a composition of - '[TargetNamespace-]Name'. - maxLength: 53 - minLength: 1 - type: string - rollback: - description: Rollback holds the configuration for Helm rollback actions - for this HelmRelease. - properties: - cleanupOnFail: - description: |- - CleanupOnFail allows deletion of new resources created during the Helm - rollback action when it fails. - type: boolean - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm rollback action. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - rollback has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - rollback has been performed. - type: boolean - force: - description: Force forces resource updates through a replacement - strategy. - type: boolean - recreate: - description: Recreate performs pod restarts for the resource if - applicable. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm rollback action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - serviceAccountName: - description: |- - The name of the Kubernetes service account to impersonate - when reconciling this HelmRelease. - maxLength: 253 - minLength: 1 - type: string - storageNamespace: - description: |- - StorageNamespace used for the Helm storage. - Defaults to the namespace of the HelmRelease. - maxLength: 63 - minLength: 1 - type: string - suspend: - description: |- - Suspend tells the controller to suspend reconciliation for this HelmRelease, - it does not apply to already started reconciliations. Defaults to false. - type: boolean - targetNamespace: - description: |- - TargetNamespace to target when performing operations for the HelmRelease. - Defaults to the namespace of the HelmRelease. - maxLength: 63 - minLength: 1 - type: string - test: - description: Test holds the configuration for Helm test actions for - this HelmRelease. - properties: - enable: - description: |- - Enable enables Helm test actions for this HelmRelease after an Helm install - or upgrade action has been performed. - type: boolean - filters: - description: Filters is a list of tests to run or exclude from - running. - items: - description: Filter holds the configuration for individual Helm - test filters. - properties: - exclude: - description: Exclude specifies whether the named test should - be excluded. - type: boolean - name: - description: Name is the name of the test. - maxLength: 253 - minLength: 1 - type: string - required: - - name - type: object - type: array - ignoreFailures: - description: |- - IgnoreFailures tells the controller to skip remediation when the Helm tests - are run but fail. Can be overwritten for tests run after install or upgrade - actions in 'Install.IgnoreTestFailures' and 'Upgrade.IgnoreTestFailures'. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation during - the performance of a Helm test action. Defaults to 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like Jobs - for hooks) during the performance of a Helm action. Defaults to '5m0s'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - uninstall: - description: Uninstall holds the configuration for Helm uninstall - actions for this HelmRelease. - properties: - deletionPropagation: - default: background - description: |- - DeletionPropagation specifies the deletion propagation policy when - a Helm uninstall is performed. - enum: - - background - - foreground - - orphan - type: string - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm rollback action. - type: boolean - disableWait: - description: |- - DisableWait disables waiting for all the resources to be deleted after - a Helm uninstall is performed. - type: boolean - keepHistory: - description: |- - KeepHistory tells Helm to remove all associated resources and mark the - release as deleted, but retain the release history. - type: boolean - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm uninstall action. Defaults - to 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - upgrade: - description: Upgrade holds the configuration for Helm upgrade actions - for this HelmRelease. - properties: - cleanupOnFail: - description: |- - CleanupOnFail allows deletion of new resources created during the Helm - upgrade action when it fails. - type: boolean - crds: - description: |- - CRDs upgrade CRDs from the Helm Chart's crds directory according - to the CRD upgrade policy provided here. Valid values are `Skip`, - `Create` or `CreateReplace`. Default is `Skip` and if omitted - CRDs are neither installed nor upgraded. - - - Skip: do neither install nor replace (update) any CRDs. - - - Create: new CRDs are created, existing CRDs are neither updated nor deleted. - - - CreateReplace: new CRDs are created, existing CRDs are updated (replaced) - but not deleted. - - - By default, CRDs are not applied during Helm upgrade action. With this - option users can opt-in to CRD upgrade, which is not (yet) natively supported by Helm. - https://helm.sh/docs/chart_best_practices/custom_resource_definitions. - enum: - - Skip - - Create - - CreateReplace - type: string - disableHooks: - description: DisableHooks prevents hooks from running during the - Helm upgrade action. - type: boolean - disableOpenAPIValidation: - description: |- - DisableOpenAPIValidation prevents the Helm upgrade action from validating - rendered templates against the Kubernetes OpenAPI Schema. - type: boolean - disableWait: - description: |- - DisableWait disables the waiting for resources to be ready after a Helm - upgrade has been performed. - type: boolean - disableWaitForJobs: - description: |- - DisableWaitForJobs disables waiting for jobs to complete after a Helm - upgrade has been performed. - type: boolean - force: - description: Force forces resource updates through a replacement - strategy. - type: boolean - preserveValues: - description: |- - PreserveValues will make Helm reuse the last release's values and merge in - overrides from 'Values'. Setting this flag makes the HelmRelease - non-declarative. - type: boolean - remediation: - description: |- - Remediation holds the remediation configuration for when the Helm upgrade - action for the HelmRelease fails. The default is to not perform any action. - properties: - ignoreTestFailures: - description: |- - IgnoreTestFailures tells the controller to skip remediation when the Helm - tests are run after an upgrade action but fail. - Defaults to 'Test.IgnoreFailures'. - type: boolean - remediateLastFailure: - description: |- - RemediateLastFailure tells the controller to remediate the last failure, when - no retries remain. Defaults to 'false' unless 'Retries' is greater than 0. - type: boolean - retries: - description: |- - Retries is the number of retries that should be attempted on failures before - bailing. Remediation, using 'Strategy', is performed between each attempt. - Defaults to '0', a negative integer equals to unlimited retries. - type: integer - strategy: - description: Strategy to use for failure remediation. Defaults - to 'rollback'. - enum: - - rollback - - uninstall - type: string - type: object - timeout: - description: |- - Timeout is the time to wait for any individual Kubernetes operation (like - Jobs for hooks) during the performance of a Helm upgrade action. Defaults to - 'HelmReleaseSpec.Timeout'. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - type: object - values: - description: Values holds the values for this Helm release. - x-kubernetes-preserve-unknown-fields: true - valuesFrom: - description: |- - ValuesFrom holds references to resources containing Helm values for this HelmRelease, - and information about how they should be merged. - items: - description: |- - ValuesReference contains a reference to a resource containing Helm values, - and optionally the key they can be found at. - properties: - kind: - description: Kind of the values referent, valid values are ('Secret', - 'ConfigMap'). - enum: - - Secret - - ConfigMap - type: string - name: - description: |- - Name of the values referent. Should reside in the same namespace as the - referring resource. - maxLength: 253 - minLength: 1 - type: string - optional: - description: |- - Optional marks this ValuesReference as optional. When set, a not found error - for the values reference is ignored, but any ValuesKey, TargetPath or - transient error will still result in a reconciliation failure. - type: boolean - targetPath: - description: |- - TargetPath is the YAML dot notation path the value should be merged at. When - set, the ValuesKey is expected to be a single flat value. Defaults to 'None', - which results in the values getting merged at the root. - maxLength: 250 - pattern: ^([a-zA-Z0-9_\-.\\\/]|\[[0-9]{1,5}\])+$ - type: string - valuesKey: - description: |- - ValuesKey is the data key where the values.yaml or a specific value can be - found at. Defaults to 'values.yaml'. - maxLength: 253 - pattern: ^[\-._a-zA-Z0-9]+$ - type: string - required: - - kind - - name - type: object - type: array - required: - - interval - type: object - x-kubernetes-validations: - - message: either chart or chartRef must be set - rule: (has(self.chart) && !has(self.chartRef)) || (!has(self.chart) - && has(self.chartRef)) - status: - default: - observedGeneration: -1 - description: HelmReleaseStatus defines the observed state of a HelmRelease. - properties: - conditions: - description: Conditions holds the conditions for the HelmRelease. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - failures: - description: |- - Failures is the reconciliation failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - helmChart: - description: |- - HelmChart is the namespaced name of the HelmChart resource created by - the controller for the HelmRelease. - type: string - history: - description: |- - History holds the history of Helm releases performed for this HelmRelease - up to the last successfully completed release. - items: - description: |- - Snapshot captures a point-in-time copy of the status information for a Helm release, - as managed by the controller. - properties: - apiVersion: - description: |- - APIVersion is the API version of the Snapshot. - Provisional: when the calculation method of the Digest field is changed, - this field will be used to distinguish between the old and new methods. - type: string - appVersion: - description: AppVersion is the chart app version of the release - object in storage. - type: string - chartName: - description: ChartName is the chart name of the release object - in storage. - type: string - chartVersion: - description: |- - ChartVersion is the chart version of the release object in - storage. - type: string - configDigest: - description: |- - ConfigDigest is the checksum of the config (better known as - "values") of the release object in storage. - It has the format of `:`. - type: string - deleted: - description: Deleted is when the release was deleted. - format: date-time - type: string - digest: - description: |- - Digest is the checksum of the release object in storage. - It has the format of `:`. - type: string - firstDeployed: - description: FirstDeployed is when the release was first deployed. - format: date-time - type: string - lastDeployed: - description: LastDeployed is when the release was last deployed. - format: date-time - type: string - name: - description: Name is the name of the release. - type: string - namespace: - description: Namespace is the namespace the release is deployed - to. - type: string - ociDigest: - description: OCIDigest is the digest of the OCI artifact associated - with the release. - type: string - status: - description: Status is the current state of the release. - type: string - testHooks: - additionalProperties: - description: |- - TestHookStatus holds the status information for a test hook as observed - to be run by the controller. - properties: - lastCompleted: - description: LastCompleted is the time the test hook last - completed. - format: date-time - type: string - lastStarted: - description: LastStarted is the time the test hook was - last started. - format: date-time - type: string - phase: - description: Phase the test hook was observed to be in. - type: string - type: object - description: |- - TestHooks is the list of test hooks for the release as observed to be - run by the controller. - type: object - version: - description: Version is the version of the release object in - storage. - type: integer - required: - - chartName - - chartVersion - - configDigest - - digest - - firstDeployed - - lastDeployed - - name - - namespace - - status - - version - type: object - type: array - installFailures: - description: |- - InstallFailures is the install failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - lastAppliedRevision: - description: |- - LastAppliedRevision is the revision of the last successfully applied - source. - Deprecated: the revision can now be found in the History. - type: string - lastAttemptedConfigDigest: - description: |- - LastAttemptedConfigDigest is the digest for the config (better known as - "values") of the last reconciliation attempt. - type: string - lastAttemptedGeneration: - description: |- - LastAttemptedGeneration is the last generation the controller attempted - to reconcile. - format: int64 - type: integer - lastAttemptedReleaseAction: - description: |- - LastAttemptedReleaseAction is the last release action performed for this - HelmRelease. It is used to determine the active remediation strategy. - enum: - - install - - upgrade - type: string - lastAttemptedRevision: - description: |- - LastAttemptedRevision is the Source revision of the last reconciliation - attempt. For OCIRepository sources, the 12 first characters of the digest are - appended to the chart version e.g. "1.2.3+1234567890ab". - type: string - lastAttemptedRevisionDigest: - description: |- - LastAttemptedRevisionDigest is the digest of the last reconciliation attempt. - This is only set for OCIRepository sources. - type: string - lastAttemptedValuesChecksum: - description: |- - LastAttemptedValuesChecksum is the SHA1 checksum for the values of the last - reconciliation attempt. - Deprecated: Use LastAttemptedConfigDigest instead. - type: string - lastHandledForceAt: - description: |- - LastHandledForceAt holds the value of the most recent force request - value, so a change of the annotation value can be detected. - type: string - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - lastHandledResetAt: - description: |- - LastHandledResetAt holds the value of the most recent reset request - value, so a change of the annotation value can be detected. - type: string - lastReleaseRevision: - description: |- - LastReleaseRevision is the revision of the last successful Helm release. - Deprecated: Use History instead. - type: integer - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - observedPostRenderersDigest: - description: |- - ObservedPostRenderersDigest is the digest for the post-renderers of - the last successful reconciliation attempt. - type: string - storageNamespace: - description: |- - StorageNamespace is the namespace of the Helm release storage for the - current release. - maxLength: 63 - minLength: 1 - type: string - upgradeFailures: - description: |- - UpgradeFailures is the upgrade failure count against the latest desired - state. It is reset after a successful reconciliation. - format: int64 - type: integer - type: object - type: object - served: true - storage: false - subresources: - status: {} ---- -apiVersion: v1 -kind: ServiceAccount -metadata: - labels: - app.kubernetes.io/component: helm-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: helm-controller - namespace: flux-system ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - labels: - app.kubernetes.io/component: helm-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - control-plane: controller - name: helm-controller - namespace: flux-system -spec: - replicas: 1 - selector: - matchLabels: - app: helm-controller - template: - metadata: - annotations: - prometheus.io/port: "8080" - prometheus.io/scrape: "true" - labels: - app: helm-controller - spec: - containers: - - args: - - --events-addr=http://notification-controller.flux-system.svc.cluster.local./ - - --watch-all-namespaces=true - - --log-level=info - - --log-encoding=json - - --enable-leader-election - env: - - name: RUNTIME_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - - name: GOMAXPROCS - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.cpu - - name: GOMEMLIMIT - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.memory - image: ghcr.io/fluxcd/helm-controller:v1.0.1 - imagePullPolicy: IfNotPresent - livenessProbe: - httpGet: - path: /healthz - port: healthz - name: manager - ports: - - containerPort: 8080 - name: http-prom - protocol: TCP - - containerPort: 9440 - name: healthz - protocol: TCP - readinessProbe: - httpGet: - path: /readyz - port: healthz - resources: - limits: - cpu: 1000m - memory: 1Gi - requests: - cpu: 100m - memory: 64Mi - securityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - readOnlyRootFilesystem: true - runAsNonRoot: true - seccompProfile: - type: RuntimeDefault - volumeMounts: - - mountPath: /tmp - name: temp - nodeSelector: - kubernetes.io/os: linux - priorityClassName: system-cluster-critical - securityContext: - fsGroup: 1337 - serviceAccountName: helm-controller - terminationGracePeriodSeconds: 600 - volumes: - - emptyDir: {} - name: temp ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: notification-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: alerts.notification.toolkit.fluxcd.io -spec: - group: notification.toolkit.fluxcd.io - names: - kind: Alert - listKind: AlertList - plural: alerts - singular: alert - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta1 Alert is deprecated, upgrade to v1beta3 - name: v1beta1 - schema: - openAPIV3Schema: - description: Alert is the Schema for the alerts API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: AlertSpec defines an alerting rule for events involving a - list of objects - properties: - eventSeverity: - default: info - description: |- - Filter events based on severity, defaults to ('info'). - If set to 'info' no events will be filtered. - enum: - - info - - error - type: string - eventSources: - description: Filter events based on the involved objects. - items: - description: |- - CrossNamespaceObjectReference contains enough information to let you locate the - typed referenced object at cluster level - properties: - apiVersion: - description: API version of the referent - type: string - kind: - description: Kind of the referent - enum: - - Bucket - - GitRepository - - Kustomization - - HelmRelease - - HelmChart - - HelmRepository - - ImageRepository - - ImagePolicy - - ImageUpdateAutomation - - OCIRepository - type: string - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - name: - description: Name of the referent - maxLength: 53 - minLength: 1 - type: string - namespace: - description: Namespace of the referent - maxLength: 53 - minLength: 1 - type: string - required: - - name - type: object - type: array - exclusionList: - description: A list of Golang regular expressions to be used for excluding - messages. - items: - type: string - type: array - providerRef: - description: Send events using this provider. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - summary: - description: Short description of the impact and affected cluster. - type: string - suspend: - description: |- - This flag tells the controller to suspend subsequent events dispatching. - Defaults to false. - type: boolean - required: - - eventSources - - providerRef - type: object - status: - default: - observedGeneration: -1 - description: AlertStatus defines the observed state of Alert - properties: - conditions: - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta2 Alert is deprecated, upgrade to v1beta3 - name: v1beta2 - schema: - openAPIV3Schema: - description: Alert is the Schema for the alerts API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: AlertSpec defines an alerting rule for events involving a - list of objects. - properties: - eventMetadata: - additionalProperties: - type: string - description: |- - EventMetadata is an optional field for adding metadata to events dispatched by the - controller. This can be used for enhancing the context of the event. If a field - would override one already present on the original event as generated by the emitter, - then the override doesn't happen, i.e. the original value is preserved, and an info - log is printed. - type: object - eventSeverity: - default: info - description: |- - EventSeverity specifies how to filter events based on severity. - If set to 'info' no events will be filtered. - enum: - - info - - error - type: string - eventSources: - description: |- - EventSources specifies how to filter events based - on the involved object kind, name and namespace. - items: - description: |- - CrossNamespaceObjectReference contains enough information to let you locate the - typed referenced object at cluster level - properties: - apiVersion: - description: API version of the referent - type: string - kind: - description: Kind of the referent - enum: - - Bucket - - GitRepository - - Kustomization - - HelmRelease - - HelmChart - - HelmRepository - - ImageRepository - - ImagePolicy - - ImageUpdateAutomation - - OCIRepository - type: string - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - MatchLabels requires the name to be set to `*`. - type: object - name: - description: |- - Name of the referent - If multiple resources are targeted `*` may be set. - maxLength: 53 - minLength: 1 - type: string - namespace: - description: Namespace of the referent - maxLength: 53 - minLength: 1 - type: string - required: - - kind - - name - type: object - type: array - exclusionList: - description: |- - ExclusionList specifies a list of Golang regular expressions - to be used for excluding messages. - items: - type: string - type: array - inclusionList: - description: |- - InclusionList specifies a list of Golang regular expressions - to be used for including messages. - items: - type: string - type: array - providerRef: - description: ProviderRef specifies which Provider this Alert should - use. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - summary: - description: Summary holds a short description of the impact and affected - cluster. - maxLength: 255 - type: string - suspend: - description: |- - Suspend tells the controller to suspend subsequent - events handling for this Alert. - type: boolean - required: - - eventSources - - providerRef - type: object - status: - default: - observedGeneration: -1 - description: AlertStatus defines the observed state of the Alert. - properties: - conditions: - description: Conditions holds the conditions for the Alert. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1beta3 - schema: - openAPIV3Schema: - description: Alert is the Schema for the alerts API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: AlertSpec defines an alerting rule for events involving a - list of objects. - properties: - eventMetadata: - additionalProperties: - type: string - description: |- - EventMetadata is an optional field for adding metadata to events dispatched by the - controller. This can be used for enhancing the context of the event. If a field - would override one already present on the original event as generated by the emitter, - then the override doesn't happen, i.e. the original value is preserved, and an info - log is printed. - type: object - eventSeverity: - default: info - description: |- - EventSeverity specifies how to filter events based on severity. - If set to 'info' no events will be filtered. - enum: - - info - - error - type: string - eventSources: - description: |- - EventSources specifies how to filter events based - on the involved object kind, name and namespace. - items: - description: |- - CrossNamespaceObjectReference contains enough information to let you locate the - typed referenced object at cluster level - properties: - apiVersion: - description: API version of the referent - type: string - kind: - description: Kind of the referent - enum: - - Bucket - - GitRepository - - Kustomization - - HelmRelease - - HelmChart - - HelmRepository - - ImageRepository - - ImagePolicy - - ImageUpdateAutomation - - OCIRepository - type: string - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - MatchLabels requires the name to be set to `*`. - type: object - name: - description: |- - Name of the referent - If multiple resources are targeted `*` may be set. - maxLength: 53 - minLength: 1 - type: string - namespace: - description: Namespace of the referent - maxLength: 53 - minLength: 1 - type: string - required: - - kind - - name - type: object - type: array - exclusionList: - description: |- - ExclusionList specifies a list of Golang regular expressions - to be used for excluding messages. - items: - type: string - type: array - inclusionList: - description: |- - InclusionList specifies a list of Golang regular expressions - to be used for including messages. - items: - type: string - type: array - providerRef: - description: ProviderRef specifies which Provider this Alert should - use. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - summary: - description: Summary holds a short description of the impact and affected - cluster. - maxLength: 255 - type: string - suspend: - description: |- - Suspend tells the controller to suspend subsequent - events handling for this Alert. - type: boolean - required: - - eventSources - - providerRef - type: object - type: object - served: true - storage: true - subresources: {} ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: notification-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: providers.notification.toolkit.fluxcd.io -spec: - group: notification.toolkit.fluxcd.io - names: - kind: Provider - listKind: ProviderList - plural: providers - singular: provider - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta1 Provider is deprecated, upgrade to v1beta3 - name: v1beta1 - schema: - openAPIV3Schema: - description: Provider is the Schema for the providers API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ProviderSpec defines the desired state of Provider - properties: - address: - description: HTTP/S webhook address of this provider - pattern: ^(http|https):// - type: string - certSecretRef: - description: |- - CertSecretRef can be given the name of a secret containing - a PEM-encoded CA certificate (`caFile`) - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - channel: - description: Alert channel for this provider - type: string - proxy: - description: HTTP/S address of the proxy - pattern: ^(http|https):// - type: string - secretRef: - description: |- - Secret reference containing the provider webhook URL - using "address" as data key - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - This flag tells the controller to suspend subsequent events handling. - Defaults to false. - type: boolean - timeout: - description: Timeout for sending alerts to the provider. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - type: - description: Type of provider - enum: - - slack - - discord - - msteams - - rocket - - generic - - generic-hmac - - github - - gitlab - - bitbucket - - azuredevops - - googlechat - - webex - - sentry - - azureeventhub - - telegram - - lark - - matrix - - opsgenie - - alertmanager - - grafana - - githubdispatch - type: string - username: - description: Bot username for this provider - type: string - required: - - type - type: object - status: - default: - observedGeneration: -1 - description: ProviderStatus defines the observed state of Provider - properties: - conditions: - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - observedGeneration: - description: ObservedGeneration is the last reconciled generation. - format: int64 - type: integer - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta2 Provider is deprecated, upgrade to v1beta3 - name: v1beta2 - schema: - openAPIV3Schema: - description: Provider is the Schema for the providers API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ProviderSpec defines the desired state of the Provider. - properties: - address: - description: |- - Address specifies the endpoint, in a generic sense, to where alerts are sent. - What kind of endpoint depends on the specific Provider type being used. - For the generic Provider, for example, this is an HTTP/S address. - For other Provider types this could be a project ID or a namespace. - maxLength: 2048 - type: string - certSecretRef: - description: |- - CertSecretRef specifies the Secret containing - a PEM-encoded CA certificate (in the `ca.crt` key). - - - Note: Support for the `caFile` key has - been deprecated. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - channel: - description: Channel specifies the destination channel where events - should be posted. - maxLength: 2048 - type: string - interval: - description: Interval at which to reconcile the Provider with its - Secret references. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - proxy: - description: Proxy the HTTP/S address of the proxy server. - maxLength: 2048 - pattern: ^(http|https)://.*$ - type: string - secretRef: - description: |- - SecretRef specifies the Secret containing the authentication - credentials for this Provider. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend subsequent - events handling for this Provider. - type: boolean - timeout: - description: Timeout for sending alerts to the Provider. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - type: - description: Type specifies which Provider implementation to use. - enum: - - slack - - discord - - msteams - - rocket - - generic - - generic-hmac - - github - - gitlab - - gitea - - bitbucketserver - - bitbucket - - azuredevops - - googlechat - - googlepubsub - - webex - - sentry - - azureeventhub - - telegram - - lark - - matrix - - opsgenie - - alertmanager - - grafana - - githubdispatch - - pagerduty - - datadog - type: string - username: - description: Username specifies the name under which events are posted. - maxLength: 2048 - type: string - required: - - type - type: object - status: - default: - observedGeneration: -1 - description: ProviderStatus defines the observed state of the Provider. - properties: - conditions: - description: Conditions holds the conditions for the Provider. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last reconciled generation. - format: int64 - type: integer - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1beta3 - schema: - openAPIV3Schema: - description: Provider is the Schema for the providers API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ProviderSpec defines the desired state of the Provider. - properties: - address: - description: |- - Address specifies the endpoint, in a generic sense, to where alerts are sent. - What kind of endpoint depends on the specific Provider type being used. - For the generic Provider, for example, this is an HTTP/S address. - For other Provider types this could be a project ID or a namespace. - maxLength: 2048 - type: string - certSecretRef: - description: |- - CertSecretRef specifies the Secret containing - a PEM-encoded CA certificate (in the `ca.crt` key). - - - Note: Support for the `caFile` key has - been deprecated. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - channel: - description: Channel specifies the destination channel where events - should be posted. - maxLength: 2048 - type: string - interval: - description: |- - Interval at which to reconcile the Provider with its Secret references. - Deprecated and not used in v1beta3. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - proxy: - description: Proxy the HTTP/S address of the proxy server. - maxLength: 2048 - pattern: ^(http|https)://.*$ - type: string - secretRef: - description: |- - SecretRef specifies the Secret containing the authentication - credentials for this Provider. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend subsequent - events handling for this Provider. - type: boolean - timeout: - description: Timeout for sending alerts to the Provider. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m))+$ - type: string - type: - description: Type specifies which Provider implementation to use. - enum: - - slack - - discord - - msteams - - rocket - - generic - - generic-hmac - - github - - gitlab - - gitea - - bitbucketserver - - bitbucket - - azuredevops - - googlechat - - googlepubsub - - webex - - sentry - - azureeventhub - - telegram - - lark - - matrix - - opsgenie - - alertmanager - - grafana - - githubdispatch - - pagerduty - - datadog - - nats - type: string - username: - description: Username specifies the name under which events are posted. - maxLength: 2048 - type: string - required: - - type - type: object - type: object - served: true - storage: true - subresources: {} ---- -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/component: notification-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: receivers.notification.toolkit.fluxcd.io -spec: - group: notification.toolkit.fluxcd.io - names: - kind: Receiver - listKind: ReceiverList - plural: receivers - singular: receiver - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - name: v1 - schema: - openAPIV3Schema: - description: Receiver is the Schema for the receivers API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ReceiverSpec defines the desired state of the Receiver. - properties: - events: - description: |- - Events specifies the list of event types to handle, - e.g. 'push' for GitHub or 'Push Hook' for GitLab. - items: - type: string - type: array - interval: - default: 10m - description: Interval at which to reconcile the Receiver with its - Secret references. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - resources: - description: A list of resources to be notified about changes. - items: - description: |- - CrossNamespaceObjectReference contains enough information to let you locate the - typed referenced object at cluster level - properties: - apiVersion: - description: API version of the referent - type: string - kind: - description: Kind of the referent - enum: - - Bucket - - GitRepository - - Kustomization - - HelmRelease - - HelmChart - - HelmRepository - - ImageRepository - - ImagePolicy - - ImageUpdateAutomation - - OCIRepository - type: string - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - MatchLabels requires the name to be set to `*`. - type: object - name: - description: |- - Name of the referent - If multiple resources are targeted `*` may be set. - maxLength: 53 - minLength: 1 - type: string - namespace: - description: Namespace of the referent - maxLength: 53 - minLength: 1 - type: string - required: - - kind - - name - type: object - type: array - secretRef: - description: |- - SecretRef specifies the Secret containing the token used - to validate the payload authenticity. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend subsequent - events handling for this receiver. - type: boolean - type: - description: |- - Type of webhook sender, used to determine - the validation procedure and payload deserialization. - enum: - - generic - - generic-hmac - - github - - gitlab - - bitbucket - - harbor - - dockerhub - - quay - - gcr - - nexus - - acr - - cdevents - type: string - required: - - resources - - secretRef - - type - type: object - status: - default: - observedGeneration: -1 - description: ReceiverStatus defines the observed state of the Receiver. - properties: - conditions: - description: Conditions holds the conditions for the Receiver. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation of - the Receiver object. - format: int64 - type: integer - webhookPath: - description: |- - WebhookPath is the generated incoming webhook address in the format - of '/hook/sha256sum(token+name+namespace)'. - type: string - type: object - type: object - served: true - storage: true - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta1 Receiver is deprecated, upgrade to v1 - name: v1beta1 - schema: - openAPIV3Schema: - description: Receiver is the Schema for the receivers API - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ReceiverSpec defines the desired state of Receiver - properties: - events: - description: |- - A list of events to handle, - e.g. 'push' for GitHub or 'Push Hook' for GitLab. - items: - type: string - type: array - resources: - description: A list of resources to be notified about changes. - items: - description: |- - CrossNamespaceObjectReference contains enough information to let you locate the - typed referenced object at cluster level - properties: - apiVersion: - description: API version of the referent - type: string - kind: - description: Kind of the referent - enum: - - Bucket - - GitRepository - - Kustomization - - HelmRelease - - HelmChart - - HelmRepository - - ImageRepository - - ImagePolicy - - ImageUpdateAutomation - - OCIRepository - type: string - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - type: object - name: - description: Name of the referent - maxLength: 53 - minLength: 1 - type: string - namespace: - description: Namespace of the referent - maxLength: 53 - minLength: 1 - type: string - required: - - name - type: object - type: array - secretRef: - description: |- - Secret reference containing the token used - to validate the payload authenticity - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - This flag tells the controller to suspend subsequent events handling. - Defaults to false. - type: boolean - type: - description: |- - Type of webhook sender, used to determine - the validation procedure and payload deserialization. - enum: - - generic - - generic-hmac - - github - - gitlab - - bitbucket - - harbor - - dockerhub - - quay - - gcr - - nexus - - acr - type: string - required: - - resources - - type - type: object - status: - default: - observedGeneration: -1 - description: ReceiverStatus defines the observed state of Receiver - properties: - conditions: - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - observedGeneration: - description: ObservedGeneration is the last observed generation. - format: int64 - type: integer - url: - description: |- - Generated webhook URL in the format - of '/hook/sha256sum(token+name+namespace)'. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - - jsonPath: .status.conditions[?(@.type=="Ready")].status - name: Ready - type: string - - jsonPath: .status.conditions[?(@.type=="Ready")].message - name: Status - type: string - deprecated: true - deprecationWarning: v1beta2 Receiver is deprecated, upgrade to v1 - name: v1beta2 - schema: - openAPIV3Schema: - description: Receiver is the Schema for the receivers API. - properties: - apiVersion: - description: |- - APIVersion defines the versioned schema of this representation of an object. - Servers should convert recognized schemas to the latest internal value, and - may reject unrecognized values. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources - type: string - kind: - description: |- - Kind is a string value representing the REST resource this object represents. - Servers may infer this from the endpoint the client submits requests to. - Cannot be updated. - In CamelCase. - More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds - type: string - metadata: - type: object - spec: - description: ReceiverSpec defines the desired state of the Receiver. - properties: - events: - description: |- - Events specifies the list of event types to handle, - e.g. 'push' for GitHub or 'Push Hook' for GitLab. - items: - type: string - type: array - interval: - description: Interval at which to reconcile the Receiver with its - Secret references. - pattern: ^([0-9]+(\.[0-9]+)?(ms|s|m|h))+$ - type: string - resources: - description: A list of resources to be notified about changes. - items: - description: |- - CrossNamespaceObjectReference contains enough information to let you locate the - typed referenced object at cluster level - properties: - apiVersion: - description: API version of the referent - type: string - kind: - description: Kind of the referent - enum: - - Bucket - - GitRepository - - Kustomization - - HelmRelease - - HelmChart - - HelmRepository - - ImageRepository - - ImagePolicy - - ImageUpdateAutomation - - OCIRepository - type: string - matchLabels: - additionalProperties: - type: string - description: |- - MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels - map is equivalent to an element of matchExpressions, whose key field is "key", the - operator is "In", and the values array contains only "value". The requirements are ANDed. - MatchLabels requires the name to be set to `*`. - type: object - name: - description: |- - Name of the referent - If multiple resources are targeted `*` may be set. - maxLength: 53 - minLength: 1 - type: string - namespace: - description: Namespace of the referent - maxLength: 53 - minLength: 1 - type: string - required: - - kind - - name - type: object - type: array - secretRef: - description: |- - SecretRef specifies the Secret containing the token used - to validate the payload authenticity. - properties: - name: - description: Name of the referent. - type: string - required: - - name - type: object - suspend: - description: |- - Suspend tells the controller to suspend subsequent - events handling for this receiver. - type: boolean - type: - description: |- - Type of webhook sender, used to determine - the validation procedure and payload deserialization. - enum: - - generic - - generic-hmac - - github - - gitlab - - bitbucket - - harbor - - dockerhub - - quay - - gcr - - nexus - - acr - type: string - required: - - resources - - type - type: object - status: - default: - observedGeneration: -1 - description: ReceiverStatus defines the observed state of the Receiver. - properties: - conditions: - description: Conditions holds the conditions for the Receiver. - items: - description: "Condition contains details for one aspect of the current - state of this API Resource.\n---\nThis struct is intended for - direct use as an array at the field path .status.conditions. For - example,\n\n\n\ttype FooStatus struct{\n\t // Represents the - observations of a foo's current state.\n\t // Known .status.conditions.type - are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // - +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t - \ // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" - patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t - \ // other fields\n\t}" - properties: - lastTransitionTime: - description: |- - lastTransitionTime is the last time the condition transitioned from one status to another. - This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. - format: date-time - type: string - message: - description: |- - message is a human readable message indicating details about the transition. - This may be an empty string. - maxLength: 32768 - type: string - observedGeneration: - description: |- - observedGeneration represents the .metadata.generation that the condition was set based upon. - For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date - with respect to the current state of the instance. - format: int64 - minimum: 0 - type: integer - reason: - description: |- - reason contains a programmatic identifier indicating the reason for the condition's last transition. - Producers of specific condition types may define expected values and meanings for this field, - and whether the values are considered a guaranteed API. - The value should be a CamelCase string. - This field may not be empty. - maxLength: 1024 - minLength: 1 - pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ - type: string - status: - description: status of the condition, one of True, False, Unknown. - enum: - - "True" - - "False" - - Unknown - type: string - type: - description: |- - type of condition in CamelCase or in foo.example.com/CamelCase. - --- - Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be - useful (see .node.status.conditions), the ability to deconflict is important. - The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) - maxLength: 316 - pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ - type: string - required: - - lastTransitionTime - - message - - reason - - status - - type - type: object - type: array - lastHandledReconcileAt: - description: |- - LastHandledReconcileAt holds the value of the most recent - reconcile request value, so a change of the annotation value - can be detected. - type: string - observedGeneration: - description: ObservedGeneration is the last observed generation of - the Receiver object. - format: int64 - type: integer - url: - description: |- - URL is the generated incoming webhook address in the format - of '/hook/sha256sum(token+name+namespace)'. - Deprecated: Replaced by WebhookPath. - type: string - webhookPath: - description: |- - WebhookPath is the generated incoming webhook address in the format - of '/hook/sha256sum(token+name+namespace)'. - type: string - type: object - type: object - served: true - storage: false - subresources: - status: {} ---- -apiVersion: v1 -kind: ServiceAccount -metadata: - labels: - app.kubernetes.io/component: notification-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - name: notification-controller - namespace: flux-system ---- -apiVersion: v1 -kind: Service -metadata: - labels: - app.kubernetes.io/component: notification-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - control-plane: controller - name: notification-controller - namespace: flux-system -spec: - ports: - - name: http - port: 80 - protocol: TCP - targetPort: http - selector: - app: notification-controller - type: ClusterIP ---- -apiVersion: v1 -kind: Service -metadata: - labels: - app.kubernetes.io/component: notification-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - control-plane: controller - name: webhook-receiver - namespace: flux-system -spec: - ports: - - name: http - port: 80 - protocol: TCP - targetPort: http-webhook - selector: - app: notification-controller - type: ClusterIP ---- -apiVersion: apps/v1 -kind: Deployment -metadata: - labels: - app.kubernetes.io/component: notification-controller - app.kubernetes.io/instance: flux-system - app.kubernetes.io/part-of: flux - app.kubernetes.io/version: v2.3.0 - control-plane: controller - name: notification-controller - namespace: flux-system -spec: - replicas: 1 - selector: - matchLabels: - app: notification-controller - template: - metadata: - annotations: - prometheus.io/port: "8080" - prometheus.io/scrape: "true" - labels: - app: notification-controller - spec: - containers: - - args: - - --watch-all-namespaces=true - - --log-level=info - - --log-encoding=json - - --enable-leader-election - env: - - name: RUNTIME_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - - name: GOMAXPROCS - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.cpu - - name: GOMEMLIMIT - valueFrom: - resourceFieldRef: - containerName: manager - resource: limits.memory - image: ghcr.io/fluxcd/notification-controller:v1.3.0 - imagePullPolicy: IfNotPresent - livenessProbe: - httpGet: - path: /healthz - port: healthz - name: manager - ports: - - containerPort: 9090 - name: http - protocol: TCP - - containerPort: 9292 - name: http-webhook - protocol: TCP - - containerPort: 8080 - name: http-prom - protocol: TCP - - containerPort: 9440 - name: healthz - protocol: TCP - readinessProbe: - httpGet: - path: /readyz - port: healthz - resources: - limits: - cpu: 1000m - memory: 1Gi - requests: - cpu: 100m - memory: 64Mi - securityContext: - allowPrivilegeEscalation: false - capabilities: - drop: - - ALL - readOnlyRootFilesystem: true - runAsNonRoot: true - seccompProfile: - type: RuntimeDefault - volumeMounts: - - mountPath: /tmp - name: temp - nodeSelector: - kubernetes.io/os: linux - securityContext: - fsGroup: 1337 - serviceAccountName: notification-controller - terminationGracePeriodSeconds: 10 - volumes: - - emptyDir: {} - name: temp diff --git a/flux/clusters/staging/flux-system/gotk-sync.yaml b/flux/clusters/staging/flux-system/gotk-sync.yaml deleted file mode 100644 index 3965b81..0000000 --- a/flux/clusters/staging/flux-system/gotk-sync.yaml +++ /dev/null @@ -1,27 +0,0 @@ -# This manifest was generated by flux. DO NOT EDIT. ---- -apiVersion: source.toolkit.fluxcd.io/v1 -kind: GitRepository -metadata: - name: flux-system - namespace: flux-system -spec: - interval: 1m0s - ref: - branch: main - secretRef: - name: flux-system - url: ssh://git@gitea.futureporn.net:2222/futureporn/fp ---- -apiVersion: kustomize.toolkit.fluxcd.io/v1 -kind: Kustomization -metadata: - name: flux-system - namespace: flux-system -spec: - interval: 10m0s - path: ./flux/clusters/staging - prune: true - sourceRef: - kind: GitRepository - name: flux-system diff --git a/flux/clusters/staging/flux-system/kustomization.yaml b/flux/clusters/staging/flux-system/kustomization.yaml deleted file mode 100644 index 3842229..0000000 --- a/flux/clusters/staging/flux-system/kustomization.yaml +++ /dev/null @@ -1,5 +0,0 @@ -apiVersion: kustomize.config.k8s.io/v1beta1 -kind: Kustomization -resources: -- gotk-components.yaml -- gotk-sync.yaml diff --git a/flux/clusters/staging/infrastructure.yaml b/flux/clusters/staging/infrastructure.yaml deleted file mode 100644 index 32d9a8b..0000000 --- a/flux/clusters/staging/infrastructure.yaml +++ /dev/null @@ -1,56 +0,0 @@ ---- -apiVersion: kustomize.toolkit.fluxcd.io/v1 -kind: Kustomization -metadata: - name: infra-controllers - namespace: flux-system -spec: - interval: 1h - retryInterval: 1m - timeout: 5m - sourceRef: - kind: GitRepository - name: flux-system - path: ./flux/infrastructure/controllers - prune: true - wait: true - ---- -apiVersion: kustomize.toolkit.fluxcd.io/v1 -kind: Kustomization -metadata: - name: infra-configs - namespace: flux-system -spec: - dependsOn: - - name: infra-controllers - interval: 1h - retryInterval: 1m - timeout: 5m - sourceRef: - kind: GitRepository - name: flux-system - path: ./flux/infrastructure/configs - prune: true - patches: - - patch: | - - op: replace - path: /spec/acme/server - value: https://acme-staging-v02.api.letsencrypt.org/directory - target: - kind: ClusterIssuer - name: letsencrypt - - patch: | - - op: replace - path: /metadata/name - value: letsencrypt-staging - target: - kind: ClusterIssuer - name: letsencrypt - - patch: | - - op: replace - path: /spec/acme/privateKeySecretRef/name - value: letsencrypt-staging - target: - kind: ClusterIssuer - name: letsencrypt \ No newline at end of file diff --git a/flux/infrastructure/configs/cluster-issuers.yaml b/flux/infrastructure/configs/cluster-issuers.yaml deleted file mode 100644 index cc25cac..0000000 --- a/flux/infrastructure/configs/cluster-issuers.yaml +++ /dev/null @@ -1,31 +0,0 @@ -## Example values are replaced using environment-specific Kuztomization patches - ---- -apiVersion: cert-manager.io/v1 -kind: ClusterIssuer -metadata: - name: letsencrypt - annotations: - meta.helm.sh/release-name: fp - meta.helm.sh/release-namespace: futureporn - labels: - app.kubernetes.io/managed-by: Helm -spec: - acme: - # Replace the email address with your own contact email - email: cj@futureporn.net - server: https://example.com - privateKeySecretRef: - name: letsencrypt - solvers: - - dns01: - webhook: - groupName: acme.exoscale.com - solverName: exoscale - config: - apiKeyRef: - name: exoscale - key: apiKey - apiSecretRef: - name: exoscale - key: apiSecret \ No newline at end of file diff --git a/flux/infrastructure/configs/kustomization.yaml b/flux/infrastructure/configs/kustomization.yaml deleted file mode 100644 index de1ebfa..0000000 --- a/flux/infrastructure/configs/kustomization.yaml +++ /dev/null @@ -1,4 +0,0 @@ -apiVersion: kustomize.config.k8s.io/v1beta1 -kind: Kustomization -resources: - - cluster-issuers.yaml diff --git a/flux/infrastructure/controllers/cert-manager.yaml b/flux/infrastructure/controllers/cert-manager.yaml deleted file mode 100644 index 096e981..0000000 --- a/flux/infrastructure/controllers/cert-manager.yaml +++ /dev/null @@ -1,36 +0,0 @@ ---- -apiVersion: v1 -kind: Namespace -metadata: - name: cert-manager - ---- -apiVersion: source.toolkit.fluxcd.io/v1 -kind: HelmRepository -metadata: - name: cert-manager - namespace: cert-manager -spec: - interval: 24h - url: https://charts.jetstack.io - ---- -apiVersion: helm.toolkit.fluxcd.io/v2 -kind: HelmRelease -metadata: - name: cert-manager - namespace: cert-manager -spec: - interval: 30m - chart: - spec: - chart: cert-manager - version: "1.x" - sourceRef: - kind: HelmRepository - name: cert-manager - namespace: cert-manager - interval: 12h - values: - installCRDs: true - diff --git a/flux/infrastructure/controllers/grafana-k8s-dashboards.yaml b/flux/infrastructure/controllers/grafana-k8s-dashboards.yaml deleted file mode 100644 index b9d2046..0000000 --- a/flux/infrastructure/controllers/grafana-k8s-dashboards.yaml +++ /dev/null @@ -1,21313 +0,0 @@ -apiVersion: v1 -data: - k8s-addons-prometheus.json: |- - { - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": [], - "__requires": [ - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "8.5.0" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "5.0.0" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - }, - { - "type": "panel", - "id": "table", - "name": "Table", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "target": { - "limit": 100, - "matchAny": false, - "tags": [], - "type": "dashboard" - }, - "type": "dashboard" - } - ] - }, - "description": "This is a modern 'Prometheus' dashboard for your Kubernetes cluster(s). Made for kube-prometheus-stack and take advantage of the latest Grafana features. GitHub repository: https://github.com/dotdc/grafana-dashboards-kubernetes", - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 1, - "links": [], - "liveNow": false, - "panels": [ - { - "collapsed": false, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 0 - }, - "id": 89, - "panels": [], - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "refId": "A" - } - ], - "title": "Information", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "?", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "orange", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 6, - "x": 0, - "y": 1 - }, - "id": 78, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "text": {}, - "textMode": "name" - }, - "pluginVersion": "10.0.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "prometheus_build_info{pod=~\"$pod\", cluster=~\"$cluster\"}", - "instant": true, - "interval": "", - "legendFormat": "{{ version }}", - "range": false, - "refId": "A" - } - ], - "title": "Prometheus version", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 1 - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 6, - "x": 6, - "y": 1 - }, - "id": 92, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "text": {}, - "textMode": "value" - }, - "pluginVersion": "10.0.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "up{pod=~\"$pod\", cluster=~\"$cluster\"} < 1", - "instant": true, - "interval": "", - "legendFormat": "__auto", - "range": false, - "refId": "A" - } - ], - "title": "Instance Down", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "text", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 6, - "x": 12, - "y": 1 - }, - "id": 72, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [], - "fields": "", - "values": false - }, - "text": {}, - "textMode": "value" - }, - "pluginVersion": "10.0.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(prometheus_tsdb_head_series{pod=~\"$pod\", cluster=~\"$cluster\"}) by (pod)", - "interval": "", - "legendFormat": "__auto", - "range": true, - "refId": "A" - } - ], - "title": "TSDB Head Series", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "blue", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 6, - "x": 18, - "y": 1 - }, - "id": 94, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "text": {}, - "textMode": "value" - }, - "pluginVersion": "10.0.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(prometheus_sd_discovered_targets{pod=~\"$pod\", cluster=~\"$cluster\"}) by (pod)", - "instant": true, - "interval": "", - "legendFormat": "__auto", - "range": false, - "refId": "A" - } - ], - "title": "Discovered Targets", - "type": "stat" - }, - { - "collapsed": false, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 5 - }, - "id": 64, - "panels": [], - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "refId": "A" - } - ], - "title": "Prometheus", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 6 - }, - "id": 93, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "up{pod=~\"$pod\", cluster=~\"$cluster\"}", - "interval": "", - "legendFormat": "{{ pod }}", - "range": true, - "refId": "A" - } - ], - "title": "Liveness by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 6 - }, - "id": 96, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(prometheus_config_last_reload_successful{pod=~\"$pod\", cluster=~\"$cluster\"}) by (pod)", - "interval": "", - "legendFormat": "{{ pod }}", - "range": true, - "refId": "A" - } - ], - "title": "Config - Last Successful Reload by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 14 - }, - "id": 74, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(prometheus_target_scrapes_exceeded_body_size_limit_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "", - "legendFormat": "{{ pod }} - Exceeded body size limit", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(rate(prometheus_target_scrapes_exceeded_sample_limit_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "hide": false, - "legendFormat": "{{ pod }} - Exceeded sample limit", - "range": true, - "refId": "B" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(rate(prometheus_target_scrapes_sample_duplicate_timestamp_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "hide": false, - "legendFormat": "{{ pod }} - Duplicate timestamp", - "range": true, - "refId": "C" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(rate(prometheus_target_scrapes_sample_out_of_bounds_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "hide": false, - "legendFormat": "{{ pod }} - Sample out of bounds", - "range": true, - "refId": "D" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(rate(prometheus_target_scrapes_sample_out_of_order_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "hide": false, - "legendFormat": "{{ pod }} - Sample out of order", - "range": true, - "refId": "E" - } - ], - "title": "Target Scrapes Errors by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 14 - }, - "id": 84, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(prometheus_sd_discovered_targets{pod=~\"$pod\", cluster=~\"$cluster\"}) by (pod)", - "interval": "", - "legendFormat": "{{ pod }}", - "range": true, - "refId": "A" - } - ], - "title": "Number of Targets by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "ms" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 22 - }, - "id": 75, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(prometheus_target_sync_length_seconds_sum{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod, scrape_job) * 1000", - "interval": "", - "legendFormat": "{{ pod }} - {{ scrape_job }}", - "range": true, - "refId": "A" - } - ], - "title": "Target Sync by pod, scrape_job", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "s" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 22 - }, - "id": 85, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "round(sum(rate(prometheus_target_interval_length_seconds_sum{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval]) / rate(prometheus_target_interval_length_seconds_count{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod))", - "interval": "", - "legendFormat": "{{ pod }}", - "range": true, - "refId": "A" - } - ], - "title": "Average Scrape Interval by pod", - "type": "timeseries" - }, - { - "collapsed": false, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 30 - }, - "id": 98, - "panels": [], - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "refId": "A" - } - ], - "title": "Prometheus TSDB / Query Engine", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 31 - }, - "id": 59, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(prometheus_tsdb_head_series{pod=~\"$pod\", cluster=~\"$cluster\"}) by (pod)", - "interval": "", - "legendFormat": "{{ pod }} - Head Series", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(prometheus_tsdb_head_chunks{pod=~\"$pod\", cluster=~\"$cluster\"}) by (pod)", - "hide": false, - "legendFormat": "{{ pod }} - Head Chunks", - "range": true, - "refId": "B" - } - ], - "title": "TSDB Head Series & Chunks by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 31 - }, - "id": 60, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(prometheus_tsdb_head_samples_appended_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "", - "legendFormat": "{{ pod }}", - "range": true, - "refId": "A" - } - ], - "title": "TSDB Head samples appended - rate by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 39 - }, - "id": 101, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(prometheus_tsdb_blocks_loaded{pod=~\"$pod\", cluster=~\"$cluster\"}) by (pod)", - "interval": "", - "legendFormat": "{{ pod }} - Head Series", - "range": true, - "refId": "A" - } - ], - "title": "TSDB Blocks Loaded by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 39 - }, - "id": 102, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(prometheus_tsdb_compactions_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "", - "legendFormat": "{{ pod }} - Total Compactions", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(rate(prometheus_tsdb_compactions_triggered_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "hide": false, - "legendFormat": "{{ pod }} - Triggered Compactions", - "range": true, - "refId": "B" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(rate(prometheus_tsdb_compactions_skipped_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "hide": false, - "legendFormat": "{{ pod }} - Skipped Compactions", - "range": true, - "refId": "C" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(rate(prometheus_tsdb_compactions_failed_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "hide": false, - "legendFormat": "{{ pod }} - Failed Compactions", - "range": true, - "refId": "D" - } - ], - "title": "TSDB Rate of Compactions by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 47 - }, - "id": 90, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(prometheus_tsdb_reloads_failures_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "", - "legendFormat": "{{ pod }}", - "range": true, - "refId": "A" - } - ], - "title": "TSDB Reload Failures by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 47 - }, - "id": 95, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(prometheus_tsdb_head_series_created_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "", - "legendFormat": "{{ pod }} - Created series", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(rate(prometheus_tsdb_head_series_removed_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "hide": false, - "legendFormat": "{{ pod }} - Deleted series", - "range": true, - "refId": "B" - } - ], - "title": "TSDB Created & Deleted series by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 55 - }, - "id": 73, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(prometheus_engine_query_duration_seconds_count{pod=~\"$pod\", slice=\"inner_eval\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "", - "legendFormat": "{{ pod }}", - "range": true, - "refId": "A" - } - ], - "title": "Engine Query Count by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "ms" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 55 - }, - "id": 86, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "max(prometheus_engine_query_duration_seconds{pod=~\"$pod\", cluster=~\"$cluster\"}) by (pod, slice) * 1000", - "interval": "", - "legendFormat": "{{ pod }} - {{ slice }}", - "range": true, - "refId": "A" - } - ], - "title": "Engine Query Duration by pod, slice", - "type": "timeseries" - }, - { - "collapsed": false, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 63 - }, - "id": 47, - "panels": [], - "targets": [ - { - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "refId": "A" - } - ], - "title": "Resources", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "CPU Cores", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": true, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 4, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "limit" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "#F2495C", - "mode": "fixed" - } - }, - { - "id": "custom.fillOpacity", - "value": 0 - } - ] - } - ] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 64 - }, - "id": 29, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(container_cpu_usage_seconds_total{pod=~\"$pod\", image!=\"\", container!=\"\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod, container)", - "interval": "$resolution", - "legendFormat": "{{ pod }} - {{ container }}", - "range": true, - "refId": "A" - } - ], - "title": "CPU Usage by pod, container", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "Bytes", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": true, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 64 - }, - "id": 51, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(container_memory_working_set_bytes{pod=~\"$pod\", image!=\"\", container!=\"\", cluster=~\"$cluster\"}) by (pod, container)", - "interval": "", - "legendFormat": "{{ pod }} - {{ container }}", - "range": true, - "refId": "A" - } - ], - "title": "Memory Usage by container", - "type": "timeseries" - }, - { - "collapsed": false, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 72 - }, - "id": 66, - "panels": [], - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "refId": "A" - } - ], - "title": "Storage", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": true, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 73 - }, - "id": 62, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(kubelet_volume_stats_used_bytes{persistentvolumeclaim=~\".*prom.*\", cluster=~\"$cluster\"}) by (persistentvolumeclaim) / sum(kubelet_volume_stats_capacity_bytes{persistentvolumeclaim=~\".*prom.*\", cluster=~\"$cluster\"}) by (persistentvolumeclaim)", - "interval": "", - "legendFormat": "{{ persistentvolumeclaim }}", - "range": true, - "refId": "A" - } - ], - "title": "Persistent Volumes - Capacity and usage in %", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": true, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 73 - }, - "id": 87, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(kubelet_volume_stats_used_bytes{persistentvolumeclaim=~\".*prom.*\", cluster=~\"$cluster\"}) by (persistentvolumeclaim)", - "interval": "", - "legendFormat": "{{ persistentvolumeclaim }} - Used", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(kubelet_volume_stats_capacity_bytes{persistentvolumeclaim=~\".*prom.*\", cluster=~\"$cluster\"}) by (persistentvolumeclaim)", - "hide": false, - "legendFormat": "{{ persistentvolumeclaim }} - Capacity", - "range": true, - "refId": "B" - } - ], - "title": "Persistent Volumes - Capacity and usage in bytes", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": true, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "links": [], - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 81 - }, - "id": 68, - "links": [], - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.4", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "1 - sum(kubelet_volume_stats_inodes_used{persistentvolumeclaim=~\".*prom.*\", cluster=~\"$cluster\"}) by (persistentvolumeclaim) / sum(kubelet_volume_stats_inodes{persistentvolumeclaim=~\".*prom.*\", cluster=~\"$cluster\"}) by (persistentvolumeclaim)", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{ persistentvolumeclaim }}", - "range": true, - "refId": "A", - "step": 240 - } - ], - "title": "Persistent Volumes - Inodes", - "type": "timeseries" - }, - { - "collapsed": false, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 89 - }, - "id": 45, - "panels": [], - "targets": [ - { - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "refId": "A" - } - ], - "title": "Network", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": true, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 90 - }, - "id": 31, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(container_network_receive_bytes_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "$resolution", - "legendFormat": "{{ pod }} - Received", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "- sum(rate(container_network_transmit_bytes_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "$resolution", - "legendFormat": "{{ pod }} - Transmitted", - "range": true, - "refId": "B" - } - ], - "title": "Network - Bandwidth by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": true, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "pps" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 90 - }, - "id": 34, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(container_network_receive_packets_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "$resolution", - "legendFormat": "{{ pod }} - Received", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "- sum(rate(container_network_transmit_packets_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "$resolution", - "legendFormat": "{{ pod }} - Transmitted", - "range": true, - "refId": "B" - } - ], - "title": "Network - Packets rate by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": true, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "pps" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 98 - }, - "id": 36, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(container_network_receive_packets_dropped_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "$resolution", - "legendFormat": "{{ pod }} - Received", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "- sum(rate(container_network_transmit_packets_dropped_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "$resolution", - "legendFormat": "{{ pod }} - Transmitted", - "range": true, - "refId": "B" - } - ], - "title": "Network - Packets Dropped by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": true, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "pps" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 98 - }, - "id": 37, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(container_network_receive_errors_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "$resolution", - "legendFormat": "{{ pod }} - Received", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "- sum(rate(container_network_transmit_errors_total{pod=~\"$pod\", cluster=~\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "$resolution", - "legendFormat": "{{ pod }} - Transmitted", - "range": true, - "refId": "B" - } - ], - "title": "Network - Errors by pod", - "type": "timeseries" - } - ], - "refresh": "30s", - "revision": 1, - "schemaVersion": 38, - "style": "dark", - "tags": [ - "Kubernetes", - "Prometheus" - ], - "templating": { - "list": [ - { - "current": { - "selected": true, - "text": "Prometheus", - "value": "Prometheus" - }, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "datasource", - "options": [], - "query": "prometheus", - "queryValue": "", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "current": { - "isNone": true, - "selected": false, - "text": "None", - "value": "" - }, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_node_info,cluster)", - "hide": 0, - "includeAll": false, - "multi": false, - "name": "cluster", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(kube_node_info,cluster)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - }, - { - "current": { - "selected": false, - "text": "All", - "value": "$__all" - }, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(prometheus_build_info{cluster=\"$cluster\"}, pod)", - "hide": 0, - "includeAll": true, - "multi": false, - "name": "pod", - "options": [], - "query": { - "query": "label_values(prometheus_build_info{cluster=\"$cluster\"}, pod)", - "refId": "StandardVariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "tagValuesQuery": "", - "tagsQuery": "", - "type": "query", - "useTags": false - }, - { - "current": { - "selected": false, - "text": "30s", - "value": "30s" - }, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "resolution", - "options": [ - { - "selected": false, - "text": "1s", - "value": "1s" - }, - { - "selected": false, - "text": "15s", - "value": "15s" - }, - { - "selected": true, - "text": "30s", - "value": "30s" - }, - { - "selected": false, - "text": "1m", - "value": "1m" - }, - { - "selected": false, - "text": "3m", - "value": "3m" - }, - { - "selected": false, - "text": "5m", - "value": "5m" - } - ], - "query": "1s, 15s, 30s, 1m, 3m, 5m", - "queryValue": "", - "skipUrlSync": false, - "type": "custom" - } - ] - }, - "time": { - "from": "now-15m", - "to": "now" - }, - "timepicker": {}, - "timezone": "", - "title": "Prometheus", - "uid": "k8s_addons_prometheus", - "version": 3, - "weekStart": "" - } -kind: ConfigMap -metadata: - annotations: - grafana_folder: Kubernetes - labels: - grafana_dashboard: "1" - name: dashboards-k8s-addons-prometheus - namespace: futureporn ---- -apiVersion: v1 -data: - k8s-addons-trivy-operator.json: | - { - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": [], - "__requires": [ - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "8.5.0" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "5.0.0" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - }, - { - "type": "panel", - "id": "table", - "name": "Table", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "target": { - "limit": 100, - "matchAny": false, - "tags": [], - "type": "dashboard" - }, - "type": "dashboard" - } - ] - }, - "description": "This is a modern dashboard for the Trivy Operator from Aqua Security. Made to take advantage of the latest Grafana features. GitHub repository: https://github.com/dotdc/grafana-dashboards-kubernetes", - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 1, - "links": [], - "liveNow": false, - "panels": [ - { - "collapsed": false, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 0 - }, - "id": 43, - "panels": [], - "title": "Vulnerabilities", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 0, - "y": 1 - }, - "id": 51, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_image_vulnerabilities{severity=\"Critical\", namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "CRITICAL", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "orange", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 4, - "y": 1 - }, - "id": 50, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_image_vulnerabilities{severity=\"High\", namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "HIGH", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "yellow", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 8, - "y": 1 - }, - "id": 49, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_image_vulnerabilities{severity=\"Medium\", namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "MEDIUM", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "blue", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 12, - "y": 1 - }, - "id": 60, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_image_vulnerabilities{severity=\"Low\", namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "LOW", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "purple", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 16, - "y": 1 - }, - "id": 52, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_image_vulnerabilities{severity=\"Unknown\", namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "UNKNOWN", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "text", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 20, - "y": 1 - }, - "id": 39, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_image_vulnerabilities{namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "TOTAL", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 15, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "linear", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "blue", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 5 - }, - "id": 58, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.5.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_image_vulnerabilities{cluster=~\"$cluster\"}) by (namespace)", - "instant": false, - "interval": "$resolution", - "legendFormat": "{{namespace}}", - "range": true, - "refId": "A" - } - ], - "title": "Total vulnerabilities by namespaces", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 15, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "linear", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "blue", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "Critical" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "red", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "High" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "orange", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Medium" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "yellow", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Low" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "blue", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Unknown" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "purple", - "mode": "fixed" - } - } - ] - } - ] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 5 - }, - "id": 61, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.5.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_image_vulnerabilities{cluster=~\"$cluster\"}) by (severity)", - "instant": false, - "interval": "$resolution", - "legendFormat": "__auto", - "range": true, - "refId": "A" - } - ], - "title": "Total vulnerabilities by severity", - "type": "timeseries" - }, - { - "collapsed": false, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 13 - }, - "id": 85, - "panels": [], - "title": "Vulnerability Details", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "", - "fieldConfig": { - "defaults": { - "custom": { - "align": "auto", - "displayMode": "auto", - "filterable": true, - "inspect": false - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "orange", - "value": 80 - } - ] - } - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "severity" - }, - "properties": [ - { - "id": "mappings", - "value": [ - { - "options": { - "Critical": { - "color": "red", - "index": 0 - }, - "High": { - "color": "orange", - "index": 1 - }, - "Low": { - "color": "blue", - "index": 3 - }, - "Medium": { - "color": "yellow", - "index": 2 - }, - "Unknown": { - "color": "purple", - "index": 4 - } - }, - "type": "value" - } - ] - }, - { - "id": "custom.displayMode", - "value": "color-text" - } - ] - } - ] - }, - "gridPos": { - "h": 12, - "w": 24, - "x": 0, - "y": 14 - }, - "id": 83, - "options": { - "footer": { - "enablePagination": true, - "fields": [ - "Value" - ], - "reducer": [ - "sum" - ], - "show": false - }, - "showHeader": true, - "sortBy": [] - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_image_vulnerabilities{namespace=~\"$namespace\", cluster=~\"$cluster\"}) by (namespace, image_registry, image_repository, image_tag, severity) > 0", - "format": "table", - "instant": false, - "legendFormat": "__auto", - "range": true, - "refId": "A" - } - ], - "title": "Vulnerability count per image and severity in $namespace namespace(s)", - "transformations": [ - { - "id": "organize", - "options": { - "excludeByName": { - "Time": true, - "Value": false - }, - "indexByName": { - "Time": 0, - "Value": 6, - "image_registry": 2, - "image_repository": 3, - "image_tag": 4, - "namespace": 1, - "severity": 5 - }, - "renameByName": { - "Value": "Nb of vulnerabilities", - "image_registry": "Image Registry", - "image_repository": "Image Repository", - "image_tag": "Image Tag", - "namespace": "Namespace", - "severity": "Severity" - } - } - }, - { - "id": "groupBy", - "options": { - "fields": { - "All values": { - "aggregations": [], - "operation": "groupby" - }, - "Count": { - "aggregations": [], - "operation": "groupby" - }, - "Image Registry": { - "aggregations": [], - "operation": "groupby" - }, - "Image Repository": { - "aggregations": [], - "operation": "groupby" - }, - "Image Tag": { - "aggregations": [], - "operation": "groupby" - }, - "Namespace": { - "aggregations": [], - "operation": "groupby" - }, - "Nb of vulnerabilities": { - "aggregations": [], - "operation": "groupby" - }, - "Severity": { - "aggregations": [], - "operation": "groupby" - }, - "Value": { - "aggregations": [], - "operation": "groupby" - }, - "image_registry": { - "aggregations": [], - "operation": "groupby" - }, - "image_repository": { - "aggregations": [], - "operation": "groupby" - }, - "image_tag": { - "aggregations": [], - "operation": "groupby" - }, - "namespace": { - "aggregations": [], - "operation": "groupby" - }, - "severity": { - "aggregations": [], - "operation": "groupby" - } - } - } - } - ], - "type": "table" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "Require operator.metricsVulnIdEnabled: true", - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "custom": { - "align": "auto", - "displayMode": "auto", - "filterable": true, - "inspect": false - }, - "links": [], - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "blue", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "severity" - }, - "properties": [ - { - "id": "mappings", - "value": [ - { - "options": { - "Critical": { - "color": "red", - "index": 0 - }, - "High": { - "color": "orange", - "index": 1 - }, - "Low": { - "color": "blue", - "index": 3 - }, - "Medium": { - "color": "yellow", - "index": 2 - }, - "Unknown": { - "color": "purple", - "index": 4 - } - }, - "type": "value" - } - ] - }, - { - "id": "custom.displayMode", - "value": "color-text" - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "vuln_id" - }, - "properties": [ - { - "id": "links", - "value": [ - { - "targetBlank": true, - "title": "https://nvd.nist.gov/vuln/detail/${__value.text}", - "url": "https://nvd.nist.gov/vuln/detail/${__value.text}" - } - ] - } - ] - } - ] - }, - "gridPos": { - "h": 12, - "w": 24, - "x": 0, - "y": 26 - }, - "id": 78, - "options": { - "footer": { - "enablePagination": true, - "fields": "", - "reducer": [ - "sum" - ], - "show": false - }, - "showHeader": true, - "sortBy": [] - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_vulnerability_id{vuln_id=~\"CVE.*\", namespace=~\"$namespace\", cluster=~\"$cluster\"}) by (namespace, image_registry, image_repository, image_tag, vuln_id, severity)", - "format": "table", - "instant": false, - "interval": "$resolution", - "legendFormat": "__auto", - "range": true, - "refId": "A" - } - ], - "title": "Detaillled CVE vulnerabilities in $namespace namespace(s)", - "transformations": [ - { - "id": "organize", - "options": { - "excludeByName": { - "Time": true, - "Value": true, - "__name__": true, - "container": true, - "endpoint": true, - "instance": true, - "job": true, - "namespace": false, - "service": true - }, - "indexByName": { - "Time": 0, - "Value": 7, - "image_registry": 2, - "image_repository": 3, - "image_tag": 4, - "namespace": 1, - "severity": 6, - "vuln_id": 5 - }, - "renameByName": { - "image_namespace": "namespace", - "image_registry": "Image Registry", - "image_repository": "Image Repository", - "image_tag": "Image Tag", - "namespace": "Namespace", - "severity": "Severity", - "vuln_id": "Vulnerability", - "vulnerability_id": "" - } - } - }, - { - "id": "groupBy", - "options": { - "fields": { - "Image Registry": { - "aggregations": [], - "operation": "groupby" - }, - "Image Repository": { - "aggregations": [], - "operation": "groupby" - }, - "Image Tag": { - "aggregations": [], - "operation": "groupby" - }, - "Namespace": { - "aggregations": [], - "operation": "groupby" - }, - "Severity": { - "aggregations": [], - "operation": "groupby" - }, - "Value": { - "aggregations": [ - "lastNotNull" - ] - }, - "Vulnerability": { - "aggregations": [], - "operation": "groupby" - }, - "image_namespace": { - "aggregations": [], - "operation": "groupby" - }, - "namespace": { - "aggregations": [], - "operation": "groupby" - }, - "severity": { - "aggregations": [], - "operation": "groupby" - }, - "vuln_id": { - "aggregations": [], - "operation": "groupby" - }, - "vulnerability_id": { - "aggregations": [], - "operation": "groupby" - } - } - } - } - ], - "type": "table" - }, - { - "collapsed": false, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 38 - }, - "id": 47, - "panels": [], - "title": "Config Audit Reports", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 0, - "y": 39 - }, - "id": 56, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_resource_configaudits{severity=\"Critical\", namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "CRITICAL", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "orange", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 4, - "y": 39 - }, - "id": 55, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_resource_configaudits{severity=\"High\", namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "HIGH", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "yellow", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 8, - "y": 39 - }, - "id": 54, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_resource_configaudits{severity=\"Medium\", namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "MEDIUM", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "blue", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 12, - "y": 39 - }, - "id": 53, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_resource_configaudits{severity=\"Low\", namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "LOW", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "text", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 16, - "y": 39 - }, - "id": 65, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_resource_configaudits{namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "TOTAL", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 15, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "linear", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "blue", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 43 - }, - "id": 62, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.5.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_resource_configaudits{cluster=~\"$cluster\"}) by (namespace)", - "instant": false, - "interval": "$resolution", - "legendFormat": "__auto", - "range": true, - "refId": "A" - } - ], - "title": "Total config audit report by namespaces", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 15, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "linear", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "blue", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "Critical" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "red", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "High" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "orange", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Medium" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "yellow", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Low" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "blue", - "mode": "fixed" - } - } - ] - } - ] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 43 - }, - "id": 63, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.5.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_resource_configaudits{cluster=~\"$cluster\"}) by (severity)", - "instant": false, - "interval": "$resolution", - "legendFormat": "__auto", - "range": true, - "refId": "A" - } - ], - "title": "Total config audit report by severity", - "type": "timeseries" - }, - { - "collapsed": false, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 51 - }, - "id": 68, - "panels": [], - "title": "RBAC Assessments", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 0, - "y": 52 - }, - "id": 72, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_role_rbacassessments{severity=\"Critical\", namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "CRITICAL", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "orange", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 4, - "y": 52 - }, - "id": 71, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_role_rbacassessments{severity=\"High\", namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "HIGH", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "yellow", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 8, - "y": 52 - }, - "id": 70, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_role_rbacassessments{severity=\"Medium\", namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "MEDIUM", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "blue", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 12, - "y": 52 - }, - "id": 69, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_role_rbacassessments{severity=\"Low\", namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "LOW", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "text", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 16, - "y": 52 - }, - "id": 73, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "textMode": "auto" - }, - "pluginVersion": "9.3.8", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_role_rbacassessments{namespace=~\"$namespace\", cluster=~\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "__auto", - "refId": "A" - } - ], - "title": "TOTAL", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 15, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "linear", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "blue", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 56 - }, - "id": 74, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.5.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_role_rbacassessments{cluster=~\"$cluster\"}) by (namespace)", - "instant": false, - "interval": "$resolution", - "legendFormat": "__auto", - "range": true, - "refId": "A" - } - ], - "title": "Total RBAC Assessments by namespaces", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 15, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "linear", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "blue", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "Critical" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "red", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "High" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "orange", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Medium" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "yellow", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Low" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "blue", - "mode": "fixed" - } - } - ] - } - ] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 56 - }, - "id": 75, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.5.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_role_rbacassessments{cluster=~\"$cluster\"}) by (severity)", - "instant": false, - "interval": "$resolution", - "legendFormat": "__auto", - "range": true, - "refId": "A" - } - ], - "title": "Total RBAC Assessments by severity", - "type": "timeseries" - }, - { - "collapsed": false, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 64 - }, - "id": 81, - "panels": [], - "title": "Exposed Secrets", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 15, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "linear", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "blue", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 24, - "x": 0, - "y": 65 - }, - "id": 76, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.5.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(trivy_image_exposedsecrets{cluster=~\"$cluster\"}) by (namespace)", - "instant": false, - "interval": "$resolution", - "legendFormat": "__auto", - "range": true, - "refId": "A" - } - ], - "title": "Total Exposed Secrets by namespaces", - "type": "timeseries" - } - ], - "refresh": "30s", - "schemaVersion": 37, - "style": "dark", - "tags": [ - "Prometheus", - "Addons", - "Trivy", - "Trivy-operator" - ], - "templating": { - "list": [ - { - "current": { - "selected": false, - "text": "Prometheus", - "value": "Prometheus" - }, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "datasource", - "options": [], - "query": "prometheus", - "queryValue": "", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "current": { - "isNone": true, - "selected": false, - "text": "None", - "value": "" - }, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_node_info,cluster)", - "hide": 0, - "includeAll": false, - "multi": false, - "name": "cluster", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(kube_node_info,cluster)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - }, - { - "current": { - "selected": true, - "text": [ - "All" - ], - "value": [ - "$__all" - ] - }, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_pod_info{cluster=\"$cluster\"}, namespace)", - "hide": 0, - "includeAll": true, - "multi": true, - "name": "namespace", - "options": [], - "query": { - "query": "label_values(kube_pod_info{cluster=\"$cluster\"}, namespace)", - "refId": "StandardVariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - }, - { - "current": { - "selected": true, - "text": "30s", - "value": "30s" - }, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "resolution", - "options": [ - { - "selected": false, - "text": "1s", - "value": "1s" - }, - { - "selected": false, - "text": "15s", - "value": "15s" - }, - { - "selected": true, - "text": "30s", - "value": "30s" - }, - { - "selected": false, - "text": "1m", - "value": "1m" - }, - { - "selected": false, - "text": "3m", - "value": "3m" - }, - { - "selected": false, - "text": "5m", - "value": "5m" - } - ], - "query": "1s, 15s, 30s, 1m, 3m, 5m", - "queryValue": "", - "skipUrlSync": false, - "type": "custom" - } - ] - }, - "time": { - "from": "now-1h", - "to": "now" - }, - "timepicker": {}, - "timezone": "", - "title": "Trivy Operator - Vulnerabilities", - "uid": "security_trivy_operator", - "version": 12, - "weekStart": "" - } -kind: ConfigMap -metadata: - annotations: - grafana_folder: Kubernetes - labels: - grafana_dashboard: "1" - name: dashboards-k8s-addons-trivy-operator - namespace: futureporn ---- -apiVersion: v1 -data: - k8s-system-api-server.json: | - { - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": [], - "__requires": [ - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "8.4.4" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "5.0.0" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "target": { - "limit": 100, - "matchAny": false, - "tags": [], - "type": "dashboard" - }, - "type": "dashboard" - } - ] - }, - "description": "This is a modern API Server dashboard for your Kubernetes cluster(s). Made for kube-prometheus-stack and take advantage of the latest Grafana features. GitHub repository: https://github.com/dotdc/grafana-dashboards-kubernetes", - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 1, - "links": [], - "liveNow": false, - "panels": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [ - { - "options": { - "0": { - "text": "DOWN" - }, - "1": { - "text": "UP" - } - }, - "type": "value" - } - ], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "red", - "value": null - }, - { - "color": "green", - "value": 1 - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 0 - }, - "id": 42, - "options": { - "colorMode": "background", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "horizontal", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "textMode": "value_and_name" - }, - "pluginVersion": "10.0.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "up{job=\"apiserver\", cluster=~\"$cluster\"}", - "interval": "", - "legendFormat": "{{ instance }}", - "refId": "A" - } - ], - "title": "API Server - Health Status", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "custom": { - "align": "auto", - "cellOptions": { - "type": "auto" - }, - "inspect": false - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - } - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "__name__" - }, - "properties": [ - { - "id": "custom.width", - "value": 188 - } - ] - } - ] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 0 - }, - "id": 60, - "options": { - "cellHeight": "sm", - "footer": { - "countRows": false, - "fields": "", - "reducer": [ - "sum" - ], - "show": false - }, - "showHeader": true, - "sortBy": [ - { - "desc": false, - "displayName": "removed_release" - } - ] - }, - "pluginVersion": "10.0.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "apiserver_requested_deprecated_apis{cluster=~\"$cluster\"}", - "interval": "", - "legendFormat": "", - "refId": "A" - } - ], - "title": "Deprecated Kubernetes Resources", - "transformations": [ - { - "id": "labelsToFields", - "options": { - "keepLabels": [ - "group", - "job", - "removed_release", - "resource", - "version", - "name" - ], - "mode": "columns" - } - }, - { - "id": "merge", - "options": {} - }, - { - "id": "organize", - "options": { - "excludeByName": { - "Time": true, - "Value": true, - "job": true - }, - "indexByName": { - "Time": 6, - "Value": 7, - "group": 1, - "job": 5, - "namespace": 0, - "removed_release": 4, - "resource": 3, - "version": 2 - }, - "renameByName": {} - } - }, - { - "id": "groupBy", - "options": { - "fields": { - "group": { - "aggregations": [ - "lastNotNull" - ], - "operation": "groupby" - }, - "job": { - "aggregations": [], - "operation": "groupby" - }, - "namespace": { - "aggregations": [ - "lastNotNull" - ], - "operation": "groupby" - }, - "removed_release": { - "aggregations": [], - "operation": "groupby" - }, - "resource": { - "aggregations": [ - "lastNotNull" - ], - "operation": "groupby" - }, - "version": { - "aggregations": [], - "operation": "groupby" - } - } - } - } - ], - "type": "table" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 8 - }, - "id": 38, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum by (code) (rate(apiserver_request_total{cluster=~\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "{{ code }}", - "refId": "A" - } - ], - "title": "API Server - HTTP Requests by code", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 8 - }, - "id": 39, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum by (verb) (rate(apiserver_request_total{cluster=~\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "{{ verb}}", - "refId": "A" - } - ], - "title": "API Server - HTTP Requests by verb", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "ms" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 16 - }, - "id": 53, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(apiserver_request_duration_seconds_sum{job=\"apiserver\", cluster=~\"$cluster\"}[$__rate_interval])) by (instance)\n/\nsum(rate(apiserver_request_duration_seconds_count{job=\"apiserver\", cluster=~\"$cluster\"}[$__rate_interval])) by (instance)", - "interval": "$resolution", - "legendFormat": "{{ instance }}", - "refId": "A" - } - ], - "title": "API Server - HTTP Requests Latency by instance", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "ms" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 16 - }, - "id": 54, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(apiserver_request_duration_seconds_sum{job=\"apiserver\", cluster=~\"$cluster\"}[$__rate_interval])) by (verb)\n/\nsum(rate(apiserver_request_duration_seconds_count{job=\"apiserver\", cluster=~\"$cluster\"}[$__rate_interval])) by (verb)", - "interval": "$resolution", - "legendFormat": "{{ verb }}", - "refId": "A" - } - ], - "title": "API Server - HTTP Requests Latency by verb", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 24 - }, - "id": 50, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum by(instance) (rate(apiserver_request_total{code=~\"5..\", job=\"apiserver\", cluster=~\"$cluster\"}[$__rate_interval]))\n / sum by(instance) (rate(apiserver_request_total{job=\"apiserver\", cluster=~\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "{{ instance }}", - "refId": "A" - } - ], - "title": "API Server - Errors by Instance", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 24 - }, - "id": 51, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum by(verb) (rate(apiserver_request_total{code=~\"5..\",job=\"apiserver\", cluster=~\"$cluster\"}[$__rate_interval]))\n / sum by(verb) (rate(apiserver_request_total{job=\"apiserver\", cluster=~\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "{{ verb }}", - "refId": "A" - } - ], - "title": "API Server - Errors by verb", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "normal" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 32 - }, - "id": 40, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(apiserver_request_total{cluster=~\"$cluster\"}[$__rate_interval])) by (instance)", - "interval": "$resolution", - "legendFormat": "{{ instance }}", - "refId": "A" - } - ], - "title": "API Server - Stacked HTTP Requests by instance", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 32 - }, - "id": 56, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(workqueue_depth{job=\"apiserver\", cluster=~\"$cluster\"}[$__rate_interval])) by (instance)", - "interval": "$resolution", - "legendFormat": "{{ instance }}", - "refId": "A" - } - ], - "title": "API Server - Work Queue by instance", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 40 - }, - "id": 47, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "rate(process_cpu_seconds_total{job=\"apiserver\", cluster=~\"$cluster\"}[$__rate_interval])", - "interval": "$resolution", - "legendFormat": "{{ instance }}", - "refId": "A" - } - ], - "title": "API Server - CPU Usage by instance", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 40 - }, - "id": 48, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "process_resident_memory_bytes{job=\"apiserver\", cluster=~\"$cluster\"}", - "interval": "$resolution", - "legendFormat": "{{ instance }}", - "refId": "A" - } - ], - "title": "API Server - Memory Usage by instance", - "type": "timeseries" - } - ], - "refresh": "30s", - "schemaVersion": 38, - "style": "dark", - "tags": [ - "Kubernetes", - "Prometheus" - ], - "templating": { - "list": [ - { - "current": { - "selected": false, - "text": "Prometheus", - "value": "Prometheus" - }, - "hide": 0, - "includeAll": false, - "label": "", - "multi": false, - "name": "datasource", - "options": [], - "query": "prometheus", - "queryValue": "", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "current": { - "isNone": true, - "selected": false, - "text": "None", - "value": "" - }, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_node_info,cluster)", - "hide": 0, - "includeAll": false, - "multi": false, - "name": "cluster", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(kube_node_info,cluster)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - }, - { - "current": { - "selected": true, - "text": "30s", - "value": "30s" - }, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "resolution", - "options": [ - { - "selected": false, - "text": "1s", - "value": "1s" - }, - { - "selected": false, - "text": "15s", - "value": "15s" - }, - { - "selected": true, - "text": "30s", - "value": "30s" - }, - { - "selected": false, - "text": "1m", - "value": "1m" - }, - { - "selected": false, - "text": "3m", - "value": "3m" - }, - { - "selected": false, - "text": "5m", - "value": "5m" - } - ], - "query": "1s, 15s, 30s, 1m, 3m, 5m", - "queryValue": "", - "skipUrlSync": false, - "type": "custom" - } - ] - }, - "time": { - "from": "now-1h", - "to": "now" - }, - "timepicker": {}, - "timezone": "", - "title": "Kubernetes / System / API Server", - "uid": "k8s_system_apisrv", - "version": 16, - "weekStart": "" - } -kind: ConfigMap -metadata: - annotations: - grafana_folder: Kubernetes - labels: - grafana_dashboard: "1" - name: dashboards-k8s-system-api-server - namespace: futureporn ---- -apiVersion: v1 -data: - k8s-system-coredns.json: | - { - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": [], - "__requires": [ - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "8.4.4" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "5.0.0" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "target": { - "limit": 100, - "matchAny": false, - "tags": [], - "type": "dashboard" - }, - "type": "dashboard" - } - ] - }, - "description": "This is a modern CoreDNS dashboard for your Kubernetes cluster(s). Made for kube-prometheus-stack and take advantage of the latest Grafana features. GitHub repository: https://github.com/dotdc/grafana-dashboards-kubernetes", - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 1, - "links": [], - "liveNow": false, - "panels": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [ - { - "options": { - "0": { - "text": "DOWN" - }, - "1": { - "text": "UP" - } - }, - "type": "value" - } - ], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "red", - "value": null - }, - { - "color": "green", - "value": 1 - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 3, - "w": 24, - "x": 0, - "y": 0 - }, - "id": 25, - "options": { - "colorMode": "background", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "vertical", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "value_and_name", - "wideLayout": true - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "up{job=~\"$job\", instance=~\"$instance\", cluster=~\"$cluster\"}", - "interval": "", - "legendFormat": "{{ instance }}", - "refId": "A" - } - ], - "title": "CoreDNS - Health Status", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 3 - }, - "id": 19, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "rate(process_cpu_seconds_total{job=~\"$job\", instance=~\"$instance\", cluster=~\"$cluster\"}[$__rate_interval])", - "interval": "$resolution", - "legendFormat": "{{ instance }}", - "refId": "A" - } - ], - "title": "CoreDNS - CPU Usage by instance", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 3 - }, - "id": 21, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "process_resident_memory_bytes{job=~\"$job\", instance=~\"$instance\", cluster=~\"$cluster\"}", - "interval": "", - "legendFormat": "{{ instance }}", - "refId": "A" - } - ], - "title": "CoreDNS - Memory Usage by instance", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 11 - }, - "id": 9, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(coredns_dns_requests_total{instance=~\"$instance\",proto=\"$protocol\", cluster=~\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "total $protocol requests", - "refId": "A" - } - ], - "title": "CoreDNS - Total DNS Requests ($protocol)", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 11 - }, - "id": 7, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(coredns_dns_request_size_bytes_sum{instance=~\"$instance\",proto=\"$protocol\", cluster=~\"$cluster\"}[$__rate_interval])) by (proto) / sum(rate(coredns_dns_request_size_bytes_count{instance=~\"$instance\",proto=\"$protocol\", cluster=~\"$cluster\"}[$__rate_interval])) by (proto)", - "interval": "$resolution", - "legendFormat": "average $protocol packet size", - "refId": "A" - } - ], - "title": "CoreDNS - Average Packet Size ($protocol)", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 19 - }, - "id": 2, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(coredns_dns_requests_total{instance=~\"$instance\", cluster=~\"$cluster\"}[$__rate_interval])) by (type)", - "interval": "$resolution", - "legendFormat": "{{ type }}", - "refId": "A" - } - ], - "title": "CoreDNS - Requests by type", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 19 - }, - "id": 4, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(coredns_dns_responses_total{instance=~\"$instance\", cluster=~\"$cluster\"}[$__rate_interval])) by (rcode)", - "interval": "$resolution", - "legendFormat": "{{ rcode }}", - "refId": "A" - } - ], - "title": "CoreDNS - Requests by return code", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 27 - }, - "id": 23, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(coredns_forward_requests_total{cluster=~\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "total forward requests", - "refId": "A" - } - ], - "title": "CoreDNS - Total Forward Requests", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 27 - }, - "id": 13, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(coredns_forward_responses_total{rcode=~\"SERVFAIL|REFUSED\", cluster=~\"$cluster\"}[$__rate_interval])) by (rcode)", - "interval": "$resolution", - "legendFormat": "{{ rcode }}", - "refId": "A" - } - ], - "title": "CoreDNS - DNS Errors", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 35 - }, - "id": 17, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(coredns_cache_hits_total{instance=~\"$instance\", cluster=~\"$cluster\"}[$__rate_interval])) by (type)", - "interval": "$resolution", - "legendFormat": "{{ type }}", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(coredns_cache_misses_total{instance=~\"$instance\", cluster=~\"$cluster\"}[$__rate_interval])) by (type)", - "interval": "$resolution", - "legendFormat": "misses", - "refId": "B" - } - ], - "title": "CoreDNS - Cache Hits / Misses", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 35 - }, - "id": 15, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(coredns_cache_entries{cluster=~\"$cluster\"}) by (type)", - "interval": "", - "legendFormat": "{{ type }}", - "refId": "A" - } - ], - "title": "CoreDNS - Cache Size", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "custom": { - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "scaleDistribution": { - "type": "linear" - } - } - }, - "overrides": [] - }, - "gridPos": { - "h": 10, - "w": 12, - "x": 0, - "y": 43 - }, - "id": 27, - "options": { - "calculate": false, - "cellGap": 1, - "color": { - "exponent": 0.5, - "fill": "dark-orange", - "mode": "scheme", - "reverse": false, - "scale": "exponential", - "scheme": "RdYlBu", - "steps": 64 - }, - "exemplars": { - "color": "rgba(255,0,255,0.7)" - }, - "filterValues": { - "le": 1e-9 - }, - "legend": { - "show": true - }, - "rowsFrame": { - "layout": "auto" - }, - "tooltip": { - "mode": "single", - "showColorScale": false, - "yHistogram": false - }, - "yAxis": { - "axisPlacement": "left", - "reverse": false, - "unit": "s" - } - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(increase(coredns_dns_request_duration_seconds_bucket{instance=~\"$instance\", cluster=~\"$cluster\"}[$__rate_interval])) by (le)", - "format": "heatmap", - "legendFormat": "{{le}}", - "range": true, - "refId": "A" - } - ], - "title": "CoreDNS - DNS request duration", - "type": "heatmap" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "custom": { - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "scaleDistribution": { - "type": "linear" - } - } - }, - "overrides": [] - }, - "gridPos": { - "h": 10, - "w": 12, - "x": 12, - "y": 43 - }, - "id": 28, - "options": { - "calculate": false, - "cellGap": 1, - "color": { - "exponent": 0.5, - "fill": "dark-orange", - "mode": "scheme", - "reverse": false, - "scale": "exponential", - "scheme": "RdYlBu", - "steps": 64 - }, - "exemplars": { - "color": "rgba(255,0,255,0.7)" - }, - "filterValues": { - "le": 1e-9 - }, - "legend": { - "show": true - }, - "rowsFrame": { - "layout": "auto" - }, - "tooltip": { - "mode": "single", - "showColorScale": false, - "yHistogram": false - }, - "yAxis": { - "axisPlacement": "left", - "reverse": false, - "unit": "decbytes" - } - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(increase(coredns_dns_request_size_bytes_bucket{instance=~\"$instance\", le!=\"0\", cluster=~\"$cluster\"}[$__rate_interval])) by (le)", - "format": "heatmap", - "legendFormat": "{{le}}", - "range": true, - "refId": "A" - } - ], - "title": "CoreDNS - DNS request size", - "type": "heatmap" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "custom": { - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "scaleDistribution": { - "type": "linear" - } - } - }, - "overrides": [] - }, - "gridPos": { - "h": 10, - "w": 12, - "x": 0, - "y": 53 - }, - "id": 29, - "options": { - "calculate": false, - "cellGap": 1, - "color": { - "exponent": 0.5, - "fill": "dark-orange", - "mode": "scheme", - "reverse": false, - "scale": "exponential", - "scheme": "RdYlBu", - "steps": 64 - }, - "exemplars": { - "color": "rgba(255,0,255,0.7)" - }, - "filterValues": { - "le": 1e-9 - }, - "legend": { - "show": true - }, - "rowsFrame": { - "layout": "auto" - }, - "tooltip": { - "mode": "single", - "showColorScale": false, - "yHistogram": false - }, - "yAxis": { - "axisPlacement": "left", - "reverse": false, - "unit": "decbytes" - } - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(increase(coredns_dns_response_size_bytes_bucket{instance=~\"$instance\", le!=\"0\", cluster=~\"$cluster\"}[$__rate_interval])) by (le)", - "format": "heatmap", - "legendFormat": "{{le}}", - "range": true, - "refId": "A" - } - ], - "title": "CoreDNS - DNS response size", - "type": "heatmap" - } - ], - "refresh": "30s", - "schemaVersion": 39, - "tags": [ - "Kubernetes", - "Prometheus" - ], - "templating": { - "list": [ - { - "current": { - "selected": false, - "text": "Prometheus", - "value": "Prometheus" - }, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "datasource", - "options": [], - "query": "prometheus", - "queryValue": "", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "current": { - "isNone": true, - "selected": false, - "text": "None", - "value": "" - }, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_node_info,cluster)", - "hide": 0, - "includeAll": false, - "multi": false, - "name": "cluster", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(kube_node_info,cluster)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - }, - { - "current": { - "selected": false, - "text": "All", - "value": "$__all" - }, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(up{job=\"$job\", cluster=\"$cluster\"},instance)", - "hide": 0, - "includeAll": true, - "label": "", - "multi": false, - "name": "instance", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(up{job=\"$job\", cluster=\"$cluster\"},instance)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "tagValuesQuery": "", - "tagsQuery": "", - "type": "query", - "useTags": false - }, - { - "allValue": "udp,tcp", - "current": { - "selected": false, - "text": "udp", - "value": "udp" - }, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(coredns_dns_requests_total{cluster=\"$cluster\"}, proto)", - "hide": 0, - "includeAll": false, - "label": "", - "multi": false, - "name": "protocol", - "options": [], - "query": { - "query": "label_values(coredns_dns_requests_total{cluster=\"$cluster\"}, proto)", - "refId": "StandardVariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "tagValuesQuery": "", - "tagsQuery": "", - "type": "query", - "useTags": false - }, - { - "current": { - "selected": false, - "text": "30s", - "value": "30s" - }, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "resolution", - "options": [ - { - "selected": false, - "text": "1s", - "value": "1s" - }, - { - "selected": false, - "text": "15s", - "value": "15s" - }, - { - "selected": true, - "text": "30s", - "value": "30s" - }, - { - "selected": false, - "text": "1m", - "value": "1m" - }, - { - "selected": false, - "text": "3m", - "value": "3m" - }, - { - "selected": false, - "text": "5m", - "value": "5m" - } - ], - "query": "1s, 15s, 30s, 1m, 3m, 5m", - "queryValue": "", - "skipUrlSync": false, - "type": "custom" - }, - { - "current": { - "selected": true, - "text": [ - "coredns" - ], - "value": [ - "coredns" - ] - }, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(coredns_build_info{cluster=\"$cluster\"},job)", - "hide": 0, - "includeAll": false, - "multi": true, - "name": "job", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(coredns_build_info{cluster=\"$cluster\"},job)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - } - ] - }, - "time": { - "from": "now-1h", - "to": "now" - }, - "timepicker": {}, - "timezone": "", - "title": "Kubernetes / System / CoreDNS", - "uid": "k8s_system_coredns", - "version": 18, - "weekStart": "" - } -kind: ConfigMap -metadata: - annotations: - grafana_folder: Kubernetes - labels: - grafana_dashboard: "1" - name: dashboards-k8s-system-coredns - namespace: futureporn ---- -apiVersion: v1 -data: - k8s-views-global.json: | - { - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": {}, - "__requires": [ - { - "type": "panel", - "id": "bargauge", - "name": "Bar gauge", - "version": "" - }, - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "10.3.1" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "1.0.0" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "target": { - "limit": 100, - "matchAny": false, - "tags": [], - "type": "dashboard" - }, - "type": "dashboard" - } - ] - }, - "description": "This is a modern 'Global View' dashboard for your Kubernetes cluster(s). Made for kube-prometheus-stack and take advantage of the latest Grafana features. GitHub repository: https://github.com/dotdc/grafana-dashboards-kubernetes", - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 1, - "links": [], - "liveNow": false, - "panels": [ - { - "collapsed": false, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 0 - }, - "id": 67, - "panels": [], - "title": "Overview", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "continuous-GrYlRd" - }, - "mappings": [], - "max": 1, - "min": 0, - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 6, - "x": 0, - "y": 1 - }, - "id": 77, - "options": { - "displayMode": "lcd", - "maxVizHeight": 300, - "minVizHeight": 10, - "minVizWidth": 0, - "namePlacement": "auto", - "orientation": "horizontal", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showUnfilled": true, - "sizing": "auto", - "valueMode": "color" - }, - "pluginVersion": "10.4.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "avg(sum by (instance, cpu) (rate(node_cpu_seconds_total{mode!~\"idle|iowait|steal\", cluster=\"$cluster\", job=\"$job\"}[$__rate_interval])))", - "interval": "", - "legendFormat": "Real Linux", - "range": true, - "refId": "Real Linux" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "avg(sum by (core) (rate(windows_cpu_time_total{mode!=\"idle\", cluster=\"$cluster\"}[$__rate_interval])))", - "hide": false, - "interval": "", - "legendFormat": "Real Windows", - "range": true, - "refId": "Real Windows" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_resource_requests{resource=\"cpu\", cluster=\"$cluster\"}) / sum(machine_cpu_cores{cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Requests", - "range": true, - "refId": "Requests" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_resource_limits{resource=\"cpu\", cluster=\"$cluster\"}) / sum(machine_cpu_cores{cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Limits", - "range": true, - "refId": "Limits" - } - ], - "title": "Global CPU Usage", - "transformations": [ - { - "id": "calculateField", - "options": { - "alias": "Real", - "mode": "reduceRow", - "reduce": { - "include": [ - "Real Linux", - "Real Windows" - ], - "reducer": "mean" - } - } - }, - { - "id": "organize", - "options": { - "excludeByName": { - "Real Linux": true, - "Real Windows": true, - "Time": true - }, - "indexByName": { - "Limits": 5, - "Real": 1, - "Real Linux": 2, - "Real Windows": 3, - "Requests": 4, - "Time": 0 - }, - "renameByName": {} - } - } - ], - "type": "bargauge" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "continuous-GrYlRd" - }, - "decimals": 2, - "mappings": [], - "max": 1, - "min": 0, - "thresholds": { - "mode": "percentage", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 6, - "x": 6, - "y": 1 - }, - "id": 78, - "options": { - "displayMode": "lcd", - "maxVizHeight": 300, - "minVizHeight": 10, - "minVizWidth": 0, - "namePlacement": "auto", - "orientation": "horizontal", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showUnfilled": true, - "sizing": "auto", - "text": {}, - "valueMode": "color" - }, - "pluginVersion": "10.4.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(node_memory_MemTotal_bytes{cluster=\"$cluster\", job=\"$job\"} - node_memory_MemAvailable_bytes{cluster=\"$cluster\", job=\"$job\"}) / sum(node_memory_MemTotal_bytes{cluster=\"$cluster\", job=\"$job\"})", - "hide": false, - "interval": "", - "legendFormat": "Real Linux", - "range": true, - "refId": "Real Linux" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(windows_memory_available_bytes{cluster=\"$cluster\"} + windows_memory_cache_bytes{cluster=\"$cluster\"}) / sum(windows_os_visible_memory_bytes{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Real Windows", - "range": true, - "refId": "Real Windows" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_resource_requests{resource=\"memory\", cluster=\"$cluster\"}) / sum(machine_memory_bytes{cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Requests", - "range": true, - "refId": "Requests" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_resource_limits{resource=\"memory\", cluster=\"$cluster\"}) / sum(machine_memory_bytes{cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Limits", - "range": true, - "refId": "Limits" - } - ], - "title": "Global RAM Usage", - "transformations": [ - { - "id": "calculateField", - "options": { - "alias": "Real", - "mode": "reduceRow", - "reduce": { - "include": [ - "Real Linux", - "Real Windows" - ], - "reducer": "mean" - } - } - }, - { - "id": "organize", - "options": { - "excludeByName": { - "Real Linux": true, - "Real Windows": true, - "Time": true - }, - "includeByName": {}, - "indexByName": { - "Limits": 5, - "Real": 3, - "Real Linux": 1, - "Real Windows": 2, - "Requests": 4, - "Time": 0 - }, - "renameByName": {} - } - } - ], - "type": "bargauge" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "blue", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 2, - "x": 12, - "y": 1 - }, - "id": 63, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "text": {}, - "textMode": "value", - "wideLayout": true - }, - "pluginVersion": "10.4.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "count(count by (node) (kube_node_info{cluster=\"$cluster\"}))", - "interval": "", - "legendFormat": "", - "range": true, - "refId": "A" - } - ], - "title": "Nodes", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 12, - "w": 10, - "x": 14, - "y": 1 - }, - "id": 52, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(kube_namespace_labels{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Namespaces", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_pod_container_status_running{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Running Containers", - "refId": "B" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_pod_status_phase{phase=\"Running\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Running Pods", - "refId": "O" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_service_info{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Services", - "refId": "C" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_endpoint_info{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Endpoints", - "refId": "D" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_ingress_info{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Ingresses", - "refId": "E" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_deployment_labels{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Deployments", - "refId": "F" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_statefulset_labels{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Statefulsets", - "refId": "G" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_daemonset_labels{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Daemonsets", - "refId": "H" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_persistentvolumeclaim_info{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Persistent Volume Claims", - "refId": "I" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_hpa_labels{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Horizontal Pod Autoscalers", - "refId": "J" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_configmap_info{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Configmaps", - "refId": "K" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_secret_info{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Secrets", - "refId": "L" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_networkpolicy_labels{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Network Policies", - "refId": "M" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "count(count by (node) (kube_node_info{cluster=\"$cluster\"}))", - "hide": false, - "interval": "", - "legendFormat": "Nodes", - "refId": "N" - } - ], - "title": "Kubernetes Resource Count", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "blue", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 2, - "x": 12, - "y": 5 - }, - "id": 59, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "text": {}, - "textMode": "value", - "wideLayout": true - }, - "pluginVersion": "10.4.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "count(kube_namespace_created{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "", - "refId": "A" - } - ], - "title": "Namespaces", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "rgb(255, 255, 255)", - "value": null - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 6, - "x": 0, - "y": 9 - }, - "id": 37, - "options": { - "colorMode": "none", - "graphMode": "none", - "justifyMode": "center", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "text": {}, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "10.4.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(node_cpu_seconds_total{mode!~\"idle|iowait|steal\", cluster=\"$cluster\", job=\"$job\"}[$__rate_interval]))", - "interval": "", - "legendFormat": "Real Linux", - "range": true, - "refId": "Real Linux" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(windows_cpu_time_total{mode!=\"idle\", cluster=\"$cluster\"}[$__rate_interval]))", - "hide": false, - "interval": "", - "legendFormat": "Real Windows", - "range": true, - "refId": "Real Windows" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_resource_requests{resource=\"cpu\", cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Requests", - "range": true, - "refId": "Requests" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_resource_limits{resource=\"cpu\", cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Limits", - "range": true, - "refId": "Limits" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(machine_cpu_cores{cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Total", - "range": true, - "refId": "Total" - } - ], - "title": "CPU Usage", - "transformations": [ - { - "id": "calculateField", - "options": { - "alias": "Real", - "mode": "reduceRow", - "reduce": { - "include": [ - "Real Linux", - "Real Windows" - ], - "reducer": "sum" - } - } - }, - { - "id": "organize", - "options": { - "excludeByName": { - "Real Linux": true, - "Real Windows": true, - "Time": true, - "Total Linux": true, - "Total Windows": true - }, - "indexByName": { - "Limits": 5, - "Real": 3, - "Real Linux": 1, - "Real Windows": 2, - "Requests": 4, - "Time": 0, - "Total": 8, - "Total Linux": 6, - "Total Windows": 7 - }, - "renameByName": {} - } - } - ], - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "rgb(255, 255, 255)", - "value": null - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 6, - "x": 6, - "y": 9 - }, - "id": 39, - "options": { - "colorMode": "none", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "text": {}, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "10.4.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(node_memory_MemTotal_bytes{cluster=\"$cluster\", job=\"$job\"} - node_memory_MemAvailable_bytes{cluster=\"$cluster\", job=\"$job\"})", - "interval": "", - "legendFormat": "Real Linux", - "range": true, - "refId": "Real Linux" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(windows_os_visible_memory_bytes{cluster=\"$cluster\"} - windows_memory_available_bytes{cluster=\"$cluster\"} - windows_memory_cache_bytes{cluster=\"$cluster\"})", - "hide": false, - "interval": "", - "legendFormat": "Real Windows", - "range": true, - "refId": "Real Windows" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_resource_requests{resource=\"memory\", cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Requests", - "range": true, - "refId": "Requests" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_resource_limits{resource=\"memory\", cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Limits", - "range": true, - "refId": "Limits" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(machine_memory_bytes{cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Total", - "range": true, - "refId": "Total" - } - ], - "title": "RAM Usage", - "transformations": [ - { - "id": "calculateField", - "options": { - "alias": "Real", - "mode": "reduceRow", - "reduce": { - "include": [ - "Real Linux", - "Real Windows" - ], - "reducer": "mean" - } - } - }, - { - "id": "organize", - "options": { - "excludeByName": { - "Real Linux": true, - "Real Windows": true, - "Time": true - }, - "includeByName": {}, - "indexByName": { - "Limits": 5, - "Real": 3, - "Real Linux": 1, - "Real Windows": 2, - "Requests": 4, - "Time": 0, - "Total": 6 - }, - "renameByName": {} - } - } - ], - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "blue", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 2, - "x": 12, - "y": 9 - }, - "id": 62, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "text": {}, - "textMode": "value", - "wideLayout": true - }, - "pluginVersion": "10.4.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_pod_status_phase{phase=\"Running\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "", - "refId": "A" - } - ], - "title": "Running Pods", - "type": "stat" - }, - { - "collapsed": false, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 13 - }, - "id": 71, - "panels": [], - "title": "Resources", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "continuous-GrYlRd", - "seriesBy": "last" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "CPU %", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 10, - "gradientMode": "scheme", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "yellow", - "value": 0.5 - }, - { - "color": "red", - "value": 0.7 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 14 - }, - "id": 72, - "options": { - "legend": { - "calcs": [], - "displayMode": "hidden", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "avg(sum by (instance, cpu) (rate(node_cpu_seconds_total{mode!~\"idle|iowait|steal\", cluster=\"$cluster\", job=\"$job\"}[$__rate_interval])))", - "interval": "$resolution", - "legendFormat": "Linux", - "range": true, - "refId": "Linux" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "1 - avg(rate(windows_cpu_time_total{cluster=\"$cluster\",mode=\"idle\"}[$__rate_interval]))", - "hide": false, - "interval": "$resolution", - "legendFormat": "Windows", - "range": true, - "refId": "Windows" - } - ], - "title": "Cluster CPU Utilization", - "transformations": [ - { - "id": "calculateField", - "options": { - "alias": "CPU usage in %", - "mode": "reduceRow", - "reduce": { - "reducer": "mean" - }, - "replaceFields": true - } - } - ], - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "continuous-GrYlRd" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "MEMORY", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 10, - "gradientMode": "scheme", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "#EAB839", - "value": 0.5 - }, - { - "color": "red", - "value": 0.7 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 14 - }, - "id": 55, - "options": { - "legend": { - "calcs": [ - "mean", - "lastNotNull", - "max", - "min" - ], - "displayMode": "hidden", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(node_memory_MemTotal_bytes{cluster=\"$cluster\", job=\"$job\"} - node_memory_MemAvailable_bytes{cluster=\"$cluster\", job=\"$job\"}) / sum(node_memory_MemTotal_bytes{cluster=\"$cluster\", job=\"$job\"})", - "interval": "$resolution", - "legendFormat": "Linux", - "range": true, - "refId": "Linux" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(windows_os_visible_memory_bytes{cluster=\"$cluster\"} - windows_memory_available_bytes{cluster=\"$cluster\"}) / sum(windows_os_visible_memory_bytes{cluster=\"$cluster\"})", - "hide": false, - "interval": "$resolution", - "legendFormat": "Windows", - "range": true, - "refId": "Windows" - } - ], - "title": "Cluster Memory Utilization", - "transformations": [ - { - "id": "calculateField", - "options": { - "alias": "Memory usage in %", - "mode": "reduceRow", - "reduce": { - "reducer": "mean" - }, - "replaceFields": true - } - } - ], - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "CPU CORES", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 22 - }, - "id": 46, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "10.4.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(container_cpu_usage_seconds_total{image!=\"\", cluster=\"$cluster\"}[$__rate_interval])) by (namespace)\n+ on (namespace)\n(sum(rate(windows_container_cpu_usage_seconds_total{container_id!=\"\", cluster=\"$cluster\"}[$__rate_interval]) * on (container_id) group_left (container, pod, namespace) max by ( container, container_id, pod, namespace) (kube_pod_container_info{container_id!=\"\", cluster=\"$cluster\"}) OR kube_namespace_created{cluster=\"$cluster\"} * 0) by (namespace))", - "format": "time_series", - "hide": false, - "interval": "$resolution", - "legendFormat": "{{ namespace }}", - "range": true, - "refId": "A" - } - ], - "title": "CPU Utilization by namespace", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 22 - }, - "id": 50, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(container_memory_working_set_bytes{image!=\"\", cluster=\"$cluster\"}) by (namespace)\n+ on (namespace)\n(sum(windows_container_memory_usage_commit_bytes{container_id!=\"\", cluster=\"$cluster\"} * on (container_id) group_left (container, pod, namespace) max by ( container, container_id, pod, namespace) (kube_pod_container_info{container_id!=\"\", cluster=\"$cluster\"}) OR kube_namespace_created{cluster=\"$cluster\"} * 0) by (namespace))", - "interval": "$resolution", - "legendFormat": "{{ namespace }}", - "range": true, - "refId": "A" - } - ], - "title": "Memory Utilization by namespace", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "CPU %", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 30 - }, - "id": 54, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "avg(sum by (instance, cpu) (rate(node_cpu_seconds_total{mode!~\"idle|iowait|steal\", cluster=\"$cluster\", job=\"$job\"}[$__rate_interval]))) by (instance)", - "interval": "$resolution", - "legendFormat": "{{ node }}", - "range": true, - "refId": "Linux" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "avg(sum by (instance,core) (rate(windows_cpu_time_total{mode!=\"idle\", cluster=\"$cluster\"}[$__rate_interval]))) by (instance)", - "hide": false, - "interval": "$resolution", - "legendFormat": "{{ node }}", - "range": true, - "refId": "Windows" - } - ], - "title": "CPU Utilization by instance", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "MEMORY", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 30 - }, - "id": 73, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(node_memory_MemTotal_bytes{cluster=\"$cluster\", job=\"$job\"} - node_memory_MemAvailable_bytes{cluster=\"$cluster\", job=\"$job\"}) by (instance)", - "hide": false, - "interval": "$resolution", - "legendFormat": "{{ instance }}", - "range": true, - "refId": "Linux" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(windows_os_visible_memory_bytes{cluster=\"$cluster\"} - windows_memory_available_bytes{cluster=\"$cluster\"}) by (instance)", - "hide": false, - "interval": "$resolution", - "legendFormat": "{{ instance }}", - "range": true, - "refId": "Windows" - } - ], - "title": "Memory Utilization by instance", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "No data is generally a good thing here.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "SECONDS", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "s" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 38 - }, - "id": 82, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(container_cpu_cfs_throttled_seconds_total{image!=\"\", cluster=\"$cluster\"}[$__rate_interval])) by (namespace) > 0", - "interval": "$resolution", - "legendFormat": "{{ namespace }}", - "range": true, - "refId": "A" - } - ], - "title": "CPU Throttled seconds by namespace", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "No data is generally a good thing here.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "NB", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 38 - }, - "id": 83, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(node_cpu_core_throttles_total{cluster=\"$cluster\", job=\"$job\"}[$__rate_interval])) by (instance)", - "interval": "$resolution", - "legendFormat": "{{ instance }}", - "range": true, - "refId": "A" - } - ], - "title": "CPU Core Throttled by instance", - "type": "timeseries" - }, - { - "collapsed": false, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 46 - }, - "id": 86, - "panels": [], - "title": "Kubernetes", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 9, - "w": 12, - "x": 0, - "y": 47 - }, - "id": 84, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(kube_pod_status_qos_class{cluster=\"$cluster\"}) by (qos_class)", - "interval": "", - "legendFormat": "{{ qos_class }} pods", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_info{cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Total pods", - "range": true, - "refId": "B" - } - ], - "title": "Kubernetes Pods QoS classes", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 9, - "w": 12, - "x": 12, - "y": 47 - }, - "id": 85, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(kube_pod_status_reason{cluster=\"$cluster\"}) by (reason)", - "interval": "", - "legendFormat": "{{ reason }}", - "range": true, - "refId": "A" - } - ], - "title": "Kubernetes Pods Status Reason", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "No data is generally a good thing here.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "points", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 9, - "w": 12, - "x": 0, - "y": 56 - }, - "id": 87, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(increase(container_oom_events_total{cluster=\"$cluster\"}[$__rate_interval])) by (namespace) > 0", - "interval": "", - "legendFormat": "{{ namespace }}", - "range": true, - "refId": "A" - } - ], - "title": "OOM Events by namespace", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "No data is generally a good thing here.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "points", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 9, - "w": 12, - "x": 12, - "y": 56 - }, - "id": 88, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(increase(kube_pod_container_status_restarts_total{cluster=\"$cluster\"}[$__rate_interval])) by (namespace) > 0", - "interval": "", - "legendFormat": "{{ namespace }}", - "range": true, - "refId": "A" - } - ], - "title": "Container Restarts by namespace", - "type": "timeseries" - }, - { - "collapsed": false, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 65 - }, - "id": 69, - "panels": [], - "title": "Network", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "Dropped noisy virtual devices for readability.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "BANDWIDTH", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 66 - }, - "id": 44, - "options": { - "legend": { - "calcs": [], - "displayMode": "hidden", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(node_network_receive_bytes_total{device!~\"(veth|azv|lxc).*\", cluster=\"$cluster\", job=\"$job\"}[$__rate_interval])) by (device)", - "interval": "$resolution", - "legendFormat": "Received : {{ device }}", - "range": true, - "refId": "Linux Received" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "- sum(rate(node_network_transmit_bytes_total{device!~\"(veth|azv|lxc).*\", cluster=\"$cluster\", job=\"$job\"}[$__rate_interval])) by (device)", - "interval": "$resolution", - "legendFormat": "Transmitted : {{ device }}", - "range": true, - "refId": "Linux Transmitted" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(windows_net_bytes_received_total{cluster=\"$cluster\"}[$__rate_interval])) by (nic)", - "hide": false, - "interval": "$resolution", - "legendFormat": "Received : {{ nic }}", - "range": true, - "refId": "Windows Received" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "- sum(rate(windows_net_bytes_sent_total{cluster=\"$cluster\"}[$__rate_interval])) by (nic)", - "hide": false, - "interval": "$resolution", - "legendFormat": "Transmitted : {{ device }}", - "range": true, - "refId": "Windows Transmitted" - } - ], - "title": "Global Network Utilization by device", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "DROPPED PACKETS", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 66 - }, - "id": 53, - "options": { - "legend": { - "calcs": [], - "displayMode": "hidden", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(node_network_receive_drop_total{cluster=\"$cluster\", job=\"$job\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "Linux Packets dropped (receive)", - "range": true, - "refId": "Linux Packets dropped (receive)" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "- sum(rate(node_network_transmit_drop_total{cluster=\"$cluster\", job=\"$job\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "Linux Packets dropped (transmit)", - "range": true, - "refId": "Linux Packets dropped (transmit)" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(windows_net_packets_received_discarded_total{cluster=\"$cluster\"}[$__rate_interval]))", - "hide": false, - "interval": "$resolution", - "legendFormat": "Windows Packets dropped (receive)", - "range": true, - "refId": "Windows Packets dropped (receive)" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "- sum(rate(windows_net_packets_outbound_discarded_total{cluster=\"$cluster\"}[$__rate_interval]))", - "hide": false, - "interval": "$resolution", - "legendFormat": "Windows Packets dropped (transmit)", - "range": true, - "refId": "Windows Packets dropped (transmit)" - } - ], - "title": "Network Saturation - Packets dropped", - "transformations": [ - { - "id": "calculateField", - "options": { - "alias": "Packets dropped (receive)", - "mode": "reduceRow", - "reduce": { - "include": [ - "Linux Packets dropped (receive)", - "Windows Packets dropped (receive)" - ], - "reducer": "mean" - } - } - }, - { - "id": "calculateField", - "options": { - "alias": "Packets dropped (transmit)", - "mode": "reduceRow", - "reduce": { - "include": [ - "Linux Packets dropped (transmit)", - "Windows Packets dropped (transmit)" - ], - "reducer": "mean" - } - } - }, - { - "id": "organize", - "options": { - "excludeByName": { - "Linux Packets dropped (receive)": true, - "Linux Packets dropped (transmit)": true, - "Time": false, - "Windows Packets dropped (receive)": true, - "Windows Packets dropped (transmit)": true - }, - "includeByName": {}, - "indexByName": {}, - "renameByName": {} - } - } - ], - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "BANDWIDTH", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 74 - }, - "id": 79, - "options": { - "legend": { - "calcs": [], - "displayMode": "hidden", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(container_network_receive_bytes_total{cluster=\"$cluster\"}[$__rate_interval])) by (namespace)\n+ on (namespace)\n(sum(rate(windows_container_network_receive_bytes_total{container_id!=\"\", cluster=\"$cluster\"}[$__rate_interval]) * on (container_id) group_left (container, pod, namespace) max by ( container, container_id, pod, namespace) (kube_pod_container_info{container_id!=\"\", cluster=\"$cluster\"}) OR kube_namespace_created{cluster=\"$cluster\"} * 0) by (namespace))", - "interval": "$resolution", - "legendFormat": "Received : {{ namespace }}", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "- (sum(rate(container_network_transmit_bytes_total{cluster=\"$cluster\"}[$__rate_interval])) by (namespace)\n+ on (namespace)\n(sum(rate(windows_container_network_transmit_bytes_total{container_id!=\"\", cluster=\"$cluster\"}[$__rate_interval]) * on (container_id) group_left (container, pod, namespace) max by ( container, container_id, pod, namespace) (kube_pod_container_info{container_id!=\"\", cluster=\"$cluster\"}) OR kube_namespace_created{cluster=\"$cluster\"} * 0) by (namespace)))", - "hide": false, - "interval": "$resolution", - "legendFormat": "Transmitted : {{ namespace }}", - "range": true, - "refId": "B" - } - ], - "title": "Network Received by namespace", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "BANDWIDTH", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 74 - }, - "id": 80, - "options": { - "legend": { - "calcs": [], - "displayMode": "hidden", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(node_network_receive_bytes_total{cluster=\"$cluster\", job=\"$job\"}[$__rate_interval])) by (instance)", - "interval": "$resolution", - "legendFormat": "Received bytes in {{ instance }}", - "range": true, - "refId": "Linux Received bytes" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "- sum(rate(node_network_transmit_bytes_total{cluster=\"$cluster\", job=\"$job\"}[$__rate_interval])) by (instance)", - "hide": false, - "interval": "$resolution", - "legendFormat": "Transmitted bytes in {{ instance }}", - "range": true, - "refId": "Linux Transmitted bytes" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(windows_net_bytes_received_total{cluster=\"$cluster\"}[$__rate_interval])) by (instance)", - "hide": false, - "interval": "$resolution", - "legendFormat": "Received bytes in {{ instance }}", - "range": true, - "refId": "Windows Received bytes" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "- sum(rate(windows_net_bytes_sent_total{cluster=\"$cluster\"}[$__rate_interval])) by (instance)", - "hide": false, - "interval": "$resolution", - "legendFormat": "Transmitted bytes in {{ instance }}", - "range": true, - "refId": "Windows Transmitted bytes" - } - ], - "title": "Total Network Received (with all virtual devices) by instance", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "Dropped noisy virtual devices for readability.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "BANDWIDTH", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 82 - }, - "id": 56, - "options": { - "legend": { - "calcs": [], - "displayMode": "hidden", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(node_network_receive_bytes_total{device!~\"(veth|azv|lxc|lo).*\", cluster=\"$cluster\", job=\"$job\"}[$__rate_interval])) by (instance)", - "interval": "$resolution", - "legendFormat": "Received bytes in {{ instance }}", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "- sum(rate(node_network_transmit_bytes_total{device!~\"(veth|azv|lxc|lo).*\", cluster=\"$cluster\", job=\"$job\"}[$__rate_interval])) by (instance)", - "hide": false, - "interval": "$resolution", - "legendFormat": "Transmitted bytes in {{ instance }}", - "range": true, - "refId": "B" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(windows_net_bytes_received_total{nic!~\".*Virtual.*\",cluster=\"$cluster\"}[$__rate_interval])) by (instance)", - "hide": false, - "interval": "$resolution", - "legendFormat": "Received bytes in {{ instance }}", - "range": true, - "refId": "C" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "- sum(rate(windows_net_bytes_sent_total{nic!~\".*Virtual.*\",cluster=\"$cluster\"}[$__rate_interval])) by (instance)", - "hide": false, - "interval": "$resolution", - "legendFormat": "Transmitted bytes in {{ instance }}", - "range": true, - "refId": "D" - } - ], - "title": "Network Received (without loopback) by instance", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "Dropped noisy virtual devices for readability.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "BANDWIDTH", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 82 - }, - "id": 81, - "options": { - "legend": { - "calcs": [], - "displayMode": "hidden", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(node_network_receive_bytes_total{device=\"lo\", cluster=\"$cluster\", job=\"$job\"}[$__rate_interval])) by (instance)", - "interval": "$resolution", - "legendFormat": "Received bytes in {{ instance }}", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "- sum(rate(node_network_transmit_bytes_total{device=\"lo\", cluster=\"$cluster\", job=\"$job\"}[$__rate_interval])) by (instance)", - "hide": false, - "interval": "$resolution", - "legendFormat": "Transmitted bytes in {{ instance }}", - "range": true, - "refId": "B" - } - ], - "title": "Network Received (loopback only) by instance", - "type": "timeseries" - } - ], - "refresh": "30s", - "schemaVersion": 39, - "tags": [ - "Kubernetes", - "Prometheus" - ], - "templating": { - "list": [ - { - "current": { - "selected": false, - "text": "", - "value": "" - }, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "datasource", - "options": [], - "query": "prometheus", - "queryValue": "", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_node_info,cluster)", - "hide": 0, - "includeAll": false, - "multi": false, - "name": "cluster", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(kube_node_info,cluster)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - }, - { - "current": { - "selected": false, - "text": "30s", - "value": "30s" - }, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "resolution", - "options": [ - { - "selected": false, - "text": "1s", - "value": "1s" - }, - { - "selected": false, - "text": "15s", - "value": "15s" - }, - { - "selected": true, - "text": "30s", - "value": "30s" - }, - { - "selected": false, - "text": "1m", - "value": "1m" - }, - { - "selected": false, - "text": "3m", - "value": "3m" - }, - { - "selected": false, - "text": "5m", - "value": "5m" - } - ], - "query": "1s, 15s, 30s, 1m, 3m, 5m", - "queryValue": "", - "skipUrlSync": false, - "type": "custom" - }, - { - "current": { - "selected": false, - "text": "", - "value": "" - }, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(node_cpu_seconds_total{cluster=\"$cluster\"},job)", - "hide": 0, - "includeAll": false, - "multi": true, - "name": "job", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(node_cpu_seconds_total{cluster=\"$cluster\"},job)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - } - ] - }, - "time": { - "from": "now-1h", - "to": "now" - }, - "timepicker": {}, - "timezone": "", - "title": "Kubernetes / Views / Global", - "uid": "k8s_views_global", - "version": 39, - "weekStart": "" - } -kind: ConfigMap -metadata: - annotations: - grafana_folder: Kubernetes - labels: - grafana_dashboard: "1" - name: dashboards-k8s-views-global - namespace: futureporn ---- -apiVersion: v1 -data: - k8s-views-namespaces.json: | - { - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": {}, - "__requires": [ - { - "type": "panel", - "id": "gauge", - "name": "Gauge", - "version": "" - }, - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "10.3.1" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "1.0.0" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "target": { - "limit": 100, - "matchAny": false, - "tags": [], - "type": "dashboard" - }, - "type": "dashboard" - } - ] - }, - "description": "This is a modern 'Namespaces View' dashboard for your Kubernetes cluster(s). Made for kube-prometheus-stack and take advantage of the latest Grafana features. GitHub repository: https://github.com/dotdc/grafana-dashboards-kubernetes", - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 1, - "links": [], - "liveNow": false, - "panels": [ - { - "collapsed": false, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 0 - }, - "id": 38, - "panels": [], - "title": "Overview", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "decimals": 2, - "mappings": [], - "max": 1, - "min": 0, - "thresholds": { - "mode": "percentage", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "orange", - "value": 50 - }, - { - "color": "red", - "value": 70 - } - ] - }, - "unit": "percentunit", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 7, - "w": 6, - "x": 0, - "y": 1 - }, - "id": 46, - "options": { - "minVizHeight": 75, - "minVizWidth": 75, - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showThresholdLabels": false, - "showThresholdMarkers": true, - "sizing": "auto", - "text": {} - }, - "pluginVersion": "10.3.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(rate(container_cpu_usage_seconds_total{namespace=~\"$namespace\", image!=\"\", cluster=\"$cluster\"}[$__rate_interval])) / sum(machine_cpu_cores{cluster=\"$cluster\"})", - "instant": true, - "interval": "", - "legendFormat": "", - "range": false, - "refId": "A" - } - ], - "title": "Namespace(s) usage on total cluster CPU in %", - "type": "gauge" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "decimals": 2, - "mappings": [], - "max": 1, - "min": 0, - "thresholds": { - "mode": "percentage", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "orange", - "value": 50 - }, - { - "color": "red", - "value": 70 - } - ] - }, - "unit": "percentunit", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 7, - "w": 6, - "x": 6, - "y": 1 - }, - "id": 48, - "options": { - "minVizHeight": 75, - "minVizWidth": 75, - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showThresholdLabels": false, - "showThresholdMarkers": true, - "sizing": "auto", - "text": {} - }, - "pluginVersion": "10.3.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(container_memory_working_set_bytes{namespace=~\"$namespace\", image!=\"\", cluster=\"$cluster\"}) / sum(machine_memory_bytes{cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "", - "refId": "A" - } - ], - "title": "Namespace(s) usage on total cluster RAM in %", - "type": "gauge" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 11, - "w": 12, - "x": 12, - "y": 1 - }, - "id": 32, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_pod_info{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Running Pods", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_service_info{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Services", - "refId": "B" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_ingress_info{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Ingresses", - "refId": "C" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_deployment_labels{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Deployments", - "refId": "D" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_statefulset_labels{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Statefulsets", - "refId": "E" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_daemonset_labels{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Daemonsets", - "refId": "F" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_persistentvolumeclaim_info{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Persistent Volume Claims", - "refId": "G" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_hpa_labels{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Horizontal Pod Autoscalers", - "refId": "H" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_configmap_info{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Configmaps", - "refId": "I" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_secret_info{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Secrets", - "refId": "J" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_networkpolicy_labels{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Network Policies", - "refId": "K" - } - ], - "title": "Kubernetes Resource Count", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "rgb(255, 255, 255)", - "value": null - } - ] - }, - "unit": "none", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 6, - "x": 0, - "y": 8 - }, - "id": 62, - "options": { - "colorMode": "none", - "graphMode": "none", - "justifyMode": "center", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "text": {}, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "10.3.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(container_cpu_usage_seconds_total{namespace=~\"$namespace\", image!=\"\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "", - "legendFormat": "Real", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_resource_requests{namespace=~\"$namespace\", resource=\"cpu\", cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Requests", - "range": true, - "refId": "B" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_resource_limits{namespace=~\"$namespace\", resource=\"cpu\", cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Limits", - "range": true, - "refId": "C" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(machine_cpu_cores{cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Cluster Total", - "range": true, - "refId": "D" - } - ], - "title": "Namespace(s) CPU Usage in cores", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "rgb(255, 255, 255)", - "value": null - } - ] - }, - "unit": "bytes", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 6, - "x": 6, - "y": 8 - }, - "id": 64, - "options": { - "colorMode": "none", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "text": {}, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "10.3.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(container_memory_working_set_bytes{namespace=~\"$namespace\", image!=\"\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Real", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_resource_requests{namespace=~\"$namespace\", resource=\"memory\", cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Requests", - "range": true, - "refId": "B" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_resource_limits{namespace=~\"$namespace\", resource=\"memory\", cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Limits", - "range": true, - "refId": "C" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(machine_memory_bytes{cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Cluster Total", - "range": true, - "refId": "D" - } - ], - "title": "Namespace(s) RAM Usage in bytes", - "type": "stat" - }, - { - "collapsed": false, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 12 - }, - "id": 40, - "panels": [], - "title": "Resources", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "CPU CORES", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 13 - }, - "id": 29, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(container_cpu_usage_seconds_total{namespace=~\"$namespace\", image!=\"\", pod=~\"${created_by}.*\", cluster=\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "$resolution", - "legendFormat": "{{ pod }}", - "range": true, - "refId": "A" - } - ], - "title": "CPU usage by Pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 13 - }, - "id": 30, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(container_memory_working_set_bytes{namespace=~\"$namespace\", image!=\"\", pod=~\"${created_by}.*\", cluster=\"$cluster\"}) by (pod)", - "interval": "$resolution", - "legendFormat": "{{ pod }}", - "range": true, - "refId": "A" - } - ], - "title": "Memory usage by Pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "SECONDS", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "s" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 21 - }, - "id": 68, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(container_cpu_cfs_throttled_seconds_total{namespace=~\"$namespace\", image!=\"\", pod=~\"${created_by}.*\", cluster=\"$cluster\"}[$__rate_interval])) by (pod) > 0", - "interval": "$resolution", - "legendFormat": "{{ pod }}", - "range": true, - "refId": "A" - } - ], - "title": "CPU Throttled seconds by pod", - "type": "timeseries" - }, - { - "collapsed": false, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 29 - }, - "id": 73, - "panels": [], - "title": "Kubernetes", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 9, - "w": 12, - "x": 0, - "y": 30 - }, - "id": 70, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(kube_pod_status_qos_class{namespace=~\"$namespace\", cluster=\"$cluster\"}) by (qos_class)", - "interval": "", - "legendFormat": "{{ qos_class }} pods", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_info{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "hide": false, - "legendFormat": "Total pods", - "range": true, - "refId": "B" - } - ], - "title": "Kubernetes Pods QoS classes", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 9, - "w": 12, - "x": 12, - "y": 30 - }, - "id": 72, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(kube_pod_status_reason{cluster=\"$cluster\"}) by (reason)", - "interval": "", - "legendFormat": "{{ reason }}", - "range": true, - "refId": "A" - } - ], - "title": "Kubernetes Pods Status Reason", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "No data is generally a good thing here.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "points", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 9, - "w": 12, - "x": 0, - "y": 39 - }, - "id": 74, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(increase(container_oom_events_total{namespace=~\"${namespace}\", cluster=\"$cluster\"}[$__rate_interval])) by (namespace, pod) > 0", - "interval": "", - "legendFormat": "namespace: {{ namespace }} - pod: {{ pod }}", - "range": true, - "refId": "A" - } - ], - "title": "OOM Events by namespace, pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "No data is generally a good thing here.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "points", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 9, - "w": 12, - "x": 12, - "y": 39 - }, - "id": 75, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(increase(kube_pod_container_status_restarts_total{namespace=~\"${namespace}\", cluster=\"$cluster\"}[$__rate_interval])) by (namespace, pod) > 0", - "interval": "", - "legendFormat": "namespace: {{ namespace }} - pod: {{ pod }}", - "range": true, - "refId": "A" - } - ], - "title": "Container Restarts by namespace, pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 0, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 48 - }, - "id": 5, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_status_ready{namespace=~\"$namespace\", pod=~\"${created_by}.*\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Ready", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_status_running{namespace=~\"$namespace\", pod=~\"${created_by}.*\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Running", - "range": true, - "refId": "B" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_pod_container_status_waiting{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Waiting", - "refId": "C" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_pod_container_status_restarts_total{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Restarts Total", - "refId": "D" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "expr": "sum(kube_pod_container_status_terminated{namespace=~\"$namespace\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "Terminated", - "refId": "E" - } - ], - "title": "Nb of pods by state", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 0, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 48 - }, - "id": 2, - "options": { - "legend": { - "calcs": [], - "displayMode": "hidden", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_pod_container_info{namespace=~\"$namespace\", pod=~\"${created_by}.*\", cluster=\"$cluster\"}) by (pod)", - "interval": "", - "legendFormat": "{{ pod }}", - "range": true, - "refId": "A" - } - ], - "title": "Nb of containers by pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 56 - }, - "id": 7, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(kube_deployment_status_replicas_available{namespace=~\"$namespace\", cluster=\"$cluster\"}) by (deployment)", - "interval": "", - "legendFormat": "{{ deployment }}", - "range": true, - "refId": "A" - } - ], - "title": "Replicas available by deployment", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 56 - }, - "id": 8, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(kube_deployment_status_replicas_unavailable{namespace=~\"$namespace\", pod=~\"${created_by}.*\", cluster=\"$cluster\"}) by (deployment)", - "interval": "", - "legendFormat": "{{ deployment }}", - "range": true, - "refId": "A" - } - ], - "title": "Replicas unavailable by deployment", - "type": "timeseries" - }, - { - "collapsed": false, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 64 - }, - "id": 42, - "panels": [], - "title": "Kubernetes Storage", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 65 - }, - "id": 65, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(kubelet_volume_stats_used_bytes{namespace=~\"$namespace\", cluster=\"$cluster\"}) by (persistentvolumeclaim) / sum(kubelet_volume_stats_capacity_bytes{namespace=~\"$namespace\", cluster=\"$cluster\"}) by (persistentvolumeclaim)", - "interval": "", - "legendFormat": "{{ persistentvolumeclaim }}", - "refId": "A" - } - ], - "title": "Persistent Volumes - Capacity and usage in %", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 65 - }, - "id": 66, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(kubelet_volume_stats_used_bytes{namespace=~\"$namespace\", cluster=\"$cluster\"}) by (persistentvolumeclaim)", - "interval": "", - "legendFormat": "{{ persistentvolumeclaim }} - Used", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(kubelet_volume_stats_capacity_bytes{namespace=~\"$namespace\", cluster=\"$cluster\"}) by (persistentvolumeclaim)", - "hide": false, - "interval": "", - "legendFormat": "{{ persistentvolumeclaim }} - Capacity", - "refId": "B" - } - ], - "title": "Persistent Volumes - Capacity and usage in bytes", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 73 - }, - "id": 27, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "1 - sum(kubelet_volume_stats_inodes_used{namespace=~\"$namespace\", cluster=\"$cluster\"}) by (persistentvolumeclaim) / sum(kubelet_volume_stats_inodes{namespace=~\"$namespace\", cluster=\"$cluster\"}) by (persistentvolumeclaim)", - "interval": "", - "legendFormat": "{{ persistentvolumeclaim }}", - "refId": "A" - } - ], - "title": "Persistent Volumes - Inodes", - "type": "timeseries" - } - ], - "refresh": "30s", - "schemaVersion": 39, - "tags": [ - "Kubernetes", - "Prometheus" - ], - "templating": { - "list": [ - { - "current": {}, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "datasource", - "options": [], - "query": "prometheus", - "queryValue": "", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_node_info,cluster)", - "hide": 0, - "includeAll": false, - "multi": false, - "name": "cluster", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(kube_node_info,cluster)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - }, - { - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_pod_info{cluster=\"$cluster\"}, namespace)", - "hide": 0, - "includeAll": true, - "multi": true, - "name": "namespace", - "options": [], - "query": { - "query": "label_values(kube_pod_info{cluster=\"$cluster\"}, namespace)", - "refId": "StandardVariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "tagValuesQuery": "", - "tagsQuery": "", - "type": "query", - "useTags": false - }, - { - "current": { - "selected": false, - "text": "30s", - "value": "30s" - }, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "resolution", - "options": [ - { - "selected": false, - "text": "1s", - "value": "1s" - }, - { - "selected": false, - "text": "15s", - "value": "15s" - }, - { - "selected": true, - "text": "30s", - "value": "30s" - }, - { - "selected": false, - "text": "1m", - "value": "1m" - }, - { - "selected": false, - "text": "3m", - "value": "3m" - }, - { - "selected": false, - "text": "5m", - "value": "5m" - } - ], - "query": "1s, 15s, 30s, 1m, 3m, 5m", - "queryValue": "", - "skipUrlSync": false, - "type": "custom" - }, - { - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_pod_info{namespace=~\"$namespace\", container!=\"\", cluster=\"$cluster\"},created_by_name)", - "description": "Can be used to filter on a specific deployment, statefulset or deamonset (only relevant panels).", - "hide": 0, - "includeAll": true, - "multi": true, - "name": "created_by", - "options": [], - "query": { - "query": "label_values(kube_pod_info{namespace=~\"$namespace\", container!=\"\", cluster=\"$cluster\"},created_by_name)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - } - ] - }, - "time": { - "from": "now-1h", - "to": "now" - }, - "timepicker": {}, - "timezone": "", - "title": "Kubernetes / Views / Namespaces", - "uid": "k8s_views_ns", - "version": 34, - "weekStart": "" - } -kind: ConfigMap -metadata: - annotations: - grafana_folder: Kubernetes - labels: - grafana_dashboard: "1" - name: dashboards-k8s-views-namespaces - namespace: futureporn ---- -apiVersion: v1 -data: - k8s-views-nodes.json: | - { - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": {}, - "__requires": [ - { - "type": "panel", - "id": "gauge", - "name": "Gauge", - "version": "" - }, - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "10.3.1" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "1.0.0" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - }, - { - "type": "panel", - "id": "table", - "name": "Table", - "version": "" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "target": { - "limit": 100, - "matchAny": false, - "tags": [], - "type": "dashboard" - }, - "type": "dashboard" - } - ] - }, - "description": "This is a modern 'Nodes View' dashboard for your Kubernetes cluster(s). Made for kube-prometheus-stack and take advantage of the latest Grafana features. GitHub repository: https://github.com/dotdc/grafana-dashboards-kubernetes", - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 1, - "links": [], - "liveNow": false, - "panels": [ - { - "collapsed": false, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 0 - }, - "id": 40, - "panels": [], - "title": "Overview", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "decimals": 2, - "mappings": [], - "max": 1, - "min": 0, - "thresholds": { - "mode": "percentage", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "orange", - "value": 50 - }, - { - "color": "red", - "value": 70 - } - ] - }, - "unit": "percentunit", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 4, - "x": 0, - "y": 1 - }, - "id": 7, - "options": { - "minVizHeight": 75, - "minVizWidth": 75, - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showThresholdLabels": false, - "showThresholdMarkers": true, - "sizing": "auto", - "text": {} - }, - "pluginVersion": "10.3.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": false, - "expr": "avg(sum by (cpu) (rate(node_cpu_seconds_total{mode!~\"idle|iowait|steal\", instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval])))", - "instant": true, - "interval": "$resolution", - "legendFormat": "", - "refId": "A" - } - ], - "title": "CPU Usage", - "type": "gauge" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "decimals": 2, - "mappings": [], - "max": 1, - "min": 0, - "thresholds": { - "mode": "percentage", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "orange", - "value": 50 - }, - { - "color": "red", - "value": 70 - } - ] - }, - "unit": "percentunit", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 4, - "x": 4, - "y": 1 - }, - "id": 13, - "options": { - "minVizHeight": 75, - "minVizWidth": 75, - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showThresholdLabels": false, - "showThresholdMarkers": true, - "sizing": "auto", - "text": {} - }, - "pluginVersion": "10.3.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": false, - "expr": "sum(node_memory_MemTotal_bytes{instance=\"$instance\", cluster=\"$cluster\"} - node_memory_MemAvailable_bytes{instance=\"$instance\", cluster=\"$cluster\"}) / sum(node_memory_MemTotal_bytes{instance=\"$instance\", cluster=\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "", - "refId": "A" - } - ], - "title": "RAM Usage", - "type": "gauge" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "blue", - "value": null - } - ] - }, - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 4, - "x": 8, - "y": 1 - }, - "id": 24, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "text": {}, - "textMode": "value", - "wideLayout": true - }, - "pluginVersion": "10.3.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(kube_pod_info{node=\"$node\", cluster=\"$cluster\"})", - "interval": "", - "legendFormat": "", - "refId": "A" - } - ], - "title": "Pods on node", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "custom": { - "align": "auto", - "cellOptions": { - "type": "auto" - }, - "inspect": false - }, - "links": [ - { - "targetBlank": true, - "title": "Pod details", - "url": "/d/k8s_views_pods/kubernetes-views-pods?${datasource:queryparam}&var-namespace=${__data.fields.namespace}&var-pod=${__data.fields.pod}&${resolution:queryparam}&${__url_time_range}" - } - ], - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unitScale": true - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "pod" - }, - "properties": [ - { - "id": "custom.width", - "value": 416 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "priority_class" - }, - "properties": [ - { - "id": "custom.width", - "value": 176 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "pod_ip" - }, - "properties": [ - { - "id": "custom.width", - "value": 157 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "created_by_kind" - }, - "properties": [ - { - "id": "custom.width", - "value": 205 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "namespace" - }, - "properties": [ - { - "id": "custom.width", - "value": 263 - } - ] - } - ] - }, - "gridPos": { - "h": 11, - "w": 12, - "x": 12, - "y": 1 - }, - "id": 5, - "options": { - "cellHeight": "sm", - "footer": { - "countRows": false, - "fields": "", - "reducer": [ - "sum" - ], - "show": false - }, - "showHeader": true, - "sortBy": [] - }, - "pluginVersion": "10.3.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "kube_pod_info{node=\"$node\", cluster=\"$cluster\"}", - "format": "table", - "interval": "", - "legendFormat": "", - "refId": "A" - } - ], - "title": "List of pods on node ($node)", - "transformations": [ - { - "id": "organize", - "options": { - "excludeByName": { - "Time": true, - "Value": true, - "__name__": true, - "container": true, - "created_by_kind": false, - "created_by_name": true, - "endpoint": true, - "env": true, - "host_ip": true, - "host_network": true, - "instance": true, - "job": true, - "node": true, - "project": true, - "prometheus_replica": true, - "service": true, - "uid": true - }, - "indexByName": { - "Time": 6, - "Value": 20, - "__name__": 7, - "container": 8, - "created_by_kind": 2, - "created_by_name": 9, - "endpoint": 10, - "env": 11, - "host_ip": 5, - "host_network": 12, - "instance": 13, - "job": 14, - "namespace": 1, - "node": 15, - "pod": 0, - "pod_ip": 3, - "priority_class": 4, - "project": 16, - "prometheus_replica": 17, - "service": 18, - "uid": 19 - }, - "renameByName": {} - } - }, - { - "id": "groupBy", - "options": { - "fields": { - "created_by_kind": { - "aggregations": [], - "operation": "groupby" - }, - "host_ip": { - "aggregations": [], - "operation": "groupby" - }, - "namespace": { - "aggregations": [ - "last" - ], - "operation": "groupby" - }, - "pod": { - "aggregations": [], - "operation": "groupby" - }, - "pod_ip": { - "aggregations": [], - "operation": "groupby" - }, - "priority_class": { - "aggregations": [], - "operation": "groupby" - } - } - } - } - ], - "type": "table" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "decimals": 3, - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "rgb(255, 255, 255)", - "value": null - } - ] - }, - "unit": "none", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 3, - "w": 2, - "x": 0, - "y": 9 - }, - "id": 9, - "options": { - "colorMode": "none", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "text": {}, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "10.3.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": false, - "expr": "sum(rate(node_cpu_seconds_total{mode!~\"idle|iowait|steal\", instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval]))", - "instant": true, - "interval": "$resolution", - "legendFormat": "", - "refId": "A" - } - ], - "title": "CPU Used", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "rgb(255, 255, 255)", - "value": null - } - ] - }, - "unit": "none", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 3, - "w": 2, - "x": 2, - "y": 9 - }, - "id": 11, - "options": { - "colorMode": "none", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "text": {}, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "10.3.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(machine_cpu_cores{node=\"$node\", cluster=\"$cluster\"})", - "interval": "$resolution", - "legendFormat": "", - "refId": "A" - } - ], - "title": "CPU Total", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "rgb(255, 255, 255)", - "value": null - } - ] - }, - "unit": "bytes", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 3, - "w": 2, - "x": 4, - "y": 9 - }, - "id": 15, - "options": { - "colorMode": "none", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "text": {}, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "10.3.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": false, - "expr": "sum(node_memory_MemTotal_bytes{instance=\"$instance\", cluster=\"$cluster\"} - node_memory_MemAvailable_bytes{instance=\"$instance\", cluster=\"$cluster\"})", - "instant": true, - "interval": "", - "legendFormat": "", - "refId": "A" - } - ], - "title": "RAM Used", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "rgb(255, 255, 255)", - "value": null - } - ] - }, - "unit": "bytes", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 3, - "w": 2, - "x": 6, - "y": 9 - }, - "id": 17, - "options": { - "colorMode": "none", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "text": {}, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "10.3.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "machine_memory_bytes{node=\"$node\", cluster=\"$cluster\"}", - "instant": false, - "interval": "", - "legendFormat": "", - "refId": "A" - } - ], - "title": "RAM Total", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "mappings": [], - "noValue": "0", - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "#EAB839", - "value": 25228800 - }, - { - "color": "red", - "value": 31536000 - } - ] - }, - "unit": "s", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 3, - "w": 4, - "x": 8, - "y": 9 - }, - "id": 18, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "text": {}, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "10.3.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "node_time_seconds{instance=\"$instance\", cluster=\"$cluster\"} - node_boot_time_seconds{instance=\"$instance\", cluster=\"$cluster\"}", - "instant": false, - "interval": "", - "legendFormat": "", - "refId": "A" - } - ], - "title": "uptime", - "type": "stat" - }, - { - "collapsed": false, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 12 - }, - "id": 38, - "panels": [], - "title": "Resources", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "max": 100, - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percent", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 9, - "w": 12, - "x": 0, - "y": 13 - }, - "id": 2, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "avg(rate(node_cpu_seconds_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval]) * 100) by (mode)", - "hide": false, - "instant": false, - "interval": "$resolution", - "legendFormat": "{{ mode }}", - "refId": "A" - } - ], - "title": "CPU Usage", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes", - "unitScale": true - }, - "overrides": [] - }, - "gridPos": { - "h": 9, - "w": 12, - "x": 12, - "y": 13 - }, - "id": 3, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "node_memory_MemTotal_bytes{instance=\"$instance\", cluster=\"$cluster\"} - node_memory_MemFree_bytes{instance=\"$instance\", cluster=\"$cluster\"} - (node_memory_Cached_bytes{instance=\"$instance\", cluster=\"$cluster\"} + node_memory_Buffers_bytes{instance=\"$instance\", cluster=\"$cluster\"})", - "instant": false, - "interval": "$resolution", - "legendFormat": "RAM Used", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "node_memory_MemTotal_bytes{instance=\"$instance\", cluster=\"$cluster\"}", - "hide": false, - "interval": "$resolution", - "legendFormat": "RAM Total", - "refId": "B" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "node_memory_Cached_bytes{instance=\"$instance\", cluster=\"$cluster\"}", - "hide": false, - "interval": "$resolution", - "legendFormat": "RAM Cache", - "refId": "C" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "node_memory_Buffers_bytes{instance=\"$instance\", cluster=\"$cluster\"}", - "hide": false, - "interval": "$resolution", - "legendFormat": "RAM Buffer", - "refId": "D" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "node_memory_MemFree_bytes{instance=\"$instance\", cluster=\"$cluster\"}", - "hide": false, - "interval": "$resolution", - "legendFormat": "RAM Free", - "refId": "E" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "node_memory_SwapTotal_bytes{instance=\"$instance\", cluster=\"$cluster\"} - node_memory_SwapFree_bytes{instance=\"$instance\", cluster=\"$cluster\"}", - "hide": false, - "interval": "$resolution", - "legendFormat": "SWAP Used", - "refId": "F" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "node_memory_SwapTotal_bytes{instance=\"$instance\", cluster=\"$cluster\"}", - "hide": false, - "interval": "$resolution", - "legendFormat": "SWAP Total", - "refId": "G" - } - ], - "title": "Memory Usage", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "CPU Cores", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 22 - }, - "id": 26, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(container_cpu_usage_seconds_total{node=\"$node\", image!=\"\", cluster=\"$cluster\"}[$__rate_interval])) by (pod)", - "interval": "$resolution", - "legendFormat": "{{ pod }}", - "refId": "A" - } - ], - "title": "CPU usage by Pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 22 - }, - "id": 28, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(container_memory_working_set_bytes{node=\"$node\", image!=\"\", cluster=\"$cluster\"}) by (pod)", - "interval": "$resolution", - "legendFormat": "{{ pod }}", - "refId": "A" - } - ], - "title": "Memory usage by Pod", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "Number of times a CPU core has been throttled on an instance.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "CPU CORES", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 30 - }, - "id": 66, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(node_cpu_core_throttles_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "Nb of cpu core throttles", - "range": true, - "refId": "A" - } - ], - "title": "Number of CPU Core Throttled", - "type": "timeseries" - }, - { - "collapsed": false, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 38 - }, - "id": 44, - "panels": [], - "title": "System", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 39 - }, - "id": 48, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "node_load1{instance=\"$instance\", cluster=\"$cluster\"}", - "interval": "$resolution", - "legendFormat": "1m", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "node_load5{instance=\"$instance\", cluster=\"$cluster\"}", - "hide": false, - "interval": "$resolution", - "legendFormat": "5m", - "range": true, - "refId": "B" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "node_load15{instance=\"$instance\", cluster=\"$cluster\"}", - "hide": false, - "interval": "$resolution", - "legendFormat": "15m", - "range": true, - "refId": "C" - } - ], - "title": "System Load", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 39 - }, - "id": 46, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "rate(node_context_switches_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval])", - "interval": "$resolution", - "intervalFactor": 1, - "legendFormat": "Context switches", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "rate(node_intr_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval])", - "hide": false, - "interval": "$resolution", - "legendFormat": "Interrupts", - "range": true, - "refId": "B" - } - ], - "title": "Context Switches & Interrupts", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 47 - }, - "id": 49, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "node_filefd_maximum{instance=\"$instance\", cluster=\"$cluster\"}", - "instant": false, - "interval": "$resolution", - "legendFormat": "Maximum file descriptors", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "node_filefd_allocated{instance=\"$instance\", cluster=\"$cluster\"}", - "hide": false, - "instant": false, - "interval": "$resolution", - "legendFormat": "Allocated file descriptors", - "refId": "B" - } - ], - "title": "File Descriptors", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "s" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 47 - }, - "id": 50, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "node_timex_estimated_error_seconds{instance=\"$instance\", cluster=\"$cluster\"}", - "instant": false, - "interval": "$resolution", - "intervalFactor": 1, - "legendFormat": "Estimated error in seconds", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "node_timex_maxerror_seconds{instance=\"$instance\", cluster=\"$cluster\"}", - "hide": false, - "interval": "$resolution", - "intervalFactor": 1, - "legendFormat": "Maximum error in seconds", - "range": true, - "refId": "B" - } - ], - "title": "Time Sync", - "type": "timeseries" - }, - { - "collapsed": false, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 55 - }, - "id": 36, - "panels": [], - "title": "Network", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "BANDWIDTH", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 56 - }, - "id": 20, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(node_network_receive_bytes_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "In", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "- sum(rate(node_network_transmit_bytes_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "Out", - "range": true, - "refId": "B" - } - ], - "title": "Network usage (bytes)", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 56 - }, - "id": 61, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(node_network_receive_errs_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "In", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "- sum(rate(node_network_transmit_errs_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "Out", - "range": true, - "refId": "B" - } - ], - "title": "Network errors", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "pps" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 64 - }, - "id": 62, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(node_network_receive_packets_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "In", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "- sum(rate(node_network_transmit_packets_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "Out", - "range": true, - "refId": "B" - } - ], - "title": "Network usage (packet/s)", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 64 - }, - "id": 64, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(node_network_receive_drop_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval]))", - "hide": false, - "interval": "$resolution", - "legendFormat": "In", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "- sum(rate(node_network_transmit_drop_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval]))", - "hide": false, - "interval": "$resolution", - "legendFormat": "Out", - "refId": "B" - } - ], - "title": "Network total drops", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 72 - }, - "id": 60, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "node_netstat_Tcp_CurrEstab{instance=\"$instance\", cluster=\"$cluster\"}", - "instant": false, - "interval": "$resolution", - "legendFormat": "TCP Currently Established", - "refId": "A" - } - ], - "title": "TCP Currently Established", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "NF Conntrack limit" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "red", - "mode": "fixed" - } - } - ] - } - ] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 72 - }, - "id": 63, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "node_nf_conntrack_entries{instance=\"$instance\", cluster=\"$cluster\"}", - "instant": false, - "interval": "$resolution", - "legendFormat": "NF Conntrack entries", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "node_nf_conntrack_entries_limit{instance=\"$instance\", cluster=\"$cluster\"}", - "hide": false, - "interval": "$resolution", - "legendFormat": "NF Conntrack limit", - "range": true, - "refId": "B" - } - ], - "title": "NF Conntrack", - "type": "timeseries" - }, - { - "collapsed": false, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 80 - }, - "id": 54, - "panels": [], - "title": "Kubernetes Storage", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 81 - }, - "id": 30, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(kubelet_volume_stats_used_bytes{node=\"$node\", cluster=\"$cluster\"}) by (persistentvolumeclaim) / sum(kubelet_volume_stats_capacity_bytes{node=\"$node\", cluster=\"$cluster\"}) by (persistentvolumeclaim)", - "interval": "$resolution", - "legendFormat": "{{ persistentvolumeclaim }}", - "range": true, - "refId": "A" - } - ], - "title": "Persistent Volumes - Usage in %", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "continuous-GrYlRd" - }, - "custom": { - "align": "auto", - "cellOptions": { - "type": "auto" - }, - "filterable": false, - "inspect": false - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "Used" - }, - "properties": [ - { - "id": "custom.width", - "value": 146 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Total" - }, - "properties": [ - { - "id": "custom.width", - "value": 167 - } - ] - } - ] - }, - "gridPos": { - "h": 16, - "w": 12, - "x": 12, - "y": 81 - }, - "id": 34, - "options": { - "cellHeight": "sm", - "footer": { - "countRows": false, - "fields": "", - "reducer": [ - "sum" - ], - "show": false - }, - "showHeader": true, - "sortBy": [] - }, - "pluginVersion": "10.2.2", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(kubelet_volume_stats_used_bytes{node=\"$node\", cluster=\"$cluster\"}) by (persistentvolumeclaim)", - "format": "table", - "hide": false, - "interval": "", - "legendFormat": "", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(kubelet_volume_stats_capacity_bytes{node=\"$node\", cluster=\"$cluster\"}) by (persistentvolumeclaim)", - "format": "table", - "hide": false, - "interval": "", - "legendFormat": "", - "refId": "B" - } - ], - "title": "Persistent Volumes - Usage in GB", - "transformations": [ - { - "id": "groupBy", - "options": { - "fields": { - "Value": { - "aggregations": [ - "lastNotNull" - ], - "operation": "aggregate" - }, - "Value #A": { - "aggregations": [ - "lastNotNull" - ], - "operation": "aggregate" - }, - "Value #B": { - "aggregations": [ - "lastNotNull" - ], - "operation": "aggregate" - }, - "persistentvolumeclaim": { - "aggregations": [], - "operation": "groupby" - } - } - } - }, - { - "id": "seriesToColumns", - "options": { - "byField": "persistentvolumeclaim" - } - }, - { - "id": "organize", - "options": { - "excludeByName": {}, - "indexByName": {}, - "renameByName": { - "Value #A (lastNotNull)": "Used", - "Value #B (lastNotNull)": "Total", - "persistentvolumeclaim": "Persistent Volume Claim" - } - } - } - ], - "type": "table" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percent" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 89 - }, - "id": 32, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(kubelet_volume_stats_inodes_used{node=\"$node\", cluster=\"$cluster\"}) by (persistentvolumeclaim) / sum(kubelet_volume_stats_inodes{node=\"$node\", cluster=\"$cluster\"}) by (persistentvolumeclaim) * 100", - "interval": "$resolution", - "legendFormat": "{{ persistentvolumeclaim }}", - "range": true, - "refId": "A" - } - ], - "title": "Persistent Volumes - Inodes", - "type": "timeseries" - }, - { - "collapsed": false, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 97 - }, - "id": 42, - "panels": [], - "title": "Node Storage", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percent" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 98 - }, - "id": 33, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "100 - ((node_filesystem_avail_bytes{instance=\"$instance\", cluster=\"$cluster\"} * 100) / node_filesystem_size_bytes{instance=\"$instance\", cluster=\"$cluster\"})", - "hide": false, - "interval": "$resolution", - "legendFormat": "{{ mountpoint }}", - "range": true, - "refId": "A" - } - ], - "title": "FS usage in %", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percent" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 98 - }, - "id": 59, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "100 - (node_filesystem_files_free{instance=\"$instance\", cluster=\"$cluster\"} / node_filesystem_files{instance=\"$instance\", cluster=\"$cluster\"} * 100)", - "hide": false, - "interval": "$resolution", - "legendFormat": "{{ mountpoint }}", - "range": true, - "refId": "A" - } - ], - "title": "FS inode usage in %", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 106 - }, - "id": 52, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "rate(node_disk_read_bytes_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval])", - "interval": "$resolution", - "legendFormat": "{{device}}", - "range": true, - "refId": "A" - } - ], - "title": "Reads by disk (bytes)", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 106 - }, - "id": 57, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "rate(node_disk_written_bytes_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval])", - "hide": false, - "interval": "$resolution", - "legendFormat": "{{device}}", - "range": true, - "refId": "A" - } - ], - "title": "Writes by disk (bytes)", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "read/s" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 114 - }, - "id": 51, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "rate(node_disk_reads_completed_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval])", - "interval": "$resolution", - "legendFormat": "{{device}}", - "range": true, - "refId": "A" - } - ], - "title": "Completed reads by disk", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "write/s" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 114 - }, - "id": 56, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "rate(node_disk_writes_completed_total{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval])", - "hide": false, - "interval": "$resolution", - "legendFormat": "{{device}}", - "range": true, - "refId": "A" - } - ], - "title": "Completed writes by disk", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "io/s" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 122 - }, - "id": 58, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "rate(node_disk_io_now{instance=\"$instance\", cluster=\"$cluster\"}[$__rate_interval]) ", - "interval": "$resolution", - "legendFormat": "{{device}}", - "range": true, - "refId": "A" - } - ], - "title": "Disk(s) io/s", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green" - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 122 - }, - "id": 55, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "right", - "showLegend": false - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(node_filesystem_device_error{instance=\"$instance\", cluster=\"$cluster\"}) by (mountpoint)", - "interval": "$resolution", - "legendFormat": "{{ mountpoint }}", - "range": true, - "refId": "A" - } - ], - "title": "FS - Device Errors", - "type": "timeseries" - } - ], - "refresh": "30s", - "schemaVersion": 39, - "tags": [ - "Kubernetes", - "Prometheus" - ], - "templating": { - "list": [ - { - "current": {}, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "datasource", - "options": [], - "query": "prometheus", - "queryValue": "", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_node_info,cluster)", - "hide": 0, - "includeAll": false, - "multi": false, - "name": "cluster", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(kube_node_info,cluster)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - }, - { - "current": { - "selected": false, - "text": "30s", - "value": "30s" - }, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "resolution", - "options": [ - { - "selected": false, - "text": "1s", - "value": "1s" - }, - { - "selected": false, - "text": "15s", - "value": "15s" - }, - { - "selected": true, - "text": "30s", - "value": "30s" - }, - { - "selected": false, - "text": "1m", - "value": "1m" - }, - { - "selected": false, - "text": "3m", - "value": "3m" - }, - { - "selected": false, - "text": "5m", - "value": "5m" - } - ], - "query": "1s, 15s, 30s, 1m, 3m, 5m", - "queryValue": "", - "skipUrlSync": false, - "type": "custom" - }, - { - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_node_info{cluster=\"$cluster\"}, node)", - "hide": 0, - "includeAll": false, - "multi": false, - "name": "node", - "options": [], - "query": { - "query": "label_values(kube_node_info{cluster=\"$cluster\"}, node)", - "refId": "StandardVariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - }, - { - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(node_uname_info{nodename=~\"(?i:($node))\"}, instance)", - "hide": 2, - "includeAll": false, - "multi": false, - "name": "instance", - "options": [], - "query": { - "query": "label_values(node_uname_info{nodename=~\"(?i:($node))\"}, instance)", - "refId": "StandardVariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - } - ] - }, - "time": { - "from": "now-1h", - "to": "now" - }, - "timepicker": {}, - "timezone": "", - "title": "Kubernetes / Views / Nodes", - "uid": "k8s_views_nodes", - "version": 29, - "weekStart": "" - } -kind: ConfigMap -metadata: - annotations: - grafana_folder: Kubernetes - labels: - grafana_dashboard: "1" - name: dashboards-k8s-views-nodes - namespace: futureporn ---- -apiVersion: v1 -data: - k8s-views-pods.json: | - { - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": [], - "__requires": [ - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "8.3.4" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "5.0.0" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - }, - { - "type": "panel", - "id": "gauge", - "name": "Gauge", - "version": "" - }, - { - "type": "panel", - "id": "table", - "name": "Table", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "target": { - "limit": 100, - "matchAny": false, - "tags": [], - "type": "dashboard" - }, - "type": "dashboard" - } - ] - }, - "description": "This is a modern 'Pods View' dashboard for your Kubernetes cluster(s). Made for kube-prometheus-stack and take advantage of the latest Grafana features. GitHub repository: https://github.com/dotdc/grafana-dashboards-kubernetes", - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 1, - "links": [], - "liveNow": false, - "panels": [ - { - "collapsed": false, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 0 - }, - "id": 43, - "panels": [], - "targets": [ - { - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "refId": "A" - } - ], - "title": "Information", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "Panel only works when a single pod is selected.", - "fieldConfig": { - "defaults": { - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "rgb(255, 255, 255)", - "value": null - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 2, - "w": 12, - "x": 0, - "y": 1 - }, - "id": 2, - "options": { - "colorMode": "none", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "mean" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "name", - "wideLayout": true - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "kube_pod_info{namespace=\"$namespace\", pod=\"$pod\", cluster=\"$cluster\"}", - "instant": true, - "interval": "", - "legendFormat": "{{ created_by_kind }}: {{ created_by_name }}", - "refId": "A" - } - ], - "title": "Created by", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "Panel only works when a single pod is selected.", - "fieldConfig": { - "defaults": { - "links": [ - { - "title": "", - "url": "/d/k8s_views_nodes/kubernetes-views-nodes?var-datasource=${datasource}&var-node=${__data.fields.node}" - } - ], - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "rgb(255, 255, 255)", - "value": null - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 2, - "w": 6, - "x": 12, - "y": 1 - }, - "id": 33, - "options": { - "colorMode": "none", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "mean" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "name", - "wideLayout": true - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "kube_pod_info{namespace=\"$namespace\", pod=\"$pod\", cluster=\"$cluster\"}", - "instant": true, - "interval": "", - "legendFormat": "{{ node }}", - "refId": "A" - } - ], - "title": "Running on", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "Panel only works when a single pod is selected.", - "fieldConfig": { - "defaults": { - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "rgb(255, 255, 255)", - "value": null - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 2, - "w": 6, - "x": 18, - "y": 1 - }, - "id": 41, - "options": { - "colorMode": "none", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "mean" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "name", - "wideLayout": true - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "kube_pod_info{namespace=\"$namespace\", pod=\"$pod\", cluster=\"$cluster\"}", - "instant": true, - "interval": "", - "legendFormat": "{{ pod_ip }}", - "refId": "A" - } - ], - "title": "Pod IP", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "Panel only works when a single pod is selected.", - "fieldConfig": { - "defaults": { - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "rgb(255, 255, 255)", - "value": null - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 2, - "w": 5, - "x": 0, - "y": 3 - }, - "id": 52, - "options": { - "colorMode": "none", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "mean" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "name", - "wideLayout": true - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "kube_pod_info{namespace=\"$namespace\", pod=\"$pod\", priority_class!=\"\", cluster=\"$cluster\"}", - "format": "time_series", - "instant": true, - "interval": "", - "legendFormat": "{{ priority_class }}", - "range": false, - "refId": "A" - } - ], - "title": "Priority Class", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "Panel only works when a single pod is selected.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "none" - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "Burstable" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "red", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "BestEffort" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "orange", - "mode": "fixed" - } - } - ] - } - ] - }, - "gridPos": { - "h": 2, - "w": 7, - "x": 5, - "y": 3 - }, - "id": 53, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "name", - "wideLayout": true - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "kube_pod_status_qos_class{namespace=\"$namespace\", pod=\"$pod\", cluster=\"$cluster\"} > 0", - "instant": true, - "interval": "", - "legendFormat": "{{ qos_class }}", - "refId": "A" - } - ], - "title": "QOS Class", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "Panel only works when a single pod is selected.", - "fieldConfig": { - "defaults": { - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "text", - "value": null - }, - { - "color": "red", - "value": 1 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 2, - "w": 6, - "x": 12, - "y": 3 - }, - "id": 56, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "name", - "wideLayout": true - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "kube_pod_container_status_last_terminated_reason{namespace=\"$namespace\", pod=\"$pod\", cluster=\"$cluster\"}", - "instant": true, - "interval": "", - "legendFormat": "{{ reason }}", - "refId": "A" - } - ], - "title": "Last Terminated Reason", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "Panel only works when a single pod is selected.", - "fieldConfig": { - "defaults": { - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "text", - "value": null - }, - { - "color": "red", - "value": 1 - }, - { - "color": "#EAB839", - "value": 2 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 2, - "w": 6, - "x": 18, - "y": 3 - }, - "id": 57, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [], - "fields": "", - "values": true - }, - "showPercentChange": false, - "textMode": "value", - "wideLayout": true - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "kube_pod_container_status_last_terminated_exitcode{namespace=\"$namespace\", pod=\"$pod\", cluster=\"$cluster\"}", - "instant": true, - "interval": "", - "legendFormat": "__auto", - "range": false, - "refId": "A" - } - ], - "title": "Last Terminated Exit Code", - "type": "stat" - }, - { - "collapsed": false, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 5 - }, - "id": 47, - "panels": [], - "targets": [ - { - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "refId": "A" - } - ], - "title": "Resources", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "blue", - "mode": "fixed" - }, - "decimals": 2, - "mappings": [], - "max": 1, - "min": 0, - "thresholds": { - "mode": "percentage", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "#EAB839", - "value": 60 - }, - { - "color": "red", - "value": 75 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 3, - "x": 0, - "y": 6 - }, - "id": 39, - "options": { - "minVizHeight": 75, - "minVizWidth": 75, - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showThresholdLabels": false, - "showThresholdMarkers": true, - "sizing": "auto" - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(rate(container_cpu_usage_seconds_total{namespace=\"$namespace\", pod=~\"$pod\", image!=\"\", cluster=\"$cluster\"}[$__rate_interval])) / sum(kube_pod_container_resource_requests{namespace=\"$namespace\", pod=~\"$pod\", resource=\"cpu\", job=~\"$job\", cluster=\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "Requests", - "refId": "A" - } - ], - "title": "Total pod CPU Requests usage", - "type": "gauge" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "decimals": 2, - "mappings": [], - "max": 1, - "min": 0, - "thresholds": { - "mode": "percentage", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "#EAB839", - "value": 60 - }, - { - "color": "red", - "value": 75 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 3, - "x": 3, - "y": 6 - }, - "id": 48, - "options": { - "minVizHeight": 75, - "minVizWidth": 75, - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showThresholdLabels": false, - "showThresholdMarkers": true, - "sizing": "auto" - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(rate(container_cpu_usage_seconds_total{namespace=\"$namespace\", pod=~\"$pod\", image!=\"\", cluster=\"$cluster\"}[$__rate_interval])) / sum(kube_pod_container_resource_limits{namespace=\"$namespace\", pod=~\"$pod\", resource=\"cpu\", job=~\"$job\", cluster=\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "Limits", - "refId": "A" - } - ], - "title": "Total pod CPU Limits usage", - "type": "gauge" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "blue", - "mode": "fixed" - }, - "decimals": 2, - "mappings": [], - "max": 1, - "min": 0, - "thresholds": { - "mode": "percentage", - "steps": [ - { - "color": "blue", - "value": null - }, - { - "color": "#EAB839", - "value": 80 - }, - { - "color": "red", - "value": 99 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 3, - "x": 6, - "y": 6 - }, - "id": 40, - "options": { - "minVizHeight": 75, - "minVizWidth": 75, - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showThresholdLabels": false, - "showThresholdMarkers": true, - "sizing": "auto" - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(container_memory_working_set_bytes{namespace=\"$namespace\", pod=~\"$pod\", image!=\"\", cluster=\"$cluster\"}) / sum(kube_pod_container_resource_requests{namespace=\"$namespace\", pod=~\"$pod\", resource=\"memory\", job=~\"$job\", cluster=\"$cluster\"})", - "instant": true, - "interval": "$resolution", - "legendFormat": "Requests", - "refId": "A" - } - ], - "title": "Total pod RAM Requests usage", - "type": "gauge" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "decimals": 2, - "mappings": [], - "max": 1, - "min": 0, - "thresholds": { - "mode": "percentage", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "#EAB839", - "value": 60 - }, - { - "color": "red", - "value": 75 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 3, - "x": 9, - "y": 6 - }, - "id": 49, - "options": { - "minVizHeight": 75, - "minVizWidth": 75, - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showThresholdLabels": false, - "showThresholdMarkers": true, - "sizing": "auto" - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(container_memory_working_set_bytes{namespace=\"$namespace\", pod=~\"$pod\", image!=\"\", cluster=\"$cluster\"}) / sum(kube_pod_container_resource_limits{namespace=\"$namespace\", pod=~\"$pod\", resource=\"memory\", job=~\"$job\", cluster=\"$cluster\"}) ", - "instant": true, - "interval": "$resolution", - "legendFormat": "Limits", - "refId": "B" - } - ], - "title": "Total pod RAM Limits usage", - "type": "gauge" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "custom": { - "align": "auto", - "cellOptions": { - "type": "auto" - }, - "filterable": false, - "inspect": false, - "minWidth": 100 - }, - "decimals": 4, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "blue", - "value": null - } - ] - }, - "unit": "none" - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "Memory Requests" - }, - "properties": [ - { - "id": "unit", - "value": "bytes" - }, - { - "id": "decimals", - "value": 2 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Memory Limits" - }, - "properties": [ - { - "id": "unit", - "value": "bytes" - }, - { - "id": "decimals", - "value": 2 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Memory Used" - }, - "properties": [ - { - "id": "unit", - "value": "bytes" - }, - { - "id": "decimals", - "value": 2 - } - ] - } - ] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 6 - }, - "id": 38, - "options": { - "cellHeight": "sm", - "footer": { - "countRows": false, - "fields": "", - "reducer": [ - "sum" - ], - "show": false - }, - "showHeader": true, - "sortBy": [] - }, - "pluginVersion": "10.4.1", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(kube_pod_container_resource_requests{namespace=\"$namespace\", pod=~\"$pod\", resource=\"cpu\", job=~\"$job\", cluster=\"$cluster\"}) by (container)", - "format": "table", - "instant": true, - "interval": "", - "intervalFactor": 1, - "legendFormat": "", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(kube_pod_container_resource_limits{namespace=\"$namespace\", pod=~\"$pod\", resource=\"cpu\", job=~\"$job\", cluster=\"$cluster\"}) by (container)", - "format": "table", - "instant": true, - "interval": "", - "intervalFactor": 1, - "legendFormat": "", - "refId": "B" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(kube_pod_container_resource_requests{namespace=\"$namespace\", pod=~\"$pod\", resource=\"memory\", job=~\"$job\", cluster=\"$cluster\"}) by (container)", - "format": "table", - "instant": true, - "interval": "", - "legendFormat": "", - "refId": "C" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(kube_pod_container_resource_limits{namespace=\"$namespace\", pod=~\"$pod\", resource=\"memory\", job=~\"$job\", cluster=\"$cluster\"}) by (container)", - "format": "table", - "instant": true, - "interval": "", - "legendFormat": "", - "refId": "D" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(rate(container_cpu_usage_seconds_total{namespace=\"$namespace\", pod=~\"$pod\", image!=\"\", container!=\"\", cluster=\"$cluster\"}[$__rate_interval])) by (container)", - "format": "table", - "hide": false, - "instant": true, - "legendFormat": "__auto", - "range": false, - "refId": "E" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(container_memory_working_set_bytes{namespace=\"$namespace\", pod=~\"$pod\", image!=\"\", container!=\"\", cluster=\"$cluster\"}) by (container)", - "format": "table", - "hide": false, - "instant": true, - "range": false, - "refId": "F" - } - ], - "title": "Resources by container", - "transformations": [ - { - "id": "seriesToColumns", - "options": { - "byField": "container" - } - }, - { - "id": "organize", - "options": { - "excludeByName": { - "Time": true, - "Time 1": true, - "Time 2": true, - "Time 4": true, - "__name__": true, - "__name__ 1": true, - "__name__ 2": true, - "__name__ 3": true, - "__name__ 4": true, - "container": false, - "endpoint": true, - "endpoint 2": true, - "endpoint 3": true, - "endpoint 4": true, - "instance": true, - "instance 2": true, - "instance 3": true, - "instance 4": true, - "job": true, - "job 2": true, - "job 3": true, - "job 4": true, - "namespace": true, - "namespace 2": true, - "namespace 3": true, - "namespace 4": true, - "node": true, - "node 2": true, - "node 3": true, - "node 4": true, - "pod": true, - "pod 2": true, - "pod 3": true, - "pod 4": true, - "resource 1": true, - "resource 2": true, - "resource 3": true, - "resource 4": true, - "service": true, - "service 2": true, - "service 3": true, - "service 4": true, - "uid 1": true, - "uid 2": true, - "uid 3": true, - "uid 4": true, - "unit 1": true, - "unit 2": true, - "unit 3": true, - "unit 4": true - }, - "indexByName": { - "Time 1": 7, - "Time 2": 8, - "Time 3": 9, - "Time 4": 10, - "Time 5": 11, - "Time 6": 12, - "Value #A": 2, - "Value #B": 3, - "Value #C": 5, - "Value #D": 6, - "Value #E": 1, - "Value #F": 4, - "container": 0 - }, - "renameByName": { - "Value #A": "CPU Requests", - "Value #B": "CPU Limits", - "Value #C": "Memory Requests", - "Value #D": "Memory Limits", - "Value #E": "CPU Used", - "Value #F": "Memory Used", - "container": "Container" - } - } - } - ], - "type": "table" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "Percent", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "area" - } - }, - "mappings": [], - "max": 1, - "min": 0, - "thresholds": { - "mode": "percentage", - "steps": [ - { - "color": "red", - "value": null - }, - { - "color": "yellow", - "value": 20 - }, - { - "color": "green", - "value": 30 - }, - { - "color": "yellow", - "value": 70 - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 14 - }, - "id": 50, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(container_cpu_usage_seconds_total{namespace=\"$namespace\", pod=~\"$pod\", image!=\"\", cluster=\"$cluster\"}[$__rate_interval])) by (container) / sum(kube_pod_container_resource_requests{namespace=\"$namespace\", pod=~\"$pod\", resource=\"cpu\", job=~\"$job\", cluster=\"$cluster\"}) by (container)", - "interval": "$resolution", - "legendFormat": "{{ container }} REQUESTS", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(rate(container_cpu_usage_seconds_total{namespace=\"$namespace\", pod=~\"$pod\", image!=\"\", cluster=\"$cluster\"}[$__rate_interval])) by (container) / sum(kube_pod_container_resource_limits{namespace=\"$namespace\", pod=~\"$pod\", resource=\"cpu\", job=~\"$job\", cluster=\"$cluster\"}) by (container)", - "hide": false, - "legendFormat": "{{ container }} LIMITS", - "range": true, - "refId": "B" - } - ], - "title": "CPU Usage / Requests & Limits by container", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "blue", - "mode": "thresholds" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "Percent", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "area" - } - }, - "mappings": [], - "max": 1, - "min": 0, - "thresholds": { - "mode": "percentage", - "steps": [ - { - "color": "red", - "value": null - }, - { - "color": "yellow", - "value": 20 - }, - { - "color": "green", - "value": 30 - }, - { - "color": "#EAB839", - "value": 70 - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 14 - }, - "id": 30, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(container_memory_working_set_bytes{namespace=\"$namespace\", pod=~\"$pod\", image!=\"\", cluster=\"$cluster\"}) by (container) / sum(kube_pod_container_resource_requests{namespace=\"$namespace\", pod=~\"$pod\", resource=\"memory\", job=~\"$job\", cluster=\"$cluster\"}) by (container)", - "interval": "", - "legendFormat": "{{ container }} REQUESTS", - "range": true, - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "expr": "sum(container_memory_working_set_bytes{namespace=\"$namespace\", pod=~\"$pod\", image!=\"\", cluster=\"$cluster\"}) by (container) / sum(kube_pod_container_resource_limits{namespace=\"$namespace\", pod=~\"$pod\", resource=\"memory\", job=~\"$job\", cluster=\"$cluster\"}) by (container)", - "hide": false, - "legendFormat": "{{ container }} LIMITS", - "range": true, - "refId": "B" - } - ], - "title": "Memory Usage / Requests & Limits by container", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "CPU Cores", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 4, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "limit" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "#F2495C", - "mode": "fixed" - } - }, - { - "id": "custom.fillOpacity", - "value": 0 - } - ] - } - ] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 22 - }, - "id": 29, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(container_cpu_usage_seconds_total{namespace=\"$namespace\", pod=~\"$pod\", image!=\"\", container!=\"\", cluster=\"$cluster\"}[$__rate_interval])) by (container)", - "interval": "$resolution", - "legendFormat": "{{ container }}", - "range": true, - "refId": "A" - } - ], - "title": "CPU Usage by container", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "Bytes", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 22 - }, - "id": 51, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(container_memory_working_set_bytes{namespace=\"$namespace\", pod=~\"$pod\", image!=\"\", container!=\"\", cluster=\"$cluster\"}) by (container)", - "interval": "", - "legendFormat": "{{ container }}", - "range": true, - "refId": "A" - } - ], - "title": "Memory Usage by container", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "description": "", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "SECONDS", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "decimals": 2, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "s" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 30 - }, - "id": 59, - "options": { - "legend": { - "calcs": [ - "min", - "max", - "mean" - ], - "displayMode": "table", - "placement": "right", - "showLegend": true, - "sortBy": "Max", - "sortDesc": true - }, - "tooltip": { - "mode": "single", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(rate(container_cpu_cfs_throttled_seconds_total{namespace=~\"$namespace\", pod=~\"$pod\", image!=\"\", container!=\"\", cluster=\"$cluster\"}[$__rate_interval])) by (container)", - "interval": "$resolution", - "legendFormat": "{{ container }}", - "range": true, - "refId": "A" - } - ], - "title": "CPU Throttled seconds by container", - "type": "timeseries" - }, - { - "collapsed": false, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 38 - }, - "id": 62, - "panels": [], - "title": "Kubernetes", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "blue", - "mode": "thresholds" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "Percent", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "area" - } - }, - "mappings": [], - "max": 1, - "min": 0, - "thresholds": { - "mode": "percentage", - "steps": [ - { - "color": "red", - "value": null - }, - { - "color": "yellow", - "value": 20 - }, - { - "color": "green", - "value": 30 - }, - { - "color": "#EAB839", - "value": 70 - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 39 - }, - "id": 60, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(increase(container_oom_events_total{namespace=\"${namespace}\", pod=\"${pod}\", container!=\"\", cluster=\"$cluster\"}[$__rate_interval])) by (container)", - "interval": "", - "legendFormat": "{{ container }}", - "range": true, - "refId": "A" - } - ], - "title": "OOM Events by container", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "blue", - "mode": "thresholds" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "Percent", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineStyle": { - "fill": "solid" - }, - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "area" - } - }, - "mappings": [], - "max": 1, - "min": 0, - "thresholds": { - "mode": "percentage", - "steps": [ - { - "color": "red", - "value": null - }, - { - "color": "yellow", - "value": 20 - }, - { - "color": "green", - "value": 30 - }, - { - "color": "#EAB839", - "value": 70 - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 39 - }, - "id": 61, - "options": { - "legend": { - "calcs": [], - "displayMode": "table", - "placement": "right", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "desc" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(increase(kube_pod_container_status_restarts_total{namespace=~\"${namespace}\", pod=\"${pod}\", container!=\"\", job=~\"$job\", cluster=\"$cluster\"}[$__rate_interval])) by (container)", - "interval": "", - "legendFormat": "{{ container }}", - "range": true, - "refId": "A" - } - ], - "title": "Container Restarts by container", - "type": "timeseries" - }, - { - "collapsed": false, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 47 - }, - "id": 45, - "panels": [], - "targets": [ - { - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "refId": "A" - } - ], - "title": "Network", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "bytes" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 48 - }, - "id": 31, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(container_network_receive_bytes_total{namespace=\"$namespace\", pod=~\"$pod\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "Received", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "- sum(rate(container_network_transmit_bytes_total{namespace=\"$namespace\", pod=~\"$pod\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "Transmitted", - "refId": "B" - } - ], - "title": "Network - Bandwidth", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "pps" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 48 - }, - "id": 34, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(container_network_receive_packets_total{namespace=\"$namespace\", pod=~\"$pod\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "Received", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "- sum(rate(container_network_transmit_packets_total{namespace=\"$namespace\", pod=~\"$pod\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "Transmitted", - "refId": "B" - } - ], - "title": "Network - Packets Rate", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "pps" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 0, - "y": 56 - }, - "id": 36, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(container_network_receive_packets_dropped_total{namespace=\"$namespace\", pod=~\"$pod\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "Received", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "- sum(rate(container_network_transmit_packets_dropped_total{namespace=\"$namespace\", pod=~\"$pod\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "Transmitted", - "refId": "B" - } - ], - "title": "Network - Packets Dropped", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "opacity", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "never", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - }, - { - "color": "red", - "value": 80 - } - ] - }, - "unit": "pps" - }, - "overrides": [] - }, - "gridPos": { - "h": 8, - "w": 12, - "x": 12, - "y": 56 - }, - "id": 37, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "mode": "multi", - "sort": "none" - } - }, - "pluginVersion": "8.3.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "sum(rate(container_network_receive_errors_total{namespace=\"$namespace\", pod=~\"$pod\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "Received", - "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "exemplar": true, - "expr": "- sum(rate(container_network_transmit_errors_total{namespace=\"$namespace\", pod=~\"$pod\", cluster=\"$cluster\"}[$__rate_interval]))", - "interval": "$resolution", - "legendFormat": "Transmitted", - "refId": "B" - } - ], - "title": "Network - Errors", - "type": "timeseries" - } - ], - "refresh": "30s", - "schemaVersion": 39, - "tags": [ - "Kubernetes", - "Prometheus" - ], - "templating": { - "list": [ - { - "current": { - "selected": false, - "text": "", - "value": "" - }, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "datasource", - "options": [], - "query": "prometheus", - "queryValue": "", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "current": { - "isNone": true, - "selected": false, - "text": "None", - "value": "" - }, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_node_info,cluster)", - "hide": 0, - "includeAll": false, - "multi": false, - "name": "cluster", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(kube_node_info,cluster)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - }, - { - "current": { - "selected": false, - "text": "monitoring", - "value": "monitoring" - }, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_pod_info{cluster=\"$cluster\"}, namespace)", - "hide": 0, - "includeAll": false, - "multi": false, - "name": "namespace", - "options": [], - "query": { - "query": "label_values(kube_pod_info{cluster=\"$cluster\"}, namespace)", - "refId": "Prometheus-namespace-Variable-Query" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "tagValuesQuery": "", - "tagsQuery": "", - "type": "query", - "useTags": false - }, - { - "current": { - "selected": false, - "text": "", - "value": "" - }, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_pod_info{namespace=\"$namespace\", cluster=\"$cluster\"}, pod)", - "hide": 0, - "includeAll": true, - "multi": true, - "name": "pod", - "options": [], - "query": { - "query": "label_values(kube_pod_info{namespace=\"$namespace\", cluster=\"$cluster\"}, pod)", - "refId": "Prometheus-pod-Variable-Query" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "tagValuesQuery": "", - "tagsQuery": "", - "type": "query", - "useTags": false - }, - { - "current": { - "selected": false, - "text": "30s", - "value": "30s" - }, - "hide": 0, - "includeAll": false, - "multi": false, - "name": "resolution", - "options": [ - { - "selected": false, - "text": "1s", - "value": "1s" - }, - { - "selected": false, - "text": "15s", - "value": "15s" - }, - { - "selected": true, - "text": "30s", - "value": "30s" - }, - { - "selected": false, - "text": "1m", - "value": "1m" - }, - { - "selected": false, - "text": "3m", - "value": "3m" - }, - { - "selected": false, - "text": "5m", - "value": "5m" - } - ], - "query": "1s, 15s, 30s, 1m, 3m, 5m", - "queryValue": "", - "skipUrlSync": false, - "type": "custom" - }, - { - "current": { - "selected": false, - "text": "kube-state-metrics", - "value": "kube-state-metrics" - }, - "datasource": { - "type": "prometheus", - "uid": "${datasource}" - }, - "definition": "label_values(kube_pod_info{namespace=\"$namespace\", cluster=\"$cluster\"},job)", - "hide": 0, - "includeAll": false, - "multi": true, - "name": "job", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(kube_pod_info{namespace=\"$namespace\", cluster=\"$cluster\"},job)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "sort": 1, - "type": "query" - } - ] - }, - "time": { - "from": "now-1h", - "to": "now" - }, - "timepicker": {}, - "timezone": "", - "title": "Kubernetes / Views / Pods", - "uid": "k8s_views_pods", - "version": 28, - "weekStart": "" - } -kind: ConfigMap -metadata: - annotations: - grafana_folder: Kubernetes - labels: - grafana_dashboard: "1" - name: dashboards-k8s-views-pods - namespace: futureporn diff --git a/flux/infrastructure/controllers/ingress-nginx.yaml b/flux/infrastructure/controllers/ingress-nginx.yaml deleted file mode 100644 index 60d386f..0000000 --- a/flux/infrastructure/controllers/ingress-nginx.yaml +++ /dev/null @@ -1,32 +0,0 @@ ---- -apiVersion: source.toolkit.fluxcd.io/v1beta2 -kind: HelmRepository -metadata: - name: ingress-nginx - namespace: futureporn -spec: - interval: 24h - url: https://kubernetes.github.io/ingress-nginx ---- -apiVersion: helm.toolkit.fluxcd.io/v2 -kind: HelmRelease -metadata: - name: ingress-nginx - namespace: futureporn -spec: - interval: 30m - chart: - spec: - chart: ingress-nginx - version: "*" - sourceRef: - kind: HelmRepository - name: ingress-nginx - namespace: futureporn - interval: 12h - values: - controller: - service: - type: "LoadBalancer" - admissionWebhooks: - enabled: false diff --git a/flux/infrastructure/controllers/kube-prometheus-stack.yaml b/flux/infrastructure/controllers/kube-prometheus-stack.yaml deleted file mode 100644 index 0b11276..0000000 --- a/flux/infrastructure/controllers/kube-prometheus-stack.yaml +++ /dev/null @@ -1,56 +0,0 @@ - - ---- -apiVersion: source.toolkit.fluxcd.io/v1 -kind: HelmRepository -metadata: - name: prometheus-community - namespace: futureporn -spec: - interval: 24h - url: https://prometheus-community.github.io/helm-charts - ---- -apiVersion: helm.toolkit.fluxcd.io/v2 -kind: HelmRelease -metadata: - name: prometheus - namespace: futureporn -spec: - interval: 30m - chart: - spec: - chart: kube-prometheus-stack - version: "60.3.0" - sourceRef: - kind: HelmRepository - name: prometheus-community - namespace: futureporn - interval: 12h - values: - prometheus: - prometheusSpec: - storageSpec: - volumeClaimTemplate: - spec: - storageClassName: vultr-block-storage-hdd - resources: - requests: - storage: 40Gi - grafana: - service: - - annotations: - - admin: - existingSecret: grafana - sidecar: - dashboards: - enabled: true - defaultFolderName: "General" - label: grafana_dashboard - labelValue: "1" - folderAnnotation: grafana_folder - searchNamespace: ALL - provider: - foldersFromFilesStructure: true diff --git a/flux/infrastructure/controllers/kustomization.yaml b/flux/infrastructure/controllers/kustomization.yaml deleted file mode 100644 index 4437653..0000000 --- a/flux/infrastructure/controllers/kustomization.yaml +++ /dev/null @@ -1,8 +0,0 @@ -apiVersion: kustomize.config.k8s.io/v1beta1 -kind: Kustomization -resources: - # - external-dns.yaml - - cert-manager.yaml - - ingress-nginx.yaml - - grafana-k8s-dashboards.yaml - - kube-prometheus-stack.yaml \ No newline at end of file diff --git a/package.json b/package.json index 511125b..a9138c4 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "futureporn", - "version": "1.0.3", + "version": "2.0.0", "description": "", "main": "index.js", "scripts": { diff --git a/packages/fetchers/README.md b/packages/fetchers/README.md deleted file mode 100644 index 85cc7df..0000000 --- a/packages/fetchers/README.md +++ /dev/null @@ -1,5 +0,0 @@ -# @futureporn/fetchers - -`fetch()` wrappers for getting/setting values in our database. - -distinct from @futureporn/scout which is fetching data from external sources. \ No newline at end of file diff --git a/packages/fetchers/package.json b/packages/fetchers/package.json deleted file mode 100644 index 33a0b29..0000000 --- a/packages/fetchers/package.json +++ /dev/null @@ -1,26 +0,0 @@ -{ - "name": "@futureporn/fetchers", - "type": "module", - "version": "1.0.0", - "description": "", - "main": "index.js", - "scripts": { - "test": "echo \"Error: no test specified\" && exit 0" - }, - "exports": { - "./*.ts": "./src/*.ts" - }, - "keywords": [], - "author": "", - "license": "Unlicense", - "dependencies": { - "@futureporn/types": "workspace:^", - "@types/qs": "^6.9.15", - "date-fns": "^3.6.0", - "qs": "^6.13.0" - }, - "devDependencies": { - "@types/chai": "^4.3.19", - "chai": "^5.1.1" - } -} diff --git a/packages/fetchers/pnpm-lock.yaml b/packages/fetchers/pnpm-lock.yaml deleted file mode 100644 index 41d8269..0000000 --- a/packages/fetchers/pnpm-lock.yaml +++ /dev/null @@ -1,221 +0,0 @@ -lockfileVersion: '9.0' - -settings: - autoInstallPeers: true - excludeLinksFromLockfile: false - -importers: - - .: - dependencies: - '@futureporn/types': - specifier: workspace:^ - version: link:../types - '@types/qs': - specifier: ^6.9.15 - version: 6.9.15 - date-fns: - specifier: ^3.6.0 - version: 3.6.0 - qs: - specifier: ^6.13.0 - version: 6.13.0 - devDependencies: - '@types/chai': - specifier: ^4.3.19 - version: 4.3.19 - chai: - specifier: ^5.1.1 - version: 5.1.1 - -packages: - - '@types/chai@4.3.19': - resolution: {integrity: sha512-2hHHvQBVE2FiSK4eN0Br6snX9MtolHaTo/batnLjlGRhoQzlCL61iVpxoqO7SfFyOw+P/pwv+0zNHzKoGWz9Cw==} - - '@types/qs@6.9.15': - resolution: {integrity: sha512-uXHQKES6DQKKCLh441Xv/dwxOq1TVS3JPUMlEqoEglvlhR6Mxnlew/Xq/LRVHpLyk7iK3zODe1qYHIMltO7XGg==} - - assertion-error@2.0.1: - resolution: {integrity: sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==} - engines: {node: '>=12'} - - call-bind@1.0.7: - resolution: {integrity: sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==} - engines: {node: '>= 0.4'} - - chai@5.1.1: - resolution: {integrity: sha512-pT1ZgP8rPNqUgieVaEY+ryQr6Q4HXNg8Ei9UnLUrjN4IA7dvQC5JB+/kxVcPNDHyBcc/26CXPkbNzq3qwrOEKA==} - engines: {node: '>=12'} - - check-error@2.1.1: - resolution: {integrity: sha512-OAlb+T7V4Op9OwdkjmguYRqncdlx5JiofwOAUkmTF+jNdHwzTaTs4sRAGpzLF3oOz5xAyDGrPgeIDFQmDOTiJw==} - engines: {node: '>= 16'} - - date-fns@3.6.0: - resolution: {integrity: sha512-fRHTG8g/Gif+kSh50gaGEdToemgfj74aRX3swtiouboip5JDLAyDE9F11nHMIcvOaXeOC6D7SpNhi7uFyB7Uww==} - - deep-eql@5.0.2: - resolution: {integrity: sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==} - engines: {node: '>=6'} - - define-data-property@1.1.4: - resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==} - engines: {node: '>= 0.4'} - - es-define-property@1.0.0: - resolution: {integrity: sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==} - engines: {node: '>= 0.4'} - - es-errors@1.3.0: - resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==} - engines: {node: '>= 0.4'} - - function-bind@1.1.2: - resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} - - get-func-name@2.0.2: - resolution: {integrity: sha512-8vXOvuE167CtIc3OyItco7N/dpRtBbYOsPsXCz7X/PMnlGjYjSGuZJgM1Y7mmew7BKf9BqvLX2tnOVy1BBUsxQ==} - - get-intrinsic@1.2.4: - resolution: {integrity: sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==} - engines: {node: '>= 0.4'} - - gopd@1.0.1: - resolution: {integrity: sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==} - - has-property-descriptors@1.0.2: - resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==} - - has-proto@1.0.3: - resolution: {integrity: sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==} - engines: {node: '>= 0.4'} - - has-symbols@1.0.3: - resolution: {integrity: sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==} - engines: {node: '>= 0.4'} - - hasown@2.0.2: - resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} - engines: {node: '>= 0.4'} - - loupe@3.1.1: - resolution: {integrity: sha512-edNu/8D5MKVfGVFRhFf8aAxiTM6Wumfz5XsaatSxlD3w4R1d/WEKUTydCdPGbl9K7QG/Ca3GnDV2sIKIpXRQcw==} - - object-inspect@1.13.2: - resolution: {integrity: sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==} - engines: {node: '>= 0.4'} - - pathval@2.0.0: - resolution: {integrity: sha512-vE7JKRyES09KiunauX7nd2Q9/L7lhok4smP9RZTDeD4MVs72Dp2qNFVz39Nz5a0FVEW0BJR6C0DYrq6unoziZA==} - engines: {node: '>= 14.16'} - - qs@6.13.0: - resolution: {integrity: sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==} - engines: {node: '>=0.6'} - - set-function-length@1.2.2: - resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==} - engines: {node: '>= 0.4'} - - side-channel@1.0.6: - resolution: {integrity: sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==} - engines: {node: '>= 0.4'} - -snapshots: - - '@types/chai@4.3.19': {} - - '@types/qs@6.9.15': {} - - assertion-error@2.0.1: {} - - call-bind@1.0.7: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - set-function-length: 1.2.2 - - chai@5.1.1: - dependencies: - assertion-error: 2.0.1 - check-error: 2.1.1 - deep-eql: 5.0.2 - loupe: 3.1.1 - pathval: 2.0.0 - - check-error@2.1.1: {} - - date-fns@3.6.0: {} - - deep-eql@5.0.2: {} - - define-data-property@1.1.4: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - gopd: 1.0.1 - - es-define-property@1.0.0: - dependencies: - get-intrinsic: 1.2.4 - - es-errors@1.3.0: {} - - function-bind@1.1.2: {} - - get-func-name@2.0.2: {} - - get-intrinsic@1.2.4: - dependencies: - es-errors: 1.3.0 - function-bind: 1.1.2 - has-proto: 1.0.3 - has-symbols: 1.0.3 - hasown: 2.0.2 - - gopd@1.0.1: - dependencies: - get-intrinsic: 1.2.4 - - has-property-descriptors@1.0.2: - dependencies: - es-define-property: 1.0.0 - - has-proto@1.0.3: {} - - has-symbols@1.0.3: {} - - hasown@2.0.2: - dependencies: - function-bind: 1.1.2 - - loupe@3.1.1: - dependencies: - get-func-name: 2.0.2 - - object-inspect@1.13.2: {} - - pathval@2.0.0: {} - - qs@6.13.0: - dependencies: - side-channel: 1.0.6 - - set-function-length@1.2.2: - dependencies: - define-data-property: 1.1.4 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - gopd: 1.0.1 - has-property-descriptors: 1.0.2 - - side-channel@1.0.6: - dependencies: - call-bind: 1.0.7 - es-errors: 1.3.0 - get-intrinsic: 1.2.4 - object-inspect: 1.13.2 diff --git a/packages/fetchers/src/config.ts b/packages/fetchers/src/config.ts deleted file mode 100644 index 3052853..0000000 --- a/packages/fetchers/src/config.ts +++ /dev/null @@ -1,27 +0,0 @@ -const requiredEnvVars = [ - 'POSTGREST_URL', - 'AUTOMATION_USER_JWT', - 'SCOUT_URL', -] as const; - -const getEnvVar = (key: typeof requiredEnvVars[number]): string => { - const value = process.env[key]; - if (!value) { - throw new Error(`Missing ${key} env var`); - } - return value; -}; - -export interface Config { - postgrestUrl: string; - automationUserJwt: string; - scoutUrl: string; -} - - -export const configs: Config = { - scoutUrl: getEnvVar('SCOUT_URL'), - postgrestUrl: getEnvVar('POSTGREST_URL'), - automationUserJwt: getEnvVar('AUTOMATION_USER_JWT') -} - diff --git a/packages/fetchers/src/createRecording.ts b/packages/fetchers/src/createRecording.ts deleted file mode 100644 index 6c8a7a6..0000000 --- a/packages/fetchers/src/createRecording.ts +++ /dev/null @@ -1,29 +0,0 @@ -import { configs } from "./config.ts"; - - -export default async function createRecording({ url, discordMessageId, date }: { url: string, discordMessageId?: string, date: Date }) { - if (!url) throw new Error('createRecording requires {string} url'); - const payload = { - url, - discord_message_id: discordMessageId, - date: date.toISOString() - } - const res = await fetch(`${configs.postgrestUrl}/recordings`, { - method: 'POST', - headers: { - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=representation', - 'Authorization': `Bearer ${configs.automationUserJwt}` - }, - body: JSON.stringify(payload) - }) - if (!res.ok) { - const body = await res.text() - const msg = `Failed to create Recording. status=${res.status}, statusText=${res.statusText}, body=${body}` - console.error(msg) - throw new Error(msg); - } - const data = await res.json() as { id: string } - return data.id -} diff --git a/packages/fetchers/src/createS3File.ts b/packages/fetchers/src/createS3File.ts deleted file mode 100644 index 11c34ca..0000000 --- a/packages/fetchers/src/createS3File.ts +++ /dev/null @@ -1,31 +0,0 @@ -import { configs } from "./config.ts" -import type { S3FileRecord, S3FileResponse } from '@futureporn/types' - -export default async function createS3File(s3File?: S3FileRecord): Promise { - if (!s3File) throw new Error(`first argument passed to createS3File must be a {S3Record}`); - console.log(s3File) - const url = `${configs.postgrestUrl}/s3_files` - const payload: any = { - s3_id: s3File.s3_id, - s3_key: s3File.s3_key, - } - const fetchOptions = { - method: 'POST', - headers: { - 'Authorization': `Bearer ${configs.automationUserJwt}`, - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=representation' - }, - body: JSON.stringify(payload) - } - const res = await fetch (url, fetchOptions) - if (!res.ok) { - const body = await res.json() - throw new Error(`Problem during createS3File. res.status=${res.status}, res.statusText=${res.statusText}, body=${JSON.stringify(body)}`) - } - const json = await res.json() as S3FileResponse[] - const s3FileRes = json[0] - if (!s3FileRes) return null - else return s3FileRes -} \ No newline at end of file diff --git a/packages/fetchers/src/createSegment.ts b/packages/fetchers/src/createSegment.ts deleted file mode 100644 index f21116e..0000000 --- a/packages/fetchers/src/createSegment.ts +++ /dev/null @@ -1,30 +0,0 @@ -import { SegmentResponse } from '@futureporn/types'; -import { configs } from './config.ts' -import querystring from 'node:querystring' - -export default async function createSegment(s3_key: string, vod_id: string): Promise { - if (!s3_key) throw new Error('getSegments requires {string} s3_key as first arg'); - const segmentPayload = { - s3_key, - vod_id - } - const res = await fetch(`${configs.postgrestUrl}/segments?select=*,vod:vods(*,recordings(*))`, { - method: 'POST', - headers: { - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=representation', - 'Authorization': `Bearer ${configs.automationUserJwt}` - }, - body: JSON.stringify(segmentPayload) - }) - if (!res.ok) { - const body = await res.text() - const msg = `failed to create Segment. status=${res.status}, statusText=${res.statusText}, body=${body}` - console.error(msg) - throw new Error(msg); - } - const data = await res.json() as SegmentResponse[] - if (!data[0]) throw new Error('failed to createSegment! body[0] was missing.'); - return data[0] -} diff --git a/packages/fetchers/src/createSegmentsVodLink.ts b/packages/fetchers/src/createSegmentsVodLink.ts deleted file mode 100644 index bff6ece..0000000 --- a/packages/fetchers/src/createSegmentsVodLink.ts +++ /dev/null @@ -1,36 +0,0 @@ -// import type { Helpers } from 'graphile-worker' -import { configs } from './config.ts' -import querystring from 'node:querystring' - -export default async function createSegmentsVodLink(vod_id: string, segment_id: string): Promise { - console.info(`createSegmentsVodLink with vod_id=${vod_id}, segment_id=${segment_id}`) - if (!vod_id) throw new Error('createSegmentsVodLink requires {string} vod_id as first arg'); - if (!segment_id) throw new Error('createSegmentsVodLink requires {string} segment_id as second arg'); - const segmentVodLinkPayload = { - vod_id, - segment_id - } - const res = await fetch(`${configs.postgrestUrl}/segments_vod_links`, { - method: 'POST', - headers: { - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=headers-only', - 'Authorization': `Bearer ${configs.automationUserJwt}`, - }, - body: JSON.stringify(segmentVodLinkPayload) - }) - if (!res.ok) { - const body = await res.text() - throw new Error(`failed to create SegmentsVodLink. status=${res.status}, statusText=${res.statusText}, body=${body}`); - } - const location = res.headers.get('location') - if (!location) throw new Error(`failed to get location header in response from postgrest`); - const parsedQuery = querystring.parse(location) - const segmentsId = parsedQuery['/segments_vod_links?id'] - if (!segmentsId) throw new Error('segments_vod_links?id was undefined which is unexpected'); - if (Array.isArray(segmentsId)) throw new Error('segments_vod_links was an array which is unexpected'); - const id = segmentsId.split('.').at(-1) - if (!id) throw new Error('failed to get id '); - return id -} diff --git a/packages/fetchers/src/createStreamInDatabase.ts b/packages/fetchers/src/createStreamInDatabase.ts deleted file mode 100644 index c9e692f..0000000 --- a/packages/fetchers/src/createStreamInDatabase.ts +++ /dev/null @@ -1,30 +0,0 @@ -import { configs } from './config.ts' - -export default async function createStreamInDatabase(url: string, discordMessageId: string): Promise { - const streamPayload = { - url, - status: 'pending_recording', - discord_message_id: discordMessageId, - date: new Date().toISOString() - } - const res = await fetch(`${configs.postgrestUrl}/streams`, { - method: 'POST', - headers: { - 'Content-Type': 'application/json', - 'Prefer': 'return=headers-only', - 'Authorization': `Bearer ${configs.automationUserJwt}`, - }, - body: JSON.stringify(streamPayload) - }) - if (!res.ok) { - const status = res.status - const statusText = res.statusText - const body = await res.text() - const msg = `Failed to create stream in database. status=${status}, statusText=${statusText}, body=${body}` - console.error(msg) - throw new Error(msg) - } - const id = res.headers.get('location')?.split('.').at(-1) - if (!id) throw new Error('id could not be parsed from location header'); - return id -} \ No newline at end of file diff --git a/packages/fetchers/src/createVod.ts b/packages/fetchers/src/createVod.ts deleted file mode 100644 index 7eb039d..0000000 --- a/packages/fetchers/src/createVod.ts +++ /dev/null @@ -1,27 +0,0 @@ -import { configs } from "./config.ts" -import type { VodResponse, VodRecord } from '@futureporn/types' - -export default async function createVod(vodPartial?: Partial): Promise { - if (!vodPartial) console.warn(`createVod() was called with no vodPartial as argument. It's recommended to supply this argument with key/values stream_id, date, and vtuber`); - if (!vodPartial?.date) throw new Error('createVod requires argument vodPartial.date'); - const url = `${configs.postgrestUrl}/vods` - const fetchOptions = { - method: 'POST', - headers: { - 'Authorization': `Bearer ${configs.automationUserJwt}`, - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=representation' - }, - body: JSON.stringify(vodPartial) - } - const res = await fetch (url, fetchOptions) - if (!res.ok) { - const body = await res.json() - throw new Error(`Problem during createVod. res.status=${res.status}, res.statusText=${res.statusText}, body=${JSON.stringify(body)}`) - } - const json = await res.json() as VodResponse[] - const vod = json[0] - if (!vod) return null - else return vod -} \ No newline at end of file diff --git a/packages/fetchers/src/findOrCreateStream.ts b/packages/fetchers/src/findOrCreateStream.ts deleted file mode 100644 index c37163c..0000000 --- a/packages/fetchers/src/findOrCreateStream.ts +++ /dev/null @@ -1,83 +0,0 @@ -import { configs } from "./config.ts" -import type { StreamResponse, VodRecord } from "@futureporn/types" -import { sub } from 'date-fns' - -export async function findStream(vtuberId: string, lteDate: Date, gteDate: Date): Promise { - const fetchUrl = `${configs.postgrestUrl}/streams?select=id&vtuber=eq.${vtuberId}&date=gte.${gteDate.toISOString()}&date=lte.${lteDate.toISOString()}` - const fetchOptions = { - method: 'GET', - headers: { - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=representation' - } - } - const res = await fetch(fetchUrl, fetchOptions) - if (!res.ok) { - const body = await res.text() - const msg = `findStream fetch failed. status=${res.status}, statusText=${res.statusText}, body=${body}` - console.error(msg) - throw new Error(msg) - } - const json = await res.json() as StreamResponse[] - if (!json || !json[0]) return null - return json[0].id -} - -export async function createStream(): Promise { - const payload = { - - } - const fetchUrl = `${configs.postgrestUrl}/streams?select=id` - const fetchOptions = { - method: 'POST', - headers: { - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=representation', - 'Authorization': `Bearer ${configs.automationUserJwt}`, - }, - body: JSON.stringify(payload) - } - const res = await fetch(fetchUrl, fetchOptions) - if (!res.ok) { - const body = await res.text() - const msg = `createStream fetch failed. status=${res.status}, statusText=${res.statusText}, body=${body}` - console.error(msg) - throw new Error(msg) - } - const json = await res.json() as StreamResponse[] - if (!json || !json[0]) return null - return json[0].id -} - - -/** - * Find or create a stream in the database. - * - * We use the vtuberId and the date to find a stream taking place within the last n minutes. - * - * The stream date range that we accept is inclusive between date and (date-minutes) - * - * This is useful for eliminating duplicates, since a stream can be created by more than one input. - * - * - manual - * - email - * - twitter - * - */ -export default async function findOrCreateStream({ vtuberId, date, minutes = 15 }: { vtuberId: string, date: Date, minutes?: number }): Promise { - console.info(`findOrCreateStream with vtuberId=${vtuberId}, date=${new Date(date).toISOString()}, minutes=${minutes}`) - if (!vtuberId) throw new Error(`findOrCreateStream requires vruberId passed in the options argument.`); - if (!date) throw new Error(`findOrCreateStream requires date passed in the options argument.`); - const gteDate = sub(date, { minutes }) - const lteDate = date - const foundStream = await findStream(vtuberId, lteDate, gteDate) - if (!foundStream) { - return createStream() - } else { - return foundStream - } - -} - diff --git a/packages/fetchers/src/findOrCreateVtuber.spec.ts b/packages/fetchers/src/findOrCreateVtuber.spec.ts deleted file mode 100644 index 65f410a..0000000 --- a/packages/fetchers/src/findOrCreateVtuber.spec.ts +++ /dev/null @@ -1,20 +0,0 @@ -import findOrCreateVtuber from "./findOrCreateVtuber.ts" -import { expect } from 'chai' - -// These tests are meant to run in docker via gitea-actions -// or https://github.com/nektos/act - -describe('integration', function () { - this.beforeAll(function () { - // set up db - // seed db - // set env - }) - describe('findOrCreateVtuber', function () { - xit('should find projektmelody when given url=https://chaturbate.com/projektmelody', async function () { - const url = 'https://chaturbate.com/projektmelody' - const vtuber = await findOrCreateVtuber({ url }, { prefer: 'return=representation' }) - expect(vtuber).to.have.property('slug', 'projektmelody') - }) - }) -}) \ No newline at end of file diff --git a/packages/fetchers/src/findOrCreateVtuber.ts b/packages/fetchers/src/findOrCreateVtuber.ts deleted file mode 100644 index cda4742..0000000 --- a/packages/fetchers/src/findOrCreateVtuber.ts +++ /dev/null @@ -1,111 +0,0 @@ -import { configs } from "./config.ts" -import type { VtuberRecord, VtuberResponse } from "@futureporn/types" -import qs from 'qs' - -interface vTuberSearchQuery { - url: string; - name: string; -} - -interface vTuberSearchOptions { - prefer: string; -} - -const columns = [ - 'chaturbate', - 'twitter', - 'patreon', - 'twitch', - 'tiktok', - 'onlyfans', - 'youtube', - 'fansly', - 'pornhub' -]; - -// @see https://bobbyhadz.com/blog/javascript-remove-trailing-slash-from-string -function removeTrailingSlash(url: string) { - return url.replace(/\/+$/, ''); -} - -function buildUrlSearchOrTemplate(columns: string[], url: string): string { - return `(${columns.map(column => `${column}.ilike.*${url}*`).join(',')})`; -} - - -async function findVtuber(query: Partial, options?: Partial): Promise { - const { url, name } = query - const queryOptions: any = {} - // When url argument exists, we try searching against the vtuber's platform urls - if (url) queryOptions.or = buildUrlSearchOrTemplate(columns, removeTrailingSlash(url)); - if (!url && !name) throw new Error(`findVtuber requires url or name passed in an object as argument.`); - const fetchUrl = `${configs.postgrestUrl}/vtubers?${qs.stringify(queryOptions, { encode: false })}` - console.info(fetchUrl) - const fetchOptions = { - method: 'GET', - headers: { - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=representation', - 'Authorization': `Bearer ${configs.automationUserJwt}`, - } - } - const res = await fetch(fetchUrl, fetchOptions) - if (!res.ok) { - const msg = `request failed. status=${res.status}, statusText=${res.statusText}` - console.error(msg) - throw new Error(msg) - } - const json = await res.json() as VtuberResponse[] - // console.info(`vtuber results as follows.`) - // console.info(json) - return json?.at(0)?.id || null -} - -async function createVtuber(vtuber: Partial): Promise { - const createVtuberPayload = vtuber - const fetchUrl = `${configs.postgrestUrl}/vtubers` - const fetchOptions = { - method: 'POST', - headers: { - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=representation', - 'Authorization': `Bearer ${configs.automationUserJwt}` - }, - body: JSON.stringify(createVtuberPayload) - } - const res = await fetch(fetchUrl, fetchOptions) - if (!res.ok) { - const body = await res.text() - const msg = `createVtuber fetch failed. status=${res.status}, statusText=${res.statusText}, body=${body}` - console.error(msg) - throw new Error(msg) - } - const json = await res.json() as VtuberResponse[] - // console.info(`createVtuber with vtuber as follows`) - // console.info(vtuber) - // console.info(json) - const vtuberData = json[0] - if (!vtuberData) throw new Error('failed to createVtuber') - return vtuberData.id -} - - -export default async function findOrCreateVtuber(query: Partial, options?: Partial): Promise { - if (!query) throw new Error(`findOrCreateVtuber requires an object as argument`); - const { url, name } = query - if (!url) throw new Error('findOrCreateVtuber was missing url which is required'); - console.info(`findOrCreateVtuber. url=${url}, name=${name}`) - new URL(url) // validate URL, throw if invalid - - const foundVtuber = await findVtuber(query) - if (!foundVtuber) { - console.info(`Failed to find vtuber, so we create one.`) - const vtuber = await fetch(`${configs.scoutUrl}/vtuber/data?url=${url}`).then((res): Promise => res.json() as any) - return createVtuber(vtuber) - } else { - return foundVtuber - } - -} \ No newline at end of file diff --git a/packages/fetchers/src/findVod.ts b/packages/fetchers/src/findVod.ts deleted file mode 100644 index c50a99f..0000000 --- a/packages/fetchers/src/findVod.ts +++ /dev/null @@ -1,25 +0,0 @@ -import type { VodResponse } from "@futureporn/types"; -import { configs } from "./config.ts"; - -export default async function findVod({ vod_id, discord_message_id }: { vod_id?: string, discord_message_id?: string }): Promise { - const fetchUrl = (!!vod_id) - ? `${configs.postgrestUrl}/vods?id=eq.${vod_id}&select=*,segments(bytes,updated_at,created_at,s3_key)` - : `${configs.postgrestUrl}/vods?discord_message_id=eq.${discord_message_id}&select=*,segments(bytes,updated_at,created_at)` - console.info(fetchUrl) - const fetchOptions = { - method: 'GET', - headers: { - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=representation' - } - } - const res = await fetch(fetchUrl, fetchOptions) - if (!res.ok) { - const msg = `request failed. status=${res.status}, statusText=${res.statusText}` - console.error(msg) - throw new Error(msg) - } - const json = await res.json() as VodResponse[] - return json?.at(0) || null -} \ No newline at end of file diff --git a/packages/fetchers/src/getPlaylistUrl.ts b/packages/fetchers/src/getPlaylistUrl.ts deleted file mode 100644 index 77be74d..0000000 --- a/packages/fetchers/src/getPlaylistUrl.ts +++ /dev/null @@ -1,23 +0,0 @@ -import { configs } from './config.ts' -import { type GenericApiResponse } from '@futureporn/types' - -export default async function getPlaylistUrl (url: string): Promise { - if (!url) throw new Error(`getPlaylistUrl requires a url, but it was undefined.`); - - const res = await fetch(`${configs.scoutUrl}/ytdlp/playlist-url?url=${url}`) - if (!res.ok) { - const body = await res.text() - console.error(`failed to getPlaylistUrl res.status=${res.status}, res.statusText=${res.statusText}, body=${body}`) - return { - error: 'PlaylistFailedError', - detail: `failed to getPlaylistUrl. res.status=${res.status}, res.statusText=${res.statusText}, body=${body}`, - data: null, - message: 'something went wrong wile fetching data from @futureporn/scout' - } - } else { - const payload = await res.json() as any - console.log(`>>>>>> getPlaylistUrl data=${payload.data}, error=${payload.error} got a data payload as follows.`) - console.log(payload) - return payload - } -} \ No newline at end of file diff --git a/packages/fetchers/src/getRecording.ts b/packages/fetchers/src/getRecording.ts deleted file mode 100644 index 045e391..0000000 --- a/packages/fetchers/src/getRecording.ts +++ /dev/null @@ -1,45 +0,0 @@ -import type { RecordingResponse } from "@futureporn/types"; -import { configs } from "./config.ts"; - -export default async function getRecording(recordingId: string): Promise { - if (!recordingId) throw new Error(`getRecording requires recordingId param, but it was missing.`); - const fetchUrl = `${configs.postgrestUrl}/recordings?id=eq.${recordingId}` - const fetchOptions = { - method: 'GET', - headers: { - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=representation' - } - } - const res = await fetch(fetchUrl, fetchOptions) - if (!res.ok) { - const msg = `request failed. status=${res.status}, statusText=${res.statusText}` - console.error(msg) - throw new Error(msg) - } - const json = await res.json() as RecordingResponse[] - return json?.at(0) || null -} - - -export async function getRecordingRelatedToVod(vodId: string): Promise { - if (!vodId) throw new Error(`getRecordingRelatedToVod requires vodId param, but it was missing.`); - const fetchUrl = `${configs.postgrestUrl}/recordings?select=*,vods!inner(id)&vods.id=eq.${vodId}` - const fetchOptions = { - method: 'GET', - headers: { - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=representation' - } - } - const res = await fetch(fetchUrl, fetchOptions) - if (!res.ok) { - const msg = `request failed. status=${res.status}, statusText=${res.statusText}` - console.error(msg) - throw new Error(msg) - } - const json = await res.json() as RecordingResponse[] - return json?.at(0) || null -} \ No newline at end of file diff --git a/packages/fetchers/src/getSegmentsFromDatabase.ts b/packages/fetchers/src/getSegmentsFromDatabase.ts deleted file mode 100644 index 40527fb..0000000 --- a/packages/fetchers/src/getSegmentsFromDatabase.ts +++ /dev/null @@ -1,37 +0,0 @@ -import { configs } from './config.ts' -import querystring from 'node:querystring' - -export default async function getSegmentsFromDatabase(s3_key: string): Promise { - if (!s3_key) throw new Error('getSegments requires {string} s3_key as first arg'); - const segmentPayload = { - s3_key - } - console.info(`Creating segment with s3_key=${s3_key}. payload as follows`) - console.info(JSON.stringify(segmentPayload)) - const res = await fetch(`${configs.postgrestUrl}/segments`, { - method: 'POST', - headers: { - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=headers-only', - 'Authorization': `Bearer ${configs.automationUserJwt}` - }, - body: JSON.stringify(segmentPayload) - }) - if (!res.ok) { - const body = await res.text() - const msg = `failed to create Segment. status=${res.status}, statusText=${res.statusText}, body=${body}` - console.error(msg) - throw new Error(msg); - } - const location = res.headers.get('location') - if (!location) throw new Error(`failed to get location header in response from postgrest`); - const parsedQuery = querystring.parse(location) - const segmentsId = parsedQuery['/segments?id'] - if (!segmentsId) throw new Error('segmentsId was undefined which is unexpected'); - if (Array.isArray(segmentsId)) throw new Error('segmentsId was an array which is unexpected'); - const id = segmentsId.split('.').at(-1) - if (!id) throw new Error('failed to get id '); - return parseInt(id) -} - diff --git a/packages/fetchers/src/getStreamFromDatabase.ts b/packages/fetchers/src/getStreamFromDatabase.ts deleted file mode 100644 index 0ca39fe..0000000 --- a/packages/fetchers/src/getStreamFromDatabase.ts +++ /dev/null @@ -1,35 +0,0 @@ -import { configs } from "./config.ts" -import type { StreamResponse } from '@futureporn/types' - - -export default async function getStreamFromDatabase(messageId: string): Promise { - - const pgRequestUrl = `${configs.postgrestUrl}/streams?discord_message_id=eq.${messageId}` - console.info(`pgRequestUrl=${pgRequestUrl}`) - const requestOptions = { - method: 'GET', - headers: { - 'Authorization': `Bearer ${configs.automationUserJwt}`, - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=representation' - } - } - try { - const res = await fetch (pgRequestUrl, requestOptions) - if (!res.ok) { - const body = await res.json() - console.error(body) - throw new Error(`Problem during getStreamFromDatabase. res.status=${res.status}, res.statusText=${res.statusText}`) - } - console.info(`we got an OK response. let's look at the json.`) - const json = await res.json() as StreamResponse[] - console.info(json) - const stream = json[0] - if (!stream) return null - else return stream - } catch (e) { - console.error(e) - throw e - } -} \ No newline at end of file diff --git a/packages/fetchers/src/getStreamIdFromMessage.ts b/packages/fetchers/src/getStreamIdFromMessage.ts deleted file mode 100644 index 678fb16..0000000 --- a/packages/fetchers/src/getStreamIdFromMessage.ts +++ /dev/null @@ -1,9 +0,0 @@ -import { type Interaction } from "discordeno" -import { isCuid } from '@paralleldrive/cuid2'; - -export default function getStreamIdFromMessage(interaction: Interaction): string|null { - const embeds = interaction.message?.embeds - const streamId = embeds?.at(0)?.title?.split(' ').at(1) - if (!streamId || !isCuid(streamId) || streamId === 'undefined') return null - else return streamId -} diff --git a/packages/fetchers/src/getUrlFromMessage.ts b/packages/fetchers/src/getUrlFromMessage.ts deleted file mode 100644 index 2b487b1..0000000 --- a/packages/fetchers/src/getUrlFromMessage.ts +++ /dev/null @@ -1,36 +0,0 @@ -import { type Interaction } from "discordeno" -import { configs } from "./config.ts" -import { type Stream } from "@futureporn/types" -import { logger } from "@discordeno/bot" - - -export default async function getUrlFromMessage(interaction: Interaction): Promise { - const messageId = interaction.message?.id - - const pgRequestUrl = `${configs.postgrestUrl}/streams?discord_message_id=eq.${messageId}` - logger.info(`pgRequestUrl=${pgRequestUrl}`) - const requestOptions = { - method: 'GET', - headers: { - 'Authorization': `Bearer ${configs.automationUserJwt}`, - 'Content-Type': 'application/json', - 'Prefer': 'return=representation' - } - } - try { - const res = await fetch (pgRequestUrl, requestOptions) - if (!res.ok) { - const body = await res.json() - logger.error(body) - throw new Error(`Problem during getUrlFromMessage. res.status=${res.status}, res.statusText=${res.statusText}`) - } - const json = await res.json() as Stream[] - const stream = json[0] - const url = stream?.url - if (!url) return null - else return url - } catch (e) { - logger.error(e) - throw e - } -} \ No newline at end of file diff --git a/packages/fetchers/src/getVod.ts b/packages/fetchers/src/getVod.ts deleted file mode 100644 index a1b06a3..0000000 --- a/packages/fetchers/src/getVod.ts +++ /dev/null @@ -1,23 +0,0 @@ -import { configs } from "./config" -import type { VodResponse } from "@futureporn/types" - -export default async function getVod(vodId: string): Promise { - const url = `${configs.postgrestUrl}/vods?select=*,segments(*),recording:recordings(is_aborted)&id=eq.${vodId}` - try { - const res = await fetch(url) - if (!res.ok) { - throw new Error(`failed fetching getVod ${vodId}. status=${res.status}, statusText=${res.statusText}`) - } - const body = await res.json() as VodResponse[] - if (!body[0]) throw new Error('body[0] was expected to be Vod data, but it was either null or undefined.'); - return body[0]; - } catch (e) { - console.error(`encountered an error during getVod()`) - if (e instanceof Error) { - console.error(e.message) - } else { - console.error(JSON.stringify(e)) - } - return null - } -} \ No newline at end of file diff --git a/packages/fetchers/src/patchRecording.ts b/packages/fetchers/src/patchRecording.ts deleted file mode 100644 index 8bc2f08..0000000 --- a/packages/fetchers/src/patchRecording.ts +++ /dev/null @@ -1,28 +0,0 @@ -import type { RecordingRecord } from "@futureporn/types"; -import { configs } from "./config.ts"; - -export default async function patchRecording(recordingId: string, payload: Partial): Promise { - const url = `${configs.postgrestUrl}/recordings?id=eq.${recordingId}` - const fetchOptions = { - method: 'PATCH', - headers: { - 'Authorization': `Bearer ${configs.automationUserJwt}`, - 'Content-Type': 'application/json', - 'Prefer': 'return=headers-only' - }, - body: JSON.stringify(payload) - } - try { - const res = await fetch(url, fetchOptions) - if (!res.ok) { - const body = await res.json() - console.error(body) - throw new Error(`Problem during patchRecording. res.status=${res.status}, res.statusText=${res.statusText}`) - } - return - } catch (e) { - console.error(e) - throw e - } -} - diff --git a/packages/fetchers/src/patchVod.ts b/packages/fetchers/src/patchVod.ts deleted file mode 100644 index a6a7d76..0000000 --- a/packages/fetchers/src/patchVod.ts +++ /dev/null @@ -1,28 +0,0 @@ -import type { StreamRecord } from "@futureporn/types"; -import { configs } from "./config.ts"; - -export default async function patchVod(stream_id: string, payload: Partial): Promise { - const url = `${configs.postgrestUrl}/vods?id=eq.${stream_id}` - const fetchOptions = { - method: 'PATCH', - headers: { - 'Authorization': `Bearer ${configs.automationUserJwt}`, - 'Content-Type': 'application/json', - 'Prefer': 'return=headers-only' - }, - body: JSON.stringify(payload) - } - try { - const res = await fetch(url, fetchOptions) - if (!res.ok) { - const body = await res.json() - console.error(body) - throw new Error(`Problem during patchVod. res.status=${res.status}, res.statusText=${res.statusText}`) - } - return - } catch (e) { - console.error(e) - throw e - } -} - diff --git a/packages/fetchers/src/patchVodInDatabase.ts b/packages/fetchers/src/patchVodInDatabase.ts deleted file mode 100644 index 2f47b55..0000000 --- a/packages/fetchers/src/patchVodInDatabase.ts +++ /dev/null @@ -1,23 +0,0 @@ -import type { VodRecord } from "@futureporn/types"; -import { configs } from "./config.ts"; - - -export default async function patchVodInDatabase(vod_id: string, payload: Partial): Promise { - const url = `${configs.postgrestUrl}/vods?id=eq.${vod_id}` - const fetchOptions = { - method: 'PATCH', - headers: { - 'Authorization': `Bearer ${configs.automationUserJwt}`, - 'Content-Type': 'application/json', - 'Prefer': 'return=headers-only' - }, - body: JSON.stringify(payload) - } - const res = await fetch(url, fetchOptions) - if (!res.ok) { - const body = await res.json() - throw new Error(`Problem during patchVodInDatabase. res.status=${res.status}, res.statusText=${res.statusText}`) - } - return -} - diff --git a/packages/fetchers/src/updateSegmentInDatabase.ts b/packages/fetchers/src/updateSegmentInDatabase.ts deleted file mode 100644 index 5473ae2..0000000 --- a/packages/fetchers/src/updateSegmentInDatabase.ts +++ /dev/null @@ -1,52 +0,0 @@ -import type { SegmentResponse } from '@futureporn/types' -import { configs } from './config.ts' - - -/** - * updateSegmentInDatabase - * - * updates the segment in the database with the new filesize - * - * resolves with the updated segment and the is_recording_aborted column of the related vod - */ -export default async function updateSegmentInDatabase({ - segment_id, - bytes, - bytes_uploaded, -}: { - segment_id: string, - bytes: number, - bytes_uploaded: number, -}): Promise { - - const payload: any = { - bytes, - bytes_uploaded - } - - const fetchUrl =`${configs.postgrestUrl}/segments?id=eq.${segment_id}&select=vod:vods(recording:recordings(is_aborted))` - // console.info(`updateSegmentInDatabase > fetchUrl=${fetchUrl}`) - const res = await fetch(fetchUrl, { - method: 'PATCH', - headers: { - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=representation', - 'Authorization': `Bearer ${configs.automationUserJwt}` - }, - body: JSON.stringify(payload) - }) - if (!res.ok) { - const body = await res.text() - const msg = `failed to updateSegmentInDatabase. status=${res.status}, statusText=${res.statusText}, body=${body}` - console.error(msg) - throw new Error(msg); - } - // console.info(`response was OK~`) - const body = await res.json() as SegmentResponse[]; - if (!body[0]) throw new Error(`failed to get a segment that matched segment_id=${segment_id}`); - const bod = body[0] - // console.info('the following was the response from PATCH-ing /segments') - // console.info(JSON.stringify(bod)) - return bod -} \ No newline at end of file diff --git a/packages/infra/README.md b/packages/infra/README.md deleted file mode 100644 index bd0b75a..0000000 --- a/packages/infra/README.md +++ /dev/null @@ -1,3 +0,0 @@ -# infra - -This module contains scripts that help with miscellaneous infrastructure tasks like cleaning up unused resources on Vultr Kubernetes Engine. diff --git a/packages/infra/package.json b/packages/infra/package.json deleted file mode 100644 index ca3589b..0000000 --- a/packages/infra/package.json +++ /dev/null @@ -1,16 +0,0 @@ -{ - "name": "infra", - "type": "module", - "version": "1.0.0", - "description": "", - "keywords": [], - "author": "", - "license": "Unlicense", - "scripts": { - "clean": "rm -rf dist", - "superclean": "rm -rf node_modules && rm -rf pnpm-lock.yaml && rm -rf dist" - }, - "dependencies": { - "dotenv": "^16.4.5" - } -} diff --git a/packages/infra/pnpm-lock.yaml b/packages/infra/pnpm-lock.yaml deleted file mode 100644 index 28a12ba..0000000 --- a/packages/infra/pnpm-lock.yaml +++ /dev/null @@ -1,23 +0,0 @@ -lockfileVersion: '9.0' - -settings: - autoInstallPeers: true - excludeLinksFromLockfile: false - -importers: - - .: - dependencies: - dotenv: - specifier: ^16.4.5 - version: 16.4.5 - -packages: - - dotenv@16.4.5: - resolution: {integrity: sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==} - engines: {node: '>=12'} - -snapshots: - - dotenv@16.4.5: {} diff --git a/packages/infra/vultr-delete-orphaned-resources.js b/packages/infra/vultr-delete-orphaned-resources.js deleted file mode 100755 index 4c8fb1b..0000000 --- a/packages/infra/vultr-delete-orphaned-resources.js +++ /dev/null @@ -1,84 +0,0 @@ -#!/usr/bin/env node - -import dotenv from 'dotenv' -import { join } from 'node:path' -dotenv.config({ path: join(import.meta.dirname, '../../.env') }) - -const apiV2Base = 'https://api.vultr.com/v2' - - -if (!process.env.VULTR_API_KEY) throw new Error('VULTR_API_KEY is missing in env'); - - - -async function deleteOrphanedLoadBalancers() { - console.log('getting load balancers') - const loadBalancersRes = await fetch(`${apiV2Base}/load-balancers`, { - method: 'GET', - headers: { - 'authorization': `Bearer ${process.env.VULTR_API_KEY}` - } - }) - const loadBalancerJson = await loadBalancersRes.json() - const orphanedLoadBalancers = loadBalancerJson.load_balancers.filter((lb) => (lb.instances.length === 0)) - console.log(`found ${orphanedLoadBalancers.length} orphaned load balancers.`) - // console.log('waiting 1 second') - await new Promise((resolve) => { setTimeout(resolve, 1000) }) - - for (const lb of orphanedLoadBalancers) { - console.log(`deleting load balancer ${lb.id}`) - - const deleteLoadBalancerRes = await fetch(`https://api.vultr.com/v2/load-balancers/${lb.id}`, { - method: 'DELETE', - headers: { - 'Authorization': `Bearer ${process.env.VULTR_API_KEY}` - } - }) - - // console.log('waiting 1 second') - await new Promise((resolve) => { setTimeout(resolve, 1000) }) - } -} - - -async function deleteOrphanedBlockStorage() { - console.log('getting block storage') - - const blocksRes = await fetch(`${apiV2Base}/blocks`, { - method: 'GET', - headers: { - 'authorization': `Bearer ${process.env.VULTR_API_KEY}` - } - }) - const blocksJson = await blocksRes.json() - - const orphanedBlocks = blocksJson.blocks.filter((b) => b.attached_to_instance === '') - console.log(`found ${orphanedBlocks.length} orphaned block storages`) - - console.log('waiting 1 second') - await new Promise((resolve) => { setTimeout(resolve, 1000) }) - for (const block of orphanedBlocks) { - console.log(`deleting block ${block.id}`) - - const deleteBlocksRes = await fetch(`${apiV2Base}/blocks/${block.id}`, { - method: 'DELETE', - headers: { - 'Authorization': `Bearer ${process.env.VULTR_API_KEY}` - } - }) - - console.log('waiting 1 second') - await new Promise((resolve) => { setTimeout(resolve, 1000) }) - - } - - -} - - -async function main() { - await deleteOrphanedLoadBalancers() - await deleteOrphanedBlockStorage() -} - -main() diff --git a/packages/storage/.mocharc.json b/packages/storage/.mocharc.json deleted file mode 100644 index 9c05d8d..0000000 --- a/packages/storage/.mocharc.json +++ /dev/null @@ -1,3 +0,0 @@ -{ - "loader": "ts-node/esm" -} \ No newline at end of file diff --git a/packages/storage/package.json b/packages/storage/package.json deleted file mode 100644 index 4f6ae1f..0000000 --- a/packages/storage/package.json +++ /dev/null @@ -1,26 +0,0 @@ -{ - "name": "@futureporn/storage", - "type": "module", - "version": "1.0.0", - "description": "", - "main": "index.js", - "exports": { - "./*.ts": "./src/*.ts" - }, - "scripts": { - "test": "echo \"Warn: no test specified\" && exit 0", - "clean": "rm -rf dist", - "superclean": "rm -rf node_modules && rm -rf pnpm-lock.yaml && rm -rf dist" - }, - "keywords": [], - "author": "@CJ_Clippy", - "license": "Unlicense", - "dependencies": { - "@aws-sdk/client-s3": "3.637.0", - "@aws-sdk/lib-storage": "^3.637.0", - "@futureporn/types": "workspace:^", - "@paralleldrive/cuid2": "^2.2.2", - "@types/node": "^22.5.2", - "dotenv": "^16.4.5" - } -} diff --git a/packages/storage/pnpm-lock.yaml b/packages/storage/pnpm-lock.yaml deleted file mode 100644 index 1e264c2..0000000 --- a/packages/storage/pnpm-lock.yaml +++ /dev/null @@ -1,1362 +0,0 @@ -lockfileVersion: '9.0' - -settings: - autoInstallPeers: true - excludeLinksFromLockfile: false - -importers: - - .: - dependencies: - '@aws-sdk/client-s3': - specifier: 3.637.0 - version: 3.637.0 - '@aws-sdk/lib-storage': - specifier: ^3.637.0 - version: 3.637.0(@aws-sdk/client-s3@3.637.0) - '@futureporn/types': - specifier: workspace:^ - version: link:../types - '@paralleldrive/cuid2': - specifier: ^2.2.2 - version: 2.2.2 - '@types/node': - specifier: ^22.5.2 - version: 22.5.2 - dotenv: - specifier: ^16.4.5 - version: 16.4.5 - -packages: - - '@aws-crypto/crc32@5.2.0': - resolution: {integrity: sha512-nLbCWqQNgUiwwtFsen1AdzAtvuLRsQS8rYgMuxCrdKf9kOssamGLuPwyTY9wyYblNr9+1XM8v6zoDTPPSIeANg==} - engines: {node: '>=16.0.0'} - - '@aws-crypto/crc32c@5.2.0': - resolution: {integrity: sha512-+iWb8qaHLYKrNvGRbiYRHSdKRWhto5XlZUEBwDjYNf+ly5SVYG6zEoYIdxvf5R3zyeP16w4PLBn3rH1xc74Rag==} - - '@aws-crypto/sha1-browser@5.2.0': - resolution: {integrity: sha512-OH6lveCFfcDjX4dbAvCFSYUjJZjDr/3XJ3xHtjn3Oj5b9RjojQo8npoLeA/bNwkOkrSQ0wgrHzXk4tDRxGKJeg==} - - '@aws-crypto/sha256-browser@5.2.0': - resolution: {integrity: sha512-AXfN/lGotSQwu6HNcEsIASo7kWXZ5HYWvfOmSNKDsEqC4OashTp8alTmaz+F7TC2L083SFv5RdB+qU3Vs1kZqw==} - - '@aws-crypto/sha256-js@5.2.0': - resolution: {integrity: sha512-FFQQyu7edu4ufvIZ+OadFpHHOt+eSTBaYaki44c+akjg7qZg9oOQeLlk77F6tSYqjDAFClrHJk9tMf0HdVyOvA==} - engines: {node: '>=16.0.0'} - - '@aws-crypto/supports-web-crypto@5.2.0': - resolution: {integrity: sha512-iAvUotm021kM33eCdNfwIN//F77/IADDSs58i+MDaOqFrVjZo9bAal0NK7HurRuWLLpF1iLX7gbWrjHjeo+YFg==} - - '@aws-crypto/util@5.2.0': - resolution: {integrity: sha512-4RkU9EsI6ZpBve5fseQlGNUWKMa1RLPQ1dnjnQoe07ldfIzcsGb5hC5W0Dm7u423KWzawlrpbjXBrXCEv9zazQ==} - - '@aws-sdk/client-s3@3.637.0': - resolution: {integrity: sha512-y6UC94fsMvhKbf0dzfnjVP1HePeGjplfcYfilZU1COIJLyTkMcUv4XcT4I407CGIrvgEafONHkiC09ygqUauNA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/client-sso-oidc@3.637.0': - resolution: {integrity: sha512-27bHALN6Qb6m6KZmPvRieJ/QRlj1lyac/GT2Rn5kJpre8Mpp+yxrtvp3h9PjNBty4lCeFEENfY4dGNSozBuBcw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.637.0 - - '@aws-sdk/client-sso@3.637.0': - resolution: {integrity: sha512-+KjLvgX5yJYROWo3TQuwBJlHCY0zz9PsLuEolmXQn0BVK1L/m9GteZHtd+rEdAoDGBpE0Xqjy1oz5+SmtsaRUw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/client-sts@3.637.0': - resolution: {integrity: sha512-xUi7x4qDubtA8QREtlblPuAcn91GS/09YVEY/RwU7xCY0aqGuFwgszAANlha4OUIqva8oVj2WO4gJuG+iaSnhw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/core@3.635.0': - resolution: {integrity: sha512-i1x/E/sgA+liUE1XJ7rj1dhyXpAKO1UKFUcTTHXok2ARjWTvszHnSXMOsB77aPbmn0fUp1JTx2kHUAZ1LVt5Bg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-env@3.620.1': - resolution: {integrity: sha512-ExuILJ2qLW5ZO+rgkNRj0xiAipKT16Rk77buvPP8csR7kkCflT/gXTyzRe/uzIiETTxM7tr8xuO9MP/DQXqkfg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-http@3.635.0': - resolution: {integrity: sha512-iJyRgEjOCQlBMXqtwPLIKYc7Bsc6nqjrZybdMDenPDa+kmLg7xh8LxHsu9088e+2/wtLicE34FsJJIfzu3L82g==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-ini@3.637.0': - resolution: {integrity: sha512-h+PFCWfZ0Q3Dx84SppET/TFpcQHmxFW8/oV9ArEvMilw4EBN+IlxgbL0CnHwjHW64szcmrM0mbebjEfHf4FXmw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.637.0 - - '@aws-sdk/credential-provider-node@3.637.0': - resolution: {integrity: sha512-yoEhoxJJfs7sPVQ6Is939BDQJZpZCoUgKr/ySse4YKOZ24t4VqgHA6+wV7rYh+7IW24Rd91UTvEzSuHYTlxlNA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-process@3.620.1': - resolution: {integrity: sha512-hWqFMidqLAkaV9G460+1at6qa9vySbjQKKc04p59OT7lZ5cO5VH5S4aI05e+m4j364MBROjjk2ugNvfNf/8ILg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-sso@3.637.0': - resolution: {integrity: sha512-Mvz+h+e62/tl+dVikLafhv+qkZJ9RUb8l2YN/LeKMWkxQylPT83CPk9aimVhCV89zth1zpREArl97+3xsfgQvA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-web-identity@3.621.0': - resolution: {integrity: sha512-w7ASSyfNvcx7+bYGep3VBgC3K6vEdLmlpjT7nSIHxxQf+WSdvy+HynwJosrpZax0sK5q0D1Jpn/5q+r5lwwW6w==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.621.0 - - '@aws-sdk/lib-storage@3.637.0': - resolution: {integrity: sha512-HiNGOP4a1QrCWwO1joKw4mCp19nLXoF9K52PislBaYDI35IlHC3DP6MeOg5zmElwtL1GtEHFBy5olfPWPsLyLg==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-s3': ^3.637.0 - - '@aws-sdk/middleware-bucket-endpoint@3.620.0': - resolution: {integrity: sha512-eGLL0W6L3HDb3OACyetZYOWpHJ+gLo0TehQKeQyy2G8vTYXqNTeqYhuI6up9HVjBzU9eQiULVQETmgQs7TFaRg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-expect-continue@3.620.0': - resolution: {integrity: sha512-QXeRFMLfyQ31nAHLbiTLtk0oHzG9QLMaof5jIfqcUwnOkO8YnQdeqzakrg1Alpy/VQ7aqzIi8qypkBe2KXZz0A==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-flexible-checksums@3.620.0': - resolution: {integrity: sha512-ftz+NW7qka2sVuwnnO1IzBku5ccP+s5qZGeRTPgrKB7OzRW85gthvIo1vQR2w+OwHFk7WJbbhhWwbCbktnP4UA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-host-header@3.620.0': - resolution: {integrity: sha512-VMtPEZwqYrII/oUkffYsNWY9PZ9xpNJpMgmyU0rlDQ25O1c0Hk3fJmZRe6pEkAJ0omD7kLrqGl1DUjQVxpd/Rg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-location-constraint@3.609.0': - resolution: {integrity: sha512-xzsdoTkszGVqGVPjUmgoP7TORiByLueMHieI1fhQL888WPdqctwAx3ES6d/bA9Q/i8jnc6hs+Fjhy8UvBTkE9A==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-logger@3.609.0': - resolution: {integrity: sha512-S62U2dy4jMDhDFDK5gZ4VxFdWzCtLzwbYyFZx2uvPYTECkepLUfzLic2BHg2Qvtu4QjX+oGE3P/7fwaGIsGNuQ==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-recursion-detection@3.620.0': - resolution: {integrity: sha512-nh91S7aGK3e/o1ck64sA/CyoFw+gAYj2BDOnoNa6ouyCrVJED96ZXWbhye/fz9SgmNUZR2g7GdVpiLpMKZoI5w==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-sdk-s3@3.635.0': - resolution: {integrity: sha512-RLdYJPEV4JL/7NBoFUs7VlP90X++5FlJdxHz0DzCjmiD3qCviKy+Cym3qg1gBgHwucs5XisuClxDrGokhAdTQw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-ssec@3.609.0': - resolution: {integrity: sha512-GZSD1s7+JswWOTamVap79QiDaIV7byJFssBW68GYjyRS5EBjNfwA/8s+6uE6g39R3ojyTbYOmvcANoZEhSULXg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-user-agent@3.637.0': - resolution: {integrity: sha512-EYo0NE9/da/OY8STDsK2LvM4kNa79DBsf4YVtaG4P5pZ615IeFsD8xOHZeuJmUrSMlVQ8ywPRX7WMucUybsKug==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/region-config-resolver@3.614.0': - resolution: {integrity: sha512-vDCeMXvic/LU0KFIUjpC3RiSTIkkvESsEfbVHiHH0YINfl8HnEqR5rj+L8+phsCeVg2+LmYwYxd5NRz4PHxt5g==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/signature-v4-multi-region@3.635.0': - resolution: {integrity: sha512-J6QY4/invOkpogCHjSaDON1hF03viPpOnsrzVuCvJMmclS/iG62R4EY0wq1alYll0YmSdmKlpJwHMWwGtqK63Q==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/token-providers@3.614.0': - resolution: {integrity: sha512-okItqyY6L9IHdxqs+Z116y5/nda7rHxLvROxtAJdLavWTYDydxrZstImNgGWTeVdmc0xX2gJCI77UYUTQWnhRw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sso-oidc': ^3.614.0 - - '@aws-sdk/types@3.609.0': - resolution: {integrity: sha512-+Tqnh9w0h2LcrUsdXyT1F8mNhXz+tVYBtP19LpeEGntmvHwa2XzvLUCWpoIAIVsHp5+HdB2X9Sn0KAtmbFXc2Q==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-arn-parser@3.568.0': - resolution: {integrity: sha512-XUKJWWo+KOB7fbnPP0+g/o5Ulku/X53t7i/h+sPHr5xxYTJJ9CYnbToo95mzxe7xWvkLrsNtJ8L+MnNn9INs2w==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-endpoints@3.637.0': - resolution: {integrity: sha512-pAqOKUHeVWHEXXDIp/qoMk/6jyxIb6GGjnK1/f8dKHtKIEs4tKsnnL563gceEvdad53OPXIt86uoevCcCzmBnw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-locate-window@3.568.0': - resolution: {integrity: sha512-3nh4TINkXYr+H41QaPelCceEB2FXP3fxp93YZXB/kqJvX0U9j0N0Uk45gvsjmEPzG8XxkPEeLIfT2I1M7A6Lig==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-user-agent-browser@3.609.0': - resolution: {integrity: sha512-fojPU+mNahzQ0YHYBsx0ZIhmMA96H+ZIZ665ObU9tl+SGdbLneVZVikGve+NmHTQwHzwkFsZYYnVKAkreJLAtA==} - - '@aws-sdk/util-user-agent-node@3.614.0': - resolution: {integrity: sha512-15ElZT88peoHnq5TEoEtZwoXTXRxNrk60TZNdpl/TUBJ5oNJ9Dqb5Z4ryb8ofN6nm9aFf59GVAerFDz8iUoHBA==} - engines: {node: '>=16.0.0'} - peerDependencies: - aws-crt: '>=1.0.0' - peerDependenciesMeta: - aws-crt: - optional: true - - '@aws-sdk/xml-builder@3.609.0': - resolution: {integrity: sha512-l9XxNcA4HX98rwCC2/KoiWcmEiRfZe4G+mYwDbCFT87JIMj6GBhLDkAzr/W8KAaA2IDr8Vc6J8fZPgVulxxfMA==} - engines: {node: '>=16.0.0'} - - '@noble/hashes@1.5.0': - resolution: {integrity: sha512-1j6kQFb7QRru7eKN3ZDvRcP13rugwdxZqCjbiAVZfIJwgj2A65UmT4TgARXGlXgnRkORLTDTrO19ZErt7+QXgA==} - engines: {node: ^14.21.3 || >=16} - - '@paralleldrive/cuid2@2.2.2': - resolution: {integrity: sha512-ZOBkgDwEdoYVlSeRbYYXs0S9MejQofiVYoTbKzy/6GQa39/q5tQU2IX46+shYnUkpEl3wc+J6wRlar7r2EK2xA==} - - '@smithy/abort-controller@3.1.1': - resolution: {integrity: sha512-MBJBiidoe+0cTFhyxT8g+9g7CeVccLM0IOKKUMCNQ1CNMJ/eIfoo0RTfVrXOONEI1UCN1W+zkiHSbzUNE9dZtQ==} - engines: {node: '>=16.0.0'} - - '@smithy/chunked-blob-reader-native@3.0.0': - resolution: {integrity: sha512-VDkpCYW+peSuM4zJip5WDfqvg2Mo/e8yxOv3VF1m11y7B8KKMKVFtmZWDe36Fvk8rGuWrPZHHXZ7rR7uM5yWyg==} - - '@smithy/chunked-blob-reader@3.0.0': - resolution: {integrity: sha512-sbnURCwjF0gSToGlsBiAmd1lRCmSn72nu9axfJu5lIx6RUEgHu6GwTMbqCdhQSi0Pumcm5vFxsi9XWXb2mTaoA==} - - '@smithy/config-resolver@3.0.5': - resolution: {integrity: sha512-SkW5LxfkSI1bUC74OtfBbdz+grQXYiPYolyu8VfpLIjEoN/sHVBlLeGXMQ1vX4ejkgfv6sxVbQJ32yF2cl1veA==} - engines: {node: '>=16.0.0'} - - '@smithy/core@2.4.0': - resolution: {integrity: sha512-cHXq+FneIF/KJbt4q4pjN186+Jf4ZB0ZOqEaZMBhT79srEyGDDBV31NqBRBjazz8ppQ1bJbDJMY9ba5wKFV36w==} - engines: {node: '>=16.0.0'} - - '@smithy/credential-provider-imds@3.2.0': - resolution: {integrity: sha512-0SCIzgd8LYZ9EJxUjLXBmEKSZR/P/w6l7Rz/pab9culE/RWuqelAKGJvn5qUOl8BgX8Yj5HWM50A5hiB/RzsgA==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-codec@3.1.2': - resolution: {integrity: sha512-0mBcu49JWt4MXhrhRAlxASNy0IjDRFU+aWNDRal9OtUJvJNiwDuyKMUONSOjLjSCeGwZaE0wOErdqULer8r7yw==} - - '@smithy/eventstream-serde-browser@3.0.6': - resolution: {integrity: sha512-2hM54UWQUOrki4BtsUI1WzmD13/SeaqT/AB3EUJKbcver/WgKNaiJ5y5F5XXuVe6UekffVzuUDrBZVAA3AWRpQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-config-resolver@3.0.3': - resolution: {integrity: sha512-NVTYjOuYpGfrN/VbRQgn31x73KDLfCXCsFdad8DiIc3IcdxL+dYA9zEQPyOP7Fy2QL8CPy2WE4WCUD+ZsLNfaQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-node@3.0.5': - resolution: {integrity: sha512-+upXvnHNyZP095s11jF5dhGw/Ihzqwl5G+/KtMnoQOpdfC3B5HYCcDVG9EmgkhJMXJlM64PyN5gjJl0uXFQehQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-universal@3.0.5': - resolution: {integrity: sha512-5u/nXbyoh1s4QxrvNre9V6vfyoLWuiVvvd5TlZjGThIikc3G+uNiG9uOTCWweSRjv1asdDIWK7nOmN7le4RYHQ==} - engines: {node: '>=16.0.0'} - - '@smithy/fetch-http-handler@3.2.4': - resolution: {integrity: sha512-kBprh5Gs5h7ug4nBWZi1FZthdqSM+T7zMmsZxx0IBvWUn7dK3diz2SHn7Bs4dQGFDk8plDv375gzenDoNwrXjg==} - - '@smithy/hash-blob-browser@3.1.2': - resolution: {integrity: sha512-hAbfqN2UbISltakCC2TP0kx4LqXBttEv2MqSPE98gVuDFMf05lU+TpC41QtqGP3Ff5A3GwZMPfKnEy0VmEUpmg==} - - '@smithy/hash-node@3.0.3': - resolution: {integrity: sha512-2ctBXpPMG+B3BtWSGNnKELJ7SH9e4TNefJS0cd2eSkOOROeBnnVBnAy9LtJ8tY4vUEoe55N4CNPxzbWvR39iBw==} - engines: {node: '>=16.0.0'} - - '@smithy/hash-stream-node@3.1.2': - resolution: {integrity: sha512-PBgDMeEdDzi6JxKwbfBtwQG9eT9cVwsf0dZzLXoJF4sHKHs5HEo/3lJWpn6jibfJwT34I1EBXpBnZE8AxAft6g==} - engines: {node: '>=16.0.0'} - - '@smithy/invalid-dependency@3.0.3': - resolution: {integrity: sha512-ID1eL/zpDULmHJbflb864k72/SNOZCADRc9i7Exq3RUNJw6raWUSlFEQ+3PX3EYs++bTxZB2dE9mEHTQLv61tw==} - - '@smithy/is-array-buffer@2.2.0': - resolution: {integrity: sha512-GGP3O9QFD24uGeAXYUjwSTXARoqpZykHadOmA8G5vfJPK0/DC67qa//0qvqrJzL1xc8WQWX7/yc7fwudjPHPhA==} - engines: {node: '>=14.0.0'} - - '@smithy/is-array-buffer@3.0.0': - resolution: {integrity: sha512-+Fsu6Q6C4RSJiy81Y8eApjEB5gVtM+oFKTffg+jSuwtvomJJrhUJBu2zS8wjXSgH/g1MKEWrzyChTBe6clb5FQ==} - engines: {node: '>=16.0.0'} - - '@smithy/md5-js@3.0.3': - resolution: {integrity: sha512-O/SAkGVwpWmelpj/8yDtsaVe6sINHLB1q8YE/+ZQbDxIw3SRLbTZuRaI10K12sVoENdnHqzPp5i3/H+BcZ3m3Q==} - - '@smithy/middleware-content-length@3.0.5': - resolution: {integrity: sha512-ILEzC2eyxx6ncej3zZSwMpB5RJ0zuqH7eMptxC4KN3f+v9bqT8ohssKbhNR78k/2tWW+KS5Spw+tbPF4Ejyqvw==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-endpoint@3.1.0': - resolution: {integrity: sha512-5y5aiKCEwg9TDPB4yFE7H6tYvGFf1OJHNczeY10/EFF8Ir8jZbNntQJxMWNfeQjC1mxPsaQ6mR9cvQbf+0YeMw==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-retry@3.0.15': - resolution: {integrity: sha512-iTMedvNt1ApdvkaoE8aSDuwaoc+BhvHqttbA/FO4Ty+y/S5hW6Ci/CTScG7vam4RYJWZxdTElc3MEfHRVH6cgQ==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-serde@3.0.3': - resolution: {integrity: sha512-puUbyJQBcg9eSErFXjKNiGILJGtiqmuuNKEYNYfUD57fUl4i9+mfmThtQhvFXU0hCVG0iEJhvQUipUf+/SsFdA==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-stack@3.0.3': - resolution: {integrity: sha512-r4klY9nFudB0r9UdSMaGSyjyQK5adUyPnQN/ZM6M75phTxOdnc/AhpvGD1fQUvgmqjQEBGCwpnPbDm8pH5PapA==} - engines: {node: '>=16.0.0'} - - '@smithy/node-config-provider@3.1.4': - resolution: {integrity: sha512-YvnElQy8HR4vDcAjoy7Xkx9YT8xZP4cBXcbJSgm/kxmiQu08DwUwj8rkGnyoJTpfl/3xYHH+d8zE+eHqoDCSdQ==} - engines: {node: '>=16.0.0'} - - '@smithy/node-http-handler@3.1.4': - resolution: {integrity: sha512-+UmxgixgOr/yLsUxcEKGH0fMNVteJFGkmRltYFHnBMlogyFdpzn2CwqWmxOrfJELhV34v0WSlaqG1UtE1uXlJg==} - engines: {node: '>=16.0.0'} - - '@smithy/property-provider@3.1.3': - resolution: {integrity: sha512-zahyOVR9Q4PEoguJ/NrFP4O7SMAfYO1HLhB18M+q+Z4KFd4V2obiMnlVoUFzFLSPeVt1POyNWneHHrZaTMoc/g==} - engines: {node: '>=16.0.0'} - - '@smithy/protocol-http@4.1.0': - resolution: {integrity: sha512-dPVoHYQ2wcHooGXg3LQisa1hH0e4y0pAddPMeeUPipI1tEOqL6A4N0/G7abeq+K8wrwSgjk4C0wnD1XZpJm5aA==} - engines: {node: '>=16.0.0'} - - '@smithy/querystring-builder@3.0.3': - resolution: {integrity: sha512-vyWckeUeesFKzCDaRwWLUA1Xym9McaA6XpFfAK5qI9DKJ4M33ooQGqvM4J+LalH4u/Dq9nFiC8U6Qn1qi0+9zw==} - engines: {node: '>=16.0.0'} - - '@smithy/querystring-parser@3.0.3': - resolution: {integrity: sha512-zahM1lQv2YjmznnfQsWbYojFe55l0SLG/988brlLv1i8z3dubloLF+75ATRsqPBboUXsW6I9CPGE5rQgLfY0vQ==} - engines: {node: '>=16.0.0'} - - '@smithy/service-error-classification@3.0.3': - resolution: {integrity: sha512-Jn39sSl8cim/VlkLsUhRFq/dKDnRUFlfRkvhOJaUbLBXUsLRLNf9WaxDv/z9BjuQ3A6k/qE8af1lsqcwm7+DaQ==} - engines: {node: '>=16.0.0'} - - '@smithy/shared-ini-file-loader@3.1.4': - resolution: {integrity: sha512-qMxS4hBGB8FY2GQqshcRUy1K6k8aBWP5vwm8qKkCT3A9K2dawUwOIJfqh9Yste/Bl0J2lzosVyrXDj68kLcHXQ==} - engines: {node: '>=16.0.0'} - - '@smithy/signature-v4@4.1.0': - resolution: {integrity: sha512-aRryp2XNZeRcOtuJoxjydO6QTaVhxx/vjaR+gx7ZjaFgrgPRyZ3HCTbfwqYj6ZWEBHkCSUfcaymKPURaByukag==} - engines: {node: '>=16.0.0'} - - '@smithy/smithy-client@3.2.0': - resolution: {integrity: sha512-pDbtxs8WOhJLJSeaF/eAbPgXg4VVYFlRcL/zoNYA5WbG3wBL06CHtBSg53ppkttDpAJ/hdiede+xApip1CwSLw==} - engines: {node: '>=16.0.0'} - - '@smithy/types@3.3.0': - resolution: {integrity: sha512-IxvBBCTFDHbVoK7zIxqA1ZOdc4QfM5HM7rGleCuHi7L1wnKv5Pn69xXJQ9hgxH60ZVygH9/JG0jRgtUncE3QUA==} - engines: {node: '>=16.0.0'} - - '@smithy/url-parser@3.0.3': - resolution: {integrity: sha512-pw3VtZtX2rg+s6HMs6/+u9+hu6oY6U7IohGhVNnjbgKy86wcIsSZwgHrFR+t67Uyxvp4Xz3p3kGXXIpTNisq8A==} - - '@smithy/util-base64@3.0.0': - resolution: {integrity: sha512-Kxvoh5Qtt0CDsfajiZOCpJxgtPHXOKwmM+Zy4waD43UoEMA+qPxxa98aE/7ZhdnBFZFXMOiBR5xbcaMhLtznQQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-body-length-browser@3.0.0': - resolution: {integrity: sha512-cbjJs2A1mLYmqmyVl80uoLTJhAcfzMOyPgjwAYusWKMdLeNtzmMz9YxNl3/jRLoxSS3wkqkf0jwNdtXWtyEBaQ==} - - '@smithy/util-body-length-node@3.0.0': - resolution: {integrity: sha512-Tj7pZ4bUloNUP6PzwhN7K386tmSmEET9QtQg0TgdNOnxhZvCssHji+oZTUIuzxECRfG8rdm2PMw2WCFs6eIYkA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-buffer-from@2.2.0': - resolution: {integrity: sha512-IJdWBbTcMQ6DA0gdNhh/BwrLkDR+ADW5Kr1aZmd4k3DIF6ezMV4R2NIAmT08wQJ3yUK82thHWmC/TnK/wpMMIA==} - engines: {node: '>=14.0.0'} - - '@smithy/util-buffer-from@3.0.0': - resolution: {integrity: sha512-aEOHCgq5RWFbP+UDPvPot26EJHjOC+bRgse5A8V3FSShqd5E5UN4qc7zkwsvJPPAVsf73QwYcHN1/gt/rtLwQA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-config-provider@3.0.0': - resolution: {integrity: sha512-pbjk4s0fwq3Di/ANL+rCvJMKM5bzAQdE5S/6RL5NXgMExFAi6UgQMPOm5yPaIWPpr+EOXKXRonJ3FoxKf4mCJQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-defaults-mode-browser@3.0.15': - resolution: {integrity: sha512-FZ4Psa3vjp8kOXcd3HJOiDPBCWtiilLl57r0cnNtq/Ga9RSDrM5ERL6xt+tO43+2af6Pn5Yp92x2n5vPuduNfg==} - engines: {node: '>= 10.0.0'} - - '@smithy/util-defaults-mode-node@3.0.15': - resolution: {integrity: sha512-KSyAAx2q6d0t6f/S4XB2+3+6aQacm3aLMhs9aLMqn18uYGUepbdssfogW5JQZpc6lXNBnp0tEnR5e9CEKmEd7A==} - engines: {node: '>= 10.0.0'} - - '@smithy/util-endpoints@2.0.5': - resolution: {integrity: sha512-ReQP0BWihIE68OAblC/WQmDD40Gx+QY1Ez8mTdFMXpmjfxSyz2fVQu3A4zXRfQU9sZXtewk3GmhfOHswvX+eNg==} - engines: {node: '>=16.0.0'} - - '@smithy/util-hex-encoding@3.0.0': - resolution: {integrity: sha512-eFndh1WEK5YMUYvy3lPlVmYY/fZcQE1D8oSf41Id2vCeIkKJXPcYDCZD+4+xViI6b1XSd7tE+s5AmXzz5ilabQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-middleware@3.0.3': - resolution: {integrity: sha512-l+StyYYK/eO3DlVPbU+4Bi06Jjal+PFLSMmlWM1BEwyLxZ3aKkf1ROnoIakfaA7mC6uw3ny7JBkau4Yc+5zfWw==} - engines: {node: '>=16.0.0'} - - '@smithy/util-retry@3.0.3': - resolution: {integrity: sha512-AFw+hjpbtVApzpNDhbjNG5NA3kyoMs7vx0gsgmlJF4s+yz1Zlepde7J58zpIRIsdjc+emhpAITxA88qLkPF26w==} - engines: {node: '>=16.0.0'} - - '@smithy/util-stream@3.1.3': - resolution: {integrity: sha512-FIv/bRhIlAxC0U7xM1BCnF2aDRPq0UaelqBHkM2lsCp26mcBbgI0tCVTv+jGdsQLUmAMybua/bjDsSu8RQHbmw==} - engines: {node: '>=16.0.0'} - - '@smithy/util-uri-escape@3.0.0': - resolution: {integrity: sha512-LqR7qYLgZTD7nWLBecUi4aqolw8Mhza9ArpNEQ881MJJIU2sE5iHCK6TdyqqzcDLy0OPe10IY4T8ctVdtynubg==} - engines: {node: '>=16.0.0'} - - '@smithy/util-utf8@2.3.0': - resolution: {integrity: sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A==} - engines: {node: '>=14.0.0'} - - '@smithy/util-utf8@3.0.0': - resolution: {integrity: sha512-rUeT12bxFnplYDe815GXbq/oixEGHfRFFtcTF3YdDi/JaENIM6aSYYLJydG83UNzLXeRI5K8abYd/8Sp/QM0kA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-waiter@3.1.2': - resolution: {integrity: sha512-4pP0EV3iTsexDx+8PPGAKCQpd/6hsQBaQhqWzU4hqKPHN5epPsxKbvUTIiYIHTxaKt6/kEaqPBpu/ufvfbrRzw==} - engines: {node: '>=16.0.0'} - - '@types/node@22.5.2': - resolution: {integrity: sha512-acJsPTEqYqulZS/Yp/S3GgeE6GZ0qYODUR8aVr/DkhHQ8l9nd4j5x1/ZJy9/gHrRlFMqkO6i0I3E27Alu4jjPg==} - - base64-js@1.5.1: - resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==} - - bowser@2.11.0: - resolution: {integrity: sha512-AlcaJBi/pqqJBIQ8U9Mcpc9i8Aqxn88Skv5d+xBX006BY5u8N3mGLHa5Lgppa7L/HfwgwLgZ6NYs+Ag6uUmJRA==} - - buffer@5.6.0: - resolution: {integrity: sha512-/gDYp/UtU0eA1ys8bOs9J6a+E/KWIY+DZ+Q2WESNUA0jFRsJOc0SNUO6xJ5SGA1xueg3NL65W6s+NY5l9cunuw==} - - dotenv@16.4.5: - resolution: {integrity: sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==} - engines: {node: '>=12'} - - events@3.3.0: - resolution: {integrity: sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==} - engines: {node: '>=0.8.x'} - - fast-xml-parser@4.4.1: - resolution: {integrity: sha512-xkjOecfnKGkSsOwtZ5Pz7Us/T6mrbPQrq0nh+aCO5V9nk5NLWmasAHumTKjiPJPWANe+kAZ84Jc8ooJkzZ88Sw==} - hasBin: true - - ieee754@1.2.1: - resolution: {integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==} - - inherits@2.0.4: - resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} - - readable-stream@3.6.2: - resolution: {integrity: sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==} - engines: {node: '>= 6'} - - safe-buffer@5.2.1: - resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} - - stream-browserify@3.0.0: - resolution: {integrity: sha512-H73RAHsVBapbim0tU2JwwOiXUj+fikfiaoYAKHF3VJfA0pe2BCzkhAHBlLG6REzE+2WNZcxOXjK7lkso+9euLA==} - - string_decoder@1.3.0: - resolution: {integrity: sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==} - - strnum@1.0.5: - resolution: {integrity: sha512-J8bbNyKKXl5qYcR36TIO8W3mVGVHrmmxsd5PAItGkmyzwJvybiw2IVq5nqd0i4LSNSkB/sx9VHllbfFdr9k1JA==} - - tslib@2.7.0: - resolution: {integrity: sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA==} - - undici-types@6.19.8: - resolution: {integrity: sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==} - - util-deprecate@1.0.2: - resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==} - - uuid@9.0.1: - resolution: {integrity: sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==} - hasBin: true - -snapshots: - - '@aws-crypto/crc32@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/crc32c@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/sha1-browser@5.2.0': - dependencies: - '@aws-crypto/supports-web-crypto': 5.2.0 - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-locate-window': 3.568.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-crypto/sha256-browser@5.2.0': - dependencies: - '@aws-crypto/sha256-js': 5.2.0 - '@aws-crypto/supports-web-crypto': 5.2.0 - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-locate-window': 3.568.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-crypto/sha256-js@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/supports-web-crypto@5.2.0': - dependencies: - tslib: 2.7.0 - - '@aws-crypto/util@5.2.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-sdk/client-s3@3.637.0': - dependencies: - '@aws-crypto/sha1-browser': 5.2.0 - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-bucket-endpoint': 3.620.0 - '@aws-sdk/middleware-expect-continue': 3.620.0 - '@aws-sdk/middleware-flexible-checksums': 3.620.0 - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-location-constraint': 3.609.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-sdk-s3': 3.635.0 - '@aws-sdk/middleware-ssec': 3.609.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/signature-v4-multi-region': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@aws-sdk/xml-builder': 3.609.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/eventstream-serde-browser': 3.0.6 - '@smithy/eventstream-serde-config-resolver': 3.0.3 - '@smithy/eventstream-serde-node': 3.0.5 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-blob-browser': 3.1.2 - '@smithy/hash-node': 3.0.3 - '@smithy/hash-stream-node': 3.1.2 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/md5-js': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-stream': 3.1.3 - '@smithy/util-utf8': 3.0.0 - '@smithy/util-waiter': 3.1.2 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sso@3.637.0': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sts@3.637.0': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/core@3.635.0': - dependencies: - '@smithy/core': 2.4.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-middleware': 3.0.3 - fast-xml-parser: 4.4.1 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-env@3.620.1': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-http@3.635.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-stream': 3.1.3 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-ini@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/credential-provider-env': 3.620.1 - '@aws-sdk/credential-provider-http': 3.635.0 - '@aws-sdk/credential-provider-process': 3.620.1 - '@aws-sdk/credential-provider-sso': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/credential-provider-web-identity': 3.621.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - aws-crt - - '@aws-sdk/credential-provider-node@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/credential-provider-env': 3.620.1 - '@aws-sdk/credential-provider-http': 3.635.0 - '@aws-sdk/credential-provider-ini': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/credential-provider-process': 3.620.1 - '@aws-sdk/credential-provider-sso': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/credential-provider-web-identity': 3.621.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - '@aws-sdk/client-sts' - - aws-crt - - '@aws-sdk/credential-provider-process@3.620.1': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-sso@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))': - dependencies: - '@aws-sdk/client-sso': 3.637.0 - '@aws-sdk/token-providers': 3.614.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - aws-crt - - '@aws-sdk/credential-provider-web-identity@3.621.0(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/lib-storage@3.637.0(@aws-sdk/client-s3@3.637.0)': - dependencies: - '@aws-sdk/client-s3': 3.637.0 - '@smithy/abort-controller': 3.1.1 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/smithy-client': 3.2.0 - buffer: 5.6.0 - events: 3.3.0 - stream-browserify: 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-bucket-endpoint@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-arn-parser': 3.568.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-expect-continue@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-flexible-checksums@3.620.0': - dependencies: - '@aws-crypto/crc32': 5.2.0 - '@aws-crypto/crc32c': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@smithy/is-array-buffer': 3.0.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-host-header@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-location-constraint@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-logger@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-recursion-detection@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-sdk-s3@3.635.0': - dependencies: - '@aws-sdk/core': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-arn-parser': 3.568.0 - '@smithy/core': 2.4.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-stream': 3.1.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-ssec@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-user-agent@3.637.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/region-config-resolver@3.614.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@aws-sdk/signature-v4-multi-region@3.635.0': - dependencies: - '@aws-sdk/middleware-sdk-s3': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/token-providers@3.614.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))': - dependencies: - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/types@3.609.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/util-arn-parser@3.568.0': - dependencies: - tslib: 2.7.0 - - '@aws-sdk/util-endpoints@3.637.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - '@smithy/util-endpoints': 2.0.5 - tslib: 2.7.0 - - '@aws-sdk/util-locate-window@3.568.0': - dependencies: - tslib: 2.7.0 - - '@aws-sdk/util-user-agent-browser@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - bowser: 2.11.0 - tslib: 2.7.0 - - '@aws-sdk/util-user-agent-node@3.614.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/xml-builder@3.609.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@noble/hashes@1.5.0': {} - - '@paralleldrive/cuid2@2.2.2': - dependencies: - '@noble/hashes': 1.5.0 - - '@smithy/abort-controller@3.1.1': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/chunked-blob-reader-native@3.0.0': - dependencies: - '@smithy/util-base64': 3.0.0 - tslib: 2.7.0 - - '@smithy/chunked-blob-reader@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/config-resolver@3.0.5': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@smithy/core@2.4.0': - dependencies: - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/credential-provider-imds@3.2.0': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - tslib: 2.7.0 - - '@smithy/eventstream-codec@3.1.2': - dependencies: - '@aws-crypto/crc32': 5.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-hex-encoding': 3.0.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-browser@3.0.6': - dependencies: - '@smithy/eventstream-serde-universal': 3.0.5 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-config-resolver@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-node@3.0.5': - dependencies: - '@smithy/eventstream-serde-universal': 3.0.5 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-universal@3.0.5': - dependencies: - '@smithy/eventstream-codec': 3.1.2 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/fetch-http-handler@3.2.4': - dependencies: - '@smithy/protocol-http': 4.1.0 - '@smithy/querystring-builder': 3.0.3 - '@smithy/types': 3.3.0 - '@smithy/util-base64': 3.0.0 - tslib: 2.7.0 - - '@smithy/hash-blob-browser@3.1.2': - dependencies: - '@smithy/chunked-blob-reader': 3.0.0 - '@smithy/chunked-blob-reader-native': 3.0.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/hash-node@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/hash-stream-node@3.1.2': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/invalid-dependency@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/is-array-buffer@2.2.0': - dependencies: - tslib: 2.7.0 - - '@smithy/is-array-buffer@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/md5-js@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/middleware-content-length@3.0.5': - dependencies: - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/middleware-endpoint@3.1.0': - dependencies: - '@smithy/middleware-serde': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@smithy/middleware-retry@3.0.15': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/service-error-classification': 3.0.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - tslib: 2.7.0 - uuid: 9.0.1 - - '@smithy/middleware-serde@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/middleware-stack@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/node-config-provider@3.1.4': - dependencies: - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/node-http-handler@3.1.4': - dependencies: - '@smithy/abort-controller': 3.1.1 - '@smithy/protocol-http': 4.1.0 - '@smithy/querystring-builder': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/property-provider@3.1.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/protocol-http@4.1.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/querystring-builder@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-uri-escape': 3.0.0 - tslib: 2.7.0 - - '@smithy/querystring-parser@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/service-error-classification@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - - '@smithy/shared-ini-file-loader@3.1.4': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/signature-v4@4.1.0': - dependencies: - '@smithy/is-array-buffer': 3.0.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-hex-encoding': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-uri-escape': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/smithy-client@3.2.0': - dependencies: - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-stack': 3.0.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-stream': 3.1.3 - tslib: 2.7.0 - - '@smithy/types@3.3.0': - dependencies: - tslib: 2.7.0 - - '@smithy/url-parser@3.0.3': - dependencies: - '@smithy/querystring-parser': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-base64@3.0.0': - dependencies: - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-body-length-browser@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-body-length-node@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-buffer-from@2.2.0': - dependencies: - '@smithy/is-array-buffer': 2.2.0 - tslib: 2.7.0 - - '@smithy/util-buffer-from@3.0.0': - dependencies: - '@smithy/is-array-buffer': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-config-provider@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-defaults-mode-browser@3.0.15': - dependencies: - '@smithy/property-provider': 3.1.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - bowser: 2.11.0 - tslib: 2.7.0 - - '@smithy/util-defaults-mode-node@3.0.15': - dependencies: - '@smithy/config-resolver': 3.0.5 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-endpoints@2.0.5': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-hex-encoding@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-middleware@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-retry@3.0.3': - dependencies: - '@smithy/service-error-classification': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-stream@3.1.3': - dependencies: - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-base64': 3.0.0 - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-hex-encoding': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-uri-escape@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-utf8@2.3.0': - dependencies: - '@smithy/util-buffer-from': 2.2.0 - tslib: 2.7.0 - - '@smithy/util-utf8@3.0.0': - dependencies: - '@smithy/util-buffer-from': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-waiter@3.1.2': - dependencies: - '@smithy/abort-controller': 3.1.1 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@types/node@22.5.2': - dependencies: - undici-types: 6.19.8 - - base64-js@1.5.1: {} - - bowser@2.11.0: {} - - buffer@5.6.0: - dependencies: - base64-js: 1.5.1 - ieee754: 1.2.1 - - dotenv@16.4.5: {} - - events@3.3.0: {} - - fast-xml-parser@4.4.1: - dependencies: - strnum: 1.0.5 - - ieee754@1.2.1: {} - - inherits@2.0.4: {} - - readable-stream@3.6.2: - dependencies: - inherits: 2.0.4 - string_decoder: 1.3.0 - util-deprecate: 1.0.2 - - safe-buffer@5.2.1: {} - - stream-browserify@3.0.0: - dependencies: - inherits: 2.0.4 - readable-stream: 3.6.2 - - string_decoder@1.3.0: - dependencies: - safe-buffer: 5.2.1 - - strnum@1.0.5: {} - - tslib@2.7.0: {} - - undici-types@6.19.8: {} - - util-deprecate@1.0.2: {} - - uuid@9.0.1: {} diff --git a/packages/storage/src/mux.ts b/packages/storage/src/mux.ts deleted file mode 100644 index e69de29..0000000 diff --git a/packages/storage/src/s3.spec.ts b/packages/storage/src/s3.spec.ts deleted file mode 100644 index dfce1cd..0000000 --- a/packages/storage/src/s3.spec.ts +++ /dev/null @@ -1,19 +0,0 @@ -import { describe } from 'mocha' -import { expect } from 'chai'; -import { uploadFile } from './s3.js' -import path from 'node:path' - -describe('s3', function () { - this.timeout(1000*30) - describe('uploadFile', function () { - it('should upload a file to specified bucket and return the storage data', async function () { - const data = await uploadFile(path.join(import.meta.dirname, './fixtures/sample.webp')) - expect(data).to.have.property('VersionId') - expect(data).to.have.property('ETag') - expect(data).to.have.property('Bucket') - expect(data).to.have.property('Key') - expect(data).to.have.property('Location') - expect(data).to.have.property('$metadata') - }) - }) -}) \ No newline at end of file diff --git a/packages/storage/src/s3.ts b/packages/storage/src/s3.ts deleted file mode 100644 index f3d67cc..0000000 --- a/packages/storage/src/s3.ts +++ /dev/null @@ -1,106 +0,0 @@ -import { S3Client, GetObjectCommand, type S3ClientConfig } from "@aws-sdk/client-s3"; -import { Upload } from "@aws-sdk/lib-storage"; -import type { S3FileResponse, S3FileRecord, StreamResponse } from '@futureporn/types'; -import { createId } from '@paralleldrive/cuid2'; -import { basename } from 'node:path'; -import fs, { createWriteStream, createReadStream } from 'node:fs'; -import { type Readable, pipeline } from "node:stream"; -import { tmpdir } from "node:os"; -import { join } from "node:path"; -import { promisify } from "node:util"; - -export interface S3FileArgs { - filePath: string; - s3BucketName: string; - s3AccessKeyId: string; - s3SecretAccessKey: string; - s3Region: string; - s3Endpoint: string; -} - -export const s3CdnMap = new Map([ - ['futureporn', 'https://futureporn-b2.b-cdn.net'], - ['fp-usc-dev', 'https://fp-usc-dev.b-cdn.net'], -]) - -function assertArguments(args: any): asserts args is S3FileArgs { - if (typeof args!== "object" ||!args) throw new Error("invalid S3FileArgs"); - if (!args.filePath) throw new Error('filePath was missing from args'); - if (typeof args.filePath!== 'string') throw new Error('filePath must be a string'); - if (!args.s3BucketName) throw new Error('s3BucketName was missing from args'); - if (typeof args.s3BucketName!== 'string') throw new Error('s3BucketName must be a string'); - if (!args.s3AccessKeyId) throw new Error('s3AccessKeyId was missing from args'); - if (typeof args.s3AccessKeyId!== 'string') throw new Error('s3AccessKeyId must be a string'); - if (!args.s3SecretAccessKey) throw new Error('s3SecretAccessKey was missing from args'); - if (typeof args.s3SecretAccessKey!== 'string') throw new Error('s3SecretAccessKey must be a string'); - if (!args.s3Region) throw new Error('s3Region was missing from args'); - if (typeof args.s3Region!== 'string') throw new Error('s3Region must be a string'); - if (!args.s3Endpoint) throw new Error('s3Endpoint was missing from args'); - if (typeof args.s3Endpoint!== 'string') throw new Error('s3Endpoint must be a string'); -} - -export function getCdnUrl(bucket: string, key: string): string { - if (!bucket) throw new Error('getCdnUrl() first arg (bucket name) was missing'); - if (!bucket) throw new Error('getCdnUrl() second arg (key) was missing'); - return `${s3CdnMap.get(bucket)}/${key}` -} - -// export async function downloadFile(args: S3FileArgs) { -// assertArguments(args) -// const { filePath, s3AccessKeyId, s3SecretAccessKey, s3BucketName, s3Endpoint, s3Region } = args -// // throw new Error('@todo') -// const download = S3Client -// } - - -export async function downloadFile (client: S3Client, bucket: string, s3_key: string): Promise { - console.log(`downloadS3File with s3File s3_key=${s3_key}, bucket=${bucket}`) - const getObjectCommand = new GetObjectCommand({ Bucket: bucket, Key: s3_key }) - const response = await client.send(getObjectCommand) - if (!response) throw new Error(`failed to receive a response while calling S3 GetObjectCommand (within downloadS3File)`); - if (!response.Body) throw new Error('S3 GetObjectCommand response did not have a Body (within downloadS3File)'); - const readStream = response.Body as Readable - const outputFilePath = join(tmpdir(), s3_key) - const writeStream = createWriteStream(outputFilePath) - const pipelinePromise = promisify(pipeline) - await pipelinePromise(readStream, writeStream) - return outputFilePath -} - -export async function uploadFile(args: unknown) { - assertArguments(args) - const { filePath, s3AccessKeyId, s3SecretAccessKey, s3BucketName, s3Endpoint, s3Region } = args; - const options: S3ClientConfig = { - endpoint: s3Endpoint, - region: s3Region, - credentials: { - accessKeyId: s3AccessKeyId, - secretAccessKey: s3SecretAccessKey, - }, - } - const client = new S3Client(options); - const key = `${createId()}-${basename(filePath)}` - const target = { - Bucket: s3BucketName, - Key: key, - Body: fs.createReadStream(filePath), - }; - - // greets https://stackoverflow.com/a/70159394/1004931 - try { - const parallelUploads3 = new Upload({ - client: client, - queueSize: 4, // optional concurrency configuration - leavePartsOnError: false, // optional manually handle dropped parts - params: target, - }); - - const res = await parallelUploads3.done(); - return res; - } catch (e) { - console.error(`while uploading a file to s3, we encountered an error`); - if (e instanceof Error) { - throw new Error(e.message); - } - } -} \ No newline at end of file diff --git a/packages/storage/tsconfig.json b/packages/storage/tsconfig.json deleted file mode 100644 index fdf87c5..0000000 --- a/packages/storage/tsconfig.json +++ /dev/null @@ -1,29 +0,0 @@ -{ - "compilerOptions": { - "noEmit": true, - "esModuleInterop": true, - "skipLibCheck": true, - "target": "es2022", - "allowJs": true, - "resolveJsonModule": true, - "moduleDetection": "force", - "isolatedModules": true, - // Enable strict type checking so you can catch bugs early - "strict": true, - "noUncheckedIndexedAccess": true, - "noImplicitOverride": true, - // Transpile our TypeScript code to JavaScript - "module": "NodeNext", - "outDir": "dist", - "lib": [ - "es2022" - ] - }, - // Include the necessary files for your project - "include": [ - "**/*.ts", - ], - "exclude": [ - "node_modules" - ] - } \ No newline at end of file diff --git a/packages/types/.eslintignore b/packages/types/.eslintignore deleted file mode 100644 index 53c37a1..0000000 --- a/packages/types/.eslintignore +++ /dev/null @@ -1 +0,0 @@ -dist \ No newline at end of file diff --git a/packages/types/.gitignore b/packages/types/.gitignore deleted file mode 100644 index 53c37a1..0000000 --- a/packages/types/.gitignore +++ /dev/null @@ -1 +0,0 @@ -dist \ No newline at end of file diff --git a/packages/types/package.json b/packages/types/package.json deleted file mode 100644 index 4ee827d..0000000 --- a/packages/types/package.json +++ /dev/null @@ -1,18 +0,0 @@ -{ - "name": "@futureporn/types", - "type": "module", - "version": "1.0.0", - "description": "", - "scripts": { - "test": "echo \"Warn: no test specified\" && exit 0", - "clean": "rm -rf dist", - "superclean": "rm -rf node_modules && rm -rf pnpm-lock.yaml && rm -rf dist" - }, - "main": "src/index.ts", - "keywords": [], - "author": "", - "license": "Unlicense", - "devDependencies": { - "typescript": "^5.5.4" - } -} diff --git a/packages/types/pnpm-lock.yaml b/packages/types/pnpm-lock.yaml deleted file mode 100644 index 2a0c411..0000000 --- a/packages/types/pnpm-lock.yaml +++ /dev/null @@ -1,24 +0,0 @@ -lockfileVersion: '9.0' - -settings: - autoInstallPeers: true - excludeLinksFromLockfile: false - -importers: - - .: - devDependencies: - typescript: - specifier: ^5.5.4 - version: 5.5.4 - -packages: - - typescript@5.5.4: - resolution: {integrity: sha512-Mtq29sKDAEYP7aljRgtPOpTvOfbwRWlS6dPRzwjdE+C0R4brX/GUyhHSecbHMFLNBLcJIPt9nl9yG5TZ1weH+Q==} - engines: {node: '>=14.17'} - hasBin: true - -snapshots: - - typescript@5.5.4: {} diff --git a/packages/types/src/index.ts b/packages/types/src/index.ts deleted file mode 100644 index d07db7a..0000000 --- a/packages/types/src/index.ts +++ /dev/null @@ -1,449 +0,0 @@ - - -export type PlatformNotificationType = 'email' | 'manual' | 'twitter' -export type ArchiveStatus = 'good' | 'issue' | 'missing' -export type RecordingState = 'recording' | 'stalled' | 'aborted' | 'failed' | 'finished' -export type ProcessingState = 'processing' -export type WaitingState = 'pending_recording' -export type Status = Partial - - -export interface IPatron { - username: string; - vanityLink?: string; -} - -// @see https://www.baeldung.com/rest-api-error-handling-best-practices -export interface GenericApiResponse { - error: string|null; - data: any; - message: string; - detail?: string; - help?: string; -} - -export interface S3FileRecord { - s3_key: string; - s3_id?: string; - bucket: string; -} - -export interface S3FileResponse { - s3_key: string; - s3_id?: string; - bucket: string; - created_at?: Date; - updated_at?: Date; -} - -export interface VtuberResponse { - id: string; -} - -export interface VtuberDataScrape { - display_name: string; - slug: string; - chaturbate_id?: string; - chaturbate?: string; - fansly_id?: string; - fansly?: string; -} - -export interface ScoutResponse { - error: boolean; - message: string; - data: any; -} - -export interface ChaturbateRoomSummary { - url: string; - name: string; -} - -export interface ChaturbateRoom { - name: string; - username: string; - room_subject: string; - tags: string[]; - is_new: boolean; - num_users: number; - num_followers: number; - current_show: string; - location?: string; - country?: string; - spoken_languages?: string; - display_name?: string; - birthday?: string; - is_hd: boolean; - age: number; - seconds_online: number; - image_url: string; - image_url_360x270: string; - chat_room_url_revshare: string; - iframe_embed_revshare: string; - chat_room_url: string; - iframe_embed: string; - slug: string; - } - -export interface VtuberRecord { - id: string; - display_name: string; - chaturbate: string; - twitter: string; - patreon: string; - twitch: string; - tiktok: string; - onlyfans: string; - youtube: string; - linktree: string; - carrd: string; - fansly: string; - pornhub: string; - discord: string; - reddit: string; - throne: string; - instagram: string; - facebook: string; - merch: string; - slug: string; - image: string; - theme_color: string; - image_blur: string; - fansly_id: string; - chaturbate_id: string; - twitter_id: string; -} - -export interface VodRecord { - id: string; - stream: StreamRecord; - stream_id: string; - created_at: string; - updated_at: string; - published_at?: string; - title?: string; - date: string; - mux_asset: MuxAssetRecord; - thumbnail?: string; - vtuber: string; - ipfs_cid?: string; - torrent?: string; - announce_title?: string; - announce_url?: string; - note?: string; - url: string; - discord_message_id: string; - s3_file: string; - recording_id: string; -} - -export interface TagRecord { - id: string -} - -export interface Timestamp { - id: string -} - -export interface User { - id: string -} - -export interface RecordingResponse { - id: string; - url: string; - date: string; - discord_interaction_id: string; - is_aborted: boolean; - vod_id: string; -} - -export interface VodResponse { - id: string; - stream: StreamResponse; - stream_id: string; - created_at: string; - updated_at: string; - published_at?: string; - title?: string; - date: string; - mux_asset: MuxAssetRecord; - thumbnail?: S3FileResponse; - vtuber: VtuberRecord; - tags?: TagRecord[]; - timestamps?: Timestamp[]; - ipfs_cid?: string; - s3_file?: S3FileResponse; - torrent?: string; - announce_title?: string; - announce_url?: string; - uploader?: User; - note?: string; - url: string; - segments?: SegmentResponse[]; - recording: RecordingRecord; - status: Status; - discord_message_id: string; -} - - -export interface StreamRecord { - id: string; - url: string; - platform_notification_type: PlatformNotificationType; - discord_message_id: string; - date: Date; - created_at: Date; - updated_at: Date; - vtuber: string; - tweet: string; - vods?: VodRecord[]; - archive_status: ArchiveStatus; - is_chaturbate_stream: Boolean; - is_fansly_stream: Boolean; - status: Status; - segments?: SegmentResponse[] -} - -export interface StreamResponse { - id: string; - url: string; - platform_notification_type: PlatformNotificationType; - discord_message_id: string; - date: Date; - created_at: Date; - updated_at: Date; - vtuber: string; - tweet: string; - vods?: VodResponse[]; - archive_status: ArchiveStatus; - is_chaturbate_stream: Boolean; - is_fansly_stream: Boolean; - is_recording_aborted: Boolean; - status: Status; - segments?: SegmentResponse[] -} - -export interface RecordingRecord { - id: number; - file_size: number; - discord_message_id: string; - is_aborted: boolean; - vod_id: string; - updated_at: Date; - created_at: Date; -} - - -export interface SegmentResponse { - id: string; - s3_key: string; - s3_id: string; - bytes: number; - bytes_uploaded: number; - checksum: string; - vod?: VodResponse; - created_at: string; - updated_at: string; -} - - - -export interface MuxAssetRecord { - id: number; - playback_id: string; - asset_id: string; -} - - -export interface IMuxAsset { - id: number; - playback_id: string; - asset_id: string; -} - -export interface IPagination { - page: number; - pageSize: number; - pageCount: number; - total: number; -} - -export interface IMuxAssetResponse { - data: IMuxAsset; - meta: IMeta; -} - -export interface IMeta { - pagination: IPagination; -} - - - -export interface IPlatformNotification { - id: number; - source: string; - platform: string; - date: string; - date_2: string; - vtuber: number; -} - -export interface IPlatformNotificationResponse { - data: IPlatformNotification; - meta: IMeta; - error?: any; -} - - -export interface ITimestamp { - id: number; - time: number; -} - -export interface ITag { - id: number; - name: string; -} - -export interface IS3File { - id: number; - url: string; - key: string; - uploadId: string; - cdn_url: string; -} -export interface IS3FileResponse { - data: IS3File; - meta: IMeta; -} - -export interface IVod { - id: number; - stream?: IStream; - published_at?: string; - uuid: string; - title?: string; - duration?: number; - date: string; - date_2: string; - mux_asset: IMuxAsset; - thumbnail?: IS3File; - vtuber: IVtuber; - s3_file: IS3File; - tag_vod_relations: ITagVodRelation[]; - timestamps: ITimestamp[]; - ipfs_cid: string; - announce_title: string; - announce_url: string; - uploader: IUserResponse; - note: string; -} - - - -export interface IUser { - id: number; - username: string; - vanityLink?: string; - image: string; -} - -export interface IUserResponse { -data: IUser; -meta: IMeta; -} - - -export interface ITagVodRelation { - id: number; - tag: ITag | IToyTag - vod: IVod; - creator_id: number; - created_at: string; -} - - -export interface IToyTag extends ITag { - toy: IToy; -} - - -export interface IToy { - id: number; - tags: ITag[]; - linkTag: ITag[]; - make: string; - model: string; - aspectRatio: string; - image2: string; - -} - - -interface IToysListProps { - toys: IToy[]; - page: number; - pageSize: number; -} - - - -export interface IStream { - id: number; - date: string; - date_2: string; - archive_status: ArchiveStatus; - vods: IVod[]; - uuid: string; - vtuber: IVtuber; - is_chaturbate_stream: boolean; - is_fansly_stream: boolean; - platform_notifications: IPlatformNotification[]; -} - - - -export interface IVtuber { - id: number; - slug: string; - display_name: string; - chaturbate?: string; - twitter?: string; - patreon?: string; - twitch?: string; - tiktok?: string; - onlyfans?: string; - youtube?: string; - linktree?: string; - carrd?: string; - fansly?: string; - pornhub?: string; - discord?: string; - reddit?: string; - throne?: string; - instagram?: string; - facebook?: string; - merch?: string; - vods: IVod[]; - description_1: string; - description_2?: string; - image: string; - image_blur?: string; - theme_color: string; - fansly_id?: string; - chaturbate_id?: string; - twitter_id?: string; -} - -export type NotificationData = { - isMatch: boolean; - url?: string; - platform?: string; - channel?: string; - displayName?: string; - date?: string; - userId?: string | null; - avatar?: string; -}; diff --git a/packages/types/tsconfig.json b/packages/types/tsconfig.json deleted file mode 100644 index a0928d2..0000000 --- a/packages/types/tsconfig.json +++ /dev/null @@ -1,36 +0,0 @@ -{ - "compilerOptions": { - // Base Options recommended for all projects - "esModuleInterop": true, - "skipLibCheck": true, - "target": "es2022", - "allowJs": false, - "resolveJsonModule": true, - "moduleDetection": "force", - "isolatedModules": true, - "verbatimModuleSyntax": true, - // Enable strict type checking so you can catch bugs early - "strict": true, - "noUncheckedIndexedAccess": true, - "noImplicitOverride": true, - // Transpile our TypeScript code to JavaScript - "module": "NodeNext", - "lib": [ - "es2022" - ], - "noImplicitAny": true, - "noImplicitThis": true, - "strictFunctionTypes": true, - "strictNullChecks": true, - "types": [], - "noEmit": true, - "forceConsistentCasingInFileNames": true - }, - // Include the necessary files for your project - "files": [ - "src/index.ts", - ], - "exclude": [ - "node_modules" - ] -} \ No newline at end of file diff --git a/packages/utils/src/image.spec.ts b/packages/utils/src/image.spec.ts index c5ec5f7..0ae64b1 100644 --- a/packages/utils/src/image.spec.ts +++ b/packages/utils/src/image.spec.ts @@ -30,7 +30,7 @@ describe('image', function () { describe('getStoryboard', function () { this.timeout(1000*60*15) it('should accept a URL and return a path to image on disk', async function () { - const url = 'http://38.242.193.246:8081/projektmelody-chaturbate-2025-01-09.mp4' + const url = 'http://38.242.193.246:8081/projektmelody-chaturbate-2025-01-11.mp4' const imagePath = await getStoryboard(url) expect(imagePath).to.match(/\.png/) }) diff --git a/scripts/build-test.sh b/scripts/build-test.sh deleted file mode 100755 index 3cd0be6..0000000 --- a/scripts/build-test.sh +++ /dev/null @@ -1,21 +0,0 @@ -#!/bin/bash - -postgres_pod_name=postgresql-primary-0 - -if [ -z $POSTGRES_PASSWORD ]; then - echo "POSTGRES_PASSWORD was missing in env. Are you executing this script via Tilt? (that is the intended method)" - exit 5 -fi - -kubectl -n futureporn exec "${postgres_pod_name}" -- env PGPASSWORD=${POSTGRES_PASSWORD} psql -U postgres -d futureporn --command " -SELECT graphile_worker.add_job( - 'combine_video_segments', - payload := json_build_object( - 's3_manifest', json_build_array( - json_build_object('id', '4_z7d53875ff1c32a1983d30b18_f118989ca296359da_d20240809_m205858_c000_v0001408_t0033_u01723237138038', 'key', 'mock-stream0.mp4'), - json_build_object('id', '4_z7d53875ff1c32a1983d30b18_f107c0649cef835e4_d20240809_m205859_c000_v0001406_t0016_u01723237139170', 'key', 'mock-stream1.mp4'), - json_build_object('id', '4_z7d53875ff1c32a1983d30b18_f10651c62f4ca1b2f_d20240809_m205900_c000_v0001076_t0022_u01723237140217', 'key', 'mock-stream2.mp4') - ) - ), - max_attempts := 3 -);" diff --git a/scripts/capture-integration.sh b/scripts/capture-integration.sh deleted file mode 100755 index f7fe5ae..0000000 --- a/scripts/capture-integration.sh +++ /dev/null @@ -1,25 +0,0 @@ -#!/bin/bash - -if [ -z "${AUTOMATION_USER_JWT}" ]; then - echo "Error: AUTOMATION_USER_JWT variable is not defined." - exit 1 -fi - -# get a random room -response=$(curl -sL --fail GET http://localhost:8134/chaturbate/random-room) -exitcode=$? -url=$(echo $response | jq -r '.url') -if [[ $exitcode -ne 0 || -z "$response" || -z "$url" ]]; then - echo "failed to get random room. exitcode=${exitcode}, response=${response}, url=${url}" - exit $exitcode -fi -echo "Random online chaturbate room url=${url}" - - - -# create a recording -curl -sL -H "Authorization: Bearer ${AUTOMATION_USER_JWT}" \ - -H "Content-Type: application/json" \ - -d '{"url": "'"${url}"'"}' \ - http://localhost:9000/recordings -echo "recording created" \ No newline at end of file diff --git a/scripts/drupal-init-wrapper.sh b/scripts/drupal-init-wrapper.sh deleted file mode 100755 index dd4c9f7..0000000 --- a/scripts/drupal-init-wrapper.sh +++ /dev/null @@ -1,8 +0,0 @@ -#!/bin/bash - - -namespace=futureporn -pod_name=$(kubectl --namespace futureporn get pods -l app.kubernetes.io/name=drupal -o custom-columns=":metadata.name" --no-headers) - -kubectl -n "${namespace}" cp ./scripts/drupal-init.sh "${pod_name}:/tmp/drupal-init.sh" -kubectl -n "${namespace}" exec "${pod_name}" -- bash -c "/tmp/drupal-init.sh" diff --git a/scripts/drupal-init.sh b/scripts/drupal-init.sh deleted file mode 100755 index e6bfc1e..0000000 --- a/scripts/drupal-init.sh +++ /dev/null @@ -1,24 +0,0 @@ - -#!/bin/bash - -# maybe https://www.drupal.org/project/extlink -# maybe https://www.drupal.org/project/seckit -# maybe https://www.drupal.org/project/slick -# maybe https://www.drupal.org/project/fontawesome - -cd /opt/bitnami/drupal -composer require \ - 'drupal/bulma:^1.0' \ - 'drupal/file_uploader_uppy:^1.0' \ - 'drupal/backup_migrate:^5.0' \ - 'drupal/cdn:^4.1' \ - 'drupal/s3fs:^3.6' \ - 'drupal/video:^3.0' \ - 'league/commonmark ^1.0' \ - 'drupal/markdown:^3.0' \ - 'drupal/color_field:^3.0' \ - 'drupal/pathauto:^1.13' \ - 'drupal/extlink:^2.0' -drush updatedb -drush theme:install bulma -drush pm:install -y file_uploader_uppy backup_migrate cdn cdn_ui s3fs video markdown diff --git a/scripts/flux-bootstrap.sh b/scripts/flux-bootstrap.sh deleted file mode 100755 index bfe9b2a..0000000 --- a/scripts/flux-bootstrap.sh +++ /dev/null @@ -1,26 +0,0 @@ -#!/bin/bash - -if [ -z "${ENV}" ]; then \ - echo "Error: ENV variable is not defined. Please set to one of development|staging|production"; exit 1; \ -fi - -if [ "${ENV}" == "development" ]; then \ - echo "Flux is not used in development environment. Skipping." - exit 0 -fi - -# flux bootstrap git \ -# --kubeconfig /home/cj/.kube/vke.yaml \ -# --url=https://gitea.futureporn.net/futureporn/fp.git \ -# --branch=main \ -# --username=cj_clippy \ -# --token-auth=true \ -# --path=clusters/production - -## --silent avoids the [Yes|no] prompt -flux bootstrap git \ - --silent \ - --url="ssh://git@gitea.futureporn.net:2222/futureporn/fp" \ - --branch=main \ - --path="flux/clusters/$ENV" \ - --private-key-file=/home/cj/.ssh/fp-flux \ No newline at end of file diff --git a/scripts/k8s-metrics.sh b/scripts/k8s-metrics.sh deleted file mode 100755 index 49f1219..0000000 --- a/scripts/k8s-metrics.sh +++ /dev/null @@ -1,12 +0,0 @@ -#!/bin/bash - -if [ -z "${ENV}" ]; then \ - echo "Error: ENV variable is not defined. Please set to one of development|staging|production"; exit 1; \ -fi - -if [ "${ENV}" == "development" ]; then \ - echo "k8s metrics are not used in development environment. Skipping." - exit 0 -fi - -kubectl apply -f https://github.com/kubernetes-sigs/metrics-server/releases/latest/download/components.yaml \ No newline at end of file diff --git a/scripts/k8s-namespaces.sh b/scripts/k8s-namespaces.sh deleted file mode 100755 index 0bf56ef..0000000 --- a/scripts/k8s-namespaces.sh +++ /dev/null @@ -1,7 +0,0 @@ -#!/bin/bash - -kubectl create namespace cert-manager -kubectl create namespace futureporn -kubectl create namespace velero - -exit 0 # important to keep the Makefile chain going even if namespaces already existed and kubectl returned non-zero \ No newline at end of file diff --git a/scripts/k8s-secrets.sh b/scripts/k8s-secrets.sh deleted file mode 100755 index ec2ad7a..0000000 --- a/scripts/k8s-secrets.sh +++ /dev/null @@ -1,210 +0,0 @@ -#!/bin/bash - - -## @todo switch to infisical -# ns=futureporn -# kubectl --namespace $ns delete secret universal-auth-credentials --ignore-not-found -# kubectl --namespace $ns create secret generic universal-auth-credentials \ -# --from-literal=clientSecret="${INFISICAL_CLIENT_SECRET}" \ -# --from-literal=clientId="${INFISICAL_CLIENT_ID}" -# echo "@todo remove all the unused secrets" -# exit 256 - -CLOUD_DATA=$(echo -e "[default]\naws_access_key_id: $VELERO_S3_KEY_ID\naws_secret_access_key: $VELERO_S3_ACCESS_KEY" | base64 -w 0) -kubectl --namespace=velero delete secret velero --ignore-not-found -## we do this so helm can adopt our pre-made secret @see https://github.com/helm/helm/pull/7649 -cat </dev/null || true)" != 'true' ]; then - docker run \ - -d --restart=always -p "127.0.0.1:${reg_port}:5000" --network bridge --name "${reg_name}" \ - registry:2 -fi - -# 2. Create kind cluster with containerd registry config dir enabled -# TODO: kind will eventually enable this by default and this patch will -# be unnecessary. -# -# See: -# https://github.com/kubernetes-sigs/kind/issues/2875 -# https://github.com/containerd/containerd/blob/main/docs/cri/config.md#registry-configuration -# See: https://github.com/containerd/containerd/blob/main/docs/hosts.md -cat <=6.9.0'} - - '@babel/helper-validator-identifier@7.24.7': - resolution: {integrity: sha512-rR+PBcQ1SMQDDyF6X0wxtG8QyLCgUB0eRAGguqRLfkCA87l7yAP7ehq8SNj96OOGTO8OBV70KhuFYcIkHXOg0w==} - engines: {node: '>=6.9.0'} - - '@babel/highlight@7.24.7': - resolution: {integrity: sha512-EStJpq4OuY8xYfhGVXngigBJRWxftKX9ksiGDnmlY3o7B/V7KIAc9X4oiK87uPJSc/vs5L869bem5fhZa8caZw==} - engines: {node: '>=6.9.0'} - - '@cspotcode/source-map-support@0.8.1': - resolution: {integrity: sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==} - engines: {node: '>=12'} - - '@discordeno/bot@19.0.0-next.b1bfe94': - resolution: {integrity: sha512-ZaPDaPM6tuxpRZCyuNB2rzqz2WYHH20IQdutu87jffW2dV26D3Nw8SJCE1LNdXdu9akes3VMXolkNsLJ5MW2Hw==} - - '@discordeno/bot@19.0.0-next.b3a8c86': - resolution: {integrity: sha512-Jm7sEmxR0MyK3kwe8CVHDQfAXpbkkPLER2nGuS2nn2e/0AXQKCCgJRMNmorJ+ZzcSw+slrw4hHkVZMBv7JML9Q==} - - '@discordeno/gateway@19.0.0-next.b1bfe94': - resolution: {integrity: sha512-rNkcCua4Inx8NkuCjbuRBAAg+AIg1qYb24Fq6/Y2+1s+iO0VSQ/ZK8PRNbI2gSpAtAbfndvxA9kyU1ggwwLjNw==} - - '@discordeno/gateway@19.0.0-next.b3a8c86': - resolution: {integrity: sha512-FGQvHcJFCyC4WCdgvRo3vAMoyvDpTkbHN4DZlEDNhN6uBMsWsd1xbjWPsJTdWGVUvHQd4+5HOv44wPhO02jytg==} - - '@discordeno/rest@19.0.0-next.b1bfe94': - resolution: {integrity: sha512-lLeP3hmMM0GFsO+VA3f9EKOwfIo1HvG1hudtlC1Er6BFWgArpz0/VizHbOyOr5SzZuzrQyfcBJJfEhC+48TsRg==} - - '@discordeno/rest@19.0.0-next.b3a8c86': - resolution: {integrity: sha512-nhhHSzibxOwxFAckgcuU8nGj+AGo3IRb1qiCcTE4gQStjqmV3ZewoVRzaRGgoY4t5ld5oeDODplXY5tloobkiw==} - - '@discordeno/types@19.0.0-next.b1bfe94': - resolution: {integrity: sha512-X1MmdPFMyzjxFEANEPrrBjdJDJBAz4RTa1vpVlMp46C3dPqTwjMwijaPhPgPnFiSwt01lJ8WaFDpMvfqOwpQNg==} - - '@discordeno/types@19.0.0-next.b3a8c86': - resolution: {integrity: sha512-1uUOpfBN3a8zDYOyL61qvWhG+NL4KMjcun7XFg7CB6wjGubUv4Uc2QXKG7SkSfNmtYzxetrrx1kyoSxkbEHLOw==} - - '@discordeno/utils@19.0.0-next.b1bfe94': - resolution: {integrity: sha512-Y+j3G83vCVXiNSEQlFR9WWJ6GZhvBmIfFjb03zFcmHa20nHExILJJBEyDpS+ptziEyF7jB+2hNw5I/SKn8Zm2w==} - - '@discordeno/utils@19.0.0-next.b3a8c86': - resolution: {integrity: sha512-W4SDymUvevihQZry9hB4lzCUNSz5QwqAzdT+3VKswjARgmQQnOjZdJ1w9rX/xoMB5FgS8Vhk6er4ABULIwPi6g==} - - '@esbuild/aix-ppc64@0.23.1': - resolution: {integrity: sha512-6VhYk1diRqrhBAqpJEdjASR/+WVRtfjpqKuNw11cLiaWpAT/Uu+nokB+UJnevzy/P9C/ty6AOe0dwueMrGh/iQ==} - engines: {node: '>=18'} - cpu: [ppc64] - os: [aix] - - '@esbuild/android-arm64@0.23.1': - resolution: {integrity: sha512-xw50ipykXcLstLeWH7WRdQuysJqejuAGPd30vd1i5zSyKK3WE+ijzHmLKxdiCMtH1pHz78rOg0BKSYOSB/2Khw==} - engines: {node: '>=18'} - cpu: [arm64] - os: [android] - - '@esbuild/android-arm@0.23.1': - resolution: {integrity: sha512-uz6/tEy2IFm9RYOyvKl88zdzZfwEfKZmnX9Cj1BHjeSGNuGLuMD1kR8y5bteYmwqKm1tj8m4cb/aKEorr6fHWQ==} - engines: {node: '>=18'} - cpu: [arm] - os: [android] - - '@esbuild/android-x64@0.23.1': - resolution: {integrity: sha512-nlN9B69St9BwUoB+jkyU090bru8L0NA3yFvAd7k8dNsVH8bi9a8cUAUSEcEEgTp2z3dbEDGJGfP6VUnkQnlReg==} - engines: {node: '>=18'} - cpu: [x64] - os: [android] - - '@esbuild/darwin-arm64@0.23.1': - resolution: {integrity: sha512-YsS2e3Wtgnw7Wq53XXBLcV6JhRsEq8hkfg91ESVadIrzr9wO6jJDMZnCQbHm1Guc5t/CdDiFSSfWP58FNuvT3Q==} - engines: {node: '>=18'} - cpu: [arm64] - os: [darwin] - - '@esbuild/darwin-x64@0.23.1': - resolution: {integrity: sha512-aClqdgTDVPSEGgoCS8QDG37Gu8yc9lTHNAQlsztQ6ENetKEO//b8y31MMu2ZaPbn4kVsIABzVLXYLhCGekGDqw==} - engines: {node: '>=18'} - cpu: [x64] - os: [darwin] - - '@esbuild/freebsd-arm64@0.23.1': - resolution: {integrity: sha512-h1k6yS8/pN/NHlMl5+v4XPfikhJulk4G+tKGFIOwURBSFzE8bixw1ebjluLOjfwtLqY0kewfjLSrO6tN2MgIhA==} - engines: {node: '>=18'} - cpu: [arm64] - os: [freebsd] - - '@esbuild/freebsd-x64@0.23.1': - resolution: {integrity: sha512-lK1eJeyk1ZX8UklqFd/3A60UuZ/6UVfGT2LuGo3Wp4/z7eRTRYY+0xOu2kpClP+vMTi9wKOfXi2vjUpO1Ro76g==} - engines: {node: '>=18'} - cpu: [x64] - os: [freebsd] - - '@esbuild/linux-arm64@0.23.1': - resolution: {integrity: sha512-/93bf2yxencYDnItMYV/v116zff6UyTjo4EtEQjUBeGiVpMmffDNUyD9UN2zV+V3LRV3/on4xdZ26NKzn6754g==} - engines: {node: '>=18'} - cpu: [arm64] - os: [linux] - - '@esbuild/linux-arm@0.23.1': - resolution: {integrity: sha512-CXXkzgn+dXAPs3WBwE+Kvnrf4WECwBdfjfeYHpMeVxWE0EceB6vhWGShs6wi0IYEqMSIzdOF1XjQ/Mkm5d7ZdQ==} - engines: {node: '>=18'} - cpu: [arm] - os: [linux] - - '@esbuild/linux-ia32@0.23.1': - resolution: {integrity: sha512-VTN4EuOHwXEkXzX5nTvVY4s7E/Krz7COC8xkftbbKRYAl96vPiUssGkeMELQMOnLOJ8k3BY1+ZY52tttZnHcXQ==} - engines: {node: '>=18'} - cpu: [ia32] - os: [linux] - - '@esbuild/linux-loong64@0.23.1': - resolution: {integrity: sha512-Vx09LzEoBa5zDnieH8LSMRToj7ir/Jeq0Gu6qJ/1GcBq9GkfoEAoXvLiW1U9J1qE/Y/Oyaq33w5p2ZWrNNHNEw==} - engines: {node: '>=18'} - cpu: [loong64] - os: [linux] - - '@esbuild/linux-mips64el@0.23.1': - resolution: {integrity: sha512-nrFzzMQ7W4WRLNUOU5dlWAqa6yVeI0P78WKGUo7lg2HShq/yx+UYkeNSE0SSfSure0SqgnsxPvmAUu/vu0E+3Q==} - engines: {node: '>=18'} - cpu: [mips64el] - os: [linux] - - '@esbuild/linux-ppc64@0.23.1': - resolution: {integrity: sha512-dKN8fgVqd0vUIjxuJI6P/9SSSe/mB9rvA98CSH2sJnlZ/OCZWO1DJvxj8jvKTfYUdGfcq2dDxoKaC6bHuTlgcw==} - engines: {node: '>=18'} - cpu: [ppc64] - os: [linux] - - '@esbuild/linux-riscv64@0.23.1': - resolution: {integrity: sha512-5AV4Pzp80fhHL83JM6LoA6pTQVWgB1HovMBsLQ9OZWLDqVY8MVobBXNSmAJi//Csh6tcY7e7Lny2Hg1tElMjIA==} - engines: {node: '>=18'} - cpu: [riscv64] - os: [linux] - - '@esbuild/linux-s390x@0.23.1': - resolution: {integrity: sha512-9ygs73tuFCe6f6m/Tb+9LtYxWR4c9yg7zjt2cYkjDbDpV/xVn+68cQxMXCjUpYwEkze2RcU/rMnfIXNRFmSoDw==} - engines: {node: '>=18'} - cpu: [s390x] - os: [linux] - - '@esbuild/linux-x64@0.23.1': - resolution: {integrity: sha512-EV6+ovTsEXCPAp58g2dD68LxoP/wK5pRvgy0J/HxPGB009omFPv3Yet0HiaqvrIrgPTBuC6wCH1LTOY91EO5hQ==} - engines: {node: '>=18'} - cpu: [x64] - os: [linux] - - '@esbuild/netbsd-x64@0.23.1': - resolution: {integrity: sha512-aevEkCNu7KlPRpYLjwmdcuNz6bDFiE7Z8XC4CPqExjTvrHugh28QzUXVOZtiYghciKUacNktqxdpymplil1beA==} - engines: {node: '>=18'} - cpu: [x64] - os: [netbsd] - - '@esbuild/openbsd-arm64@0.23.1': - resolution: {integrity: sha512-3x37szhLexNA4bXhLrCC/LImN/YtWis6WXr1VESlfVtVeoFJBRINPJ3f0a/6LV8zpikqoUg4hyXw0sFBt5Cr+Q==} - engines: {node: '>=18'} - cpu: [arm64] - os: [openbsd] - - '@esbuild/openbsd-x64@0.23.1': - resolution: {integrity: sha512-aY2gMmKmPhxfU+0EdnN+XNtGbjfQgwZj43k8G3fyrDM/UdZww6xrWxmDkuz2eCZchqVeABjV5BpildOrUbBTqA==} - engines: {node: '>=18'} - cpu: [x64] - os: [openbsd] - - '@esbuild/sunos-x64@0.23.1': - resolution: {integrity: sha512-RBRT2gqEl0IKQABT4XTj78tpk9v7ehp+mazn2HbUeZl1YMdaGAQqhapjGTCe7uw7y0frDi4gS0uHzhvpFuI1sA==} - engines: {node: '>=18'} - cpu: [x64] - os: [sunos] - - '@esbuild/win32-arm64@0.23.1': - resolution: {integrity: sha512-4O+gPR5rEBe2FpKOVyiJ7wNDPA8nGzDuJ6gN4okSA1gEOYZ67N8JPk58tkWtdtPeLz7lBnY6I5L3jdsr3S+A6A==} - engines: {node: '>=18'} - cpu: [arm64] - os: [win32] - - '@esbuild/win32-ia32@0.23.1': - resolution: {integrity: sha512-BcaL0Vn6QwCwre3Y717nVHZbAa4UBEigzFm6VdsVdT/MbZ38xoj1X9HPkZhbmaBGUD1W8vxAfffbDe8bA6AKnQ==} - engines: {node: '>=18'} - cpu: [ia32] - os: [win32] - - '@esbuild/win32-x64@0.23.1': - resolution: {integrity: sha512-BHpFFeslkWrXWyUPnbKm+xYYVYruCinGcftSBaa8zoF9hZO4BcSCFUvHVTtzpIY6YzUnYtuEhZ+C9iEXjxnasg==} - engines: {node: '>=18'} - cpu: [x64] - os: [win32] - - '@graphile/logger@0.2.0': - resolution: {integrity: sha512-jjcWBokl9eb1gVJ85QmoaQ73CQ52xAaOCF29ukRbYNl6lY+ts0ErTaDYOBlejcbUs2OpaiqYLO5uDhyLFzWw4w==} - - '@jridgewell/resolve-uri@3.1.2': - resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==} - engines: {node: '>=6.0.0'} - - '@jridgewell/sourcemap-codec@1.5.0': - resolution: {integrity: sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==} - - '@jridgewell/trace-mapping@0.3.9': - resolution: {integrity: sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==} - - '@noble/hashes@1.5.0': - resolution: {integrity: sha512-1j6kQFb7QRru7eKN3ZDvRcP13rugwdxZqCjbiAVZfIJwgj2A65UmT4TgARXGlXgnRkORLTDTrO19ZErt7+QXgA==} - engines: {node: ^14.21.3 || >=16} - - '@paralleldrive/cuid2@2.2.2': - resolution: {integrity: sha512-ZOBkgDwEdoYVlSeRbYYXs0S9MejQofiVYoTbKzy/6GQa39/q5tQU2IX46+shYnUkpEl3wc+J6wRlar7r2EK2xA==} - - '@tsconfig/node10@1.0.11': - resolution: {integrity: sha512-DcRjDCujK/kCk/cUe8Xz8ZSpm8mS3mNNpta+jGCA6USEDfktlNvm1+IuZ9eTcDbNk41BHwpHHeW+N1lKCz4zOw==} - - '@tsconfig/node12@1.0.11': - resolution: {integrity: sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==} - - '@tsconfig/node14@1.0.3': - resolution: {integrity: sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==} - - '@tsconfig/node16@1.0.4': - resolution: {integrity: sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==} - - '@types/chai@4.3.19': - resolution: {integrity: sha512-2hHHvQBVE2FiSK4eN0Br6snX9MtolHaTo/batnLjlGRhoQzlCL61iVpxoqO7SfFyOw+P/pwv+0zNHzKoGWz9Cw==} - - '@types/debug@4.1.12': - resolution: {integrity: sha512-vIChWdVG3LG1SMxEvI/AK+FWJthlrqlTu7fbrlywTkkaONwk/UAGaULXRlf8vkzFBLVm0zkMdCquhL5aOjhXPQ==} - - '@types/interpret@1.1.3': - resolution: {integrity: sha512-uBaBhj/BhilG58r64mtDb/BEdH51HIQLgP5bmWzc5qCtFMja8dCk/IOJmk36j0lbi9QHwI6sbtUNGuqXdKCAtQ==} - - '@types/mocha@10.0.7': - resolution: {integrity: sha512-GN8yJ1mNTcFcah/wKEFIJckJx9iJLoMSzWcfRRuxz/Jk+U6KQNnml+etbtxFK8lPjzOw3zp4Ha/kjSst9fsHYw==} - - '@types/ms@0.7.34': - resolution: {integrity: sha512-nG96G3Wp6acyAgJqGasjODb+acrI7KltPiRxzHPXnP3NgI28bpQDRv53olbqGXbfcgF5aiiHmO3xpwEpS5Ld9g==} - - '@types/node@20.16.3': - resolution: {integrity: sha512-/wdGiWRkMOm53gAsSyFMXFZHbVg7C6CbkrzHNpaHoYfsUWPg7m6ZRKtvQjgvQ9i8WT540a3ydRlRQbxjY30XxQ==} - - '@types/node@22.5.2': - resolution: {integrity: sha512-acJsPTEqYqulZS/Yp/S3GgeE6GZ0qYODUR8aVr/DkhHQ8l9nd4j5x1/ZJy9/gHrRlFMqkO6i0I3E27Alu4jjPg==} - - '@types/pg@8.11.8': - resolution: {integrity: sha512-IqpCf8/569txXN/HoP5i1LjXfKZWL76Yr2R77xgeIICUbAYHeoaEZFhYHo2uDftecLWrTJUq63JvQu8q3lnDyA==} - - '@types/qs@6.9.15': - resolution: {integrity: sha512-uXHQKES6DQKKCLh441Xv/dwxOq1TVS3JPUMlEqoEglvlhR6Mxnlew/Xq/LRVHpLyk7iK3zODe1qYHIMltO7XGg==} - - '@types/retry@0.12.2': - resolution: {integrity: sha512-XISRgDJ2Tc5q4TRqvgJtzsRkFYNJzZrhTdtMoGVBttwzzQJkPnS3WWTFc7kuDRoPtPakl+T+OfdEUjYJj7Jbow==} - - '@types/semver@7.5.8': - resolution: {integrity: sha512-I8EUhyrgfLrcTkzV3TSsGyl1tSuPrEDzr0yd5m90UgNxQkyDXULk3b6MlQqTCpZpNtWe1K0hzclnZkTcLBe2UQ==} - - acorn-walk@8.3.3: - resolution: {integrity: sha512-MxXdReSRhGO7VlFe1bRG/oI7/mdLV9B9JJT0N8vZOhF7gFRR5l3M8W9G8JxmKV+JC5mGqJ0QvqfSOLsCPa4nUw==} - engines: {node: '>=0.4.0'} - - acorn@8.12.1: - resolution: {integrity: sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==} - engines: {node: '>=0.4.0'} - hasBin: true - - ansi-colors@4.1.3: - resolution: {integrity: sha512-/6w/C21Pm1A7aZitlI5Ni/2J6FFQN8i1Cvz3kHABAAbw93v/NlvKdVOqz7CCWz/3iv/JplRSEEZ83XION15ovw==} - engines: {node: '>=6'} - - ansi-regex@5.0.1: - resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==} - engines: {node: '>=8'} - - ansi-styles@3.2.1: - resolution: {integrity: sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==} - engines: {node: '>=4'} - - ansi-styles@4.3.0: - resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} - engines: {node: '>=8'} - - anymatch@3.1.3: - resolution: {integrity: sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==} - engines: {node: '>= 8'} - - arg@4.1.3: - resolution: {integrity: sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==} - - argparse@2.0.1: - resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==} - - assertion-error@2.0.1: - resolution: {integrity: sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==} - engines: {node: '>=12'} - - balanced-match@1.0.2: - resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} - - binary-extensions@2.3.0: - resolution: {integrity: sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==} - engines: {node: '>=8'} - - brace-expansion@1.1.11: - resolution: {integrity: sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==} - - brace-expansion@2.0.1: - resolution: {integrity: sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==} - - braces@3.0.3: - resolution: {integrity: sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==} - engines: {node: '>=8'} - - browser-stdout@1.3.1: - resolution: {integrity: sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==} - - call-bind@1.0.7: - resolution: {integrity: sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==} - engines: {node: '>= 0.4'} - - callsites@3.1.0: - resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==} - engines: {node: '>=6'} - - camelcase@6.3.0: - resolution: {integrity: sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==} - engines: {node: '>=10'} - - chai@5.1.1: - resolution: {integrity: sha512-pT1ZgP8rPNqUgieVaEY+ryQr6Q4HXNg8Ei9UnLUrjN4IA7dvQC5JB+/kxVcPNDHyBcc/26CXPkbNzq3qwrOEKA==} - engines: {node: '>=12'} - - chalk@2.4.2: - resolution: {integrity: sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==} - engines: {node: '>=4'} - - chalk@4.1.2: - resolution: {integrity: sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==} - engines: {node: '>=10'} - - check-error@2.1.1: - resolution: {integrity: sha512-OAlb+T7V4Op9OwdkjmguYRqncdlx5JiofwOAUkmTF+jNdHwzTaTs4sRAGpzLF3oOz5xAyDGrPgeIDFQmDOTiJw==} - engines: {node: '>= 16'} - - chokidar@3.6.0: - resolution: {integrity: sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==} - engines: {node: '>= 8.10.0'} - - cliui@7.0.4: - resolution: {integrity: sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==} - - cliui@8.0.1: - resolution: {integrity: sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==} - engines: {node: '>=12'} - - color-convert@1.9.3: - resolution: {integrity: sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==} - - color-convert@2.0.1: - resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} - engines: {node: '>=7.0.0'} - - color-name@1.1.3: - resolution: {integrity: sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==} - - color-name@1.1.4: - resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} - - commander@12.1.0: - resolution: {integrity: sha512-Vw8qHK3bZM9y/P10u3Vib8o/DdkvA2OtPtZvD871QKjy74Wj1WSKFILMPRPSdUSx5RFK1arlJzEtA4PkFgnbuA==} - engines: {node: '>=18'} - - concat-map@0.0.1: - resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} - - cosmiconfig@8.3.6: - resolution: {integrity: sha512-kcZ6+W5QzcJ3P1Mt+83OUv/oHFqZHIx8DuxG6eZ5RGMERoLqp4BuGjhHLYGK+Kf5XVkQvqBSmAy/nGWN3qDgEA==} - engines: {node: '>=14'} - peerDependencies: - typescript: '>=4.9.5' - peerDependenciesMeta: - typescript: - optional: true - - create-require@1.1.1: - resolution: {integrity: sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==} - - data-uri-to-buffer@4.0.1: - resolution: {integrity: sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A==} - engines: {node: '>= 12'} - - date-fns@3.6.0: - resolution: {integrity: sha512-fRHTG8g/Gif+kSh50gaGEdToemgfj74aRX3swtiouboip5JDLAyDE9F11nHMIcvOaXeOC6D7SpNhi7uFyB7Uww==} - - dd-cache-proxy@2.1.2: - resolution: {integrity: sha512-mdu5mIW3d4tyQPH1CakAsjbw7ScYrFBdN+tEs6ChHJLU0NTj5LlKpU6Dx4ymGUALweylruxclcrW/5rr5wbNUg==} - peerDependencies: - '@discordeno/bot': 19.0.0-next.352887c - - debug@4.3.6: - resolution: {integrity: sha512-O/09Bd4Z1fBrU4VzkhFqVgpPzaGbw6Sm9FEkBT1A/YBXQFGuuSxa1dN2nxgxS34JmKXqYx8CZAwEVoJFImUXIg==} - engines: {node: '>=6.0'} - peerDependencies: - supports-color: '*' - peerDependenciesMeta: - supports-color: - optional: true - - decamelize@4.0.0: - resolution: {integrity: sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==} - engines: {node: '>=10'} - - deep-eql@5.0.2: - resolution: {integrity: sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==} - engines: {node: '>=6'} - - define-data-property@1.1.4: - resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==} - engines: {node: '>= 0.4'} - - diff@4.0.2: - resolution: {integrity: sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==} - engines: {node: '>=0.3.1'} - - diff@5.2.0: - resolution: {integrity: sha512-uIFDxqpRZGZ6ThOk84hEfqWoHx2devRFvpTZcTHur85vImfaxUbTW9Ryh4CpCuDnToOP1CEtXKIgytHBPVff5A==} - engines: {node: '>=0.3.1'} - - discordeno@19.0.0-next.b3a8c86: - resolution: {integrity: sha512-n0IF5nlP5ECPaNsBHpuvmqgmV4AFlNr7DmbTNRX5jnh01b5z6YvSMmzAT1Um3YAIb/JI3krvJjg6+C1Mxy2AEA==} - hasBin: true - - dotenv@16.4.5: - resolution: {integrity: sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==} - engines: {node: '>=12'} - - emoji-regex@8.0.0: - resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==} - - error-ex@1.3.2: - resolution: {integrity: sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==} - - es-define-property@1.0.0: - resolution: {integrity: sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==} - engines: {node: '>= 0.4'} - - es-errors@1.3.0: - resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==} - engines: {node: '>= 0.4'} - - esbuild@0.23.1: - resolution: {integrity: sha512-VVNz/9Sa0bs5SELtn3f7qhJCDPCF5oMEl5cO9/SSinpE9hbPVvxbd572HH5AKiP7WD8INO53GgfDDhRjkylHEg==} - engines: {node: '>=18'} - hasBin: true - - escalade@3.2.0: - resolution: {integrity: sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==} - engines: {node: '>=6'} - - escape-string-regexp@1.0.5: - resolution: {integrity: sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==} - engines: {node: '>=0.8.0'} - - escape-string-regexp@4.0.0: - resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} - engines: {node: '>=10'} - - fetch-blob@3.2.0: - resolution: {integrity: sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ==} - engines: {node: ^12.20 || >= 14.13} - - fill-range@7.1.1: - resolution: {integrity: sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==} - engines: {node: '>=8'} - - find-up@5.0.0: - resolution: {integrity: sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==} - engines: {node: '>=10'} - - find-up@7.0.0: - resolution: {integrity: sha512-YyZM99iHrqLKjmt4LJDj58KI+fYyufRLBSYcqycxf//KpBk9FoewoGX0450m9nB44qrZnovzC2oeP5hUibxc/g==} - engines: {node: '>=18'} - - flat@5.0.2: - resolution: {integrity: sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==} - hasBin: true - - formdata-polyfill@4.0.10: - resolution: {integrity: sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g==} - engines: {node: '>=12.20.0'} - - fs.realpath@1.0.0: - resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==} - - fsevents@2.3.3: - resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} - engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} - os: [darwin] - - function-bind@1.1.2: - resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} - - get-caller-file@2.0.5: - resolution: {integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==} - engines: {node: 6.* || 8.* || >= 10.*} - - get-func-name@2.0.2: - resolution: {integrity: sha512-8vXOvuE167CtIc3OyItco7N/dpRtBbYOsPsXCz7X/PMnlGjYjSGuZJgM1Y7mmew7BKf9BqvLX2tnOVy1BBUsxQ==} - - get-intrinsic@1.2.4: - resolution: {integrity: sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==} - engines: {node: '>= 0.4'} - - get-tsconfig@4.8.0: - resolution: {integrity: sha512-Pgba6TExTZ0FJAn1qkJAjIeKoDJ3CsI2ChuLohJnZl/tTU8MVrq3b+2t5UOPfRa4RMsorClBjJALkJUMjG1PAw==} - - glob-parent@5.1.2: - resolution: {integrity: sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==} - engines: {node: '>= 6'} - - glob@8.1.0: - resolution: {integrity: sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==} - engines: {node: '>=12'} - deprecated: Glob versions prior to v9 are no longer supported - - gopd@1.0.1: - resolution: {integrity: sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==} - - graphile-config@0.0.1-beta.9: - resolution: {integrity: sha512-7vNxXZ24OAgXxDKXYi9JtgWPMuNbBL3057Yf32Ux+/rVP4+EePgySCc+NNnn0tORi8qwqVreN8bdWqGIcSwNXg==} - engines: {node: '>=16'} - - graphile-worker@0.16.6: - resolution: {integrity: sha512-e7gGYDmGqzju2l83MpzX8vNG/lOtVJiSzI3eZpAFubSxh/cxs7sRrRGBGjzBP1kNG0H+c95etPpNRNlH65PYhw==} - engines: {node: '>=14.0.0'} - hasBin: true - - has-flag@3.0.0: - resolution: {integrity: sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==} - engines: {node: '>=4'} - - has-flag@4.0.0: - resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} - engines: {node: '>=8'} - - has-property-descriptors@1.0.2: - resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==} - - has-proto@1.0.3: - resolution: {integrity: sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==} - engines: {node: '>= 0.4'} - - has-symbols@1.0.3: - resolution: {integrity: sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==} - engines: {node: '>= 0.4'} - - hasown@2.0.2: - resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} - engines: {node: '>= 0.4'} - - he@1.2.0: - resolution: {integrity: sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==} - hasBin: true - - ignore-by-default@1.0.1: - resolution: {integrity: sha512-Ius2VYcGNk7T90CppJqcIkS5ooHUZyIQK+ClZfMfMNFEF9VSE73Fq+906u/CWu92x4gzZMWOwfFYckPObzdEbA==} - - import-fresh@3.3.0: - resolution: {integrity: sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==} - engines: {node: '>=6'} - - inflight@1.0.6: - resolution: {integrity: sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==} - deprecated: This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful. - - inherits@2.0.4: - resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} - - interpret@3.1.1: - resolution: {integrity: sha512-6xwYfHbajpoF0xLW+iwLkhwgvLoZDfjYfoFNu8ftMoXINzwuymNLd9u/KmwtdT2GbR+/Cz66otEGEVVUHX9QLQ==} - engines: {node: '>=10.13.0'} - - is-arrayish@0.2.1: - resolution: {integrity: sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==} - - is-binary-path@2.1.0: - resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==} - engines: {node: '>=8'} - - is-extglob@2.1.1: - resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} - engines: {node: '>=0.10.0'} - - is-fullwidth-code-point@3.0.0: - resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==} - engines: {node: '>=8'} - - is-glob@4.0.3: - resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==} - engines: {node: '>=0.10.0'} - - is-network-error@1.1.0: - resolution: {integrity: sha512-tUdRRAnhT+OtCZR/LxZelH/C7QtjtFrTu5tXCA8pl55eTUElUHT+GPYV8MBMBvea/j+NxQqVt3LbWMRir7Gx9g==} - engines: {node: '>=16'} - - is-number@7.0.0: - resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==} - engines: {node: '>=0.12.0'} - - is-plain-obj@2.1.0: - resolution: {integrity: sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==} - engines: {node: '>=8'} - - is-unicode-supported@0.1.0: - resolution: {integrity: sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==} - engines: {node: '>=10'} - - js-tokens@4.0.0: - resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} - - js-yaml@4.1.0: - resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==} - hasBin: true - - json-parse-even-better-errors@2.3.1: - resolution: {integrity: sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==} - - json5@2.2.3: - resolution: {integrity: sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==} - engines: {node: '>=6'} - hasBin: true - - lines-and-columns@1.2.4: - resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} - - locate-path@6.0.0: - resolution: {integrity: sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==} - engines: {node: '>=10'} - - locate-path@7.2.0: - resolution: {integrity: sha512-gvVijfZvn7R+2qyPX8mAuKcFGDf6Nc61GdvGafQsHL0sBIxfKzA+usWn4GFC/bk+QdwPUD4kWFJLhElipq+0VA==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - log-symbols@4.1.0: - resolution: {integrity: sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==} - engines: {node: '>=10'} - - loupe@3.1.1: - resolution: {integrity: sha512-edNu/8D5MKVfGVFRhFf8aAxiTM6Wumfz5XsaatSxlD3w4R1d/WEKUTydCdPGbl9K7QG/Ca3GnDV2sIKIpXRQcw==} - - make-error@1.3.6: - resolution: {integrity: sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==} - - minimatch@3.1.2: - resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==} - - minimatch@5.1.6: - resolution: {integrity: sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==} - engines: {node: '>=10'} - - mocha@10.7.3: - resolution: {integrity: sha512-uQWxAu44wwiACGqjbPYmjo7Lg8sFrS3dQe7PP2FQI+woptP4vZXSMcfMyFL/e1yFEeEpV4RtyTpZROOKmxis+A==} - engines: {node: '>= 14.0.0'} - hasBin: true - - ms@2.1.2: - resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==} - - ms@2.1.3: - resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} - - node-domexception@1.0.0: - resolution: {integrity: sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==} - engines: {node: '>=10.5.0'} - - node-fetch@3.3.2: - resolution: {integrity: sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - nodemon@3.1.4: - resolution: {integrity: sha512-wjPBbFhtpJwmIeY2yP7QF+UKzPfltVGtfce1g/bB15/8vCGZj8uxD62b/b9M9/WVgme0NZudpownKN+c0plXlQ==} - engines: {node: '>=10'} - hasBin: true - - normalize-path@3.0.0: - resolution: {integrity: sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==} - engines: {node: '>=0.10.0'} - - object-inspect@1.13.2: - resolution: {integrity: sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==} - engines: {node: '>= 0.4'} - - obuf@1.1.2: - resolution: {integrity: sha512-PX1wu0AmAdPqOL1mWhqmlOd8kOIZQwGZw6rh7uby9fTc5lhaOWFLX3I6R1hrF9k3zUY40e6igsLGkDXK92LJNg==} - - once@1.4.0: - resolution: {integrity: sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==} - - p-limit@3.1.0: - resolution: {integrity: sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==} - engines: {node: '>=10'} - - p-limit@4.0.0: - resolution: {integrity: sha512-5b0R4txpzjPWVw/cXXUResoD4hb6U/x9BH08L7nw+GN1sezDzPdxeRvpc9c433fZhBan/wusjbCsqwqm4EIBIQ==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - p-locate@5.0.0: - resolution: {integrity: sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==} - engines: {node: '>=10'} - - p-locate@6.0.0: - resolution: {integrity: sha512-wPrq66Llhl7/4AGC6I+cqxT07LhXvWL08LNXz1fENOw0Ap4sRZZ/gZpTTJ5jpurzzzfS2W/Ge9BY3LgLjCShcw==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - p-retry@6.2.0: - resolution: {integrity: sha512-JA6nkq6hKyWLLasXQXUrO4z8BUZGUt/LjlJxx8Gb2+2ntodU/SS63YZ8b0LUTbQ8ZB9iwOfhEPhg4ykKnn2KsA==} - engines: {node: '>=16.17'} - - parent-module@1.0.1: - resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} - engines: {node: '>=6'} - - parse-json@5.2.0: - resolution: {integrity: sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==} - engines: {node: '>=8'} - - path-exists@4.0.0: - resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==} - engines: {node: '>=8'} - - path-exists@5.0.0: - resolution: {integrity: sha512-RjhtfwJOxzcFmNOi6ltcbcu4Iu+FL3zEj83dk4kAS+fVpTxXLO1b38RvJgT/0QwvV/L3aY9TAnyv0EOqW4GoMQ==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - path-type@4.0.0: - resolution: {integrity: sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==} - engines: {node: '>=8'} - - pathval@2.0.0: - resolution: {integrity: sha512-vE7JKRyES09KiunauX7nd2Q9/L7lhok4smP9RZTDeD4MVs72Dp2qNFVz39Nz5a0FVEW0BJR6C0DYrq6unoziZA==} - engines: {node: '>= 14.16'} - - pg-cloudflare@1.1.1: - resolution: {integrity: sha512-xWPagP/4B6BgFO+EKz3JONXv3YDgvkbVrGw2mTo3D6tVDQRh1e7cqVGvyR3BE+eQgAvx1XhW/iEASj4/jCWl3Q==} - - pg-connection-string@2.6.4: - resolution: {integrity: sha512-v+Z7W/0EO707aNMaAEfiGnGL9sxxumwLl2fJvCQtMn9Fxsg+lPpPkdcyBSv/KFgpGdYkMfn+EI1Or2EHjpgLCA==} - - pg-int8@1.0.1: - resolution: {integrity: sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw==} - engines: {node: '>=4.0.0'} - - pg-numeric@1.0.2: - resolution: {integrity: sha512-BM/Thnrw5jm2kKLE5uJkXqqExRUY/toLHda65XgFTBTFYZyopbKjBe29Ii3RbkvlsMoFwD+tHeGaCjjv0gHlyw==} - engines: {node: '>=4'} - - pg-pool@3.6.2: - resolution: {integrity: sha512-Htjbg8BlwXqSBQ9V8Vjtc+vzf/6fVUuak/3/XXKA9oxZprwW3IMDQTGHP+KDmVL7rtd+R1QjbnCFPuTHm3G4hg==} - peerDependencies: - pg: '>=8.0' - - pg-protocol@1.6.1: - resolution: {integrity: sha512-jPIlvgoD63hrEuihvIg+tJhoGjUsLPn6poJY9N5CnlPd91c2T18T/9zBtLxZSb1EhYxBRoZJtzScCaWlYLtktg==} - - pg-types@2.2.0: - resolution: {integrity: sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==} - engines: {node: '>=4'} - - pg-types@4.0.2: - resolution: {integrity: sha512-cRL3JpS3lKMGsKaWndugWQoLOCoP+Cic8oseVcbr0qhPzYD5DWXK+RZ9LY9wxRf7RQia4SCwQlXk0q6FCPrVng==} - engines: {node: '>=10'} - - pg@8.12.0: - resolution: {integrity: sha512-A+LHUSnwnxrnL/tZ+OLfqR1SxLN3c/pgDztZ47Rpbsd4jUytsTtwQo/TLPRzPJMp/1pbhYVhH9cuSZLAajNfjQ==} - engines: {node: '>= 8.0.0'} - peerDependencies: - pg-native: '>=3.0.1' - peerDependenciesMeta: - pg-native: - optional: true - - pgpass@1.0.5: - resolution: {integrity: sha512-FdW9r/jQZhSeohs1Z3sI1yxFQNFvMcnmfuj4WBMUTxOrAyLMaTcE1aAMBiTlbMNaXvBCQuVi0R7hd8udDSP7ug==} - - picocolors@1.1.0: - resolution: {integrity: sha512-TQ92mBOW0l3LeMeyLV6mzy/kWr8lkd/hp3mTg7wYK7zJhuBStmGMBG0BdeDZS/dZx1IukaX6Bk11zcln25o1Aw==} - - picomatch@2.3.1: - resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==} - engines: {node: '>=8.6'} - - postgres-array@2.0.0: - resolution: {integrity: sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA==} - engines: {node: '>=4'} - - postgres-array@3.0.2: - resolution: {integrity: sha512-6faShkdFugNQCLwucjPcY5ARoW1SlbnrZjmGl0IrrqewpvxvhSLHimCVzqeuULCbG0fQv7Dtk1yDbG3xv7Veog==} - engines: {node: '>=12'} - - postgres-bytea@1.0.0: - resolution: {integrity: sha512-xy3pmLuQqRBZBXDULy7KbaitYqLcmxigw14Q5sj8QBVLqEwXfeybIKVWiqAXTlcvdvb0+xkOtDbfQMOf4lST1w==} - engines: {node: '>=0.10.0'} - - postgres-bytea@3.0.0: - resolution: {integrity: sha512-CNd4jim9RFPkObHSjVHlVrxoVQXz7quwNFpz7RY1okNNme49+sVyiTvTRobiLV548Hx/hb1BG+iE7h9493WzFw==} - engines: {node: '>= 6'} - - postgres-date@1.0.7: - resolution: {integrity: sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q==} - engines: {node: '>=0.10.0'} - - postgres-date@2.1.0: - resolution: {integrity: sha512-K7Juri8gtgXVcDfZttFKVmhglp7epKb1K4pgrkLxehjqkrgPhfG6OO8LHLkfaqkbpjNRnra018XwAr1yQFWGcA==} - engines: {node: '>=12'} - - postgres-interval@1.2.0: - resolution: {integrity: sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==} - engines: {node: '>=0.10.0'} - - postgres-interval@3.0.0: - resolution: {integrity: sha512-BSNDnbyZCXSxgA+1f5UU2GmwhoI0aU5yMxRGO8CdFEcY2BQF9xm/7MqKnYoM1nJDk8nONNWDk9WeSmePFhQdlw==} - engines: {node: '>=12'} - - postgres-range@1.1.4: - resolution: {integrity: sha512-i/hbxIE9803Alj/6ytL7UHQxRvZkI9O4Sy+J3HGc4F4oo/2eQAjTSNJ0bfxyse3bH0nuVesCk+3IRLaMtG3H6w==} - - pretty-bytes@6.1.1: - resolution: {integrity: sha512-mQUvGU6aUFQ+rNvTIAcZuWGRT9a6f6Yrg9bHs4ImKF+HZCEK+plBvnAZYSIQztknZF2qnzNtr6F8s0+IuptdlQ==} - engines: {node: ^14.13.1 || >=16.0.0} - - pstree.remy@1.1.8: - resolution: {integrity: sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==} - - qs@6.13.0: - resolution: {integrity: sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==} - engines: {node: '>=0.6'} - - randombytes@2.1.0: - resolution: {integrity: sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==} - - rate-limiter-flexible@5.0.3: - resolution: {integrity: sha512-lWx2y8NBVlTOLPyqs+6y7dxfEpT6YFqKy3MzWbCy95sTTOhOuxufP2QvRyOHpfXpB9OUJPbVLybw3z3AVAS5fA==} - - readdirp@3.6.0: - resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==} - engines: {node: '>=8.10.0'} - - require-directory@2.1.1: - resolution: {integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==} - engines: {node: '>=0.10.0'} - - resolve-from@4.0.0: - resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==} - engines: {node: '>=4'} - - resolve-pkg-maps@1.0.0: - resolution: {integrity: sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==} - - retry@0.13.1: - resolution: {integrity: sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg==} - engines: {node: '>= 4'} - - safe-buffer@5.2.1: - resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} - - semver@7.6.3: - resolution: {integrity: sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==} - engines: {node: '>=10'} - hasBin: true - - serialize-javascript@6.0.2: - resolution: {integrity: sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g==} - - set-function-length@1.2.2: - resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==} - engines: {node: '>= 0.4'} - - side-channel@1.0.6: - resolution: {integrity: sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==} - engines: {node: '>= 0.4'} - - simple-update-notifier@2.0.0: - resolution: {integrity: sha512-a2B9Y0KlNXl9u/vsW6sTIu9vGEpfKu2wRV6l1H3XEas/0gUIzGzBoP/IouTcUQbm9JWZLH3COxyn03TYlFax6w==} - engines: {node: '>=10'} - - split2@4.2.0: - resolution: {integrity: sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==} - engines: {node: '>= 10.x'} - - string-width@4.2.3: - resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} - engines: {node: '>=8'} - - strip-ansi@6.0.1: - resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} - engines: {node: '>=8'} - - strip-json-comments@3.1.1: - resolution: {integrity: sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==} - engines: {node: '>=8'} - - supports-color@5.5.0: - resolution: {integrity: sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==} - engines: {node: '>=4'} - - supports-color@7.2.0: - resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==} - engines: {node: '>=8'} - - supports-color@8.1.1: - resolution: {integrity: sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==} - engines: {node: '>=10'} - - to-regex-range@5.0.1: - resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} - engines: {node: '>=8.0'} - - touch@3.1.1: - resolution: {integrity: sha512-r0eojU4bI8MnHr8c5bNo7lJDdI2qXlWWJk6a9EAFG7vbhTjElYhBVS3/miuE0uOuoLdb8Mc/rVfsmm6eo5o9GA==} - hasBin: true - - ts-node@10.9.2: - resolution: {integrity: sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==} - hasBin: true - peerDependencies: - '@swc/core': '>=1.2.50' - '@swc/wasm': '>=1.2.50' - '@types/node': '*' - typescript: '>=2.7' - peerDependenciesMeta: - '@swc/core': - optional: true - '@swc/wasm': - optional: true - - tslib@2.7.0: - resolution: {integrity: sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA==} - - tsx@4.19.0: - resolution: {integrity: sha512-bV30kM7bsLZKZIOCHeMNVMJ32/LuJzLVajkQI/qf92J2Qr08ueLQvW00PUZGiuLPP760UINwupgUj8qrSCPUKg==} - engines: {node: '>=18.0.0'} - hasBin: true - - typescript@5.5.4: - resolution: {integrity: sha512-Mtq29sKDAEYP7aljRgtPOpTvOfbwRWlS6dPRzwjdE+C0R4brX/GUyhHSecbHMFLNBLcJIPt9nl9yG5TZ1weH+Q==} - engines: {node: '>=14.17'} - hasBin: true - - undefsafe@2.0.5: - resolution: {integrity: sha512-WxONCrssBM8TSPRqN5EmsjVrsv4A8X12J4ArBiiayv3DyyG3ZlIg6yysuuSYdZsVz3TKcTg2fd//Ujd4CHV1iA==} - - undici-types@6.19.8: - resolution: {integrity: sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==} - - unicorn-magic@0.1.0: - resolution: {integrity: sha512-lRfVq8fE8gz6QMBuDM6a+LO3IAzTi05H6gCVaUpir2E1Rwpo4ZUog45KpNXKC/Mn3Yb9UDuHumeFTo9iV/D9FQ==} - engines: {node: '>=18'} - - v8-compile-cache-lib@3.0.1: - resolution: {integrity: sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==} - - web-streams-polyfill@3.3.3: - resolution: {integrity: sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw==} - engines: {node: '>= 8'} - - workerpool@6.5.1: - resolution: {integrity: sha512-Fs4dNYcsdpYSAfVxhnl1L5zTksjvOJxtC5hzMNl+1t9B8hTJTdKDyZ5ju7ztgPy+ft9tBFXoOlDNiOT9WUXZlA==} - - wrap-ansi@7.0.0: - resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} - engines: {node: '>=10'} - - wrappy@1.0.2: - resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==} - - ws@8.18.0: - resolution: {integrity: sha512-8VbfWfHLbbwu3+N6OKsOMpBdT4kXPDDB9cJk2bJ6mh9ucxdlnNvH1e+roYkKmN9Nxw2yjz7VzeO9oOz2zJ04Pw==} - engines: {node: '>=10.0.0'} - peerDependencies: - bufferutil: ^4.0.1 - utf-8-validate: '>=5.0.2' - peerDependenciesMeta: - bufferutil: - optional: true - utf-8-validate: - optional: true - - xtend@4.0.2: - resolution: {integrity: sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==} - engines: {node: '>=0.4'} - - y18n@5.0.8: - resolution: {integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==} - engines: {node: '>=10'} - - yargs-parser@20.2.9: - resolution: {integrity: sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==} - engines: {node: '>=10'} - - yargs-parser@21.1.1: - resolution: {integrity: sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==} - engines: {node: '>=12'} - - yargs-unparser@2.0.0: - resolution: {integrity: sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==} - engines: {node: '>=10'} - - yargs@16.2.0: - resolution: {integrity: sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==} - engines: {node: '>=10'} - - yargs@17.7.2: - resolution: {integrity: sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==} - engines: {node: '>=12'} - - yn@3.1.1: - resolution: {integrity: sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==} - engines: {node: '>=6'} - - yocto-queue@0.1.0: - resolution: {integrity: sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==} - engines: {node: '>=10'} - - yocto-queue@1.1.1: - resolution: {integrity: sha512-b4JR1PFR10y1mKjhHY9LaGo6tmrgjit7hxVIeAmyMw3jegXR4dhYqLaQF5zMXZxY7tLpMyJeLjr1C4rLmkVe8g==} - engines: {node: '>=12.20'} - -snapshots: - - '@babel/code-frame@7.24.7': - dependencies: - '@babel/highlight': 7.24.7 - picocolors: 1.1.0 - - '@babel/helper-validator-identifier@7.24.7': {} - - '@babel/highlight@7.24.7': - dependencies: - '@babel/helper-validator-identifier': 7.24.7 - chalk: 2.4.2 - js-tokens: 4.0.0 - picocolors: 1.1.0 - - '@cspotcode/source-map-support@0.8.1': - dependencies: - '@jridgewell/trace-mapping': 0.3.9 - - '@discordeno/bot@19.0.0-next.b1bfe94': - dependencies: - '@discordeno/gateway': 19.0.0-next.b1bfe94 - '@discordeno/rest': 19.0.0-next.b1bfe94 - '@discordeno/types': 19.0.0-next.b1bfe94 - '@discordeno/utils': 19.0.0-next.b1bfe94 - transitivePeerDependencies: - - bufferutil - - utf-8-validate - - '@discordeno/bot@19.0.0-next.b3a8c86': - dependencies: - '@discordeno/gateway': 19.0.0-next.b3a8c86 - '@discordeno/rest': 19.0.0-next.b3a8c86 - '@discordeno/types': 19.0.0-next.b3a8c86 - '@discordeno/utils': 19.0.0-next.b3a8c86 - transitivePeerDependencies: - - bufferutil - - utf-8-validate - - '@discordeno/gateway@19.0.0-next.b1bfe94': - dependencies: - '@discordeno/types': 19.0.0-next.b1bfe94 - '@discordeno/utils': 19.0.0-next.b1bfe94 - ws: 8.18.0 - transitivePeerDependencies: - - bufferutil - - utf-8-validate - - '@discordeno/gateway@19.0.0-next.b3a8c86': - dependencies: - '@discordeno/types': 19.0.0-next.b3a8c86 - '@discordeno/utils': 19.0.0-next.b3a8c86 - ws: 8.18.0 - transitivePeerDependencies: - - bufferutil - - utf-8-validate - - '@discordeno/rest@19.0.0-next.b1bfe94': - dependencies: - '@discordeno/types': 19.0.0-next.b1bfe94 - '@discordeno/utils': 19.0.0-next.b1bfe94 - - '@discordeno/rest@19.0.0-next.b3a8c86': - dependencies: - '@discordeno/types': 19.0.0-next.b3a8c86 - '@discordeno/utils': 19.0.0-next.b3a8c86 - - '@discordeno/types@19.0.0-next.b1bfe94': {} - - '@discordeno/types@19.0.0-next.b3a8c86': {} - - '@discordeno/utils@19.0.0-next.b1bfe94': - dependencies: - '@discordeno/types': 19.0.0-next.b1bfe94 - - '@discordeno/utils@19.0.0-next.b3a8c86': - dependencies: - '@discordeno/types': 19.0.0-next.b3a8c86 - - '@esbuild/aix-ppc64@0.23.1': - optional: true - - '@esbuild/android-arm64@0.23.1': - optional: true - - '@esbuild/android-arm@0.23.1': - optional: true - - '@esbuild/android-x64@0.23.1': - optional: true - - '@esbuild/darwin-arm64@0.23.1': - optional: true - - '@esbuild/darwin-x64@0.23.1': - optional: true - - '@esbuild/freebsd-arm64@0.23.1': - optional: true - - '@esbuild/freebsd-x64@0.23.1': - optional: true - - '@esbuild/linux-arm64@0.23.1': - optional: true - - '@esbuild/linux-arm@0.23.1': - optional: true - - '@esbuild/linux-ia32@0.23.1': - optional: true - - '@esbuild/linux-loong64@0.23.1': - optional: true - - '@esbuild/linux-mips64el@0.23.1': - optional: true - - '@esbuild/linux-ppc64@0.23.1': - optional: true - - '@esbuild/linux-riscv64@0.23.1': - optional: true - - '@esbuild/linux-s390x@0.23.1': - optional: true - - '@esbuild/linux-x64@0.23.1': - optional: true - - '@esbuild/netbsd-x64@0.23.1': - optional: true - - '@esbuild/openbsd-arm64@0.23.1': - optional: true - - '@esbuild/openbsd-x64@0.23.1': - optional: true - - '@esbuild/sunos-x64@0.23.1': - optional: true - - '@esbuild/win32-arm64@0.23.1': - optional: true - - '@esbuild/win32-ia32@0.23.1': - optional: true - - '@esbuild/win32-x64@0.23.1': - optional: true - - '@graphile/logger@0.2.0': {} - - '@jridgewell/resolve-uri@3.1.2': {} - - '@jridgewell/sourcemap-codec@1.5.0': {} - - '@jridgewell/trace-mapping@0.3.9': - dependencies: - '@jridgewell/resolve-uri': 3.1.2 - '@jridgewell/sourcemap-codec': 1.5.0 - - '@noble/hashes@1.5.0': {} - - '@paralleldrive/cuid2@2.2.2': - dependencies: - '@noble/hashes': 1.5.0 - - '@tsconfig/node10@1.0.11': {} - - '@tsconfig/node12@1.0.11': {} - - '@tsconfig/node14@1.0.3': {} - - '@tsconfig/node16@1.0.4': {} - - '@types/chai@4.3.19': {} - - '@types/debug@4.1.12': - dependencies: - '@types/ms': 0.7.34 - - '@types/interpret@1.1.3': - dependencies: - '@types/node': 22.5.2 - - '@types/mocha@10.0.7': {} - - '@types/ms@0.7.34': {} - - '@types/node@20.16.3': - dependencies: - undici-types: 6.19.8 - - '@types/node@22.5.2': - dependencies: - undici-types: 6.19.8 - - '@types/pg@8.11.8': - dependencies: - '@types/node': 22.5.2 - pg-protocol: 1.6.1 - pg-types: 4.0.2 - - '@types/qs@6.9.15': {} - - '@types/retry@0.12.2': {} - - '@types/semver@7.5.8': {} - - acorn-walk@8.3.3: - dependencies: - acorn: 8.12.1 - - acorn@8.12.1: {} - - ansi-colors@4.1.3: {} - - ansi-regex@5.0.1: {} - - ansi-styles@3.2.1: - dependencies: - color-convert: 1.9.3 - - ansi-styles@4.3.0: - dependencies: - color-convert: 2.0.1 - - anymatch@3.1.3: - dependencies: - normalize-path: 3.0.0 - picomatch: 2.3.1 - - arg@4.1.3: {} - - argparse@2.0.1: {} - - assertion-error@2.0.1: {} - - balanced-match@1.0.2: {} - - binary-extensions@2.3.0: {} - - brace-expansion@1.1.11: - dependencies: - balanced-match: 1.0.2 - concat-map: 0.0.1 - - brace-expansion@2.0.1: - dependencies: - balanced-match: 1.0.2 - - braces@3.0.3: - dependencies: - fill-range: 7.1.1 - - browser-stdout@1.3.1: {} - - call-bind@1.0.7: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - set-function-length: 1.2.2 - - callsites@3.1.0: {} - - camelcase@6.3.0: {} - - chai@5.1.1: - dependencies: - assertion-error: 2.0.1 - check-error: 2.1.1 - deep-eql: 5.0.2 - loupe: 3.1.1 - pathval: 2.0.0 - - chalk@2.4.2: - dependencies: - ansi-styles: 3.2.1 - escape-string-regexp: 1.0.5 - supports-color: 5.5.0 - - chalk@4.1.2: - dependencies: - ansi-styles: 4.3.0 - supports-color: 7.2.0 - - check-error@2.1.1: {} - - chokidar@3.6.0: - dependencies: - anymatch: 3.1.3 - braces: 3.0.3 - glob-parent: 5.1.2 - is-binary-path: 2.1.0 - is-glob: 4.0.3 - normalize-path: 3.0.0 - readdirp: 3.6.0 - optionalDependencies: - fsevents: 2.3.3 - - cliui@7.0.4: - dependencies: - string-width: 4.2.3 - strip-ansi: 6.0.1 - wrap-ansi: 7.0.0 - - cliui@8.0.1: - dependencies: - string-width: 4.2.3 - strip-ansi: 6.0.1 - wrap-ansi: 7.0.0 - - color-convert@1.9.3: - dependencies: - color-name: 1.1.3 - - color-convert@2.0.1: - dependencies: - color-name: 1.1.4 - - color-name@1.1.3: {} - - color-name@1.1.4: {} - - commander@12.1.0: {} - - concat-map@0.0.1: {} - - cosmiconfig@8.3.6(typescript@5.5.4): - dependencies: - import-fresh: 3.3.0 - js-yaml: 4.1.0 - parse-json: 5.2.0 - path-type: 4.0.0 - optionalDependencies: - typescript: 5.5.4 - - create-require@1.1.1: {} - - data-uri-to-buffer@4.0.1: {} - - date-fns@3.6.0: {} - - dd-cache-proxy@2.1.2(@discordeno/bot@19.0.0-next.b1bfe94): - dependencies: - '@discordeno/bot': 19.0.0-next.b1bfe94 - - debug@4.3.6: - dependencies: - ms: 2.1.2 - - debug@4.3.6(supports-color@5.5.0): - dependencies: - ms: 2.1.2 - optionalDependencies: - supports-color: 5.5.0 - - debug@4.3.6(supports-color@8.1.1): - dependencies: - ms: 2.1.2 - optionalDependencies: - supports-color: 8.1.1 - - decamelize@4.0.0: {} - - deep-eql@5.0.2: {} - - define-data-property@1.1.4: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - gopd: 1.0.1 - - diff@4.0.2: {} - - diff@5.2.0: {} - - discordeno@19.0.0-next.b3a8c86: - dependencies: - '@discordeno/bot': 19.0.0-next.b3a8c86 - '@discordeno/gateway': 19.0.0-next.b3a8c86 - '@discordeno/rest': 19.0.0-next.b3a8c86 - '@discordeno/types': 19.0.0-next.b3a8c86 - '@discordeno/utils': 19.0.0-next.b3a8c86 - commander: 12.1.0 - find-up: 7.0.0 - typescript: 5.5.4 - transitivePeerDependencies: - - bufferutil - - utf-8-validate - - dotenv@16.4.5: {} - - emoji-regex@8.0.0: {} - - error-ex@1.3.2: - dependencies: - is-arrayish: 0.2.1 - - es-define-property@1.0.0: - dependencies: - get-intrinsic: 1.2.4 - - es-errors@1.3.0: {} - - esbuild@0.23.1: - optionalDependencies: - '@esbuild/aix-ppc64': 0.23.1 - '@esbuild/android-arm': 0.23.1 - '@esbuild/android-arm64': 0.23.1 - '@esbuild/android-x64': 0.23.1 - '@esbuild/darwin-arm64': 0.23.1 - '@esbuild/darwin-x64': 0.23.1 - '@esbuild/freebsd-arm64': 0.23.1 - '@esbuild/freebsd-x64': 0.23.1 - '@esbuild/linux-arm': 0.23.1 - '@esbuild/linux-arm64': 0.23.1 - '@esbuild/linux-ia32': 0.23.1 - '@esbuild/linux-loong64': 0.23.1 - '@esbuild/linux-mips64el': 0.23.1 - '@esbuild/linux-ppc64': 0.23.1 - '@esbuild/linux-riscv64': 0.23.1 - '@esbuild/linux-s390x': 0.23.1 - '@esbuild/linux-x64': 0.23.1 - '@esbuild/netbsd-x64': 0.23.1 - '@esbuild/openbsd-arm64': 0.23.1 - '@esbuild/openbsd-x64': 0.23.1 - '@esbuild/sunos-x64': 0.23.1 - '@esbuild/win32-arm64': 0.23.1 - '@esbuild/win32-ia32': 0.23.1 - '@esbuild/win32-x64': 0.23.1 - - escalade@3.2.0: {} - - escape-string-regexp@1.0.5: {} - - escape-string-regexp@4.0.0: {} - - fetch-blob@3.2.0: - dependencies: - node-domexception: 1.0.0 - web-streams-polyfill: 3.3.3 - - fill-range@7.1.1: - dependencies: - to-regex-range: 5.0.1 - - find-up@5.0.0: - dependencies: - locate-path: 6.0.0 - path-exists: 4.0.0 - - find-up@7.0.0: - dependencies: - locate-path: 7.2.0 - path-exists: 5.0.0 - unicorn-magic: 0.1.0 - - flat@5.0.2: {} - - formdata-polyfill@4.0.10: - dependencies: - fetch-blob: 3.2.0 - - fs.realpath@1.0.0: {} - - fsevents@2.3.3: - optional: true - - function-bind@1.1.2: {} - - get-caller-file@2.0.5: {} - - get-func-name@2.0.2: {} - - get-intrinsic@1.2.4: - dependencies: - es-errors: 1.3.0 - function-bind: 1.1.2 - has-proto: 1.0.3 - has-symbols: 1.0.3 - hasown: 2.0.2 - - get-tsconfig@4.8.0: - dependencies: - resolve-pkg-maps: 1.0.0 - - glob-parent@5.1.2: - dependencies: - is-glob: 4.0.3 - - glob@8.1.0: - dependencies: - fs.realpath: 1.0.0 - inflight: 1.0.6 - inherits: 2.0.4 - minimatch: 5.1.6 - once: 1.4.0 - - gopd@1.0.1: - dependencies: - get-intrinsic: 1.2.4 - - graphile-config@0.0.1-beta.9: - dependencies: - '@types/interpret': 1.1.3 - '@types/node': 20.16.3 - '@types/semver': 7.5.8 - chalk: 4.1.2 - debug: 4.3.6 - interpret: 3.1.1 - semver: 7.6.3 - tslib: 2.7.0 - yargs: 17.7.2 - transitivePeerDependencies: - - supports-color - - graphile-worker@0.16.6(typescript@5.5.4): - dependencies: - '@graphile/logger': 0.2.0 - '@types/debug': 4.1.12 - '@types/pg': 8.11.8 - cosmiconfig: 8.3.6(typescript@5.5.4) - graphile-config: 0.0.1-beta.9 - json5: 2.2.3 - pg: 8.12.0 - tslib: 2.7.0 - yargs: 17.7.2 - transitivePeerDependencies: - - pg-native - - supports-color - - typescript - - has-flag@3.0.0: {} - - has-flag@4.0.0: {} - - has-property-descriptors@1.0.2: - dependencies: - es-define-property: 1.0.0 - - has-proto@1.0.3: {} - - has-symbols@1.0.3: {} - - hasown@2.0.2: - dependencies: - function-bind: 1.1.2 - - he@1.2.0: {} - - ignore-by-default@1.0.1: {} - - import-fresh@3.3.0: - dependencies: - parent-module: 1.0.1 - resolve-from: 4.0.0 - - inflight@1.0.6: - dependencies: - once: 1.4.0 - wrappy: 1.0.2 - - inherits@2.0.4: {} - - interpret@3.1.1: {} - - is-arrayish@0.2.1: {} - - is-binary-path@2.1.0: - dependencies: - binary-extensions: 2.3.0 - - is-extglob@2.1.1: {} - - is-fullwidth-code-point@3.0.0: {} - - is-glob@4.0.3: - dependencies: - is-extglob: 2.1.1 - - is-network-error@1.1.0: {} - - is-number@7.0.0: {} - - is-plain-obj@2.1.0: {} - - is-unicode-supported@0.1.0: {} - - js-tokens@4.0.0: {} - - js-yaml@4.1.0: - dependencies: - argparse: 2.0.1 - - json-parse-even-better-errors@2.3.1: {} - - json5@2.2.3: {} - - lines-and-columns@1.2.4: {} - - locate-path@6.0.0: - dependencies: - p-locate: 5.0.0 - - locate-path@7.2.0: - dependencies: - p-locate: 6.0.0 - - log-symbols@4.1.0: - dependencies: - chalk: 4.1.2 - is-unicode-supported: 0.1.0 - - loupe@3.1.1: - dependencies: - get-func-name: 2.0.2 - - make-error@1.3.6: {} - - minimatch@3.1.2: - dependencies: - brace-expansion: 1.1.11 - - minimatch@5.1.6: - dependencies: - brace-expansion: 2.0.1 - - mocha@10.7.3: - dependencies: - ansi-colors: 4.1.3 - browser-stdout: 1.3.1 - chokidar: 3.6.0 - debug: 4.3.6(supports-color@8.1.1) - diff: 5.2.0 - escape-string-regexp: 4.0.0 - find-up: 5.0.0 - glob: 8.1.0 - he: 1.2.0 - js-yaml: 4.1.0 - log-symbols: 4.1.0 - minimatch: 5.1.6 - ms: 2.1.3 - serialize-javascript: 6.0.2 - strip-json-comments: 3.1.1 - supports-color: 8.1.1 - workerpool: 6.5.1 - yargs: 16.2.0 - yargs-parser: 20.2.9 - yargs-unparser: 2.0.0 - - ms@2.1.2: {} - - ms@2.1.3: {} - - node-domexception@1.0.0: {} - - node-fetch@3.3.2: - dependencies: - data-uri-to-buffer: 4.0.1 - fetch-blob: 3.2.0 - formdata-polyfill: 4.0.10 - - nodemon@3.1.4: - dependencies: - chokidar: 3.6.0 - debug: 4.3.6(supports-color@5.5.0) - ignore-by-default: 1.0.1 - minimatch: 3.1.2 - pstree.remy: 1.1.8 - semver: 7.6.3 - simple-update-notifier: 2.0.0 - supports-color: 5.5.0 - touch: 3.1.1 - undefsafe: 2.0.5 - - normalize-path@3.0.0: {} - - object-inspect@1.13.2: {} - - obuf@1.1.2: {} - - once@1.4.0: - dependencies: - wrappy: 1.0.2 - - p-limit@3.1.0: - dependencies: - yocto-queue: 0.1.0 - - p-limit@4.0.0: - dependencies: - yocto-queue: 1.1.1 - - p-locate@5.0.0: - dependencies: - p-limit: 3.1.0 - - p-locate@6.0.0: - dependencies: - p-limit: 4.0.0 - - p-retry@6.2.0: - dependencies: - '@types/retry': 0.12.2 - is-network-error: 1.1.0 - retry: 0.13.1 - - parent-module@1.0.1: - dependencies: - callsites: 3.1.0 - - parse-json@5.2.0: - dependencies: - '@babel/code-frame': 7.24.7 - error-ex: 1.3.2 - json-parse-even-better-errors: 2.3.1 - lines-and-columns: 1.2.4 - - path-exists@4.0.0: {} - - path-exists@5.0.0: {} - - path-type@4.0.0: {} - - pathval@2.0.0: {} - - pg-cloudflare@1.1.1: - optional: true - - pg-connection-string@2.6.4: {} - - pg-int8@1.0.1: {} - - pg-numeric@1.0.2: {} - - pg-pool@3.6.2(pg@8.12.0): - dependencies: - pg: 8.12.0 - - pg-protocol@1.6.1: {} - - pg-types@2.2.0: - dependencies: - pg-int8: 1.0.1 - postgres-array: 2.0.0 - postgres-bytea: 1.0.0 - postgres-date: 1.0.7 - postgres-interval: 1.2.0 - - pg-types@4.0.2: - dependencies: - pg-int8: 1.0.1 - pg-numeric: 1.0.2 - postgres-array: 3.0.2 - postgres-bytea: 3.0.0 - postgres-date: 2.1.0 - postgres-interval: 3.0.0 - postgres-range: 1.1.4 - - pg@8.12.0: - dependencies: - pg-connection-string: 2.6.4 - pg-pool: 3.6.2(pg@8.12.0) - pg-protocol: 1.6.1 - pg-types: 2.2.0 - pgpass: 1.0.5 - optionalDependencies: - pg-cloudflare: 1.1.1 - - pgpass@1.0.5: - dependencies: - split2: 4.2.0 - - picocolors@1.1.0: {} - - picomatch@2.3.1: {} - - postgres-array@2.0.0: {} - - postgres-array@3.0.2: {} - - postgres-bytea@1.0.0: {} - - postgres-bytea@3.0.0: - dependencies: - obuf: 1.1.2 - - postgres-date@1.0.7: {} - - postgres-date@2.1.0: {} - - postgres-interval@1.2.0: - dependencies: - xtend: 4.0.2 - - postgres-interval@3.0.0: {} - - postgres-range@1.1.4: {} - - pretty-bytes@6.1.1: {} - - pstree.remy@1.1.8: {} - - qs@6.13.0: - dependencies: - side-channel: 1.0.6 - - randombytes@2.1.0: - dependencies: - safe-buffer: 5.2.1 - - rate-limiter-flexible@5.0.3: {} - - readdirp@3.6.0: - dependencies: - picomatch: 2.3.1 - - require-directory@2.1.1: {} - - resolve-from@4.0.0: {} - - resolve-pkg-maps@1.0.0: {} - - retry@0.13.1: {} - - safe-buffer@5.2.1: {} - - semver@7.6.3: {} - - serialize-javascript@6.0.2: - dependencies: - randombytes: 2.1.0 - - set-function-length@1.2.2: - dependencies: - define-data-property: 1.1.4 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - gopd: 1.0.1 - has-property-descriptors: 1.0.2 - - side-channel@1.0.6: - dependencies: - call-bind: 1.0.7 - es-errors: 1.3.0 - get-intrinsic: 1.2.4 - object-inspect: 1.13.2 - - simple-update-notifier@2.0.0: - dependencies: - semver: 7.6.3 - - split2@4.2.0: {} - - string-width@4.2.3: - dependencies: - emoji-regex: 8.0.0 - is-fullwidth-code-point: 3.0.0 - strip-ansi: 6.0.1 - - strip-ansi@6.0.1: - dependencies: - ansi-regex: 5.0.1 - - strip-json-comments@3.1.1: {} - - supports-color@5.5.0: - dependencies: - has-flag: 3.0.0 - - supports-color@7.2.0: - dependencies: - has-flag: 4.0.0 - - supports-color@8.1.1: - dependencies: - has-flag: 4.0.0 - - to-regex-range@5.0.1: - dependencies: - is-number: 7.0.0 - - touch@3.1.1: {} - - ts-node@10.9.2(@types/node@22.5.2)(typescript@5.5.4): - dependencies: - '@cspotcode/source-map-support': 0.8.1 - '@tsconfig/node10': 1.0.11 - '@tsconfig/node12': 1.0.11 - '@tsconfig/node14': 1.0.3 - '@tsconfig/node16': 1.0.4 - '@types/node': 22.5.2 - acorn: 8.12.1 - acorn-walk: 8.3.3 - arg: 4.1.3 - create-require: 1.1.1 - diff: 4.0.2 - make-error: 1.3.6 - typescript: 5.5.4 - v8-compile-cache-lib: 3.0.1 - yn: 3.1.1 - - tslib@2.7.0: {} - - tsx@4.19.0: - dependencies: - esbuild: 0.23.1 - get-tsconfig: 4.8.0 - optionalDependencies: - fsevents: 2.3.3 - - typescript@5.5.4: {} - - undefsafe@2.0.5: {} - - undici-types@6.19.8: {} - - unicorn-magic@0.1.0: {} - - v8-compile-cache-lib@3.0.1: {} - - web-streams-polyfill@3.3.3: {} - - workerpool@6.5.1: {} - - wrap-ansi@7.0.0: - dependencies: - ansi-styles: 4.3.0 - string-width: 4.2.3 - strip-ansi: 6.0.1 - - wrappy@1.0.2: {} - - ws@8.18.0: {} - - xtend@4.0.2: {} - - y18n@5.0.8: {} - - yargs-parser@20.2.9: {} - - yargs-parser@21.1.1: {} - - yargs-unparser@2.0.0: - dependencies: - camelcase: 6.3.0 - decamelize: 4.0.0 - flat: 5.0.2 - is-plain-obj: 2.1.0 - - yargs@16.2.0: - dependencies: - cliui: 7.0.4 - escalade: 3.2.0 - get-caller-file: 2.0.5 - require-directory: 2.1.1 - string-width: 4.2.3 - y18n: 5.0.8 - yargs-parser: 20.2.9 - - yargs@17.7.2: - dependencies: - cliui: 8.0.1 - escalade: 3.2.0 - get-caller-file: 2.0.5 - require-directory: 2.1.1 - string-width: 4.2.3 - y18n: 5.0.8 - yargs-parser: 21.1.1 - - yn@3.1.1: {} - - yocto-queue@0.1.0: {} - - yocto-queue@1.1.1: {} diff --git a/services/bot/src/bot.ts b/services/bot/src/bot.ts deleted file mode 100644 index 445684c..0000000 --- a/services/bot/src/bot.ts +++ /dev/null @@ -1,57 +0,0 @@ -import { createBot, createGatewayManager, createRestManager, Intents, type Bot } from '@discordeno/bot' -import { createProxyCache, } from 'dd-cache-proxy'; -import { configs } from './config.ts' - -export const bot = createProxyCache( - createBot({ - token: configs.token, - intents: Intents.Guilds | Intents.GuildMessages, - rest: createRestManager({ token: configs.token, applicationId: configs.discordApplicationId }), - gateway: createGatewayManager({ token: configs.token }) - }), - { - desiredProps: { - guilds: ['id', 'name', 'roles'], - roles: ['id', 'guildId', 'permissions'], - }, - cacheInMemory: { - guilds: true, - roles: true, - default: false, - }, - }, -) - -// Setup desired properties -bot.transformers.desiredProperties.interaction.id = true -bot.transformers.desiredProperties.interaction.type = true -bot.transformers.desiredProperties.interaction.data = true -bot.transformers.desiredProperties.interaction.token = true -bot.transformers.desiredProperties.interaction.guildId = true -bot.transformers.desiredProperties.interaction.member = true -bot.transformers.desiredProperties.interaction.message = true -bot.transformers.desiredProperties.interaction.user = true -bot.transformers.desiredProperties.interaction.channelId = true - -bot.transformers.desiredProperties.message.activity = true -bot.transformers.desiredProperties.message.id = true -bot.transformers.desiredProperties.message.referencedMessage = true - -bot.transformers.desiredProperties.guild.id = true -bot.transformers.desiredProperties.guild.name = true -bot.transformers.desiredProperties.guild.roles = true - -bot.transformers.desiredProperties.role.id = true -bot.transformers.desiredProperties.role.guildId = true -bot.transformers.desiredProperties.role.permissions = true - -bot.transformers.desiredProperties.member.id = true -bot.transformers.desiredProperties.member.roles = true - -bot.transformers.desiredProperties.channel.id = true - -bot.transformers.desiredProperties.user.id = true -bot.transformers.desiredProperties.user.username = true -bot.transformers.desiredProperties.user.discriminator = true - - diff --git a/services/bot/src/collector.ts b/services/bot/src/collector.ts deleted file mode 100644 index fd25474..0000000 --- a/services/bot/src/collector.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { EventEmitter } from 'node:events' -import type { Interaction } from '@discordeno/bot' - -export class ItemCollector extends EventEmitter { - onItem(callback: (item: Interaction) => unknown): void { - this.on('item', callback) - } - - collect(item: Interaction): void { - this.emit('item', item) - } -} - -export default ItemCollector \ No newline at end of file diff --git a/services/bot/src/commands.ts b/services/bot/src/commands.ts deleted file mode 100644 index 54fe19c..0000000 --- a/services/bot/src/commands.ts +++ /dev/null @@ -1,22 +0,0 @@ -// @see https://github.com/discordeno/discordeno/blob/main/examples/advanced/src/commands.ts - -import { type ApplicationCommandOption, type ApplicationCommandTypes, Collection, type Interaction } from '@discordeno/bot' - -export const commands = new Collection() - -export function createCommand(command: Command): void { - commands.set(command.name, command) -} - -export interface Command { - /** The name of this command. */ - name: string - /** What does this command do? */ - description: string - /** The type of command this is. */ - type: ApplicationCommandTypes - /** The options for this command */ - options?: ApplicationCommandOption[] - /** This will be executed when the command is run. */ - execute: (interaction: Interaction, options: Record) => unknown -} \ No newline at end of file diff --git a/services/bot/src/commands/cancel.ts b/services/bot/src/commands/cancel.ts deleted file mode 100644 index bd5faeb..0000000 --- a/services/bot/src/commands/cancel.ts +++ /dev/null @@ -1,53 +0,0 @@ - -import { ApplicationCommandTypes, type Interaction } from '@discordeno/bot' -import type { Status } from '@futureporn/types' -import { createCommand } from '../commands.ts' -import { bot } from '../bot.ts' -import { configs } from '../config.ts' -import { rbacAllow } from '../middlewares/rbac.ts' - -createCommand({ - name: 'cancel', - description: 'Cancel a recording', - type: ApplicationCommandTypes.ChatInput, - async execute(interaction: Interaction) { - bot.logger.info(`cancel command is executing now.`) - const message = interaction.message - if (!message) return bot.logger.error('interaction.message was missing'); - if (!message.id) return bot.logger.error(`interaction.message.id was missing`); - - const url = `${configs.postgrestUrl}/recordings?discord_message_id=eq.${message.id}`; - const options = { - method: 'PATCH', - headers: { - 'Content-Type': 'application/json', - 'Accept': 'application/json', - 'Prefer': 'return=representation', - 'Authorization': `Bearer ${configs.automationUserJwt}` - }, - body: JSON.stringify({ - is_aborted: true, - status: 'aborted' as Status - }) - }; - - let vodId: string; - try { - await rbacAllow(['admin', 'moderator', 'testAdmin'], interaction) - const response = await fetch(url, options); - - bot.logger.info(`response.ok=${response.ok}`) - const data: any = await response.json(); - vodId = data?.at(0).id - bot.logger.info(interaction.user); - interaction.respond(`<@${interaction.user.id}> cancelled recording on VOD ${vodId}`, { isPrivate: false }) - bot.logger.info(`Cancel command successfully ran on message.id=${message.id}`) - - } catch (error) { - bot.logger.error('error encountered while cancelling job') - bot.logger.error(error); - } - - - }, -}) diff --git a/services/bot/src/commands/donger.ts b/services/bot/src/commands/donger.ts deleted file mode 100644 index d342f60..0000000 --- a/services/bot/src/commands/donger.ts +++ /dev/null @@ -1,49 +0,0 @@ - -import { ApplicationCommandTypes, type Interaction } from '@discordeno/bot' -import { createCommand } from '../commands.ts' - - - -const dongers: string[] = [ - '( ͡ᵔ ͜ʖ ͡ᵔ )', - '¯\_(ツ)_/¯', - '(๑>ᴗ<๑)', - '(̿▀̿ ̿Ĺ̯̿̿▀̿ ̿)', - '( ͡° ͜ʖ ͡°)', - '٩(͡๏̯͡๏)۶', - 'ლ(´◉❥◉`ლ)', - '( ゚Д゚)', - 'ԅ( ͒ ۝ ͒ )ᕤ', - '( ͡ᵔ ͜ʖ ͡°)', - '( ͠° ͟ʖ ͡°)╭∩╮', - '༼ つ ❦౪❦ ༽つ', - '( ͡↑ ͜ʖ ͡↑)', - '(ভ_ ভ) ރ // ┊ \\', - 'ヽ(⌐□益□)ノ', - '༼ つ ◕‿◕ ༽つ', - 'ヽ(⚆෴⚆)ノ', - '(つ .•́ _ʖ •̀.)つ', - '༼⌐■ل͟■༽', - '┬─┬ノ( ͡° ͜ʖ ͡°ノ)', - '༼⁰o⁰;༽꒳ᵒ꒳ᵎᵎᵎ', - '( -_・) ▄︻̷̿┻̿═━一', - '【 º ᗜ º 】', - 'ᕦ(✧╭╮✧)ᕥ', - '┗( T﹏T )┛', - '(Φ ᆺ Φ)', - '(TдT)', - '☞(◉▽◉)☞' -]; - -createCommand({ - name: 'donger', - description: 'Get a free donger!', - type: ApplicationCommandTypes.ChatInput, - async execute(interaction: Interaction) { - const selectedDonger = dongers[Math.floor(Math.random()*dongers.length)] - console.log(`selectedDonger=${selectedDonger}`) - await interaction.respond({ - content: selectedDonger - }) - }, -}) \ No newline at end of file diff --git a/services/bot/src/commands/ping.ts b/services/bot/src/commands/ping.ts deleted file mode 100644 index bf448a5..0000000 --- a/services/bot/src/commands/ping.ts +++ /dev/null @@ -1,13 +0,0 @@ -import { ApplicationCommandTypes, createEmbeds, snowflakeToTimestamp, type Interaction } from '@discordeno/bot' -import { createCommand } from '../commands.ts' - -createCommand({ - name: 'ping', - description: 'See if the bot latency is okay', - type: ApplicationCommandTypes.ChatInput, - async execute(interaction: Interaction) { - const ping = Date.now() - snowflakeToTimestamp(interaction.id) - const embeds = createEmbeds().setTitle(`The bot ping is ${ping}ms`) - await interaction.respond({ embeds }) - }, -}) \ No newline at end of file diff --git a/services/bot/src/commands/process.ts b/services/bot/src/commands/process.ts deleted file mode 100644 index 6f16237..0000000 --- a/services/bot/src/commands/process.ts +++ /dev/null @@ -1,31 +0,0 @@ - -import { ApplicationCommandTypes, logger, type Interaction } from '@discordeno/bot' -import { createCommand } from '../commands.ts' -import getStreamFromDatabase from '@futureporn/fetchers/getStreamFromDatabase.ts' -import patchVod from '@futureporn/fetchers/patchVod.ts' -import { quickAddJob, type WorkerUtilsOptions } from 'graphile-worker' -import { configs } from '../config.ts' -import findVod from '@futureporn/fetchers/findVod.ts' - -function throwErr(msg: string) { - logger.error(msg) - throw new Error(msg) -} - -createCommand({ - name: 'process', - description: 'Process a vod for publishing', - type: ApplicationCommandTypes.ChatInput, - async execute(interaction: Interaction) { - const discord_message_id = String(interaction?.message?.id) - logger.info(`process command begins.`) - if (!discord_message_id) return throwErr('failed to get discord message id'); - const vod = await findVod({ discord_message_id }) - if (!vod) return throwErr('failed to get vod while finding vod to process'); - const options: WorkerUtilsOptions = { connectionString: configs.connectionString } - logger.info(`now we will quickAddJob process_video`) - await quickAddJob(options, 'process_video', { vod_id: vod.id }) - logger.info(`now we will patchVodInDatabase`) - await patchVod(vod.id, { status: 'processing' }) - }, -}) diff --git a/services/bot/src/commands/record.ts b/services/bot/src/commands/record.ts deleted file mode 100644 index 90e8b72..0000000 --- a/services/bot/src/commands/record.ts +++ /dev/null @@ -1,148 +0,0 @@ -import { - type Interaction, - type InteractionCallbackData, - ApplicationCommandOptionTypes, - ApplicationCommandTypes, - EmbedsBuilder, - logger, -} from '@discordeno/bot' -import { rbacAllow } from '../middlewares/rbac.ts' -import { createCommand } from '../commands.ts' -import { configs } from '../config.ts' -import type { StreamResponse } from '@futureporn/types' -import createVod from '@futureporn/fetchers/createVod.ts' -import findOrCreateVtuber from '@futureporn/fetchers/findOrCreateVtuber.ts' -import findOrCreateStream from '@futureporn/fetchers/findOrCreateStream.ts' -import createRecording from '@futureporn/fetchers/createRecording.ts' - - -/** - * - * Get the livestream URL from the pre-existing discord message. IDK why we do this - */ -async function getUrlFromMessage(interaction: Interaction): Promise { - const messageId = interaction.message?.id - - const pgRequestUrl = `${configs.postgrestUrl}/vods?discord_message_id=eq.${messageId}` - logger.info(`pgRequestUrl=${pgRequestUrl}`) - const requestOptions = { - method: 'GET', - headers: { - 'Authorization': `Bearer ${configs.automationUserJwt}`, - 'Content-Type': 'application/json', - 'Prefer': 'return=representation' - } - } - try { - const res = await fetch (pgRequestUrl, requestOptions) - if (!res.ok) { - const body = await res.json() - logger.error(body) - throw new Error(`Problem during getUrlFromMessage. res.status=${res.status}, res.statusText=${res.statusText}`) - } - const json = await res.json() as StreamResponse[] - const vod = json[0] - const url = vod?.url - if (!url) return null - else return url - } catch (e) { - logger.error(e) - throw e - } - - -} - -async function getUrlFromData(interaction: Interaction): Promise { - if (!interaction) throw new Error('interaction arg passed to getOptions was missing'); - const url = (interaction.data?.options?.find(o => o.name === 'url'))?.value - if (!url) return null; - return String(url) -} - - -createCommand({ - name: 'record', - description: 'Record a livestream.', - type: ApplicationCommandTypes.ChatInput, - options: [ - { - name: 'url', - description: 'URL of the livestream', - type: ApplicationCommandOptionTypes.String - }, - ], - async execute(interaction: Interaction) { - // logger.info('record command is running now`)') - // logger.info(interaction) - await interaction.defer() - - - try { - // logger.info('we are at the top of the record command. start of the try{} block. we are about to run rbacAllow.') - await rbacAllow(['admin', 'patron', 'moderator', 'testAdmin'], interaction) - // The url can come from one of two places. - // interaction.data.options, or interaction.message?.embeds - let url - url = await getUrlFromData(interaction) - logger.info(`getUrlFromData url=${url}`) - if (!url) { - url = await getUrlFromMessage(interaction) - logger.info(`getUrlFromMessage url=${url}`) - } - - logger.info(`url=${url}`) - if (!url) throw new Error('Neither the interaction data nor the message embed contained a URL.'); - - - // respond to the interaction and get a message ID which we will then add to the database Record - const embeds = new EmbedsBuilder() - .setTitle(`VOD ⋅`) - .setDescription('Waiting for a worker to start the job.') - .setFields([ - { name: 'Status', value: 'Pending', inline: true }, - { name: 'URL', value: url, inline: true } - ]) - .setColor('#808080') - - const response: InteractionCallbackData = { embeds } - const message = await interaction.edit(response) - - if (!message?.id) { - const msg = `message.id was empty, ruh roh raggy` - console.error(msg) - throw new Error(msg) - } - - const discord_message_id = message.id.toString() - const discordMessageId = discord_message_id - const date = new Date() - - await createRecording({ url, discordMessageId, date }) - - } catch (e) { - const message = `Record failed due to the following error.\n${e}` - logger.error(message) - // acknowledged interactions cannot be responded to. Instead, the message gets sent as a followup, which is a public message. - // we want the message to be ephemeral/isPrivate, so we send our own message flagged as ephemeral. - // Nevermind, the following flagged message doesn't really work because I don't think regular messages can be ephemeral. (only interaction responses can be ephemeral) - // - // if (interaction.acknowledged) { - // const messageOptions: CreateMessageOptions = { - // content: message, - // flags: MessageFlags.Ephemeral - // } - logger.info(`User ${interaction.user.username} (${interaction.user.id})`) - // const dmChannel = await bot.rest.getDmChannel() - // logger.info(`dmChannel as follows`) - // logger.info(dmChannel) - // await bot.rest.sendMessage(dmChannel.id, messageOptions) - // } else { - await interaction.edit(message) - // } - - } - - - } -}) \ No newline at end of file diff --git a/services/bot/src/commands/yeah.ts b/services/bot/src/commands/yeah.ts deleted file mode 100644 index 5555aee..0000000 --- a/services/bot/src/commands/yeah.ts +++ /dev/null @@ -1,20 +0,0 @@ - -import { ApplicationCommandTypes, type Interaction } from '@discordeno/bot' -import { createCommand } from '../commands.ts' -import { bot } from '../bot.ts' - - - -createCommand({ - name: 'yeah', - description: 'Yeah! a message', - type: ApplicationCommandTypes.ChatInput, - async execute(interaction: Interaction) { - // interaction.message.id - const message = interaction.message - if (!message) return bot.logger.error('interaction.message was missing'); - if (!message.id) return bot.logger.error(`interaction.message.id was missing`); - interaction.respond('https://futureporn-b2.b-cdn.net/yeah_nobg.png', { isPrivate: true }) - bot.logger.info(`Yeah! command successfully ran with message.id=${message.id}`) - }, -}) diff --git a/services/bot/src/config.ts b/services/bot/src/config.ts deleted file mode 100644 index 9db60d5..0000000 --- a/services/bot/src/config.ts +++ /dev/null @@ -1,47 +0,0 @@ -import 'dotenv/config' - -const requiredEnvVars = [ - 'HTTP_PROXY', - 'WORKER_CONNECTION_STRING', - 'POSTGREST_URL', - 'DISCORD_TOKEN', - 'DISCORD_CHANNEL_ID', - 'DISCORD_GUILD_ID', - 'DISCORD_APPLICATION_ID', - 'AUTOMATION_USER_JWT', - 'SCOUT_URL' -] as const; - -const getEnvVar = (key: typeof requiredEnvVars[number]): string => { - const value = process.env[key]; - if (!value) { - throw new Error(`Missing ${key} env var`); - } - return value; -}; - -export interface Config { - postgrestUrl: string; - httpProxy: string; - token: string; - automationUserJwt: string; - discordGuildId: string; - discordChannelId: string; - connectionString: string; - discordApplicationId: string; - scoutUrl: string; -} - - -export const configs: Config = { - scoutUrl: getEnvVar('SCOUT_URL'), - connectionString: getEnvVar('WORKER_CONNECTION_STRING'), - httpProxy: getEnvVar('HTTP_PROXY'), - postgrestUrl: getEnvVar('POSTGREST_URL'), - token: getEnvVar('DISCORD_TOKEN'), - automationUserJwt: getEnvVar('AUTOMATION_USER_JWT'), - discordGuildId: getEnvVar('DISCORD_GUILD_ID'), - discordChannelId: getEnvVar('DISCORD_CHANNEL_ID'), - discordApplicationId: getEnvVar('DISCORD_APPLICATION_ID'), -} - diff --git a/services/bot/src/events/interactionCreate.ts b/services/bot/src/events/interactionCreate.ts deleted file mode 100644 index 2d0cbf7..0000000 --- a/services/bot/src/events/interactionCreate.ts +++ /dev/null @@ -1,51 +0,0 @@ -import { InteractionTypes, commandOptionsParser, type Interaction } from '@discordeno/bot' -import { bot } from '../bot.ts' -import { commands, type Command } from '../commands.ts' -import ItemCollector from '../collector.ts' - -export const collectors = new Set() - -const execCommand = async function execCommand(command: Command, interaction: Interaction) { - const options = commandOptionsParser(interaction) - - try { - await command.execute(interaction, options) - } catch (error) { - bot.logger.error(`There was an error running the ${command.name} command.`) - bot.logger.error(error) - } -} - -const handleApplicationCommand = async function handleApplicationCommand (interaction: Interaction) { - - if (!interaction.data) return - const command = commands.get(interaction.data.name) - - if (!command) { - bot.logger.error(`Command ${interaction.data.name} not found`) - return - } - - execCommand(command, interaction) - -} - -const handleMessageComponent = async function handleMessageComponent (interaction: Interaction) { - if (!interaction.data) return - if (!interaction.data.customId) return - const command = commands.get(interaction.data.customId) - if (!command) return bot.logger.error(`Command customId=${interaction.data.customId} not found`); - execCommand(command, interaction) -} - -bot.events.interactionCreate = async (interaction: Interaction) => { - - if (interaction.type === InteractionTypes.ApplicationCommand) { - await handleApplicationCommand(interaction) - } else if (interaction.type === InteractionTypes.MessageComponent) { - await handleMessageComponent(interaction) - } else { - bot.logger.info(`received interaction of type=${interaction.type}`) - } - -} \ No newline at end of file diff --git a/services/bot/src/events/ready.ts b/services/bot/src/events/ready.ts deleted file mode 100644 index 2f42fde..0000000 --- a/services/bot/src/events/ready.ts +++ /dev/null @@ -1,20 +0,0 @@ - - -import { ActivityTypes } from '@discordeno/bot' -import { bot } from '../bot.ts' - -bot.events.ready = async ({ shardId }) => { - - await bot.gateway.editShardStatus(shardId, { - status: 'online', - activities: [ - { - name: 'chat', - type: ActivityTypes.Watching, - timestamps: { - start: Date.now(), - }, - }, - ], - }) -} \ No newline at end of file diff --git a/services/bot/src/index.ts b/services/bot/src/index.ts deleted file mode 100644 index 742dbd7..0000000 --- a/services/bot/src/index.ts +++ /dev/null @@ -1,63 +0,0 @@ - -import { type WorkerUtils, type RunnerOptions, run } from 'graphile-worker' -import { bot } from './bot.ts' -import type { Interaction } from '@discordeno/bot' -import { importDirectory } from './utils/loader.ts' -import { join, dirname } from 'node:path' -import { fileURLToPath } from 'url' -import { configs } from './config.ts' -import { updateApplicationCommands } from './utils/update-commands.ts' - - -const __dirname = dirname(fileURLToPath(import.meta.url)); - - -export interface ExecuteArguments { - interaction: Interaction; - workerUtils: WorkerUtils; -} - - -async function setupGraphileWorker() { - const preset: GraphileConfig.Preset = { - worker: { - connectionString: configs.connectionString, - concurrentJobs: 3, - fileExtensions: [".js", ".ts"], - taskDirectory: join(__dirname, 'tasks') - }, - }; - const runnerOptions: RunnerOptions = { - preset - } - const runner = await run(runnerOptions) - if (!runner) throw new Error('failed to initialize graphile worker'); - await runner.promise -} - - - -async function setupBot() { - bot.logger.info('Starting @futureporn/bot.') - bot.logger.info('Loading commands...') - await importDirectory(join(__dirname, './commands')) - bot.logger.info('Loading events...') - await importDirectory(join(__dirname, './events')) - await bot.start() -} - - -async function main() { - console.log('setting up bot') - await setupBot() - console.log('setting up graphile worker') - await setupGraphileWorker() // this needs to run after setupBot() has run - console.log('updating discord application commands') - await updateApplicationCommands() // this needs to run after importDirectory() has run -} - -main().catch((e) => { - console.error("error during main() function") - console.error(e) - process.exit(3) -}) diff --git a/services/bot/src/middlewares/rbac.ts b/services/bot/src/middlewares/rbac.ts deleted file mode 100644 index 1802bf7..0000000 --- a/services/bot/src/middlewares/rbac.ts +++ /dev/null @@ -1,28 +0,0 @@ -import type { Interaction } from "@discordeno/bot"; -import { logger } from "discordeno"; - - -const roleMap = new Map([ - ['patron', BigInt('1084677850180882555')], - ['admin', BigInt('1084928253699039282')], - ['moderator', BigInt('1084935803324612729')], - ['testAdmin', BigInt('1275625364382552169')] -]); - -export async function rbacAllow (roleNames: string[], interaction: Interaction) { - const roleIds = roleNames.map((role) => roleMap.get(role)).filter((roleId): roleId is bigint => roleId !== undefined); - - if (!interaction.member?.roles || interaction.member.roles.length < 1 ) { - throw new Error(`User has no roles.`) - } - logger.info(`rbacAllow ${roleNames.join(',')} begin! the user responsible for the interaction has the following roles-- ${interaction.member.roles.join(',')}`) - - const hasRequiredRole = roleIds.some(roleId => interaction.member!.roles.includes(roleId)); - - if (!hasRequiredRole) { - throw new Error(`User lacks the role to run this command. One of the following roles is required: ${roleNames.join(', ')}`) - } - - logger.info(`rbacAllow ${roleNames.join(',')} success!`) - return -} \ No newline at end of file diff --git a/services/bot/src/register-commands.ts b/services/bot/src/register-commands.ts deleted file mode 100644 index 170d514..0000000 --- a/services/bot/src/register-commands.ts +++ /dev/null @@ -1,7 +0,0 @@ -import 'dotenv/config' - -import { bot } from './bot.js' -import { updateApplicationCommands } from './utils/update-commands.ts' - -bot.logger.info('Updating commands...') -await updateApplicationCommands() \ No newline at end of file diff --git a/services/bot/src/tasks/README.md b/services/bot/src/tasks/README.md deleted file mode 100644 index 2dba732..0000000 --- a/services/bot/src/tasks/README.md +++ /dev/null @@ -1,30 +0,0 @@ -Task names uses underscores because graphile_worker expects them to be that way because graphile_worker interfaces with Postgresql which uses lowercase and numberscores. - -here are some administrative functions for clearing all tasks. Also see https://worker.graphile.org/docs/admin-functions - -(search tags, for easily finding this file by content) -administrative tasks -clear all -delete all -jobs -addJob() - -## Add job via SQL - -```sql -SELECT graphile_worker.add_job('process_stream_recording', max_attempts := 3); -``` - -## complete/cancel a job via SQL - -```sql -SELECT * FROM graphile_worker.complete_jobs(ARRAY[7, 99, 38674, ...]); -``` - -## cancel all jobs - -```sql -SELECT * FROM graphile_worker.complete_jobs( - ARRAY(SELECT id FROM graphile_worker.jobs) -); -``` \ No newline at end of file diff --git a/services/bot/src/tasks/restart_failed_recordings.ts b/services/bot/src/tasks/restart_failed_recordings.ts deleted file mode 100644 index 7ef9c2e..0000000 --- a/services/bot/src/tasks/restart_failed_recordings.ts +++ /dev/null @@ -1,85 +0,0 @@ -import type { Task, Helpers } from "graphile-worker" -import { sub } from 'date-fns' -import type { RecordingRecord } from "@futureporn/types" -import qs from 'qs' -import fetch from 'node-fetch' -import { configs } from '../config.ts' - -interface Payload { - idle_minutes: number; -} - -function assertPayload(payload: any): asserts payload is Payload { - if (typeof payload !== "object" || !payload) throw new Error("invalid payload"); - if (!payload.idle_minutes) throw new Error('idle_minutes was missing from payload'); - if (typeof payload.idle_minutes !== 'number') throw new Error('idle_minutes must be a number'); -} - -export const restart_failed_recordings: Task = async function (payload: unknown, helpers: Helpers) { - assertPayload(payload) - const { idle_minutes } = payload - helpers.logger.info(`restart_failed_recordings has begun. Expring 'recording' and 'pending' records that haven't been updated in ${idle_minutes} minutes.`) - - const url = 'http://postgrest.futureporn.svc.cluster.local:9000/records' - let records: RecordingRecord[] = [] - - try { - // 1. identify failed /records - // Any record that was updated earlier than n minute ago AND is in 'pending' or 'recording' state is marked as stalled. - const timestamp = sub(new Date(), { minutes: idle_minutes }).toISOString() - const queryOptions = { - updated_at: `lt.${timestamp}`, - or: '(recording_state.eq.pending,recording_state.eq.recording)' - } - const updatePayload = { - updated_at: new Date().toISOString(), - recording_state: 'stalled' - } - helpers.logger.info(JSON.stringify(updatePayload)) - const query = qs.stringify(queryOptions) - const res = await fetch (`${url}?${query}`, { - method: 'PATCH', - headers: { - 'Content-Type': 'application/json', - 'Authorization': `Bearer ${configs.automationUserJwt}`, - 'Prefer': 'return=headers-only' - }, - body: JSON.stringify(updatePayload) - }) - if (!res.ok) { - const body = await res.text() - helpers.logger.info(JSON.stringify(res.headers)) - helpers.logger.error(`Response code was not 200. status=${res.status}, statusText=${res.statusText}`) - helpers.logger.error(body) - return; - } - - const body = await res.text() - helpers.logger.info('body as follows') - helpers.logger.info(body) - - // const data = await res.json() as RecordingRecord[] - - // if (data.length < 1) return; - // records = data - - } catch (e: any) { - if (e instanceof Error) { - helpers.logger.error(`hi there we encountered an error while fetching /records`) - helpers.logger.error(e.message) - } else { - helpers.logger.error(e) - } - - } - - // // 2. identify and update - // for (const record of records) { - // const res = await fetch(`${url}?`) - // } - - - // // 3. done -} - -export default restart_failed_recordings \ No newline at end of file diff --git a/services/bot/src/tasks/update_discord_message.ts b/services/bot/src/tasks/update_discord_message.ts deleted file mode 100644 index e463354..0000000 --- a/services/bot/src/tasks/update_discord_message.ts +++ /dev/null @@ -1,226 +0,0 @@ -import 'dotenv/config' -import type { Status, SegmentResponse, VodResponse } from '@futureporn/types' -import { type Task, type Helpers } from 'graphile-worker' -import { intervalToDuration, formatDuration } from 'date-fns' -import prettyBytes from 'pretty-bytes' -import { - EmbedsBuilder, - ButtonStyles, - type ActionRow, - MessageComponentTypes, - type ButtonComponent, - type EditMessage, -} from '@discordeno/bot' -import { bot } from '../bot.ts' -import { configs } from '../config.ts' -import findVod from '@futureporn/fetchers/findVod.ts' - -const yeahEmojiId = BigInt('1253191939461873756') - - -interface Payload { - vod_id: string; -} - - - -function assertPayload(payload: any): asserts payload is Payload { - if (typeof payload !== "object" || !payload) throw new Error("invalid payload"); - if (!payload.vod_id) throw new Error(`vod_id was absent in the payload`); - if (typeof payload.vod_id !== 'string') throw new Error(`vod_id was not a string`); -} - - - - -async function editDiscordMessage({ helpers, vod }: { vod: VodResponse, helpers: Helpers }) { - // bot.logger.info(`editDiscordmessage with vod status=${vod.status} `) - // bot.logger.info(vod) - const discordMessageId = vod.discord_message_id - if (!discordMessageId) throw new Error(`discordMessageId was missing!`); - if (typeof discordMessageId !== 'string') throw new Error(`discordMessageId was not a string!`); - - const channelId = BigInt(configs.discordChannelId) - - const updatedMessage: EditMessage = { - embeds: getEmbeds(vod, helpers), - components: getButtonRow(vod.status) - } - bot.helpers.editMessage(channelId, discordMessageId, updatedMessage) -} - - - - -/** - * update_discord_message is the task where we edit a previously sent discord message to display - * the most up-to-date status information from the database - * - * Sometimes the update is changing the state, one of Pending|Recording|Aborted|Ended. - * Sometimes the update is adding a thumbnail image to the message - */ -export const update_discord_message: Task = async function (payload, helpers: Helpers) { - try { - assertPayload(payload) - const { vod_id } = payload - const vodId = vod_id - - const vod = await findVod({ vod_id: vodId }) - if (!vod) throw new Error('failed to get vod from database'); - // helpers.logger.info(`update_discord_message got the following vod`) - // helpers.logger.info(JSON.stringify(vod, null, 2)) - await editDiscordMessage({ helpers, vod }) - } catch (e) { - helpers.logger.error(`caught an error during update_discord_message. e=${e}`) - } -} - - - -function getEmbeds(vod: VodResponse, helpers: Helpers) { - const vodId = vod.id - const url = vod.url - const segments = vod?.segments - const status = vod.status || 'unknown' - bot.logger.info(`getEmbeds with vodId=${vodId}, status=${vod.status}, segments.length=${segments?.length}`) - const embeds = new EmbedsBuilder() - .setTitle(`VOD ${vodId}`) - .setFields([ - { name: 'Status', value: status.charAt(0).toUpperCase()+status.slice(1), inline: true }, - // { name: 'Filesize', value: prettyBytes(fileSize), inline: true }, // filesize isn't on stream. filesize is on segment. keeping for reference. @todo - { name: 'URL', value: url, inline: false }, - { name: 'CDN2', value: '@todo', inline: false }, - ]) - if (status === 'pending_recording') { - embeds - .setDescription("Waiting for a worker to accept the job.") - .setColor(2326507) - } else if (status === 'recording') { - embeds - .setDescription('The vod is being recorded.') - .setColor(392960) - } else if (status === 'aborted') { - embeds - .setDescription("The recording was stopped by the user.") - .setColor(8289651) - } else if (status === 'finished') { - embeds - .setDescription("The recording has ended nominally.") - .setColor(10855845) - } else if (status === 'failed') { - embeds - .setDescription("The recording has ended abnorminally.") - .setColor(8289651) - } else if (status === 'processing') { - embeds - .setDescription("The recording is being prepared for publishing.") - .setColor(392960) - } else if (status === 'stalled') { - embeds - .setDescription("We have not received a progress update recently.") - .setColor(8289651) - } else { - embeds - .setDescription(`The recording is in an unknown state? (streamStatus=${status} this is a bug.)`) - .setColor(10855845) - } - - // Add an Embed for segments - if (segments) { - const getDuration = (s: SegmentResponse) => formatDuration(intervalToDuration({ start: s.created_at, end: s.updated_at })) - embeds.newEmbed() - .setTitle(`Segments`) - .setFields( - (segments.length === 0) - ? ([{ value: 'There are no segments', name: '', inline: false }]) - : segments.sort((a, b) => new Date(a.created_at).getTime() - new Date(b.created_at).getTime()).map((s, i) => ( - { - name: `Segment ${i+1}`, - value: `${getDuration(s)} (${prettyBytes(s.bytes)}) (${s.s3_key})`, - inline: false - } - ))) - } - - // Add an embed for S3 files - - // Add an Embed for processing tasks - if (status === 'processing') { - const tasks = [ - { name: 'combine_video_segments', complete: false }, - { name: 'generate_thumbnail', complete: false }, - { name: 'queue_moderator_review', complete: false }, - { name: 'create_mux_asset', complete: false }, - { name: 'create_torrent', complete: false }, - ] - embeds.newEmbed() - .setTitle('Processing Tasks') - .setDescription(tasks.map((task) => `* ${task.complete ? '🗹' : '☐'} ${task.name}`).join('\n')) } - - - return embeds -} - - - -function getButtonRow(streamStatus: Status): ActionRow[] { - const components: ButtonComponent[] = [] - - const yeahButton: ButtonComponent = { - type: MessageComponentTypes.Button, - customId: 'yeah', - label: "Yeah!", - emoji: { - id: yeahEmojiId - }, - style: ButtonStyles.Success - } - const processButton: ButtonComponent = { - type: MessageComponentTypes.Button, - customId: 'process', - label: 'Process Recording', - style: ButtonStyles.Success - } - const cancelButton: ButtonComponent = { - type: MessageComponentTypes.Button, - customId: 'cancel', - label: 'Cancel', - style: ButtonStyles.Danger - } - - const retryButton: ButtonComponent = { - type: MessageComponentTypes.Button, - customId: 'record', - label: 'Retry Recording', - style: ButtonStyles.Secondary - } - - - if (streamStatus === 'pending_recording' || streamStatus === 'recording') { - components.push(cancelButton) - components.push(yeahButton) - } else if (streamStatus === 'aborted') { - components.push(retryButton) - components.push(processButton) // @todo this is only for testing. normally the process button is hidden if the stream was aborted. - } else if (streamStatus === 'finished') { - components.push(processButton) - } else if (streamStatus === 'processing') { - // @todo redo this section which is only this way for testing - components.push(retryButton) - components.push(processButton) - } else { - components.push(retryButton) - components.push(processButton) - } - - - const actionRow: ActionRow = { - type: MessageComponentTypes.ActionRow, - components: components as [ButtonComponent] - } - - return [actionRow] -} - - -export default update_discord_message \ No newline at end of file diff --git a/services/bot/src/tasks/update_vod_statuses.test.ts b/services/bot/src/tasks/update_vod_statuses.test.ts deleted file mode 100644 index 4dc70c2..0000000 --- a/services/bot/src/tasks/update_vod_statuses.test.ts +++ /dev/null @@ -1,11 +0,0 @@ -import { updateStalledVods } from "./update_vod_statuses.ts" - -describe('update_vod_statuses', function () { - describe('updateStalledVods', function () { - describe('integration', function () { - xit("Should fetch a list of VODs that haven't been updated in the past 2 minutes", function () { - // const - }) - }) - }) -}) \ No newline at end of file diff --git a/services/bot/src/tasks/update_vod_statuses.ts b/services/bot/src/tasks/update_vod_statuses.ts deleted file mode 100644 index f8022a9..0000000 --- a/services/bot/src/tasks/update_vod_statuses.ts +++ /dev/null @@ -1,127 +0,0 @@ -import type { Task, Helpers } from "graphile-worker" -import { sub } from 'date-fns' -import type { Status } from "@futureporn/types" -import qs from 'qs' -import fetch from 'node-fetch' -import { configs } from '../config.ts' - -interface Payload { - stalled_minutes: number; - finished_minutes: number; -} - -function assertPayload(payload: any): asserts payload is Payload { - if (typeof payload !== "object" || !payload) throw new Error("invalid payload"); - if (!payload.stalled_minutes) throw new Error(`stalled_minutes was absent in the payload`); - if (!payload.finished_minutes) throw new Error(`finished_minutes was absent in the payload`); - if (typeof payload.stalled_minutes !== 'number') throw new Error(`stalled_minutes parameter was not a number`); - if (typeof payload.finished_minutes !== 'number') throw new Error(`finished_minutes parameter was not a number`); -} - -export async function updateFinishedVods({ - helpers, - finished_minutes, - url -}: { - helpers: Helpers, - finished_minutes: number, - url: string -}) { - helpers.logger.info(`updateFinishedVods with finished_minutes=${finished_minutes}, url=${url}`) - - // 1. identify and update stalled /vods - // Any vods that was updated earlier than n minute ago AND is in 'pending_recording' or 'recording' state is marked as stalled. - const timestamp = sub(new Date(), { minutes: finished_minutes }).toISOString() - const queryOptions = { - updated_at: `lt.${timestamp}`, - or: '(status.eq.pending_recording,status.eq.recording)' - } - const updatePayload = { - updated_at: new Date().toISOString(), - status: 'stalled' as Status - } - // helpers.logger.info(JSON.stringify(updatePayload)) - const query = qs.stringify(queryOptions) - const res = await fetch (`${url}?${query}`, { - method: 'PATCH', - headers: { - 'Content-Type': 'application/json', - 'Authorization': `Bearer ${configs.automationUserJwt}`, - 'Prefer': 'return=headers-only' - }, - body: JSON.stringify(updatePayload) - }) - if (!res.ok) { - const body = await res.text() - helpers.logger.info(JSON.stringify(res.headers)) - helpers.logger.error(`Response code was not 200. status=${res.status}, statusText=${res.statusText}`) - helpers.logger.error(body) - return; - } -} - -export async function updateStalledVods({ - helpers, - url, - stalled_minutes = 1, -}: { - helpers: Helpers, - url: string, - stalled_minutes: number, -}) { - // Identify and update stalled vods - // note: we are checking /vods update_at which gets updated whenever a /segments gets updated (thanks to postgres trigger functions) - const stalledTimestamp = sub(new Date(), { minutes: stalled_minutes }).toISOString(); - const stalledQueryOptions = { - 'select': 'status,id', - 'updated_at': `lt.${stalledTimestamp}`, - 'or': '(status.eq.pending_recording,status.eq.recording)', - }; - const stalledUpdatePayload = { - status: 'stalled', - }; - - const stalledQuery = qs.stringify(stalledQueryOptions, { encode: false }); - const stalledFetchUrl = `${url}?${stalledQuery}`; - helpers.logger.info(`updateStalledVods with stalledFetchUrl=${stalledFetchUrl}, stalled_minutes=${stalled_minutes}, url=${url}`); - const stalledOptions = { - method: 'PATCH', - headers: { - 'Content-Type': 'application/json', - 'Authorization': `Bearer ${configs.automationUserJwt}`, - 'Prefer': 'return=headers-only', - }, - body: JSON.stringify(stalledUpdatePayload), - }; - const stalledRes = await fetch(stalledFetchUrl, stalledOptions); - if (!stalledRes.ok) { - const stalledBody = await stalledRes.text(); - helpers.logger.info(JSON.stringify(stalledRes.headers)); - helpers.logger.error(`Stalled response code was not 200. status=${stalledRes.status}, statusText=${stalledRes.statusText}`); - helpers.logger.error(stalledBody); - return; - } - // const stalledResHeaders = stalledRes.headers.get('') - // console.log(stalledRes) - // console.log(stalledResBody) -} - -export const update_vod_statuses: Task = async function (payload: unknown, helpers: Helpers) { - assertPayload(payload) - const { stalled_minutes, finished_minutes } = payload - helpers.logger.info(` & & update_vod_statuses has begun.`) - const url = 'http://postgrest.futureporn.svc.cluster.local:9000/vods' - try { - await updateStalledVods({ helpers, url, stalled_minutes }) - await updateFinishedVods({ helpers, url, finished_minutes }) - } catch (e: any) { - if (e instanceof Error) { - helpers.logger.error(`hi there we encountered an error while fetching /vods`) - helpers.logger.error(e.message) - } else { - helpers.logger.error(e) - } - } -} - -export default update_vod_statuses \ No newline at end of file diff --git a/services/bot/src/utils/loader.ts b/services/bot/src/utils/loader.ts deleted file mode 100644 index 3926b45..0000000 --- a/services/bot/src/utils/loader.ts +++ /dev/null @@ -1,23 +0,0 @@ -// @see https://github.com/discordeno/discordeno/blob/main/examples/advanced/src/utils/loader.ts - -import { readdir } from 'node:fs/promises' -import { logger } from '@discordeno/bot' -import { join } from 'node:path' - -export async function importDirectory(folder: string): Promise { - const files = await readdir(folder, { recursive: true }) - - // bot.logger.info(files) - for (const filename of files) { - if (!filename.endsWith('.js') && !filename.endsWith('.ts')) continue - logger.info(`loading ${filename}`) - - // Using `file://` and `process.cwd()` to avoid weird issues with relative paths and/or Windows - // await import(`file://${process.cwd()}/${folder}/${filename}`).catch((x) => - await import(join(folder, filename)).catch((x) => - // console.error(x) - logger.error(`cannot import ${filename} for reason: ${x}`) - // logger.fatal(`Cannot import file (${folder}/${filename}) for reason: ${x}`), - ) - } -} \ No newline at end of file diff --git a/services/bot/src/utils/update-commands.ts b/services/bot/src/utils/update-commands.ts deleted file mode 100644 index 06e1429..0000000 --- a/services/bot/src/utils/update-commands.ts +++ /dev/null @@ -1,8 +0,0 @@ -import { bot } from '../bot.ts' -import { commands } from '../commands.ts' - -export async function updateApplicationCommands(): Promise { - bot.logger.info(`updating application commands. commands=${commands.array()}`) - console.log(commands) - await bot.helpers.upsertGlobalApplicationCommands(commands.array()) -} \ No newline at end of file diff --git a/services/bot/tsconfig.json b/services/bot/tsconfig.json deleted file mode 100644 index fbafd15..0000000 --- a/services/bot/tsconfig.json +++ /dev/null @@ -1,32 +0,0 @@ -{ - "compilerOptions": { - // Base Options recommended for all projects - "esModuleInterop": true, - "skipLibCheck": true, - "target": "es2022", - "allowJs": true, - "noEmit": true, - "allowImportingTsExtensions": true, - "resolveJsonModule": true, - "moduleDetection": "force", - "isolatedModules": true, - "verbatimModuleSyntax": true, - // Enable strict type checking so you can catch bugs early - "strict": true, - "noUncheckedIndexedAccess": true, - "noImplicitOverride": true, - // Transpile our TypeScript code to JavaScript - "module": "NodeNext", - "outDir": "dist", - "lib": [ - "es2022" - ] - }, - // Include the necessary files for your project - "include": [ - "**/*.ts" -, "src/events/interactionCreate.ts.old", "../../packages/fetchers/createStreamInDatabase.ts", "../../packages/fetchers/createVod.ts", "../../packages/fetchers/findOrCreateStream.ts", "../../packages/fetchers/findOrCreateVtuber.spec.ts", "../../packages/fetchers/findOrCreateVtuber.ts", "../../packages/fetchers/findVod.ts", "../../packages/fetchers/getStreamFromDatabase.ts", "../../packages/fetchers/getStreamIdFromMessage.ts", "../../packages/fetchers/getUrlFromMessage.ts", "../../packages/fetchers/patchVod.ts" ], - "exclude": [ - "node_modules" - ] -} \ No newline at end of file diff --git a/services/bright/config/dev.exs b/services/bright/config/dev.exs index 866fd54..9405361 100644 --- a/services/bright/config/dev.exs +++ b/services/bright/config/dev.exs @@ -16,7 +16,7 @@ config :bright, Bright.Repo, config :bright, BrightWeb.Endpoint, # Binding to loopback ipv4 address prevents access from other machines. # Change to `ip: {0, 0, 0, 0}` to allow access from other machines. - http: [ip: {0, 0, 0, 0}, port: "#{System.get_env("PORT")}"], + http: [ip: {0, 0, 0, 0}, port: "#{System.get_env("PORT") || 4000}"], check_origin: false, code_reloader: true, debug_errors: true, diff --git a/services/bright/lib/bright/jobs/create_hls_playlist.ex b/services/bright/lib/bright/jobs/create_hls_playlist.ex index a960851..8b859e2 100644 --- a/services/bright/lib/bright/jobs/create_hls_playlist.ex +++ b/services/bright/lib/bright/jobs/create_hls_playlist.ex @@ -26,6 +26,7 @@ defmodule Bright.Jobs.CreateHlsPlaylist do {:ok, package_job_id} <- start_package(transcode_job_id), {:ok, asset_id} <- poll_job_completion(package_job_id) do update_vod_with_playlist_url(vod, package_job_id) + Logger.info("HLS playlist created and updated for VOD ID #{vod_id}") else {:error, reason} -> diff --git a/services/bright/lib/bright/platforms/platform.ex b/services/bright/lib/bright/platforms/platform.ex index dcc915f..d39237b 100644 --- a/services/bright/lib/bright/platforms/platform.ex +++ b/services/bright/lib/bright/platforms/platform.ex @@ -7,8 +7,6 @@ defmodule Bright.Platforms.Platform do field :url, :string field :icon, :string - many_to_many :streams, Bright.Streams.Stream, join_through: "streams_platforms" - timestamps(type: :utc_datetime) end @@ -19,12 +17,3 @@ defmodule Bright.Platforms.Platform do |> validate_required([:name, :url, :icon]) end end - - - -defimpl Phoenix.HTML.Safe, for: Bright.Platforms.Platform do - def to_iodata(platform) do - # platform.icon - platform.name - end -end diff --git a/services/bright/lib/bright_web/components/core_components.ex b/services/bright/lib/bright_web/components/core_components.ex index 7f11275..ca2d3de 100644 --- a/services/bright/lib/bright_web/components/core_components.ex +++ b/services/bright/lib/bright_web/components/core_components.ex @@ -115,7 +115,7 @@ defmodule BrightWeb.CoreComponents do phx-click={JS.push("lv:clear-flash", value: %{key: @kind}) |> hide("##{@id}")} role="alert" class={[ - "notification", + "mb-5 notification", @kind == :info && "is-success", @kind == :error && "is-danger" ]} diff --git a/services/bright/lib/bright_web/components/layouts/root.html.heex b/services/bright/lib/bright_web/components/layouts/root.html.heex index 72c95c8..9c4a422 100644 --- a/services/bright/lib/bright_web/components/layouts/root.html.heex +++ b/services/bright/lib/bright_web/components/layouts/root.html.heex @@ -20,50 +20,8 @@ %> - -
    - <%= if @current_platform do %> -
  • - {@current_platform.email} -
  • -
  • - <.link - href={~p"/platforms/settings"} - class="text-[0.8125rem] leading-6 text-zinc-900 font-semibold hover:text-zinc-700" - > - Settings - -
  • -
  • - <.link - href={~p"/platforms/log_out"} - method="delete" - class="text-[0.8125rem] leading-6 text-zinc-900 font-semibold hover:text-zinc-700" - > - Log out - -
  • - <% else %> -
  • - <.link - href={~p"/platforms/register"} - class="text-[0.8125rem] leading-6 text-zinc-900 font-semibold hover:text-zinc-700" - > - Register - -
  • -
  • - <.link - href={~p"/platforms/log_in"} - class="text-[0.8125rem] leading-6 text-zinc-900 font-semibold hover:text-zinc-700" - > - Log in - -
  • - <% end %> -
<%= @inner_content %> diff --git a/services/bright/lib/bright_web/components/navigation_components.ex b/services/bright/lib/bright_web/components/navigation_components.ex index 8a72aca..8fc2a1a 100644 --- a/services/bright/lib/bright_web/components/navigation_components.ex +++ b/services/bright/lib/bright_web/components/navigation_components.ex @@ -61,6 +61,12 @@ defmodule BrightWeb.NavigationComponents do Streams Archive + <.link + href={~p"/vods"} + class="navbar-item"> + Vods + + <.link href={~p"/vtubers"} class="navbar-item"> @@ -80,7 +86,7 @@ defmodule BrightWeb.NavigationComponents do Tags - <.link + - """ diff --git a/services/bright/lib/bright_web/controllers/page_controller.ex b/services/bright/lib/bright_web/controllers/page_controller.ex index 12a72e3..e95330a 100644 --- a/services/bright/lib/bright_web/controllers/page_controller.ex +++ b/services/bright/lib/bright_web/controllers/page_controller.ex @@ -8,8 +8,6 @@ defmodule BrightWeb.PageController do # send_resp(conn, 201, "") conn - |> put_flash(:info, "You are beautiful!") - |> put_flash(:error, "Test error!") |> put_status(202) |> render(:home, layout: false) # redirect(conn, to: ~p"/redirect_test") @@ -24,6 +22,11 @@ defmodule BrightWeb.PageController do render(conn, :api, layout: false) end + def health(conn, _params) do + data = %{message: "OK", status: "success"} + json(conn, data) + end + def redirect_test(conn, _params) do render(conn, :home, layout: false) end diff --git a/services/bright/lib/bright_web/controllers/page_html/about.html.heex b/services/bright/lib/bright_web/controllers/page_html/about.html.heex index b3884c3..5fdd3a3 100644 --- a/services/bright/lib/bright_web/controllers/page_html/about.html.heex +++ b/services/bright/lib/bright_web/controllers/page_html/about.html.heex @@ -2,7 +2,23 @@ <.navbar /> -
-

About

-

@todo

+
+ +
+

About

+ +

Welcome to Futureporn, a platform built by fans, for fans, dedicated to preserving the moments that matter in the world of live streaming. It all started with a simple need: capturing ProjektMelody's streams on Chaturbate. Chaturbate doesn’t save VODs, and sometimes we missed the magic. Other times, creators like ProjektMelody faced unnecessary de-platforming for simply being unique. We wanted to create a space where this content could endure, unshaken by the tides of censorship or fleeting platforms.

+
+ +
+

Our Mission

+

FuturePorn is more than an archive; it's a movement. We aim to document who streamed, when they streamed, and celebrate the creativity and passion of our favorite H-streamers. We're building a community-driven platform to ensure that these moments withstand the test of time.

+
+ +
+

Our Vision

+

Imagine a world where you can revisit your favorite streams, sync interactive toys with VOD playback, and relive the excitement as it happened. FuturePorn isn’t just about preserving memories; it’s about innovating how we connect with the content and creators we love.

+

Join us as we redefine archiving and fandom, ensuring that no stream is ever lost again. Together, we can create a platform that stands as a testament to creativity, individuality, and the fans who make it all possible.

+
+
diff --git a/services/bright/lib/bright_web/controllers/page_html/home.html.heex b/services/bright/lib/bright_web/controllers/page_html/home.html.heex index 5af5234..04ae375 100644 --- a/services/bright/lib/bright_web/controllers/page_html/home.html.heex +++ b/services/bright/lib/bright_web/controllers/page_html/home.html.heex @@ -15,44 +15,18 @@
-

Latest VODs

- <%= for number <- 1..10 do %> - - <%= number %> - <%= number * number %> - - <% end %> + + +

This is a pre-alpha test of Futureporn version 2.0

+

Work is in progress, check back each week for updates.

+
+ Streams Archive +
-
- -
- - - - Hello {@current_uuid} - - Archive - - <.back navigate={~p"/posts"}>Back -
diff --git a/services/bright/lib/bright_web/controllers/platform_html/index.html.heex b/services/bright/lib/bright_web/controllers/platform_html/index.html.heex index 3d643c4..790c6dc 100644 --- a/services/bright/lib/bright_web/controllers/platform_html/index.html.heex +++ b/services/bright/lib/bright_web/controllers/platform_html/index.html.heex @@ -10,7 +10,7 @@ <.table id="platforms" rows={@platforms} row_click={&JS.navigate(~p"/platforms/#{&1}")}> <:col :let={platform} label="Name">{platform.name} <:col :let={platform} label="Url">{platform.url} - <:col :let={platform} label="Icon">{raw(platform.icon)} + <:col :let={platform} label="Icon">{raw platform.icon} <:action :let={platform}>
<.link navigate={~p"/platforms/#{platform}"}>Show diff --git a/services/bright/lib/bright_web/controllers/platform_html/show.html.heex b/services/bright/lib/bright_web/controllers/platform_html/show.html.heex index c94efa7..4066f62 100644 --- a/services/bright/lib/bright_web/controllers/platform_html/show.html.heex +++ b/services/bright/lib/bright_web/controllers/platform_html/show.html.heex @@ -11,7 +11,7 @@ <.list> <:item title="Name">{@platform.name} <:item title="Url">{@platform.url} - <:item title="Icon">{raw(@platform.icon)} + <:item title="Icon">{raw @platform.icon} <.back navigate={~p"/platforms"}>Back to platforms diff --git a/services/bright/lib/bright_web/controllers/platform_session_controller.ex b/services/bright/lib/bright_web/controllers/platform_session_controller.ex deleted file mode 100644 index a2a9b59..0000000 --- a/services/bright/lib/bright_web/controllers/platform_session_controller.ex +++ /dev/null @@ -1,42 +0,0 @@ -defmodule BrightWeb.PlatformSessionController do - use BrightWeb, :controller - - alias Bright.Platforms - alias BrightWeb.PlatformAuth - - def create(conn, %{"_action" => "registered"} = params) do - create(conn, params, "Account created successfully!") - end - - def create(conn, %{"_action" => "password_updated"} = params) do - conn - |> put_session(:platform_return_to, ~p"/platforms/settings") - |> create(params, "Password updated successfully!") - end - - def create(conn, params) do - create(conn, params, "Welcome back!") - end - - defp create(conn, %{"platform" => platform_params}, info) do - %{"email" => email, "password" => password} = platform_params - - if platform = Platforms.get_platform_by_email_and_password(email, password) do - conn - |> put_flash(:info, info) - |> PlatformAuth.log_in_platform(platform, platform_params) - else - # In order to prevent user enumeration attacks, don't disclose whether the email is registered. - conn - |> put_flash(:error, "Invalid email or password") - |> put_flash(:email, String.slice(email, 0, 160)) - |> redirect(to: ~p"/platforms/log_in") - end - end - - def delete(conn, _params) do - conn - |> put_flash(:info, "Logged out successfully.") - |> PlatformAuth.log_out_platform() - end -end diff --git a/services/bright/lib/bright_web/controllers/vtuber_html/vtuber_form.html.heex b/services/bright/lib/bright_web/controllers/vtuber_html/vtuber_form.html.heex index 65a1e82..30c72e8 100644 --- a/services/bright/lib/bright_web/controllers/vtuber_html/vtuber_form.html.heex +++ b/services/bright/lib/bright_web/controllers/vtuber_html/vtuber_form.html.heex @@ -3,11 +3,10 @@ Oops, something went wrong! Please check the errors below. - <.input field={f[:slug]} type="text" label="Slug" /> <.input field={f[:display_name]} type="text" label="Display Name" /> + <.input field={f[:slug]} type="text" label="Slug" /> <.input field={f[:image]} type="text" label="Image URL" /> <.input field={f[:theme_color]} type="color" label="Theme Color" /> - <.input field={f[:chaturbate]} type="text" label="Chaturbate" /> <.input field={f[:twitter]} type="text" label="Twitter" /> <.input field={f[:patreon]} type="text" label="Patreon" /> diff --git a/services/bright/lib/bright_web/live/platform_confirmation_instructions_live.ex b/services/bright/lib/bright_web/live/platform_confirmation_instructions_live.ex deleted file mode 100644 index 780c42a..0000000 --- a/services/bright/lib/bright_web/live/platform_confirmation_instructions_live.ex +++ /dev/null @@ -1,51 +0,0 @@ -defmodule BrightWeb.PlatformConfirmationInstructionsLive do - use BrightWeb, :live_view - - alias Bright.Platforms - - def render(assigns) do - ~H""" -
- <.header class="text-center"> - No confirmation instructions received? - <:subtitle>We'll send a new confirmation link to your inbox - - - <.simple_form for={@form} id="resend_confirmation_form" phx-submit="send_instructions"> - <.input field={@form[:email]} type="email" placeholder="Email" required /> - <:actions> - <.button phx-disable-with="Sending..." class="w-full"> - Resend confirmation instructions - - - - -

- <.link href={~p"/platforms/register"}>Register - | <.link href={~p"/platforms/log_in"}>Log in -

-
- """ - end - - def mount(_params, _session, socket) do - {:ok, assign(socket, form: to_form(%{}, as: "platform"))} - end - - def handle_event("send_instructions", %{"platform" => %{"email" => email}}, socket) do - if platform = Platforms.get_platform_by_email(email) do - Platforms.deliver_platform_confirmation_instructions( - platform, - &url(~p"/platforms/confirm/#{&1}") - ) - end - - info = - "If your email is in our system and it has not been confirmed yet, you will receive an email with instructions shortly." - - {:noreply, - socket - |> put_flash(:info, info) - |> redirect(to: ~p"/")} - end -end diff --git a/services/bright/lib/bright_web/live/platform_confirmation_live.ex b/services/bright/lib/bright_web/live/platform_confirmation_live.ex deleted file mode 100644 index 1dfaec4..0000000 --- a/services/bright/lib/bright_web/live/platform_confirmation_live.ex +++ /dev/null @@ -1,58 +0,0 @@ -defmodule BrightWeb.PlatformConfirmationLive do - use BrightWeb, :live_view - - alias Bright.Platforms - - def render(%{live_action: :edit} = assigns) do - ~H""" -
- <.header class="text-center">Confirm Account - - <.simple_form for={@form} id="confirmation_form" phx-submit="confirm_account"> - - <:actions> - <.button phx-disable-with="Confirming..." class="w-full">Confirm my account - - - -

- <.link href={~p"/platforms/register"}>Register - | <.link href={~p"/platforms/log_in"}>Log in -

-
- """ - end - - def mount(%{"token" => token}, _session, socket) do - form = to_form(%{"token" => token}, as: "platform") - {:ok, assign(socket, form: form), temporary_assigns: [form: nil]} - end - - # Do not log in the platform after confirmation to avoid a - # leaked token giving the platform access to the account. - def handle_event("confirm_account", %{"platform" => %{"token" => token}}, socket) do - case Platforms.confirm_platform(token) do - {:ok, _} -> - {:noreply, - socket - |> put_flash(:info, "Platform confirmed successfully.") - |> redirect(to: ~p"/")} - - :error -> - # If there is a current platform and the account was already confirmed, - # then odds are that the confirmation link was already visited, either - # by some automation or by the platform themselves, so we redirect without - # a warning message. - case socket.assigns do - %{current_platform: %{confirmed_at: confirmed_at}} when not is_nil(confirmed_at) -> - {:noreply, redirect(socket, to: ~p"/")} - - %{} -> - {:noreply, - socket - |> put_flash(:error, "Platform confirmation link is invalid or it has expired.") - |> redirect(to: ~p"/")} - end - end - end -end diff --git a/services/bright/lib/bright_web/live/platform_forgot_password_live.ex b/services/bright/lib/bright_web/live/platform_forgot_password_live.ex deleted file mode 100644 index 99402d4..0000000 --- a/services/bright/lib/bright_web/live/platform_forgot_password_live.ex +++ /dev/null @@ -1,50 +0,0 @@ -defmodule BrightWeb.PlatformForgotPasswordLive do - use BrightWeb, :live_view - - alias Bright.Platforms - - def render(assigns) do - ~H""" -
- <.header class="text-center"> - Forgot your password? - <:subtitle>We'll send a password reset link to your inbox - - - <.simple_form for={@form} id="reset_password_form" phx-submit="send_email"> - <.input field={@form[:email]} type="email" placeholder="Email" required /> - <:actions> - <.button phx-disable-with="Sending..." class="w-full"> - Send password reset instructions - - - -

- <.link href={~p"/platforms/register"}>Register - | <.link href={~p"/platforms/log_in"}>Log in -

-
- """ - end - - def mount(_params, _session, socket) do - {:ok, assign(socket, form: to_form(%{}, as: "platform"))} - end - - def handle_event("send_email", %{"platform" => %{"email" => email}}, socket) do - if platform = Platforms.get_platform_by_email(email) do - Platforms.deliver_platform_reset_password_instructions( - platform, - &url(~p"/platforms/reset_password/#{&1}") - ) - end - - info = - "If your email is in our system, you will receive instructions to reset your password shortly." - - {:noreply, - socket - |> put_flash(:info, info) - |> redirect(to: ~p"/")} - end -end diff --git a/services/bright/lib/bright_web/live/platform_login_live.ex b/services/bright/lib/bright_web/live/platform_login_live.ex deleted file mode 100644 index af55de9..0000000 --- a/services/bright/lib/bright_web/live/platform_login_live.ex +++ /dev/null @@ -1,43 +0,0 @@ -defmodule BrightWeb.PlatformLoginLive do - use BrightWeb, :live_view - - def render(assigns) do - ~H""" -
- <.header class="text-center"> - Log in to account - <:subtitle> - Don't have an account? - <.link navigate={~p"/platforms/register"} class="font-semibold text-brand hover:underline"> - Sign up - - for an account now. - - - - <.simple_form for={@form} id="login_form" action={~p"/platforms/log_in"} phx-update="ignore"> - <.input field={@form[:email]} type="email" label="Email" required /> - <.input field={@form[:password]} type="password" label="Password" required /> - - <:actions> - <.input field={@form[:remember_me]} type="checkbox" label="Keep me logged in" /> - <.link href={~p"/platforms/reset_password"} class="text-sm font-semibold"> - Forgot your password? - - - <:actions> - <.button phx-disable-with="Logging in..." class="w-full"> - Log in - - - -
- """ - end - - def mount(_params, _session, socket) do - email = Phoenix.Flash.get(socket.assigns.flash, :email) - form = to_form(%{"email" => email}, as: "platform") - {:ok, assign(socket, form: form), temporary_assigns: [form: form]} - end -end diff --git a/services/bright/lib/bright_web/live/platform_registration_live.ex b/services/bright/lib/bright_web/live/platform_registration_live.ex deleted file mode 100644 index 591e36b..0000000 --- a/services/bright/lib/bright_web/live/platform_registration_live.ex +++ /dev/null @@ -1,87 +0,0 @@ -defmodule BrightWeb.PlatformRegistrationLive do - use BrightWeb, :live_view - - alias Bright.Platforms - alias Bright.Platforms.Platform - - def render(assigns) do - ~H""" -
- <.header class="text-center"> - Register for an account - <:subtitle> - Already registered? - <.link navigate={~p"/platforms/log_in"} class="font-semibold text-brand hover:underline"> - Log in - - to your account now. - - - - <.simple_form - for={@form} - id="registration_form" - phx-submit="save" - phx-change="validate" - phx-trigger-action={@trigger_submit} - action={~p"/platforms/log_in?_action=registered"} - method="post" - > - <.error :if={@check_errors}> - Oops, something went wrong! Please check the errors below. - - - <.input field={@form[:email]} type="email" label="Email" required /> - <.input field={@form[:password]} type="password" label="Password" required /> - - <:actions> - <.button phx-disable-with="Creating account..." class="w-full">Create an account - - -
- """ - end - - def mount(_params, _session, socket) do - changeset = Platforms.change_platform_registration(%Platform{}) - - socket = - socket - |> assign(trigger_submit: false, check_errors: false) - |> assign_form(changeset) - - {:ok, socket, temporary_assigns: [form: nil]} - end - - def handle_event("save", %{"platform" => platform_params}, socket) do - case Platforms.register_platform(platform_params) do - {:ok, platform} -> - {:ok, _} = - Platforms.deliver_platform_confirmation_instructions( - platform, - &url(~p"/platforms/confirm/#{&1}") - ) - - changeset = Platforms.change_platform_registration(platform) - {:noreply, socket |> assign(trigger_submit: true) |> assign_form(changeset)} - - {:error, %Ecto.Changeset{} = changeset} -> - {:noreply, socket |> assign(check_errors: true) |> assign_form(changeset)} - end - end - - def handle_event("validate", %{"platform" => platform_params}, socket) do - changeset = Platforms.change_platform_registration(%Platform{}, platform_params) - {:noreply, assign_form(socket, Map.put(changeset, :action, :validate))} - end - - defp assign_form(socket, %Ecto.Changeset{} = changeset) do - form = to_form(changeset, as: "platform") - - if changeset.valid? do - assign(socket, form: form, check_errors: false) - else - assign(socket, form: form) - end - end -end diff --git a/services/bright/lib/bright_web/live/platform_reset_password_live.ex b/services/bright/lib/bright_web/live/platform_reset_password_live.ex deleted file mode 100644 index 7bce43e..0000000 --- a/services/bright/lib/bright_web/live/platform_reset_password_live.ex +++ /dev/null @@ -1,89 +0,0 @@ -defmodule BrightWeb.PlatformResetPasswordLive do - use BrightWeb, :live_view - - alias Bright.Platforms - - def render(assigns) do - ~H""" -
- <.header class="text-center">Reset Password - - <.simple_form - for={@form} - id="reset_password_form" - phx-submit="reset_password" - phx-change="validate" - > - <.error :if={@form.errors != []}> - Oops, something went wrong! Please check the errors below. - - - <.input field={@form[:password]} type="password" label="New password" required /> - <.input - field={@form[:password_confirmation]} - type="password" - label="Confirm new password" - required - /> - <:actions> - <.button phx-disable-with="Resetting..." class="w-full">Reset Password - - - -

- <.link href={~p"/platforms/register"}>Register - | <.link href={~p"/platforms/log_in"}>Log in -

-
- """ - end - - def mount(params, _session, socket) do - socket = assign_platform_and_token(socket, params) - - form_source = - case socket.assigns do - %{platform: platform} -> - Platforms.change_platform_password(platform) - - _ -> - %{} - end - - {:ok, assign_form(socket, form_source), temporary_assigns: [form: nil]} - end - - # Do not log in the platform after reset password to avoid a - # leaked token giving the platform access to the account. - def handle_event("reset_password", %{"platform" => platform_params}, socket) do - case Platforms.reset_platform_password(socket.assigns.platform, platform_params) do - {:ok, _} -> - {:noreply, - socket - |> put_flash(:info, "Password reset successfully.") - |> redirect(to: ~p"/platforms/log_in")} - - {:error, changeset} -> - {:noreply, assign_form(socket, Map.put(changeset, :action, :insert))} - end - end - - def handle_event("validate", %{"platform" => platform_params}, socket) do - changeset = Platforms.change_platform_password(socket.assigns.platform, platform_params) - {:noreply, assign_form(socket, Map.put(changeset, :action, :validate))} - end - - defp assign_platform_and_token(socket, %{"token" => token}) do - if platform = Platforms.get_platform_by_reset_password_token(token) do - assign(socket, platform: platform, token: token) - else - socket - |> put_flash(:error, "Reset password link is invalid or it has expired.") - |> redirect(to: ~p"/") - end - end - - defp assign_form(socket, %{} = source) do - assign(socket, :form, to_form(source, as: "platform")) - end -end diff --git a/services/bright/lib/bright_web/live/platform_settings_live.ex b/services/bright/lib/bright_web/live/platform_settings_live.ex deleted file mode 100644 index d1fa883..0000000 --- a/services/bright/lib/bright_web/live/platform_settings_live.ex +++ /dev/null @@ -1,167 +0,0 @@ -defmodule BrightWeb.PlatformSettingsLive do - use BrightWeb, :live_view - - alias Bright.Platforms - - def render(assigns) do - ~H""" - <.header class="text-center"> - Account Settings - <:subtitle>Manage your account email address and password settings - - -
-
- <.simple_form - for={@email_form} - id="email_form" - phx-submit="update_email" - phx-change="validate_email" - > - <.input field={@email_form[:email]} type="email" label="Email" required /> - <.input - field={@email_form[:current_password]} - name="current_password" - id="current_password_for_email" - type="password" - label="Current password" - value={@email_form_current_password} - required - /> - <:actions> - <.button phx-disable-with="Changing...">Change Email - - -
-
- <.simple_form - for={@password_form} - id="password_form" - action={~p"/platforms/log_in?_action=password_updated"} - method="post" - phx-change="validate_password" - phx-submit="update_password" - phx-trigger-action={@trigger_submit} - > - - <.input field={@password_form[:password]} type="password" label="New password" required /> - <.input - field={@password_form[:password_confirmation]} - type="password" - label="Confirm new password" - /> - <.input - field={@password_form[:current_password]} - name="current_password" - type="password" - label="Current password" - id="current_password_for_password" - value={@current_password} - required - /> - <:actions> - <.button phx-disable-with="Changing...">Change Password - - -
-
- """ - end - - def mount(%{"token" => token}, _session, socket) do - socket = - case Platforms.update_platform_email(socket.assigns.current_platform, token) do - :ok -> - put_flash(socket, :info, "Email changed successfully.") - - :error -> - put_flash(socket, :error, "Email change link is invalid or it has expired.") - end - - {:ok, push_navigate(socket, to: ~p"/platforms/settings")} - end - - def mount(_params, _session, socket) do - platform = socket.assigns.current_platform - email_changeset = Platforms.change_platform_email(platform) - password_changeset = Platforms.change_platform_password(platform) - - socket = - socket - |> assign(:current_password, nil) - |> assign(:email_form_current_password, nil) - |> assign(:current_email, platform.email) - |> assign(:email_form, to_form(email_changeset)) - |> assign(:password_form, to_form(password_changeset)) - |> assign(:trigger_submit, false) - - {:ok, socket} - end - - def handle_event("validate_email", params, socket) do - %{"current_password" => password, "platform" => platform_params} = params - - email_form = - socket.assigns.current_platform - |> Platforms.change_platform_email(platform_params) - |> Map.put(:action, :validate) - |> to_form() - - {:noreply, assign(socket, email_form: email_form, email_form_current_password: password)} - end - - def handle_event("update_email", params, socket) do - %{"current_password" => password, "platform" => platform_params} = params - platform = socket.assigns.current_platform - - case Platforms.apply_platform_email(platform, password, platform_params) do - {:ok, applied_platform} -> - Platforms.deliver_platform_update_email_instructions( - applied_platform, - platform.email, - &url(~p"/platforms/settings/confirm_email/#{&1}") - ) - - info = "A link to confirm your email change has been sent to the new address." - {:noreply, socket |> put_flash(:info, info) |> assign(email_form_current_password: nil)} - - {:error, changeset} -> - {:noreply, assign(socket, :email_form, to_form(Map.put(changeset, :action, :insert)))} - end - end - - def handle_event("validate_password", params, socket) do - %{"current_password" => password, "platform" => platform_params} = params - - password_form = - socket.assigns.current_platform - |> Platforms.change_platform_password(platform_params) - |> Map.put(:action, :validate) - |> to_form() - - {:noreply, assign(socket, password_form: password_form, current_password: password)} - end - - def handle_event("update_password", params, socket) do - %{"current_password" => password, "platform" => platform_params} = params - platform = socket.assigns.current_platform - - case Platforms.update_platform_password(platform, password, platform_params) do - {:ok, platform} -> - password_form = - platform - |> Platforms.change_platform_password(platform_params) - |> to_form() - - {:noreply, assign(socket, trigger_submit: true, password_form: password_form)} - - {:error, changeset} -> - {:noreply, assign(socket, password_form: to_form(changeset))} - end - end -end diff --git a/services/bright/lib/bright_web/router.ex b/services/bright/lib/bright_web/router.ex index 7cae201..c991da5 100644 --- a/services/bright/lib/bright_web/router.ex +++ b/services/bright/lib/bright_web/router.ex @@ -1,8 +1,7 @@ defmodule BrightWeb.Router do use BrightWeb, :router - import BrightWeb.PlatformAuth - alias Bright.ShoppingCart + pipeline :browser do plug :accepts, ["html", "json"] @@ -11,14 +10,9 @@ defmodule BrightWeb.Router do plug :put_root_layout, html: {BrightWeb.Layouts, :root} plug :protect_from_forgery plug :put_secure_browser_headers - plug :fetch_current_platform - plug :fetch_current_user - plug :fetch_current_cart end - pipeline :auth do - # plug :ensure_authenticated - end + defp fetch_current_user(conn, _) do @@ -32,29 +26,20 @@ defmodule BrightWeb.Router do end end - defp fetch_current_cart(conn, _opts) do - if cart = ShoppingCart.get_cart_by_user_uuid(conn.assigns.current_uuid) do - assign(conn, :cart, cart) - else - {:ok, new_cart} = ShoppingCart.create_cart(conn.assigns.current_uuid) - assign(conn, :cart, new_cart) - end - end + pipeline :api do plug :accepts, ["json"] end scope "/" do - pipe_through [:browser, :auth] + pipe_through [:browser] - get "/posts/new", PostController, :new - post "/posts", PostController, :create - - get "/streams/new", StreamController, :new - post "/streams", StreamController, :create + # get "/streams/new", StreamController, :new + # post "/streams", StreamController, :create end + scope "/", BrightWeb do pipe_through :browser @@ -64,16 +49,16 @@ defmodule BrightWeb.Router do get "/patrons", PatronController, :index get "/about", PageController, :about get "/api", PageController, :api + get "/up", PageController, :health - resources "/cart_items", CartItemController, only: [:create, :delete] - get "/cart", CartController, :show - put "/cart", CartController, :update resources "/orders", OrderController, only: [:create, :show] resources "/archive", StreamController get "/streams", StreamController, :index - get "/stream", StreamController, :show + get "/streams/:id", StreamController, :show + get "/streams/new", StreamController, :new + post "/streams", StreamController, :create resources "/vods", VodController @@ -83,6 +68,8 @@ defmodule BrightWeb.Router do end + + # resources "/users", UserController resources "/tags", TagController @@ -128,39 +115,11 @@ defmodule BrightWeb.Router do ## Authentication routes - scope "/", BrightWeb do - pipe_through [:browser, :redirect_if_platform_is_authenticated] - - live_session :redirect_if_platform_is_authenticated, - on_mount: [{BrightWeb.PlatformAuth, :redirect_if_platform_is_authenticated}] do - live "/platforms/register", PlatformRegistrationLive, :new - live "/platforms/log_in", PlatformLoginLive, :new - live "/platforms/reset_password", PlatformForgotPasswordLive, :new - live "/platforms/reset_password/:token", PlatformResetPasswordLive, :edit - end - - post "/platforms/log_in", PlatformSessionController, :create - end - - scope "/", BrightWeb do - pipe_through [:browser, :require_authenticated_platform] - - live_session :require_authenticated_platform, - on_mount: [{BrightWeb.PlatformAuth, :ensure_authenticated}] do - live "/platforms/settings", PlatformSettingsLive, :edit - live "/platforms/settings/confirm_email/:token", PlatformSettingsLive, :confirm_email - end - end - scope "/", BrightWeb do pipe_through [:browser] - delete "/platforms/log_out", PlatformSessionController, :delete - - live_session :current_platform, - on_mount: [{BrightWeb.PlatformAuth, :mount_current_platform}] do - live "/platforms/confirm/:token", PlatformConfirmationLive, :edit - live "/platforms/confirm", PlatformConfirmationInstructionsLive, :new - end end + + + end diff --git a/services/bright/mix.exs b/services/bright/mix.exs index 2660234..0efccb2 100644 --- a/services/bright/mix.exs +++ b/services/bright/mix.exs @@ -56,12 +56,6 @@ defmodule Bright.MixProject do {:oban, "~> 2.17"}, {:mox, "~> 0.5.0", only: :test}, {:httpoison, "~> 2.0"} - # {:superstreamer_player, - # github: "superstreamerapp/superstreamer", - # app: false, - # compile: false, - # sparse: "packages/player" - # } ] end diff --git a/services/bright/mix.lock b/services/bright/mix.lock index 7deac38..cdfd8b7 100644 --- a/services/bright/mix.lock +++ b/services/bright/mix.lock @@ -1,14 +1,17 @@ %{ "bandit": {:hex, :bandit, "1.6.1", "9e01b93d72ddc21d8c576a704949e86ee6cde7d11270a1d3073787876527a48f", [:mix], [{:hpax, "~> 1.0", [hex: :hpax, repo: "hexpm", optional: false]}, {:plug, "~> 1.14", [hex: :plug, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}, {:thousand_island, "~> 1.0", [hex: :thousand_island, repo: "hexpm", optional: false]}, {:websock, "~> 0.5", [hex: :websock, repo: "hexpm", optional: false]}], "hexpm", "5a904bf010ea24b67979835e0507688e31ac873d4ffc8ed0e5413e8d77455031"}, + "bcrypt_elixir": {:hex, :bcrypt_elixir, "3.2.0", "feab711974beba4cb348147170346fe097eea2e840db4e012a145e180ed4ab75", [:make, :mix], [{:comeonin, "~> 5.3", [hex: :comeonin, repo: "hexpm", optional: false]}, {:elixir_make, "~> 0.6", [hex: :elixir_make, repo: "hexpm", optional: false]}], "hexpm", "563e92a6c77d667b19c5f4ba17ab6d440a085696bdf4c68b9b0f5b30bc5422b8"}, "bulma": {:hex, :bulma, "1.0.2", "50dfffe8d28b0bd527418560223b407f9e80e990e187e1653b17eff818f8fcbe", [:mix], [], "hexpm", "27745727ff7f451d140a2438c0ca4448bc8ca73e0a6d2d4f24e1b5b9ced8a774"}, "castore": {:hex, :castore, "1.0.10", "43bbeeac820f16c89f79721af1b3e092399b3a1ecc8df1a472738fd853574911", [:mix], [], "hexpm", "1b0b7ea14d889d9ea21202c43a4fa015eb913021cb535e8ed91946f4b77a8848"}, "certifi": {:hex, :certifi, "2.12.0", "2d1cca2ec95f59643862af91f001478c9863c2ac9cb6e2f89780bfd8de987329", [:rebar3], [], "hexpm", "ee68d85df22e554040cdb4be100f33873ac6051387baf6a8f6ce82272340ff1c"}, + "comeonin": {:hex, :comeonin, "5.5.0", "364d00df52545c44a139bad919d7eacb55abf39e86565878e17cebb787977368", [:mix], [], "hexpm", "6287fc3ba0aad34883cbe3f7949fc1d1e738e5ccdce77165bc99490aa69f47fb"}, "dart_sass": {:hex, :dart_sass, "0.7.0", "7979e056cb74fd6843e1c72db763cffc7726a9192a657735b7d24c0d9c26a1ce", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}], "hexpm", "4a8e70bca41aa00846398abdf5ad8a64d7907a0f7bf40145cd2e40d5971629f2"}, "db_connection": {:hex, :db_connection, "2.7.0", "b99faa9291bb09892c7da373bb82cba59aefa9b36300f6145c5f201c7adf48ec", [:mix], [{:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "dcf08f31b2701f857dfc787fbad78223d61a32204f217f15e881dd93e4bdd3ff"}, "decimal": {:hex, :decimal, "2.3.0", "3ad6255aa77b4a3c4f818171b12d237500e63525c2fd056699967a3e7ea20f62", [:mix], [], "hexpm", "a4d66355cb29cb47c3cf30e71329e58361cfcb37c34235ef3bf1d7bf3773aeac"}, "dns_cluster": {:hex, :dns_cluster, "0.1.3", "0bc20a2c88ed6cc494f2964075c359f8c2d00e1bf25518a6a6c7fd277c9b0c66", [:mix], [], "hexpm", "46cb7c4a1b3e52c7ad4cbe33ca5079fbde4840dedeafca2baf77996c2da1bc33"}, "ecto": {:hex, :ecto, "3.12.5", "4a312960ce612e17337e7cefcf9be45b95a3be6b36b6f94dfb3d8c361d631866", [:mix], [{:decimal, "~> 2.0", [hex: :decimal, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "6eb18e80bef8bb57e17f5a7f068a1719fbda384d40fc37acb8eb8aeca493b6ea"}, "ecto_sql": {:hex, :ecto_sql, "3.12.1", "c0d0d60e85d9ff4631f12bafa454bc392ce8b9ec83531a412c12a0d415a3a4d0", [:mix], [{:db_connection, "~> 2.4.1 or ~> 2.5", [hex: :db_connection, repo: "hexpm", optional: false]}, {:ecto, "~> 3.12", [hex: :ecto, repo: "hexpm", optional: false]}, {:myxql, "~> 0.7", [hex: :myxql, repo: "hexpm", optional: true]}, {:postgrex, "~> 0.19 or ~> 1.0", [hex: :postgrex, repo: "hexpm", optional: true]}, {:tds, "~> 2.1.1 or ~> 2.2", [hex: :tds, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4.0 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "aff5b958a899762c5f09028c847569f7dfb9cc9d63bdb8133bff8a5546de6bf5"}, + "elixir_make": {:hex, :elixir_make, "0.9.0", "6484b3cd8c0cee58f09f05ecaf1a140a8c97670671a6a0e7ab4dc326c3109726", [:mix], [], "hexpm", "db23d4fd8b757462ad02f8aa73431a426fe6671c80b200d9710caf3d1dd0ffdb"}, "esbuild": {:hex, :esbuild, "0.8.2", "5f379dfa383ef482b738e7771daf238b2d1cfb0222bef9d3b20d4c8f06c7a7ac", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}, {:jason, "~> 1.4", [hex: :jason, repo: "hexpm", optional: false]}], "hexpm", "558a8a08ed78eb820efbfda1de196569d8bfa9b51e8371a1934fbb31345feda7"}, "expo": {:hex, :expo, "1.1.0", "f7b9ed7fb5745ebe1eeedf3d6f29226c5dd52897ac67c0f8af62a07e661e5c75", [:mix], [], "hexpm", "fbadf93f4700fb44c331362177bdca9eeb8097e8b0ef525c9cc501cb9917c960"}, "file_system": {:hex, :file_system, "1.0.1", "79e8ceaddb0416f8b8cd02a0127bdbababe7bf4a23d2a395b983c1f8b3f73edd", [:mix], [], "hexpm", "4414d1f38863ddf9120720cd976fce5bdde8e91d8283353f0e31850fa89feb9e"}, diff --git a/services/bright/priv/repo/migrations/20250109070300_create_platforms_auth_tables.exs b/services/bright/priv/repo/migrations/20250109070300_create_platforms_auth_tables.exs deleted file mode 100644 index 97c54c1..0000000 --- a/services/bright/priv/repo/migrations/20250109070300_create_platforms_auth_tables.exs +++ /dev/null @@ -1,29 +0,0 @@ -defmodule Bright.Repo.Migrations.CreatePlatformsAuthTables do - use Ecto.Migration - - def change do - execute "CREATE EXTENSION IF NOT EXISTS citext", "" - - create table(:platforms) do - add :email, :citext, null: false - add :hashed_password, :string, null: false - add :confirmed_at, :utc_datetime - - timestamps(type: :utc_datetime) - end - - create unique_index(:platforms, [:email]) - - create table(:platforms_tokens) do - add :platform_id, references(:platforms, on_delete: :delete_all), null: false - add :token, :binary, null: false - add :context, :string, null: false - add :sent_to, :string - - timestamps(type: :utc_datetime, updated_at: false) - end - - create index(:platforms_tokens, [:platform_id]) - create unique_index(:platforms_tokens, [:context, :token]) - end -end diff --git a/services/bright/test/bright/platforms_test.exs b/services/bright/test/bright/platforms_test.exs index 28b2ccc..a6fc77f 100644 --- a/services/bright/test/bright/platforms_test.exs +++ b/services/bright/test/bright/platforms_test.exs @@ -3,58 +3,6 @@ defmodule Bright.PlatformsTest do alias Bright.Platforms - describe "platforms" do - alias Bright.Platforms.Platform - - import Bright.PlatformsFixtures - - @invalid_attrs %{} - - test "list_platforms/0 returns all platforms" do - platform = platform_fixture() - assert Platforms.list_platforms() == [platform] - end - - test "get_platform!/1 returns the platform with given id" do - platform = platform_fixture() - assert Platforms.get_platform!(platform.id) == platform - end - - test "create_platform/1 with valid data creates a platform" do - valid_attrs = %{} - - assert {:ok, %Platform{} = platform} = Platforms.create_platform(valid_attrs) - end - - test "create_platform/1 with invalid data returns error changeset" do - assert {:error, %Ecto.Changeset{}} = Platforms.create_platform(@invalid_attrs) - end - - test "update_platform/2 with valid data updates the platform" do - platform = platform_fixture() - update_attrs = %{} - - assert {:ok, %Platform{} = platform} = Platforms.update_platform(platform, update_attrs) - end - - test "update_platform/2 with invalid data returns error changeset" do - platform = platform_fixture() - assert {:error, %Ecto.Changeset{}} = Platforms.update_platform(platform, @invalid_attrs) - assert platform == Platforms.get_platform!(platform.id) - end - - test "delete_platform/1 deletes the platform" do - platform = platform_fixture() - assert {:ok, %Platform{}} = Platforms.delete_platform(platform) - assert_raise Ecto.NoResultsError, fn -> Platforms.get_platform!(platform.id) end - end - - test "change_platform/1 returns a platform changeset" do - platform = platform_fixture() - assert %Ecto.Changeset{} = Platforms.change_platform(platform) - end - end - describe "platforms" do alias Bright.Platforms.Platform @@ -73,12 +21,12 @@ defmodule Bright.PlatformsTest do end test "create_platform/1 with valid data creates a platform" do - valid_attrs = %{name: "some name", url: "some url", icon: "some icon"} + valid_attrs = %{name: "some name", url: true, icon: ""} assert {:ok, %Platform{} = platform} = Platforms.create_platform(valid_attrs) assert platform.name == "some name" - assert platform.url == "some url" - assert platform.icon == "some icon" + assert platform.url == true + assert platform.icon == "" end test "create_platform/1 with invalid data returns error changeset" do @@ -87,12 +35,12 @@ defmodule Bright.PlatformsTest do test "update_platform/2 with valid data updates the platform" do platform = platform_fixture() - update_attrs = %{name: "some updated name", url: "some updated url", icon: "some updated icon"} + update_attrs = %{name: "some updated name", url: false, icon: "blah"} assert {:ok, %Platform{} = platform} = Platforms.update_platform(platform, update_attrs) assert platform.name == "some updated name" - assert platform.url == "some updated url" - assert platform.icon == "some updated icon" + assert platform.url == false + assert platform.icon == "blah" end test "update_platform/2 with invalid data returns error changeset" do @@ -112,507 +60,4 @@ defmodule Bright.PlatformsTest do assert %Ecto.Changeset{} = Platforms.change_platform(platform) end end - - import Bright.PlatformsFixtures - alias Bright.Platforms.{Platform, PlatformToken} - - describe "get_platform_by_email/1" do - test "does not return the platform if the email does not exist" do - refute Platforms.get_platform_by_email("unknown@example.com") - end - - test "returns the platform if the email exists" do - %{id: id} = platform = platform_fixture() - assert %Platform{id: ^id} = Platforms.get_platform_by_email(platform.email) - end - end - - describe "get_platform_by_email_and_password/2" do - test "does not return the platform if the email does not exist" do - refute Platforms.get_platform_by_email_and_password("unknown@example.com", "hello world!") - end - - test "does not return the platform if the password is not valid" do - platform = platform_fixture() - refute Platforms.get_platform_by_email_and_password(platform.email, "invalid") - end - - test "returns the platform if the email and password are valid" do - %{id: id} = platform = platform_fixture() - - assert %Platform{id: ^id} = - Platforms.get_platform_by_email_and_password(platform.email, valid_platform_password()) - end - end - - describe "get_platform!/1" do - test "raises if id is invalid" do - assert_raise Ecto.NoResultsError, fn -> - Platforms.get_platform!(-1) - end - end - - test "returns the platform with the given id" do - %{id: id} = platform = platform_fixture() - assert %Platform{id: ^id} = Platforms.get_platform!(platform.id) - end - end - - describe "register_platform/1" do - test "requires email and password to be set" do - {:error, changeset} = Platforms.register_platform(%{}) - - assert %{ - password: ["can't be blank"], - email: ["can't be blank"] - } = errors_on(changeset) - end - - test "validates email and password when given" do - {:error, changeset} = Platforms.register_platform(%{email: "not valid", password: "not valid"}) - - assert %{ - email: ["must have the @ sign and no spaces"], - password: ["should be at least 12 character(s)"] - } = errors_on(changeset) - end - - test "validates maximum values for email and password for security" do - too_long = String.duplicate("db", 100) - {:error, changeset} = Platforms.register_platform(%{email: too_long, password: too_long}) - assert "should be at most 160 character(s)" in errors_on(changeset).email - assert "should be at most 72 character(s)" in errors_on(changeset).password - end - - test "validates email uniqueness" do - %{email: email} = platform_fixture() - {:error, changeset} = Platforms.register_platform(%{email: email}) - assert "has already been taken" in errors_on(changeset).email - - # Now try with the upper cased email too, to check that email case is ignored. - {:error, changeset} = Platforms.register_platform(%{email: String.upcase(email)}) - assert "has already been taken" in errors_on(changeset).email - end - - test "registers platforms with a hashed password" do - email = unique_platform_email() - {:ok, platform} = Platforms.register_platform(valid_platform_attributes(email: email)) - assert platform.email == email - assert is_binary(platform.hashed_password) - assert is_nil(platform.confirmed_at) - assert is_nil(platform.password) - end - end - - describe "change_platform_registration/2" do - test "returns a changeset" do - assert %Ecto.Changeset{} = changeset = Platforms.change_platform_registration(%Platform{}) - assert changeset.required == [:password, :email] - end - - test "allows fields to be set" do - email = unique_platform_email() - password = valid_platform_password() - - changeset = - Platforms.change_platform_registration( - %Platform{}, - valid_platform_attributes(email: email, password: password) - ) - - assert changeset.valid? - assert get_change(changeset, :email) == email - assert get_change(changeset, :password) == password - assert is_nil(get_change(changeset, :hashed_password)) - end - end - - describe "change_platform_email/2" do - test "returns a platform changeset" do - assert %Ecto.Changeset{} = changeset = Platforms.change_platform_email(%Platform{}) - assert changeset.required == [:email] - end - end - - describe "apply_platform_email/3" do - setup do - %{platform: platform_fixture()} - end - - test "requires email to change", %{platform: platform} do - {:error, changeset} = Platforms.apply_platform_email(platform, valid_platform_password(), %{}) - assert %{email: ["did not change"]} = errors_on(changeset) - end - - test "validates email", %{platform: platform} do - {:error, changeset} = - Platforms.apply_platform_email(platform, valid_platform_password(), %{email: "not valid"}) - - assert %{email: ["must have the @ sign and no spaces"]} = errors_on(changeset) - end - - test "validates maximum value for email for security", %{platform: platform} do - too_long = String.duplicate("db", 100) - - {:error, changeset} = - Platforms.apply_platform_email(platform, valid_platform_password(), %{email: too_long}) - - assert "should be at most 160 character(s)" in errors_on(changeset).email - end - - test "validates email uniqueness", %{platform: platform} do - %{email: email} = platform_fixture() - password = valid_platform_password() - - {:error, changeset} = Platforms.apply_platform_email(platform, password, %{email: email}) - - assert "has already been taken" in errors_on(changeset).email - end - - test "validates current password", %{platform: platform} do - {:error, changeset} = - Platforms.apply_platform_email(platform, "invalid", %{email: unique_platform_email()}) - - assert %{current_password: ["is not valid"]} = errors_on(changeset) - end - - test "applies the email without persisting it", %{platform: platform} do - email = unique_platform_email() - {:ok, platform} = Platforms.apply_platform_email(platform, valid_platform_password(), %{email: email}) - assert platform.email == email - assert Platforms.get_platform!(platform.id).email != email - end - end - - describe "deliver_platform_update_email_instructions/3" do - setup do - %{platform: platform_fixture()} - end - - test "sends token through notification", %{platform: platform} do - token = - extract_platform_token(fn url -> - Platforms.deliver_platform_update_email_instructions(platform, "current@example.com", url) - end) - - {:ok, token} = Base.url_decode64(token, padding: false) - assert platform_token = Repo.get_by(PlatformToken, token: :crypto.hash(:sha256, token)) - assert platform_token.platform_id == platform.id - assert platform_token.sent_to == platform.email - assert platform_token.context == "change:current@example.com" - end - end - - describe "update_platform_email/2" do - setup do - platform = platform_fixture() - email = unique_platform_email() - - token = - extract_platform_token(fn url -> - Platforms.deliver_platform_update_email_instructions(%{platform | email: email}, platform.email, url) - end) - - %{platform: platform, token: token, email: email} - end - - test "updates the email with a valid token", %{platform: platform, token: token, email: email} do - assert Platforms.update_platform_email(platform, token) == :ok - changed_platform = Repo.get!(Platform, platform.id) - assert changed_platform.email != platform.email - assert changed_platform.email == email - assert changed_platform.confirmed_at - assert changed_platform.confirmed_at != platform.confirmed_at - refute Repo.get_by(PlatformToken, platform_id: platform.id) - end - - test "does not update email with invalid token", %{platform: platform} do - assert Platforms.update_platform_email(platform, "oops") == :error - assert Repo.get!(Platform, platform.id).email == platform.email - assert Repo.get_by(PlatformToken, platform_id: platform.id) - end - - test "does not update email if platform email changed", %{platform: platform, token: token} do - assert Platforms.update_platform_email(%{platform | email: "current@example.com"}, token) == :error - assert Repo.get!(Platform, platform.id).email == platform.email - assert Repo.get_by(PlatformToken, platform_id: platform.id) - end - - test "does not update email if token expired", %{platform: platform, token: token} do - {1, nil} = Repo.update_all(PlatformToken, set: [inserted_at: ~N[2020-01-01 00:00:00]]) - assert Platforms.update_platform_email(platform, token) == :error - assert Repo.get!(Platform, platform.id).email == platform.email - assert Repo.get_by(PlatformToken, platform_id: platform.id) - end - end - - describe "change_platform_password/2" do - test "returns a platform changeset" do - assert %Ecto.Changeset{} = changeset = Platforms.change_platform_password(%Platform{}) - assert changeset.required == [:password] - end - - test "allows fields to be set" do - changeset = - Platforms.change_platform_password(%Platform{}, %{ - "password" => "new valid password" - }) - - assert changeset.valid? - assert get_change(changeset, :password) == "new valid password" - assert is_nil(get_change(changeset, :hashed_password)) - end - end - - describe "update_platform_password/3" do - setup do - %{platform: platform_fixture()} - end - - test "validates password", %{platform: platform} do - {:error, changeset} = - Platforms.update_platform_password(platform, valid_platform_password(), %{ - password: "not valid", - password_confirmation: "another" - }) - - assert %{ - password: ["should be at least 12 character(s)"], - password_confirmation: ["does not match password"] - } = errors_on(changeset) - end - - test "validates maximum values for password for security", %{platform: platform} do - too_long = String.duplicate("db", 100) - - {:error, changeset} = - Platforms.update_platform_password(platform, valid_platform_password(), %{password: too_long}) - - assert "should be at most 72 character(s)" in errors_on(changeset).password - end - - test "validates current password", %{platform: platform} do - {:error, changeset} = - Platforms.update_platform_password(platform, "invalid", %{password: valid_platform_password()}) - - assert %{current_password: ["is not valid"]} = errors_on(changeset) - end - - test "updates the password", %{platform: platform} do - {:ok, platform} = - Platforms.update_platform_password(platform, valid_platform_password(), %{ - password: "new valid password" - }) - - assert is_nil(platform.password) - assert Platforms.get_platform_by_email_and_password(platform.email, "new valid password") - end - - test "deletes all tokens for the given platform", %{platform: platform} do - _ = Platforms.generate_platform_session_token(platform) - - {:ok, _} = - Platforms.update_platform_password(platform, valid_platform_password(), %{ - password: "new valid password" - }) - - refute Repo.get_by(PlatformToken, platform_id: platform.id) - end - end - - describe "generate_platform_session_token/1" do - setup do - %{platform: platform_fixture()} - end - - test "generates a token", %{platform: platform} do - token = Platforms.generate_platform_session_token(platform) - assert platform_token = Repo.get_by(PlatformToken, token: token) - assert platform_token.context == "session" - - # Creating the same token for another platform should fail - assert_raise Ecto.ConstraintError, fn -> - Repo.insert!(%PlatformToken{ - token: platform_token.token, - platform_id: platform_fixture().id, - context: "session" - }) - end - end - end - - describe "get_platform_by_session_token/1" do - setup do - platform = platform_fixture() - token = Platforms.generate_platform_session_token(platform) - %{platform: platform, token: token} - end - - test "returns platform by token", %{platform: platform, token: token} do - assert session_platform = Platforms.get_platform_by_session_token(token) - assert session_platform.id == platform.id - end - - test "does not return platform for invalid token" do - refute Platforms.get_platform_by_session_token("oops") - end - - test "does not return platform for expired token", %{token: token} do - {1, nil} = Repo.update_all(PlatformToken, set: [inserted_at: ~N[2020-01-01 00:00:00]]) - refute Platforms.get_platform_by_session_token(token) - end - end - - describe "delete_platform_session_token/1" do - test "deletes the token" do - platform = platform_fixture() - token = Platforms.generate_platform_session_token(platform) - assert Platforms.delete_platform_session_token(token) == :ok - refute Platforms.get_platform_by_session_token(token) - end - end - - describe "deliver_platform_confirmation_instructions/2" do - setup do - %{platform: platform_fixture()} - end - - test "sends token through notification", %{platform: platform} do - token = - extract_platform_token(fn url -> - Platforms.deliver_platform_confirmation_instructions(platform, url) - end) - - {:ok, token} = Base.url_decode64(token, padding: false) - assert platform_token = Repo.get_by(PlatformToken, token: :crypto.hash(:sha256, token)) - assert platform_token.platform_id == platform.id - assert platform_token.sent_to == platform.email - assert platform_token.context == "confirm" - end - end - - describe "confirm_platform/1" do - setup do - platform = platform_fixture() - - token = - extract_platform_token(fn url -> - Platforms.deliver_platform_confirmation_instructions(platform, url) - end) - - %{platform: platform, token: token} - end - - test "confirms the email with a valid token", %{platform: platform, token: token} do - assert {:ok, confirmed_platform} = Platforms.confirm_platform(token) - assert confirmed_platform.confirmed_at - assert confirmed_platform.confirmed_at != platform.confirmed_at - assert Repo.get!(Platform, platform.id).confirmed_at - refute Repo.get_by(PlatformToken, platform_id: platform.id) - end - - test "does not confirm with invalid token", %{platform: platform} do - assert Platforms.confirm_platform("oops") == :error - refute Repo.get!(Platform, platform.id).confirmed_at - assert Repo.get_by(PlatformToken, platform_id: platform.id) - end - - test "does not confirm email if token expired", %{platform: platform, token: token} do - {1, nil} = Repo.update_all(PlatformToken, set: [inserted_at: ~N[2020-01-01 00:00:00]]) - assert Platforms.confirm_platform(token) == :error - refute Repo.get!(Platform, platform.id).confirmed_at - assert Repo.get_by(PlatformToken, platform_id: platform.id) - end - end - - describe "deliver_platform_reset_password_instructions/2" do - setup do - %{platform: platform_fixture()} - end - - test "sends token through notification", %{platform: platform} do - token = - extract_platform_token(fn url -> - Platforms.deliver_platform_reset_password_instructions(platform, url) - end) - - {:ok, token} = Base.url_decode64(token, padding: false) - assert platform_token = Repo.get_by(PlatformToken, token: :crypto.hash(:sha256, token)) - assert platform_token.platform_id == platform.id - assert platform_token.sent_to == platform.email - assert platform_token.context == "reset_password" - end - end - - describe "get_platform_by_reset_password_token/1" do - setup do - platform = platform_fixture() - - token = - extract_platform_token(fn url -> - Platforms.deliver_platform_reset_password_instructions(platform, url) - end) - - %{platform: platform, token: token} - end - - test "returns the platform with valid token", %{platform: %{id: id}, token: token} do - assert %Platform{id: ^id} = Platforms.get_platform_by_reset_password_token(token) - assert Repo.get_by(PlatformToken, platform_id: id) - end - - test "does not return the platform with invalid token", %{platform: platform} do - refute Platforms.get_platform_by_reset_password_token("oops") - assert Repo.get_by(PlatformToken, platform_id: platform.id) - end - - test "does not return the platform if token expired", %{platform: platform, token: token} do - {1, nil} = Repo.update_all(PlatformToken, set: [inserted_at: ~N[2020-01-01 00:00:00]]) - refute Platforms.get_platform_by_reset_password_token(token) - assert Repo.get_by(PlatformToken, platform_id: platform.id) - end - end - - describe "reset_platform_password/2" do - setup do - %{platform: platform_fixture()} - end - - test "validates password", %{platform: platform} do - {:error, changeset} = - Platforms.reset_platform_password(platform, %{ - password: "not valid", - password_confirmation: "another" - }) - - assert %{ - password: ["should be at least 12 character(s)"], - password_confirmation: ["does not match password"] - } = errors_on(changeset) - end - - test "validates maximum values for password for security", %{platform: platform} do - too_long = String.duplicate("db", 100) - {:error, changeset} = Platforms.reset_platform_password(platform, %{password: too_long}) - assert "should be at most 72 character(s)" in errors_on(changeset).password - end - - test "updates the password", %{platform: platform} do - {:ok, updated_platform} = Platforms.reset_platform_password(platform, %{password: "new valid password"}) - assert is_nil(updated_platform.password) - assert Platforms.get_platform_by_email_and_password(platform.email, "new valid password") - end - - test "deletes all tokens for the given platform", %{platform: platform} do - _ = Platforms.generate_platform_session_token(platform) - {:ok, _} = Platforms.reset_platform_password(platform, %{password: "new valid password"}) - refute Repo.get_by(PlatformToken, platform_id: platform.id) - end - end - - describe "inspect/2 for the Platform module" do - test "does not include password" do - refute inspect(%Platform{password: "123456"}) =~ "password: \"123456\"" - end - end end diff --git a/services/bright/test/bright_web/live/platform_confirmation_instructions_live_test.exs b/services/bright/test/bright_web/live/platform_confirmation_instructions_live_test.exs deleted file mode 100644 index 6f7df7c..0000000 --- a/services/bright/test/bright_web/live/platform_confirmation_instructions_live_test.exs +++ /dev/null @@ -1,67 +0,0 @@ -defmodule BrightWeb.PlatformConfirmationInstructionsLiveTest do - use BrightWeb.ConnCase, async: true - - import Phoenix.LiveViewTest - import Bright.PlatformsFixtures - - alias Bright.Platforms - alias Bright.Repo - - setup do - %{platform: platform_fixture()} - end - - describe "Resend confirmation" do - test "renders the resend confirmation page", %{conn: conn} do - {:ok, _lv, html} = live(conn, ~p"/platforms/confirm") - assert html =~ "Resend confirmation instructions" - end - - test "sends a new confirmation token", %{conn: conn, platform: platform} do - {:ok, lv, _html} = live(conn, ~p"/platforms/confirm") - - {:ok, conn} = - lv - |> form("#resend_confirmation_form", platform: %{email: platform.email}) - |> render_submit() - |> follow_redirect(conn, ~p"/") - - assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ - "If your email is in our system" - - assert Repo.get_by!(Platforms.PlatformToken, platform_id: platform.id).context == "confirm" - end - - test "does not send confirmation token if platform is confirmed", %{conn: conn, platform: platform} do - Repo.update!(Platforms.Platform.confirm_changeset(platform)) - - {:ok, lv, _html} = live(conn, ~p"/platforms/confirm") - - {:ok, conn} = - lv - |> form("#resend_confirmation_form", platform: %{email: platform.email}) - |> render_submit() - |> follow_redirect(conn, ~p"/") - - assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ - "If your email is in our system" - - refute Repo.get_by(Platforms.PlatformToken, platform_id: platform.id) - end - - test "does not send confirmation token if email is invalid", %{conn: conn} do - {:ok, lv, _html} = live(conn, ~p"/platforms/confirm") - - {:ok, conn} = - lv - |> form("#resend_confirmation_form", platform: %{email: "unknown@example.com"}) - |> render_submit() - |> follow_redirect(conn, ~p"/") - - assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ - "If your email is in our system" - - assert Repo.all(Platforms.PlatformToken) == [] - end - end -end diff --git a/services/bright/test/bright_web/live/platform_confirmation_live_test.exs b/services/bright/test/bright_web/live/platform_confirmation_live_test.exs deleted file mode 100644 index f58f27a..0000000 --- a/services/bright/test/bright_web/live/platform_confirmation_live_test.exs +++ /dev/null @@ -1,89 +0,0 @@ -defmodule BrightWeb.PlatformConfirmationLiveTest do - use BrightWeb.ConnCase, async: true - - import Phoenix.LiveViewTest - import Bright.PlatformsFixtures - - alias Bright.Platforms - alias Bright.Repo - - setup do - %{platform: platform_fixture()} - end - - describe "Confirm platform" do - test "renders confirmation page", %{conn: conn} do - {:ok, _lv, html} = live(conn, ~p"/platforms/confirm/some-token") - assert html =~ "Confirm Account" - end - - test "confirms the given token once", %{conn: conn, platform: platform} do - token = - extract_platform_token(fn url -> - Platforms.deliver_platform_confirmation_instructions(platform, url) - end) - - {:ok, lv, _html} = live(conn, ~p"/platforms/confirm/#{token}") - - result = - lv - |> form("#confirmation_form") - |> render_submit() - |> follow_redirect(conn, "/") - - assert {:ok, conn} = result - - assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ - "Platform confirmed successfully" - - assert Platforms.get_platform!(platform.id).confirmed_at - refute get_session(conn, :platform_token) - assert Repo.all(Platforms.PlatformToken) == [] - - # when not logged in - {:ok, lv, _html} = live(conn, ~p"/platforms/confirm/#{token}") - - result = - lv - |> form("#confirmation_form") - |> render_submit() - |> follow_redirect(conn, "/") - - assert {:ok, conn} = result - - assert Phoenix.Flash.get(conn.assigns.flash, :error) =~ - "Platform confirmation link is invalid or it has expired" - - # when logged in - conn = - build_conn() - |> log_in_platform(platform) - - {:ok, lv, _html} = live(conn, ~p"/platforms/confirm/#{token}") - - result = - lv - |> form("#confirmation_form") - |> render_submit() - |> follow_redirect(conn, "/") - - assert {:ok, conn} = result - refute Phoenix.Flash.get(conn.assigns.flash, :error) - end - - test "does not confirm email with invalid token", %{conn: conn, platform: platform} do - {:ok, lv, _html} = live(conn, ~p"/platforms/confirm/invalid-token") - - {:ok, conn} = - lv - |> form("#confirmation_form") - |> render_submit() - |> follow_redirect(conn, ~p"/") - - assert Phoenix.Flash.get(conn.assigns.flash, :error) =~ - "Platform confirmation link is invalid or it has expired" - - refute Platforms.get_platform!(platform.id).confirmed_at - end - end -end diff --git a/services/bright/test/bright_web/live/platform_forgot_password_live_test.exs b/services/bright/test/bright_web/live/platform_forgot_password_live_test.exs deleted file mode 100644 index bbcb069..0000000 --- a/services/bright/test/bright_web/live/platform_forgot_password_live_test.exs +++ /dev/null @@ -1,63 +0,0 @@ -defmodule BrightWeb.PlatformForgotPasswordLiveTest do - use BrightWeb.ConnCase, async: true - - import Phoenix.LiveViewTest - import Bright.PlatformsFixtures - - alias Bright.Platforms - alias Bright.Repo - - describe "Forgot password page" do - test "renders email page", %{conn: conn} do - {:ok, lv, html} = live(conn, ~p"/platforms/reset_password") - - assert html =~ "Forgot your password?" - assert has_element?(lv, ~s|a[href="#{~p"/platforms/register"}"]|, "Register") - assert has_element?(lv, ~s|a[href="#{~p"/platforms/log_in"}"]|, "Log in") - end - - test "redirects if already logged in", %{conn: conn} do - result = - conn - |> log_in_platform(platform_fixture()) - |> live(~p"/platforms/reset_password") - |> follow_redirect(conn, ~p"/") - - assert {:ok, _conn} = result - end - end - - describe "Reset link" do - setup do - %{platform: platform_fixture()} - end - - test "sends a new reset password token", %{conn: conn, platform: platform} do - {:ok, lv, _html} = live(conn, ~p"/platforms/reset_password") - - {:ok, conn} = - lv - |> form("#reset_password_form", platform: %{"email" => platform.email}) - |> render_submit() - |> follow_redirect(conn, "/") - - assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ "If your email is in our system" - - assert Repo.get_by!(Platforms.PlatformToken, platform_id: platform.id).context == - "reset_password" - end - - test "does not send reset password token if email is invalid", %{conn: conn} do - {:ok, lv, _html} = live(conn, ~p"/platforms/reset_password") - - {:ok, conn} = - lv - |> form("#reset_password_form", platform: %{"email" => "unknown@example.com"}) - |> render_submit() - |> follow_redirect(conn, "/") - - assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ "If your email is in our system" - assert Repo.all(Platforms.PlatformToken) == [] - end - end -end diff --git a/services/bright/test/bright_web/live/platform_login_live_test.exs b/services/bright/test/bright_web/live/platform_login_live_test.exs deleted file mode 100644 index 0aaa7d7..0000000 --- a/services/bright/test/bright_web/live/platform_login_live_test.exs +++ /dev/null @@ -1,87 +0,0 @@ -defmodule BrightWeb.PlatformLoginLiveTest do - use BrightWeb.ConnCase, async: true - - import Phoenix.LiveViewTest - import Bright.PlatformsFixtures - - describe "Log in page" do - test "renders log in page", %{conn: conn} do - {:ok, _lv, html} = live(conn, ~p"/platforms/log_in") - - assert html =~ "Log in" - assert html =~ "Register" - assert html =~ "Forgot your password?" - end - - test "redirects if already logged in", %{conn: conn} do - result = - conn - |> log_in_platform(platform_fixture()) - |> live(~p"/platforms/log_in") - |> follow_redirect(conn, "/") - - assert {:ok, _conn} = result - end - end - - describe "platform login" do - test "redirects if platform login with valid credentials", %{conn: conn} do - password = "123456789abcd" - platform = platform_fixture(%{password: password}) - - {:ok, lv, _html} = live(conn, ~p"/platforms/log_in") - - form = - form(lv, "#login_form", platform: %{email: platform.email, password: password, remember_me: true}) - - conn = submit_form(form, conn) - - assert redirected_to(conn) == ~p"/" - end - - test "redirects to login page with a flash error if there are no valid credentials", %{ - conn: conn - } do - {:ok, lv, _html} = live(conn, ~p"/platforms/log_in") - - form = - form(lv, "#login_form", - platform: %{email: "test@email.com", password: "123456", remember_me: true} - ) - - conn = submit_form(form, conn) - - assert Phoenix.Flash.get(conn.assigns.flash, :error) == "Invalid email or password" - - assert redirected_to(conn) == "/platforms/log_in" - end - end - - describe "login navigation" do - test "redirects to registration page when the Register button is clicked", %{conn: conn} do - {:ok, lv, _html} = live(conn, ~p"/platforms/log_in") - - {:ok, _login_live, login_html} = - lv - |> element(~s|main a:fl-contains("Sign up")|) - |> render_click() - |> follow_redirect(conn, ~p"/platforms/register") - - assert login_html =~ "Register" - end - - test "redirects to forgot password page when the Forgot Password button is clicked", %{ - conn: conn - } do - {:ok, lv, _html} = live(conn, ~p"/platforms/log_in") - - {:ok, conn} = - lv - |> element(~s|main a:fl-contains("Forgot your password?")|) - |> render_click() - |> follow_redirect(conn, ~p"/platforms/reset_password") - - assert conn.resp_body =~ "Forgot your password?" - end - end -end diff --git a/services/bright/test/bright_web/live/platform_registration_live_test.exs b/services/bright/test/bright_web/live/platform_registration_live_test.exs deleted file mode 100644 index 9a6e82c..0000000 --- a/services/bright/test/bright_web/live/platform_registration_live_test.exs +++ /dev/null @@ -1,87 +0,0 @@ -defmodule BrightWeb.PlatformRegistrationLiveTest do - use BrightWeb.ConnCase, async: true - - import Phoenix.LiveViewTest - import Bright.PlatformsFixtures - - describe "Registration page" do - test "renders registration page", %{conn: conn} do - {:ok, _lv, html} = live(conn, ~p"/platforms/register") - - assert html =~ "Register" - assert html =~ "Log in" - end - - test "redirects if already logged in", %{conn: conn} do - result = - conn - |> log_in_platform(platform_fixture()) - |> live(~p"/platforms/register") - |> follow_redirect(conn, "/") - - assert {:ok, _conn} = result - end - - test "renders errors for invalid data", %{conn: conn} do - {:ok, lv, _html} = live(conn, ~p"/platforms/register") - - result = - lv - |> element("#registration_form") - |> render_change(platform: %{"email" => "with spaces", "password" => "too short"}) - - assert result =~ "Register" - assert result =~ "must have the @ sign and no spaces" - assert result =~ "should be at least 12 character" - end - end - - describe "register platform" do - test "creates account and logs the platform in", %{conn: conn} do - {:ok, lv, _html} = live(conn, ~p"/platforms/register") - - email = unique_platform_email() - form = form(lv, "#registration_form", platform: valid_platform_attributes(email: email)) - render_submit(form) - conn = follow_trigger_action(form, conn) - - assert redirected_to(conn) == ~p"/" - - # Now do a logged in request and assert on the menu - conn = get(conn, "/") - response = html_response(conn, 200) - assert response =~ email - assert response =~ "Settings" - assert response =~ "Log out" - end - - test "renders errors for duplicated email", %{conn: conn} do - {:ok, lv, _html} = live(conn, ~p"/platforms/register") - - platform = platform_fixture(%{email: "test@email.com"}) - - result = - lv - |> form("#registration_form", - platform: %{"email" => platform.email, "password" => "valid_password"} - ) - |> render_submit() - - assert result =~ "has already been taken" - end - end - - describe "registration navigation" do - test "redirects to login page when the Log in button is clicked", %{conn: conn} do - {:ok, lv, _html} = live(conn, ~p"/platforms/register") - - {:ok, _login_live, login_html} = - lv - |> element(~s|main a:fl-contains("Log in")|) - |> render_click() - |> follow_redirect(conn, ~p"/platforms/log_in") - - assert login_html =~ "Log in" - end - end -end diff --git a/services/bright/test/bright_web/live/platform_reset_password_live_test.exs b/services/bright/test/bright_web/live/platform_reset_password_live_test.exs deleted file mode 100644 index f10ea66..0000000 --- a/services/bright/test/bright_web/live/platform_reset_password_live_test.exs +++ /dev/null @@ -1,118 +0,0 @@ -defmodule BrightWeb.PlatformResetPasswordLiveTest do - use BrightWeb.ConnCase, async: true - - import Phoenix.LiveViewTest - import Bright.PlatformsFixtures - - alias Bright.Platforms - - setup do - platform = platform_fixture() - - token = - extract_platform_token(fn url -> - Platforms.deliver_platform_reset_password_instructions(platform, url) - end) - - %{token: token, platform: platform} - end - - describe "Reset password page" do - test "renders reset password with valid token", %{conn: conn, token: token} do - {:ok, _lv, html} = live(conn, ~p"/platforms/reset_password/#{token}") - - assert html =~ "Reset Password" - end - - test "does not render reset password with invalid token", %{conn: conn} do - {:error, {:redirect, to}} = live(conn, ~p"/platforms/reset_password/invalid") - - assert to == %{ - flash: %{"error" => "Reset password link is invalid or it has expired."}, - to: ~p"/" - } - end - - test "renders errors for invalid data", %{conn: conn, token: token} do - {:ok, lv, _html} = live(conn, ~p"/platforms/reset_password/#{token}") - - result = - lv - |> element("#reset_password_form") - |> render_change( - platform: %{"password" => "secret12", "password_confirmation" => "secret123456"} - ) - - assert result =~ "should be at least 12 character" - assert result =~ "does not match password" - end - end - - describe "Reset Password" do - test "resets password once", %{conn: conn, token: token, platform: platform} do - {:ok, lv, _html} = live(conn, ~p"/platforms/reset_password/#{token}") - - {:ok, conn} = - lv - |> form("#reset_password_form", - platform: %{ - "password" => "new valid password", - "password_confirmation" => "new valid password" - } - ) - |> render_submit() - |> follow_redirect(conn, ~p"/platforms/log_in") - - refute get_session(conn, :platform_token) - assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ "Password reset successfully" - assert Platforms.get_platform_by_email_and_password(platform.email, "new valid password") - end - - test "does not reset password on invalid data", %{conn: conn, token: token} do - {:ok, lv, _html} = live(conn, ~p"/platforms/reset_password/#{token}") - - result = - lv - |> form("#reset_password_form", - platform: %{ - "password" => "too short", - "password_confirmation" => "does not match" - } - ) - |> render_submit() - - assert result =~ "Reset Password" - assert result =~ "should be at least 12 character(s)" - assert result =~ "does not match password" - end - end - - describe "Reset password navigation" do - test "redirects to login page when the Log in button is clicked", %{conn: conn, token: token} do - {:ok, lv, _html} = live(conn, ~p"/platforms/reset_password/#{token}") - - {:ok, conn} = - lv - |> element(~s|main a:fl-contains("Log in")|) - |> render_click() - |> follow_redirect(conn, ~p"/platforms/log_in") - - assert conn.resp_body =~ "Log in" - end - - test "redirects to registration page when the Register button is clicked", %{ - conn: conn, - token: token - } do - {:ok, lv, _html} = live(conn, ~p"/platforms/reset_password/#{token}") - - {:ok, conn} = - lv - |> element(~s|main a:fl-contains("Register")|) - |> render_click() - |> follow_redirect(conn, ~p"/platforms/register") - - assert conn.resp_body =~ "Register" - end - end -end diff --git a/services/bright/test/bright_web/live/platform_settings_live_test.exs b/services/bright/test/bright_web/live/platform_settings_live_test.exs deleted file mode 100644 index d7e7853..0000000 --- a/services/bright/test/bright_web/live/platform_settings_live_test.exs +++ /dev/null @@ -1,210 +0,0 @@ -defmodule BrightWeb.PlatformSettingsLiveTest do - use BrightWeb.ConnCase, async: true - - alias Bright.Platforms - import Phoenix.LiveViewTest - import Bright.PlatformsFixtures - - describe "Settings page" do - test "renders settings page", %{conn: conn} do - {:ok, _lv, html} = - conn - |> log_in_platform(platform_fixture()) - |> live(~p"/platforms/settings") - - assert html =~ "Change Email" - assert html =~ "Change Password" - end - - test "redirects if platform is not logged in", %{conn: conn} do - assert {:error, redirect} = live(conn, ~p"/platforms/settings") - - assert {:redirect, %{to: path, flash: flash}} = redirect - assert path == ~p"/platforms/log_in" - assert %{"error" => "You must log in to access this page."} = flash - end - end - - describe "update email form" do - setup %{conn: conn} do - password = valid_platform_password() - platform = platform_fixture(%{password: password}) - %{conn: log_in_platform(conn, platform), platform: platform, password: password} - end - - test "updates the platform email", %{conn: conn, password: password, platform: platform} do - new_email = unique_platform_email() - - {:ok, lv, _html} = live(conn, ~p"/platforms/settings") - - result = - lv - |> form("#email_form", %{ - "current_password" => password, - "platform" => %{"email" => new_email} - }) - |> render_submit() - - assert result =~ "A link to confirm your email" - assert Platforms.get_platform_by_email(platform.email) - end - - test "renders errors with invalid data (phx-change)", %{conn: conn} do - {:ok, lv, _html} = live(conn, ~p"/platforms/settings") - - result = - lv - |> element("#email_form") - |> render_change(%{ - "action" => "update_email", - "current_password" => "invalid", - "platform" => %{"email" => "with spaces"} - }) - - assert result =~ "Change Email" - assert result =~ "must have the @ sign and no spaces" - end - - test "renders errors with invalid data (phx-submit)", %{conn: conn, platform: platform} do - {:ok, lv, _html} = live(conn, ~p"/platforms/settings") - - result = - lv - |> form("#email_form", %{ - "current_password" => "invalid", - "platform" => %{"email" => platform.email} - }) - |> render_submit() - - assert result =~ "Change Email" - assert result =~ "did not change" - assert result =~ "is not valid" - end - end - - describe "update password form" do - setup %{conn: conn} do - password = valid_platform_password() - platform = platform_fixture(%{password: password}) - %{conn: log_in_platform(conn, platform), platform: platform, password: password} - end - - test "updates the platform password", %{conn: conn, platform: platform, password: password} do - new_password = valid_platform_password() - - {:ok, lv, _html} = live(conn, ~p"/platforms/settings") - - form = - form(lv, "#password_form", %{ - "current_password" => password, - "platform" => %{ - "email" => platform.email, - "password" => new_password, - "password_confirmation" => new_password - } - }) - - render_submit(form) - - new_password_conn = follow_trigger_action(form, conn) - - assert redirected_to(new_password_conn) == ~p"/platforms/settings" - - assert get_session(new_password_conn, :platform_token) != get_session(conn, :platform_token) - - assert Phoenix.Flash.get(new_password_conn.assigns.flash, :info) =~ - "Password updated successfully" - - assert Platforms.get_platform_by_email_and_password(platform.email, new_password) - end - - test "renders errors with invalid data (phx-change)", %{conn: conn} do - {:ok, lv, _html} = live(conn, ~p"/platforms/settings") - - result = - lv - |> element("#password_form") - |> render_change(%{ - "current_password" => "invalid", - "platform" => %{ - "password" => "too short", - "password_confirmation" => "does not match" - } - }) - - assert result =~ "Change Password" - assert result =~ "should be at least 12 character(s)" - assert result =~ "does not match password" - end - - test "renders errors with invalid data (phx-submit)", %{conn: conn} do - {:ok, lv, _html} = live(conn, ~p"/platforms/settings") - - result = - lv - |> form("#password_form", %{ - "current_password" => "invalid", - "platform" => %{ - "password" => "too short", - "password_confirmation" => "does not match" - } - }) - |> render_submit() - - assert result =~ "Change Password" - assert result =~ "should be at least 12 character(s)" - assert result =~ "does not match password" - assert result =~ "is not valid" - end - end - - describe "confirm email" do - setup %{conn: conn} do - platform = platform_fixture() - email = unique_platform_email() - - token = - extract_platform_token(fn url -> - Platforms.deliver_platform_update_email_instructions(%{platform | email: email}, platform.email, url) - end) - - %{conn: log_in_platform(conn, platform), token: token, email: email, platform: platform} - end - - test "updates the platform email once", %{conn: conn, platform: platform, token: token, email: email} do - {:error, redirect} = live(conn, ~p"/platforms/settings/confirm_email/#{token}") - - assert {:live_redirect, %{to: path, flash: flash}} = redirect - assert path == ~p"/platforms/settings" - assert %{"info" => message} = flash - assert message == "Email changed successfully." - refute Platforms.get_platform_by_email(platform.email) - assert Platforms.get_platform_by_email(email) - - # use confirm token again - {:error, redirect} = live(conn, ~p"/platforms/settings/confirm_email/#{token}") - assert {:live_redirect, %{to: path, flash: flash}} = redirect - assert path == ~p"/platforms/settings" - assert %{"error" => message} = flash - assert message == "Email change link is invalid or it has expired." - end - - test "does not update email with invalid token", %{conn: conn, platform: platform} do - {:error, redirect} = live(conn, ~p"/platforms/settings/confirm_email/oops") - assert {:live_redirect, %{to: path, flash: flash}} = redirect - assert path == ~p"/platforms/settings" - assert %{"error" => message} = flash - assert message == "Email change link is invalid or it has expired." - assert Platforms.get_platform_by_email(platform.email) - end - - test "redirects if platform is not logged in", %{token: token} do - conn = build_conn() - {:error, redirect} = live(conn, ~p"/platforms/settings/confirm_email/#{token}") - assert {:redirect, %{to: path, flash: flash}} = redirect - assert path == ~p"/platforms/log_in" - assert %{"error" => message} = flash - assert message == "You must log in to access this page." - end - end -end diff --git a/services/bright/test/bright_web/platform_auth_test.exs b/services/bright/test/bright_web/platform_auth_test.exs deleted file mode 100644 index d041a63..0000000 --- a/services/bright/test/bright_web/platform_auth_test.exs +++ /dev/null @@ -1,272 +0,0 @@ -defmodule BrightWeb.PlatformAuthTest do - use BrightWeb.ConnCase, async: true - - alias Phoenix.LiveView - alias Bright.Platforms - alias BrightWeb.PlatformAuth - import Bright.PlatformsFixtures - - @remember_me_cookie "_bright_web_platform_remember_me" - - setup %{conn: conn} do - conn = - conn - |> Map.replace!(:secret_key_base, BrightWeb.Endpoint.config(:secret_key_base)) - |> init_test_session(%{}) - - %{platform: platform_fixture(), conn: conn} - end - - describe "log_in_platform/3" do - test "stores the platform token in the session", %{conn: conn, platform: platform} do - conn = PlatformAuth.log_in_platform(conn, platform) - assert token = get_session(conn, :platform_token) - assert get_session(conn, :live_socket_id) == "platforms_sessions:#{Base.url_encode64(token)}" - assert redirected_to(conn) == ~p"/" - assert Platforms.get_platform_by_session_token(token) - end - - test "clears everything previously stored in the session", %{conn: conn, platform: platform} do - conn = conn |> put_session(:to_be_removed, "value") |> PlatformAuth.log_in_platform(platform) - refute get_session(conn, :to_be_removed) - end - - test "redirects to the configured path", %{conn: conn, platform: platform} do - conn = conn |> put_session(:platform_return_to, "/hello") |> PlatformAuth.log_in_platform(platform) - assert redirected_to(conn) == "/hello" - end - - test "writes a cookie if remember_me is configured", %{conn: conn, platform: platform} do - conn = conn |> fetch_cookies() |> PlatformAuth.log_in_platform(platform, %{"remember_me" => "true"}) - assert get_session(conn, :platform_token) == conn.cookies[@remember_me_cookie] - - assert %{value: signed_token, max_age: max_age} = conn.resp_cookies[@remember_me_cookie] - assert signed_token != get_session(conn, :platform_token) - assert max_age == 5_184_000 - end - end - - describe "logout_platform/1" do - test "erases session and cookies", %{conn: conn, platform: platform} do - platform_token = Platforms.generate_platform_session_token(platform) - - conn = - conn - |> put_session(:platform_token, platform_token) - |> put_req_cookie(@remember_me_cookie, platform_token) - |> fetch_cookies() - |> PlatformAuth.log_out_platform() - - refute get_session(conn, :platform_token) - refute conn.cookies[@remember_me_cookie] - assert %{max_age: 0} = conn.resp_cookies[@remember_me_cookie] - assert redirected_to(conn) == ~p"/" - refute Platforms.get_platform_by_session_token(platform_token) - end - - test "broadcasts to the given live_socket_id", %{conn: conn} do - live_socket_id = "platforms_sessions:abcdef-token" - BrightWeb.Endpoint.subscribe(live_socket_id) - - conn - |> put_session(:live_socket_id, live_socket_id) - |> PlatformAuth.log_out_platform() - - assert_receive %Phoenix.Socket.Broadcast{event: "disconnect", topic: ^live_socket_id} - end - - test "works even if platform is already logged out", %{conn: conn} do - conn = conn |> fetch_cookies() |> PlatformAuth.log_out_platform() - refute get_session(conn, :platform_token) - assert %{max_age: 0} = conn.resp_cookies[@remember_me_cookie] - assert redirected_to(conn) == ~p"/" - end - end - - describe "fetch_current_platform/2" do - test "authenticates platform from session", %{conn: conn, platform: platform} do - platform_token = Platforms.generate_platform_session_token(platform) - conn = conn |> put_session(:platform_token, platform_token) |> PlatformAuth.fetch_current_platform([]) - assert conn.assigns.current_platform.id == platform.id - end - - test "authenticates platform from cookies", %{conn: conn, platform: platform} do - logged_in_conn = - conn |> fetch_cookies() |> PlatformAuth.log_in_platform(platform, %{"remember_me" => "true"}) - - platform_token = logged_in_conn.cookies[@remember_me_cookie] - %{value: signed_token} = logged_in_conn.resp_cookies[@remember_me_cookie] - - conn = - conn - |> put_req_cookie(@remember_me_cookie, signed_token) - |> PlatformAuth.fetch_current_platform([]) - - assert conn.assigns.current_platform.id == platform.id - assert get_session(conn, :platform_token) == platform_token - - assert get_session(conn, :live_socket_id) == - "platforms_sessions:#{Base.url_encode64(platform_token)}" - end - - test "does not authenticate if data is missing", %{conn: conn, platform: platform} do - _ = Platforms.generate_platform_session_token(platform) - conn = PlatformAuth.fetch_current_platform(conn, []) - refute get_session(conn, :platform_token) - refute conn.assigns.current_platform - end - end - - describe "on_mount :mount_current_platform" do - test "assigns current_platform based on a valid platform_token", %{conn: conn, platform: platform} do - platform_token = Platforms.generate_platform_session_token(platform) - session = conn |> put_session(:platform_token, platform_token) |> get_session() - - {:cont, updated_socket} = - PlatformAuth.on_mount(:mount_current_platform, %{}, session, %LiveView.Socket{}) - - assert updated_socket.assigns.current_platform.id == platform.id - end - - test "assigns nil to current_platform assign if there isn't a valid platform_token", %{conn: conn} do - platform_token = "invalid_token" - session = conn |> put_session(:platform_token, platform_token) |> get_session() - - {:cont, updated_socket} = - PlatformAuth.on_mount(:mount_current_platform, %{}, session, %LiveView.Socket{}) - - assert updated_socket.assigns.current_platform == nil - end - - test "assigns nil to current_platform assign if there isn't a platform_token", %{conn: conn} do - session = conn |> get_session() - - {:cont, updated_socket} = - PlatformAuth.on_mount(:mount_current_platform, %{}, session, %LiveView.Socket{}) - - assert updated_socket.assigns.current_platform == nil - end - end - - describe "on_mount :ensure_authenticated" do - test "authenticates current_platform based on a valid platform_token", %{conn: conn, platform: platform} do - platform_token = Platforms.generate_platform_session_token(platform) - session = conn |> put_session(:platform_token, platform_token) |> get_session() - - {:cont, updated_socket} = - PlatformAuth.on_mount(:ensure_authenticated, %{}, session, %LiveView.Socket{}) - - assert updated_socket.assigns.current_platform.id == platform.id - end - - test "redirects to login page if there isn't a valid platform_token", %{conn: conn} do - platform_token = "invalid_token" - session = conn |> put_session(:platform_token, platform_token) |> get_session() - - socket = %LiveView.Socket{ - endpoint: BrightWeb.Endpoint, - assigns: %{__changed__: %{}, flash: %{}} - } - - {:halt, updated_socket} = PlatformAuth.on_mount(:ensure_authenticated, %{}, session, socket) - assert updated_socket.assigns.current_platform == nil - end - - test "redirects to login page if there isn't a platform_token", %{conn: conn} do - session = conn |> get_session() - - socket = %LiveView.Socket{ - endpoint: BrightWeb.Endpoint, - assigns: %{__changed__: %{}, flash: %{}} - } - - {:halt, updated_socket} = PlatformAuth.on_mount(:ensure_authenticated, %{}, session, socket) - assert updated_socket.assigns.current_platform == nil - end - end - - describe "on_mount :redirect_if_platform_is_authenticated" do - test "redirects if there is an authenticated platform ", %{conn: conn, platform: platform} do - platform_token = Platforms.generate_platform_session_token(platform) - session = conn |> put_session(:platform_token, platform_token) |> get_session() - - assert {:halt, _updated_socket} = - PlatformAuth.on_mount( - :redirect_if_platform_is_authenticated, - %{}, - session, - %LiveView.Socket{} - ) - end - - test "doesn't redirect if there is no authenticated platform", %{conn: conn} do - session = conn |> get_session() - - assert {:cont, _updated_socket} = - PlatformAuth.on_mount( - :redirect_if_platform_is_authenticated, - %{}, - session, - %LiveView.Socket{} - ) - end - end - - describe "redirect_if_platform_is_authenticated/2" do - test "redirects if platform is authenticated", %{conn: conn, platform: platform} do - conn = conn |> assign(:current_platform, platform) |> PlatformAuth.redirect_if_platform_is_authenticated([]) - assert conn.halted - assert redirected_to(conn) == ~p"/" - end - - test "does not redirect if platform is not authenticated", %{conn: conn} do - conn = PlatformAuth.redirect_if_platform_is_authenticated(conn, []) - refute conn.halted - refute conn.status - end - end - - describe "require_authenticated_platform/2" do - test "redirects if platform is not authenticated", %{conn: conn} do - conn = conn |> fetch_flash() |> PlatformAuth.require_authenticated_platform([]) - assert conn.halted - - assert redirected_to(conn) == ~p"/platforms/log_in" - - assert Phoenix.Flash.get(conn.assigns.flash, :error) == - "You must log in to access this page." - end - - test "stores the path to redirect to on GET", %{conn: conn} do - halted_conn = - %{conn | path_info: ["foo"], query_string: ""} - |> fetch_flash() - |> PlatformAuth.require_authenticated_platform([]) - - assert halted_conn.halted - assert get_session(halted_conn, :platform_return_to) == "/foo" - - halted_conn = - %{conn | path_info: ["foo"], query_string: "bar=baz"} - |> fetch_flash() - |> PlatformAuth.require_authenticated_platform([]) - - assert halted_conn.halted - assert get_session(halted_conn, :platform_return_to) == "/foo?bar=baz" - - halted_conn = - %{conn | path_info: ["foo"], query_string: "bar", method: "POST"} - |> fetch_flash() - |> PlatformAuth.require_authenticated_platform([]) - - assert halted_conn.halted - refute get_session(halted_conn, :platform_return_to) - end - - test "does not redirect if platform is authenticated", %{conn: conn, platform: platform} do - conn = conn |> assign(:current_platform, platform) |> PlatformAuth.require_authenticated_platform([]) - refute conn.halted - refute conn.status - end - end -end diff --git a/services/capture/.gitignore b/services/capture/.gitignore deleted file mode 100644 index c4273d6..0000000 --- a/services/capture/.gitignore +++ /dev/null @@ -1,153 +0,0 @@ -.pnp.* -.yarn/* -!.yarn/patches -!.yarn/plugins -!.yarn/releases -!.yarn/sdks -!.yarn/versions - - -# Created by https://www.toptal.com/developers/gitignore/api/node -# Edit at https://www.toptal.com/developers/gitignore?templates=node - -### Node ### -# Logs -logs -*.log -npm-debug.log* -yarn-debug.log* -yarn-error.log* -lerna-debug.log* -.pnpm-debug.log* - -# Diagnostic reports (https://nodejs.org/api/report.html) -report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json - -# Runtime data -pids -*.pid -*.seed -*.pid.lock - -# Directory for instrumented libs generated by jscoverage/JSCover -lib-cov - -# Coverage directory used by tools like istanbul -coverage -*.lcov - -# nyc test coverage -.nyc_output - -# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) -.grunt - -# Bower dependency directory (https://bower.io/) -bower_components - -# node-waf configuration -.lock-wscript - -# Compiled binary addons (https://nodejs.org/api/addons.html) -build/Release - -# Dependency directories -node_modules/ -jspm_packages/ - -# Snowpack dependency directory (https://snowpack.dev/) -web_modules/ - -# TypeScript cache -*.tsbuildinfo - -# Optional npm cache directory -.npm - -# Optional eslint cache -.eslintcache - -# Optional stylelint cache -.stylelintcache - -# Microbundle cache -.rpt2_cache/ -.rts2_cache_cjs/ -.rts2_cache_es/ -.rts2_cache_umd/ - -# Optional REPL history -.node_repl_history - -# Output of 'npm pack' -*.tgz - -# Yarn Integrity file -.yarn-integrity - -# dotenv environment variable files -.env -.env.development.local -.env.test.local -.env.production.local -.env.local - -# parcel-bundler cache (https://parceljs.org/) -.cache -.parcel-cache - -# Next.js build output -.next -out - -# Nuxt.js build / generate output -.nuxt -dist - -# Gatsby files -.cache/ -# Comment in the public line in if your project uses Gatsby and not Next.js -# https://nextjs.org/blog/next-9-1#public-directory-support -# public - -# vuepress build output -.vuepress/dist - -# vuepress v2.x temp and cache directory -.temp - -# Docusaurus cache and generated files -.docusaurus - -# Serverless directories -.serverless/ - -# FuseBox cache -.fusebox/ - -# DynamoDB Local files -.dynamodb/ - -# TernJS port file -.tern-port - -# Stores VSCode versions used for testing VSCode extensions -.vscode-test - -# yarn v2 -.yarn/cache -.yarn/unplugged -.yarn/build-state.yml -.yarn/install-state.gz -.pnp.* - -### Node Patch ### -# Serverless Webpack directories -.webpack/ - -# Optional stylelint cache - -# SvelteKit build / generate output -.svelte-kit - -# End of https://www.toptal.com/developers/gitignore/api/node diff --git a/services/capture/.mocharc.json b/services/capture/.mocharc.json deleted file mode 100644 index a169700..0000000 --- a/services/capture/.mocharc.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "$schema": "https://json.schemastore.org/mocharc.json", - "extensions": ["ts"], - "spec": "./src/**/*.spec.ts", - "require": "tsx" -} \ No newline at end of file diff --git a/services/capture/README.md b/services/capture/README.md deleted file mode 100644 index f423bc9..0000000 --- a/services/capture/README.md +++ /dev/null @@ -1,48 +0,0 @@ -# @futureporn/capture - -Cloud based wrapper around yt-dlp - -There are two components in this package. api and worker. - -## API components - -API container serves a REST API. Use this API to start/stop recordings. The API component doesn't actually record, it just queues the record task which a worker then picks up. When run in k8s, this container is meant to have 1 replication. - -## Worker components - -Worker container runs a Graphile Worker which listens for specific tasks related to recording. This container is meant to have 1 or more replications. - -## Issues - -* [ ] Initial Capture worker UI updates are too slow; the worker begins work without notifying the admin. -* [ ] Too much RAM usage! Array buffer size just keeps growing until OOMKilled while the upload is in progress. - -## Misc dev notes - -### idea for taking snapshots of stream in progress - -https://ofarukcaki.medium.com/producing-real-time-video-with-node-js-and-ffmpeg-a59ac27461a1 - -### youtube-dl end of stream output - -The end-of-stream output from yt-dlp when recording CB looks like this - -``` -[https @ 0x5646887f1580] Opening 'https://edge11-lax.live.mmcdn.com/live-hls/amlst:hotfallingdevil-sd-fdf87e5b6c880e52d38e8c94f8ebf8728c980a91d56fb4ace13748ba59012336_trns_h264/chunklist_w881713853_b5128000_t64RlBTOjMwLjA=.m3u8' for reading -[hls @ 0x564687dd0980] Skip ('#EXT-X-VERSION:4') -[hls @ 0x564687dd0980] Skip ('#EXT-X-DISCONTINUITY-SEQUENCE:0') -[hls @ 0x564687dd0980] Skip ('#EXT-X-PROGRAM-DATE-TIME:2023-01-31T17:48:45.947+00:00') -[https @ 0x5646880bf880] Opening 'https://edge11-lax.live.mmcdn.com/live-hls/amlst:hotfallingdevil-sd-fdf87e5b6c880e52d38e8c94f8ebf8728c980a91d56fb4ace13748ba59012336_trns_h264/media_w881713853_b5128000_t64RlBTOjMwLjA=_18912.ts' for reading -[https @ 0x564688097d00] Opening 'https://edge11-lax.live.mmcdn.com/live-hls/amlst:hotfallingdevil-sd-fdf87e5b6c880e52d38e8c94f8ebf8728c980a91d56fb4ace13748ba59012336_trns_h264/media_w881713853_b5128000_t64RlBTOjMwLjA=_18913.ts' for reading -[https @ 0x5646887f1580] Opening 'https://edge11-lax.live.mmcdn.com/live-hls/amlst:hotfallingdevil-sd-fdf87e5b6c880e52d38e8c94f8ebf8728c980a91d56fb4ace13748ba59012336_trns_h264/chunklist_w881713853_b5128000_t64RlBTOjMwLjA=.m3u8' for reading -[https @ 0x5646886e8580] HTTP error 403 Forbidden -[hls @ 0x564687dd0980] keepalive request failed for 'https://edge11-lax.live.mmcdn.com/live-hls/amlst:hotfallingdevil-sd-fdf87e5b6c880e52d38e8c94f8ebf8728c980a91d56fb4ace13748ba59012336_trns_h264/chunklist_w881713853_b5128000_t64RlBTOjMwLjA=.m3u8' with error: 'Server returned 403 Forbidden (access denied)' when parsing playlist -[https @ 0x5646886ccfc0] HTTP error 403 Forbidden -[hls @ 0x564687dd0980] Failed to reload playlist 0 -[https @ 0x5646886bf680] HTTP error 403 Forbidden -[hls @ 0x564687dd0980] Failed to reload playlist 0 -frame= 5355 fps= 31 q=-1.0 Lsize= 71404kB time=00:02:58.50 bitrate=3277.0kbits/s speed=1.02x -video:68484kB audio:2790kB subtitle:0kB other streams:0kB global headers:0kB muxing overhead: 0.181873% -[ffmpeg] Downloaded 73117881 bytes -[download] 100% of 69.73MiB in 02:57 -``` \ No newline at end of file diff --git a/services/capture/package.json b/services/capture/package.json deleted file mode 100644 index df096d4..0000000 --- a/services/capture/package.json +++ /dev/null @@ -1,87 +0,0 @@ -{ - "name": "@futureporn/capture", - "type": "module", - "version": "0.3.5", - "license": "Unlicense", - "private": true, - "packageManager": "pnpm@9.6.0", - "scripts": { - "start": "node dist/index.js", - "build": "tsup", - "test": "mocha", - "integration": "FUTUREPORN_WORKDIR=/home/cj/Downloads mocha ./integration/**/*.test.js", - "dev": "pnpm run dev.nodemon # yes this is crazy to have nodemon execute tsx, but it's the only way I have found to get live reloading in TS/ESM/docker with Graphile Worker's way of loading tasks", - "dev.tsx": "tsx ./src/index.ts", - "dev.nodemon": "nodemon --ext ts --exec \"pnpm run dev.tsx\"", - "dev.node": "node --no-warnings=ExperimentalWarning --loader ts-node/esm src/index.ts", - "clean": "rm -rf dist", - "superclean": "rm -rf node_modules && rm -rf pnpm-lock.yaml && rm -rf dist" - }, - "dependencies": { - "@aws-sdk/client-s3": "^3.637.0", - "@aws-sdk/lib-storage": "^3.637.0", - "@aws-sdk/types": "^3.609.0", - "@futureporn/fetchers": "workspace:^", - "@futureporn/types": "workspace:^", - "@futureporn/utils": "workspace:^", - "@paralleldrive/cuid2": "^2.2.2", - "@types/chai": "^4.3.19", - "@types/chai-as-promised": "^8.0.0", - "@types/fluent-ffmpeg": "^2.1.26", - "@types/mocha": "^10.0.7", - "@types/qs": "^6.9.15", - "date-fns": "^3.6.0", - "discord.js": "^14.16.1", - "diskusage": "^1.2.0", - "dotenv": "^16.4.5", - "execa": "^9.3.1", - "exponential-backoff": "^3.1.1", - "fastify": "^4.28.1", - "fastify-plugin": "^4.5.1", - "fastq": "^1.17.1", - "faye": "^1.4.0", - "faye-websocket": "^0.11.4", - "fluent-ffmpeg": "^2.1.3", - "graphile-config": "0.0.1-beta.9", - "graphile-worker": "^0.16.6", - "https": "^1.0.0", - "ioredis": "^5.4.1", - "minimatch": "^10.0.1", - "nanoid": "^5.0.7", - "p-retry": "^6.2.0", - "pg-boss": "^10.1.1", - "pino-pretty": "^11.2.2", - "postgres": "^3.4.4", - "qs": "^6.13.0", - "rxjs": "^7.8.1", - "sql": "^0.78.0", - "winston": "^3.14.2", - "youtube-dl-wrap": "github:insanity54/youtube-dl-wrap" - }, - "devDependencies": { - "@smithy/util-stream": "^3.1.3", - "@types/node": "^22.5.2", - "@types/sinon": "^17.0.3", - "@types/sinon-chai": "^3.2.12", - "aws-sdk": "^2.1687.0", - "aws-sdk-client-mock": "^4.0.1", - "aws-sdk-mock": "^6.1.1", - "chai": "^5.1.1", - "chai-as-promised": "^8.0.0", - "cheerio": "1.0.0", - "mocha": "^10.7.3", - "multiformats": "^13.2.2", - "node-abort-controller": "^3.1.1", - "node-fetch": "^3.3.2", - "nodemon": "^3.1.4", - "pretty-bytes": "^6.1.1", - "s3": "link:aws-sdk/clients/s3", - "sinon": "^18.0.0", - "sinon-chai": "^4.0.0", - "sinon-test": "^3.1.6", - "ts-node": "^10.9.2", - "tsup": "^8.2.4", - "tsx": "^4.19.0", - "typescript": "^5.5.4" - } -} diff --git a/services/capture/pnpm-lock.yaml b/services/capture/pnpm-lock.yaml deleted file mode 100644 index 8ca66dc..0000000 --- a/services/capture/pnpm-lock.yaml +++ /dev/null @@ -1,5836 +0,0 @@ -lockfileVersion: '9.0' - -settings: - autoInstallPeers: true - excludeLinksFromLockfile: false - -importers: - - .: - dependencies: - '@aws-sdk/client-s3': - specifier: ^3.637.0 - version: 3.637.0 - '@aws-sdk/lib-storage': - specifier: ^3.637.0 - version: 3.637.0(@aws-sdk/client-s3@3.637.0) - '@aws-sdk/types': - specifier: ^3.609.0 - version: 3.609.0 - '@futureporn/fetchers': - specifier: workspace:^ - version: link:../../packages/fetchers - '@futureporn/types': - specifier: workspace:^ - version: link:../../packages/types - '@futureporn/utils': - specifier: workspace:^ - version: link:../../packages/utils - '@paralleldrive/cuid2': - specifier: ^2.2.2 - version: 2.2.2 - '@types/chai': - specifier: ^4.3.19 - version: 4.3.19 - '@types/chai-as-promised': - specifier: ^8.0.0 - version: 8.0.0 - '@types/fluent-ffmpeg': - specifier: ^2.1.26 - version: 2.1.26 - '@types/mocha': - specifier: ^10.0.7 - version: 10.0.7 - '@types/qs': - specifier: ^6.9.15 - version: 6.9.15 - date-fns: - specifier: ^3.6.0 - version: 3.6.0 - discord.js: - specifier: ^14.16.1 - version: 14.16.1 - diskusage: - specifier: ^1.2.0 - version: 1.2.0 - dotenv: - specifier: ^16.4.5 - version: 16.4.5 - execa: - specifier: ^9.3.1 - version: 9.3.1 - exponential-backoff: - specifier: ^3.1.1 - version: 3.1.1 - fastify: - specifier: ^4.28.1 - version: 4.28.1 - fastify-plugin: - specifier: ^4.5.1 - version: 4.5.1 - fastq: - specifier: ^1.17.1 - version: 1.17.1 - faye: - specifier: ^1.4.0 - version: 1.4.0 - faye-websocket: - specifier: ^0.11.4 - version: 0.11.4 - fluent-ffmpeg: - specifier: ^2.1.3 - version: 2.1.3 - graphile-config: - specifier: 0.0.1-beta.9 - version: 0.0.1-beta.9 - graphile-worker: - specifier: ^0.16.6 - version: 0.16.6(typescript@5.5.4) - https: - specifier: ^1.0.0 - version: 1.0.0 - ioredis: - specifier: ^5.4.1 - version: 5.4.1 - minimatch: - specifier: ^10.0.1 - version: 10.0.1 - nanoid: - specifier: ^5.0.7 - version: 5.0.7 - p-retry: - specifier: ^6.2.0 - version: 6.2.0 - pg-boss: - specifier: ^10.1.1 - version: 10.1.1 - pino-pretty: - specifier: ^11.2.2 - version: 11.2.2 - postgres: - specifier: ^3.4.4 - version: 3.4.4 - qs: - specifier: ^6.13.0 - version: 6.13.0 - rxjs: - specifier: ^7.8.1 - version: 7.8.1 - sql: - specifier: ^0.78.0 - version: 0.78.0 - winston: - specifier: ^3.14.2 - version: 3.14.2 - youtube-dl-wrap: - specifier: github:insanity54/youtube-dl-wrap - version: https://codeload.github.com/insanity54/youtube-dl-wrap/tar.gz/b47e9388063b4ef516624112ef59b7d11f7775d9 - devDependencies: - '@smithy/util-stream': - specifier: ^3.1.3 - version: 3.1.3 - '@types/node': - specifier: ^22.5.2 - version: 22.5.2 - '@types/sinon': - specifier: ^17.0.3 - version: 17.0.3 - '@types/sinon-chai': - specifier: ^3.2.12 - version: 3.2.12 - aws-sdk: - specifier: ^2.1687.0 - version: 2.1687.0 - aws-sdk-client-mock: - specifier: ^4.0.1 - version: 4.0.1 - aws-sdk-mock: - specifier: ^6.1.1 - version: 6.1.1 - chai: - specifier: ^5.1.1 - version: 5.1.1 - chai-as-promised: - specifier: ^8.0.0 - version: 8.0.0(chai@5.1.1) - cheerio: - specifier: 1.0.0 - version: 1.0.0 - mocha: - specifier: ^10.7.3 - version: 10.7.3 - multiformats: - specifier: ^13.2.2 - version: 13.2.2 - node-abort-controller: - specifier: ^3.1.1 - version: 3.1.1 - node-fetch: - specifier: ^3.3.2 - version: 3.3.2 - nodemon: - specifier: ^3.1.4 - version: 3.1.4 - pretty-bytes: - specifier: ^6.1.1 - version: 6.1.1 - s3: - specifier: link:aws-sdk/clients/s3 - version: link:aws-sdk/clients/s3 - sinon: - specifier: ^18.0.0 - version: 18.0.0 - sinon-chai: - specifier: ^4.0.0 - version: 4.0.0(chai@5.1.1)(sinon@18.0.0) - sinon-test: - specifier: ^3.1.6 - version: 3.1.6(sinon@18.0.0) - ts-node: - specifier: ^10.9.2 - version: 10.9.2(@types/node@22.5.2)(typescript@5.5.4) - tsup: - specifier: ^8.2.4 - version: 8.2.4(tsx@4.19.0)(typescript@5.5.4) - tsx: - specifier: ^4.19.0 - version: 4.19.0 - typescript: - specifier: ^5.5.4 - version: 5.5.4 - -packages: - - '@aws-crypto/crc32@5.2.0': - resolution: {integrity: sha512-nLbCWqQNgUiwwtFsen1AdzAtvuLRsQS8rYgMuxCrdKf9kOssamGLuPwyTY9wyYblNr9+1XM8v6zoDTPPSIeANg==} - engines: {node: '>=16.0.0'} - - '@aws-crypto/crc32c@5.2.0': - resolution: {integrity: sha512-+iWb8qaHLYKrNvGRbiYRHSdKRWhto5XlZUEBwDjYNf+ly5SVYG6zEoYIdxvf5R3zyeP16w4PLBn3rH1xc74Rag==} - - '@aws-crypto/sha1-browser@5.2.0': - resolution: {integrity: sha512-OH6lveCFfcDjX4dbAvCFSYUjJZjDr/3XJ3xHtjn3Oj5b9RjojQo8npoLeA/bNwkOkrSQ0wgrHzXk4tDRxGKJeg==} - - '@aws-crypto/sha256-browser@5.2.0': - resolution: {integrity: sha512-AXfN/lGotSQwu6HNcEsIASo7kWXZ5HYWvfOmSNKDsEqC4OashTp8alTmaz+F7TC2L083SFv5RdB+qU3Vs1kZqw==} - - '@aws-crypto/sha256-js@5.2.0': - resolution: {integrity: sha512-FFQQyu7edu4ufvIZ+OadFpHHOt+eSTBaYaki44c+akjg7qZg9oOQeLlk77F6tSYqjDAFClrHJk9tMf0HdVyOvA==} - engines: {node: '>=16.0.0'} - - '@aws-crypto/supports-web-crypto@5.2.0': - resolution: {integrity: sha512-iAvUotm021kM33eCdNfwIN//F77/IADDSs58i+MDaOqFrVjZo9bAal0NK7HurRuWLLpF1iLX7gbWrjHjeo+YFg==} - - '@aws-crypto/util@5.2.0': - resolution: {integrity: sha512-4RkU9EsI6ZpBve5fseQlGNUWKMa1RLPQ1dnjnQoe07ldfIzcsGb5hC5W0Dm7u423KWzawlrpbjXBrXCEv9zazQ==} - - '@aws-sdk/client-s3@3.637.0': - resolution: {integrity: sha512-y6UC94fsMvhKbf0dzfnjVP1HePeGjplfcYfilZU1COIJLyTkMcUv4XcT4I407CGIrvgEafONHkiC09ygqUauNA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/client-sso-oidc@3.637.0': - resolution: {integrity: sha512-27bHALN6Qb6m6KZmPvRieJ/QRlj1lyac/GT2Rn5kJpre8Mpp+yxrtvp3h9PjNBty4lCeFEENfY4dGNSozBuBcw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.637.0 - - '@aws-sdk/client-sso@3.637.0': - resolution: {integrity: sha512-+KjLvgX5yJYROWo3TQuwBJlHCY0zz9PsLuEolmXQn0BVK1L/m9GteZHtd+rEdAoDGBpE0Xqjy1oz5+SmtsaRUw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/client-sts@3.637.0': - resolution: {integrity: sha512-xUi7x4qDubtA8QREtlblPuAcn91GS/09YVEY/RwU7xCY0aqGuFwgszAANlha4OUIqva8oVj2WO4gJuG+iaSnhw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/core@3.635.0': - resolution: {integrity: sha512-i1x/E/sgA+liUE1XJ7rj1dhyXpAKO1UKFUcTTHXok2ARjWTvszHnSXMOsB77aPbmn0fUp1JTx2kHUAZ1LVt5Bg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-env@3.620.1': - resolution: {integrity: sha512-ExuILJ2qLW5ZO+rgkNRj0xiAipKT16Rk77buvPP8csR7kkCflT/gXTyzRe/uzIiETTxM7tr8xuO9MP/DQXqkfg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-http@3.635.0': - resolution: {integrity: sha512-iJyRgEjOCQlBMXqtwPLIKYc7Bsc6nqjrZybdMDenPDa+kmLg7xh8LxHsu9088e+2/wtLicE34FsJJIfzu3L82g==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-ini@3.637.0': - resolution: {integrity: sha512-h+PFCWfZ0Q3Dx84SppET/TFpcQHmxFW8/oV9ArEvMilw4EBN+IlxgbL0CnHwjHW64szcmrM0mbebjEfHf4FXmw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.637.0 - - '@aws-sdk/credential-provider-node@3.637.0': - resolution: {integrity: sha512-yoEhoxJJfs7sPVQ6Is939BDQJZpZCoUgKr/ySse4YKOZ24t4VqgHA6+wV7rYh+7IW24Rd91UTvEzSuHYTlxlNA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-process@3.620.1': - resolution: {integrity: sha512-hWqFMidqLAkaV9G460+1at6qa9vySbjQKKc04p59OT7lZ5cO5VH5S4aI05e+m4j364MBROjjk2ugNvfNf/8ILg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-sso@3.637.0': - resolution: {integrity: sha512-Mvz+h+e62/tl+dVikLafhv+qkZJ9RUb8l2YN/LeKMWkxQylPT83CPk9aimVhCV89zth1zpREArl97+3xsfgQvA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-web-identity@3.621.0': - resolution: {integrity: sha512-w7ASSyfNvcx7+bYGep3VBgC3K6vEdLmlpjT7nSIHxxQf+WSdvy+HynwJosrpZax0sK5q0D1Jpn/5q+r5lwwW6w==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.621.0 - - '@aws-sdk/lib-storage@3.637.0': - resolution: {integrity: sha512-HiNGOP4a1QrCWwO1joKw4mCp19nLXoF9K52PislBaYDI35IlHC3DP6MeOg5zmElwtL1GtEHFBy5olfPWPsLyLg==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-s3': ^3.637.0 - - '@aws-sdk/middleware-bucket-endpoint@3.620.0': - resolution: {integrity: sha512-eGLL0W6L3HDb3OACyetZYOWpHJ+gLo0TehQKeQyy2G8vTYXqNTeqYhuI6up9HVjBzU9eQiULVQETmgQs7TFaRg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-expect-continue@3.620.0': - resolution: {integrity: sha512-QXeRFMLfyQ31nAHLbiTLtk0oHzG9QLMaof5jIfqcUwnOkO8YnQdeqzakrg1Alpy/VQ7aqzIi8qypkBe2KXZz0A==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-flexible-checksums@3.620.0': - resolution: {integrity: sha512-ftz+NW7qka2sVuwnnO1IzBku5ccP+s5qZGeRTPgrKB7OzRW85gthvIo1vQR2w+OwHFk7WJbbhhWwbCbktnP4UA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-host-header@3.620.0': - resolution: {integrity: sha512-VMtPEZwqYrII/oUkffYsNWY9PZ9xpNJpMgmyU0rlDQ25O1c0Hk3fJmZRe6pEkAJ0omD7kLrqGl1DUjQVxpd/Rg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-location-constraint@3.609.0': - resolution: {integrity: sha512-xzsdoTkszGVqGVPjUmgoP7TORiByLueMHieI1fhQL888WPdqctwAx3ES6d/bA9Q/i8jnc6hs+Fjhy8UvBTkE9A==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-logger@3.609.0': - resolution: {integrity: sha512-S62U2dy4jMDhDFDK5gZ4VxFdWzCtLzwbYyFZx2uvPYTECkepLUfzLic2BHg2Qvtu4QjX+oGE3P/7fwaGIsGNuQ==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-recursion-detection@3.620.0': - resolution: {integrity: sha512-nh91S7aGK3e/o1ck64sA/CyoFw+gAYj2BDOnoNa6ouyCrVJED96ZXWbhye/fz9SgmNUZR2g7GdVpiLpMKZoI5w==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-sdk-s3@3.635.0': - resolution: {integrity: sha512-RLdYJPEV4JL/7NBoFUs7VlP90X++5FlJdxHz0DzCjmiD3qCviKy+Cym3qg1gBgHwucs5XisuClxDrGokhAdTQw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-ssec@3.609.0': - resolution: {integrity: sha512-GZSD1s7+JswWOTamVap79QiDaIV7byJFssBW68GYjyRS5EBjNfwA/8s+6uE6g39R3ojyTbYOmvcANoZEhSULXg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-user-agent@3.637.0': - resolution: {integrity: sha512-EYo0NE9/da/OY8STDsK2LvM4kNa79DBsf4YVtaG4P5pZ615IeFsD8xOHZeuJmUrSMlVQ8ywPRX7WMucUybsKug==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/region-config-resolver@3.614.0': - resolution: {integrity: sha512-vDCeMXvic/LU0KFIUjpC3RiSTIkkvESsEfbVHiHH0YINfl8HnEqR5rj+L8+phsCeVg2+LmYwYxd5NRz4PHxt5g==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/signature-v4-multi-region@3.635.0': - resolution: {integrity: sha512-J6QY4/invOkpogCHjSaDON1hF03viPpOnsrzVuCvJMmclS/iG62R4EY0wq1alYll0YmSdmKlpJwHMWwGtqK63Q==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/token-providers@3.614.0': - resolution: {integrity: sha512-okItqyY6L9IHdxqs+Z116y5/nda7rHxLvROxtAJdLavWTYDydxrZstImNgGWTeVdmc0xX2gJCI77UYUTQWnhRw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sso-oidc': ^3.614.0 - - '@aws-sdk/types@3.609.0': - resolution: {integrity: sha512-+Tqnh9w0h2LcrUsdXyT1F8mNhXz+tVYBtP19LpeEGntmvHwa2XzvLUCWpoIAIVsHp5+HdB2X9Sn0KAtmbFXc2Q==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-arn-parser@3.568.0': - resolution: {integrity: sha512-XUKJWWo+KOB7fbnPP0+g/o5Ulku/X53t7i/h+sPHr5xxYTJJ9CYnbToo95mzxe7xWvkLrsNtJ8L+MnNn9INs2w==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-endpoints@3.637.0': - resolution: {integrity: sha512-pAqOKUHeVWHEXXDIp/qoMk/6jyxIb6GGjnK1/f8dKHtKIEs4tKsnnL563gceEvdad53OPXIt86uoevCcCzmBnw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-locate-window@3.568.0': - resolution: {integrity: sha512-3nh4TINkXYr+H41QaPelCceEB2FXP3fxp93YZXB/kqJvX0U9j0N0Uk45gvsjmEPzG8XxkPEeLIfT2I1M7A6Lig==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-user-agent-browser@3.609.0': - resolution: {integrity: sha512-fojPU+mNahzQ0YHYBsx0ZIhmMA96H+ZIZ665ObU9tl+SGdbLneVZVikGve+NmHTQwHzwkFsZYYnVKAkreJLAtA==} - - '@aws-sdk/util-user-agent-node@3.614.0': - resolution: {integrity: sha512-15ElZT88peoHnq5TEoEtZwoXTXRxNrk60TZNdpl/TUBJ5oNJ9Dqb5Z4ryb8ofN6nm9aFf59GVAerFDz8iUoHBA==} - engines: {node: '>=16.0.0'} - peerDependencies: - aws-crt: '>=1.0.0' - peerDependenciesMeta: - aws-crt: - optional: true - - '@aws-sdk/xml-builder@3.609.0': - resolution: {integrity: sha512-l9XxNcA4HX98rwCC2/KoiWcmEiRfZe4G+mYwDbCFT87JIMj6GBhLDkAzr/W8KAaA2IDr8Vc6J8fZPgVulxxfMA==} - engines: {node: '>=16.0.0'} - - '@babel/code-frame@7.24.7': - resolution: {integrity: sha512-BcYH1CVJBO9tvyIZ2jVeXgSIMvGZ2FDRvDdOIVQyuklNKSsx+eppDEBq/g47Ayw+RqNFE+URvOShmf+f/qwAlA==} - engines: {node: '>=6.9.0'} - - '@babel/helper-validator-identifier@7.24.7': - resolution: {integrity: sha512-rR+PBcQ1SMQDDyF6X0wxtG8QyLCgUB0eRAGguqRLfkCA87l7yAP7ehq8SNj96OOGTO8OBV70KhuFYcIkHXOg0w==} - engines: {node: '>=6.9.0'} - - '@babel/highlight@7.24.7': - resolution: {integrity: sha512-EStJpq4OuY8xYfhGVXngigBJRWxftKX9ksiGDnmlY3o7B/V7KIAc9X4oiK87uPJSc/vs5L869bem5fhZa8caZw==} - engines: {node: '>=6.9.0'} - - '@colors/colors@1.6.0': - resolution: {integrity: sha512-Ir+AOibqzrIsL6ajt3Rz3LskB7OiMVHqltZmspbW/TJuTVuyOMirVqAkjfY6JISiLHgyNqicAC8AyHHGzNd/dA==} - engines: {node: '>=0.1.90'} - - '@cspotcode/source-map-support@0.8.1': - resolution: {integrity: sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==} - engines: {node: '>=12'} - - '@dabh/diagnostics@2.0.3': - resolution: {integrity: sha512-hrlQOIi7hAfzsMqlGSFyVucrx38O+j6wiGOf//H2ecvIEqYN4ADBSS2iLMh5UFyDunCNniUIPk/q3riFv45xRA==} - - '@discordjs/builders@1.9.0': - resolution: {integrity: sha512-0zx8DePNVvQibh5ly5kCEei5wtPBIUbSoE9n+91Rlladz4tgtFbJ36PZMxxZrTEOQ7AHMZ/b0crT/0fCy6FTKg==} - engines: {node: '>=18'} - - '@discordjs/collection@1.5.3': - resolution: {integrity: sha512-SVb428OMd3WO1paV3rm6tSjM4wC+Kecaa1EUGX7vc6/fddvw/6lg90z4QtCqm21zvVe92vMMDt9+DkIvjXImQQ==} - engines: {node: '>=16.11.0'} - - '@discordjs/collection@2.1.1': - resolution: {integrity: sha512-LiSusze9Tc7qF03sLCujF5iZp7K+vRNEDBZ86FT9aQAv3vxMLihUvKvpsCWiQ2DJq1tVckopKm1rxomgNUc9hg==} - engines: {node: '>=18'} - - '@discordjs/formatters@0.5.0': - resolution: {integrity: sha512-98b3i+Y19RFq1Xke4NkVY46x8KjJQjldHUuEbCqMvp1F5Iq9HgnGpu91jOi/Ufazhty32eRsKnnzS8n4c+L93g==} - engines: {node: '>=18'} - - '@discordjs/rest@2.4.0': - resolution: {integrity: sha512-Xb2irDqNcq+O8F0/k/NaDp7+t091p+acb51iA4bCKfIn+WFWd6HrNvcsSbMMxIR9NjcMZS6NReTKygqiQN+ntw==} - engines: {node: '>=18'} - - '@discordjs/util@1.1.1': - resolution: {integrity: sha512-eddz6UnOBEB1oITPinyrB2Pttej49M9FZQY8NxgEvc3tq6ZICZ19m70RsmzRdDHk80O9NoYN/25AqJl8vPVf/g==} - engines: {node: '>=18'} - - '@discordjs/ws@1.1.1': - resolution: {integrity: sha512-PZ+vLpxGCRtmr2RMkqh8Zp+BenUaJqlS6xhgWKEZcgC/vfHLEzpHtKkB0sl3nZWpwtcKk6YWy+pU3okL2I97FA==} - engines: {node: '>=16.11.0'} - - '@esbuild/aix-ppc64@0.23.1': - resolution: {integrity: sha512-6VhYk1diRqrhBAqpJEdjASR/+WVRtfjpqKuNw11cLiaWpAT/Uu+nokB+UJnevzy/P9C/ty6AOe0dwueMrGh/iQ==} - engines: {node: '>=18'} - cpu: [ppc64] - os: [aix] - - '@esbuild/android-arm64@0.23.1': - resolution: {integrity: sha512-xw50ipykXcLstLeWH7WRdQuysJqejuAGPd30vd1i5zSyKK3WE+ijzHmLKxdiCMtH1pHz78rOg0BKSYOSB/2Khw==} - engines: {node: '>=18'} - cpu: [arm64] - os: [android] - - '@esbuild/android-arm@0.23.1': - resolution: {integrity: sha512-uz6/tEy2IFm9RYOyvKl88zdzZfwEfKZmnX9Cj1BHjeSGNuGLuMD1kR8y5bteYmwqKm1tj8m4cb/aKEorr6fHWQ==} - engines: {node: '>=18'} - cpu: [arm] - os: [android] - - '@esbuild/android-x64@0.23.1': - resolution: {integrity: sha512-nlN9B69St9BwUoB+jkyU090bru8L0NA3yFvAd7k8dNsVH8bi9a8cUAUSEcEEgTp2z3dbEDGJGfP6VUnkQnlReg==} - engines: {node: '>=18'} - cpu: [x64] - os: [android] - - '@esbuild/darwin-arm64@0.23.1': - resolution: {integrity: sha512-YsS2e3Wtgnw7Wq53XXBLcV6JhRsEq8hkfg91ESVadIrzr9wO6jJDMZnCQbHm1Guc5t/CdDiFSSfWP58FNuvT3Q==} - engines: {node: '>=18'} - cpu: [arm64] - os: [darwin] - - '@esbuild/darwin-x64@0.23.1': - resolution: {integrity: sha512-aClqdgTDVPSEGgoCS8QDG37Gu8yc9lTHNAQlsztQ6ENetKEO//b8y31MMu2ZaPbn4kVsIABzVLXYLhCGekGDqw==} - engines: {node: '>=18'} - cpu: [x64] - os: [darwin] - - '@esbuild/freebsd-arm64@0.23.1': - resolution: {integrity: sha512-h1k6yS8/pN/NHlMl5+v4XPfikhJulk4G+tKGFIOwURBSFzE8bixw1ebjluLOjfwtLqY0kewfjLSrO6tN2MgIhA==} - engines: {node: '>=18'} - cpu: [arm64] - os: [freebsd] - - '@esbuild/freebsd-x64@0.23.1': - resolution: {integrity: sha512-lK1eJeyk1ZX8UklqFd/3A60UuZ/6UVfGT2LuGo3Wp4/z7eRTRYY+0xOu2kpClP+vMTi9wKOfXi2vjUpO1Ro76g==} - engines: {node: '>=18'} - cpu: [x64] - os: [freebsd] - - '@esbuild/linux-arm64@0.23.1': - resolution: {integrity: sha512-/93bf2yxencYDnItMYV/v116zff6UyTjo4EtEQjUBeGiVpMmffDNUyD9UN2zV+V3LRV3/on4xdZ26NKzn6754g==} - engines: {node: '>=18'} - cpu: [arm64] - os: [linux] - - '@esbuild/linux-arm@0.23.1': - resolution: {integrity: sha512-CXXkzgn+dXAPs3WBwE+Kvnrf4WECwBdfjfeYHpMeVxWE0EceB6vhWGShs6wi0IYEqMSIzdOF1XjQ/Mkm5d7ZdQ==} - engines: {node: '>=18'} - cpu: [arm] - os: [linux] - - '@esbuild/linux-ia32@0.23.1': - resolution: {integrity: sha512-VTN4EuOHwXEkXzX5nTvVY4s7E/Krz7COC8xkftbbKRYAl96vPiUssGkeMELQMOnLOJ8k3BY1+ZY52tttZnHcXQ==} - engines: {node: '>=18'} - cpu: [ia32] - os: [linux] - - '@esbuild/linux-loong64@0.23.1': - resolution: {integrity: sha512-Vx09LzEoBa5zDnieH8LSMRToj7ir/Jeq0Gu6qJ/1GcBq9GkfoEAoXvLiW1U9J1qE/Y/Oyaq33w5p2ZWrNNHNEw==} - engines: {node: '>=18'} - cpu: [loong64] - os: [linux] - - '@esbuild/linux-mips64el@0.23.1': - resolution: {integrity: sha512-nrFzzMQ7W4WRLNUOU5dlWAqa6yVeI0P78WKGUo7lg2HShq/yx+UYkeNSE0SSfSure0SqgnsxPvmAUu/vu0E+3Q==} - engines: {node: '>=18'} - cpu: [mips64el] - os: [linux] - - '@esbuild/linux-ppc64@0.23.1': - resolution: {integrity: sha512-dKN8fgVqd0vUIjxuJI6P/9SSSe/mB9rvA98CSH2sJnlZ/OCZWO1DJvxj8jvKTfYUdGfcq2dDxoKaC6bHuTlgcw==} - engines: {node: '>=18'} - cpu: [ppc64] - os: [linux] - - '@esbuild/linux-riscv64@0.23.1': - resolution: {integrity: sha512-5AV4Pzp80fhHL83JM6LoA6pTQVWgB1HovMBsLQ9OZWLDqVY8MVobBXNSmAJi//Csh6tcY7e7Lny2Hg1tElMjIA==} - engines: {node: '>=18'} - cpu: [riscv64] - os: [linux] - - '@esbuild/linux-s390x@0.23.1': - resolution: {integrity: sha512-9ygs73tuFCe6f6m/Tb+9LtYxWR4c9yg7zjt2cYkjDbDpV/xVn+68cQxMXCjUpYwEkze2RcU/rMnfIXNRFmSoDw==} - engines: {node: '>=18'} - cpu: [s390x] - os: [linux] - - '@esbuild/linux-x64@0.23.1': - resolution: {integrity: sha512-EV6+ovTsEXCPAp58g2dD68LxoP/wK5pRvgy0J/HxPGB009omFPv3Yet0HiaqvrIrgPTBuC6wCH1LTOY91EO5hQ==} - engines: {node: '>=18'} - cpu: [x64] - os: [linux] - - '@esbuild/netbsd-x64@0.23.1': - resolution: {integrity: sha512-aevEkCNu7KlPRpYLjwmdcuNz6bDFiE7Z8XC4CPqExjTvrHugh28QzUXVOZtiYghciKUacNktqxdpymplil1beA==} - engines: {node: '>=18'} - cpu: [x64] - os: [netbsd] - - '@esbuild/openbsd-arm64@0.23.1': - resolution: {integrity: sha512-3x37szhLexNA4bXhLrCC/LImN/YtWis6WXr1VESlfVtVeoFJBRINPJ3f0a/6LV8zpikqoUg4hyXw0sFBt5Cr+Q==} - engines: {node: '>=18'} - cpu: [arm64] - os: [openbsd] - - '@esbuild/openbsd-x64@0.23.1': - resolution: {integrity: sha512-aY2gMmKmPhxfU+0EdnN+XNtGbjfQgwZj43k8G3fyrDM/UdZww6xrWxmDkuz2eCZchqVeABjV5BpildOrUbBTqA==} - engines: {node: '>=18'} - cpu: [x64] - os: [openbsd] - - '@esbuild/sunos-x64@0.23.1': - resolution: {integrity: sha512-RBRT2gqEl0IKQABT4XTj78tpk9v7ehp+mazn2HbUeZl1YMdaGAQqhapjGTCe7uw7y0frDi4gS0uHzhvpFuI1sA==} - engines: {node: '>=18'} - cpu: [x64] - os: [sunos] - - '@esbuild/win32-arm64@0.23.1': - resolution: {integrity: sha512-4O+gPR5rEBe2FpKOVyiJ7wNDPA8nGzDuJ6gN4okSA1gEOYZ67N8JPk58tkWtdtPeLz7lBnY6I5L3jdsr3S+A6A==} - engines: {node: '>=18'} - cpu: [arm64] - os: [win32] - - '@esbuild/win32-ia32@0.23.1': - resolution: {integrity: sha512-BcaL0Vn6QwCwre3Y717nVHZbAa4UBEigzFm6VdsVdT/MbZ38xoj1X9HPkZhbmaBGUD1W8vxAfffbDe8bA6AKnQ==} - engines: {node: '>=18'} - cpu: [ia32] - os: [win32] - - '@esbuild/win32-x64@0.23.1': - resolution: {integrity: sha512-BHpFFeslkWrXWyUPnbKm+xYYVYruCinGcftSBaa8zoF9hZO4BcSCFUvHVTtzpIY6YzUnYtuEhZ+C9iEXjxnasg==} - engines: {node: '>=18'} - cpu: [x64] - os: [win32] - - '@fastify/ajv-compiler@3.6.0': - resolution: {integrity: sha512-LwdXQJjmMD+GwLOkP7TVC68qa+pSSogeWWmznRJ/coyTcfe9qA05AHFSe1eZFwK6q+xVRpChnvFUkf1iYaSZsQ==} - - '@fastify/error@3.4.1': - resolution: {integrity: sha512-wWSvph+29GR783IhmvdwWnN4bUxTD01Vm5Xad4i7i1VuAOItLvbPAb69sb0IQ2N57yprvhNIwAP5B6xfKTmjmQ==} - - '@fastify/fast-json-stringify-compiler@4.3.0': - resolution: {integrity: sha512-aZAXGYo6m22Fk1zZzEUKBvut/CIIQe/BapEORnxiD5Qr0kPHqqI69NtEMCme74h+at72sPhbkb4ZrLd1W3KRLA==} - - '@fastify/merge-json-schemas@0.1.1': - resolution: {integrity: sha512-fERDVz7topgNjtXsJTTW1JKLy0rhuLRcquYqNR9rF7OcVpCa2OVW49ZPDIhaRRCaUuvVxI+N416xUoF76HNSXA==} - - '@graphile/logger@0.2.0': - resolution: {integrity: sha512-jjcWBokl9eb1gVJ85QmoaQ73CQ52xAaOCF29ukRbYNl6lY+ts0ErTaDYOBlejcbUs2OpaiqYLO5uDhyLFzWw4w==} - - '@ioredis/commands@1.2.0': - resolution: {integrity: sha512-Sx1pU8EM64o2BrqNpEO1CNLtKQwyhuXuqyfH7oGKCk+1a33d2r5saW8zNwm3j6BTExtjrv2BxTgzzkMwts6vGg==} - - '@isaacs/cliui@8.0.2': - resolution: {integrity: sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==} - engines: {node: '>=12'} - - '@jridgewell/gen-mapping@0.3.5': - resolution: {integrity: sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==} - engines: {node: '>=6.0.0'} - - '@jridgewell/resolve-uri@3.1.2': - resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==} - engines: {node: '>=6.0.0'} - - '@jridgewell/set-array@1.2.1': - resolution: {integrity: sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==} - engines: {node: '>=6.0.0'} - - '@jridgewell/sourcemap-codec@1.5.0': - resolution: {integrity: sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==} - - '@jridgewell/trace-mapping@0.3.25': - resolution: {integrity: sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==} - - '@jridgewell/trace-mapping@0.3.9': - resolution: {integrity: sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==} - - '@noble/hashes@1.5.0': - resolution: {integrity: sha512-1j6kQFb7QRru7eKN3ZDvRcP13rugwdxZqCjbiAVZfIJwgj2A65UmT4TgARXGlXgnRkORLTDTrO19ZErt7+QXgA==} - engines: {node: ^14.21.3 || >=16} - - '@nodelib/fs.scandir@2.1.5': - resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==} - engines: {node: '>= 8'} - - '@nodelib/fs.stat@2.0.5': - resolution: {integrity: sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==} - engines: {node: '>= 8'} - - '@nodelib/fs.walk@1.2.8': - resolution: {integrity: sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==} - engines: {node: '>= 8'} - - '@paralleldrive/cuid2@2.2.2': - resolution: {integrity: sha512-ZOBkgDwEdoYVlSeRbYYXs0S9MejQofiVYoTbKzy/6GQa39/q5tQU2IX46+shYnUkpEl3wc+J6wRlar7r2EK2xA==} - - '@pkgjs/parseargs@0.11.0': - resolution: {integrity: sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==} - engines: {node: '>=14'} - - '@rollup/rollup-android-arm-eabi@4.21.2': - resolution: {integrity: sha512-fSuPrt0ZO8uXeS+xP3b+yYTCBUd05MoSp2N/MFOgjhhUhMmchXlpTQrTpI8T+YAwAQuK7MafsCOxW7VrPMrJcg==} - cpu: [arm] - os: [android] - - '@rollup/rollup-android-arm64@4.21.2': - resolution: {integrity: sha512-xGU5ZQmPlsjQS6tzTTGwMsnKUtu0WVbl0hYpTPauvbRAnmIvpInhJtgjj3mcuJpEiuUw4v1s4BimkdfDWlh7gA==} - cpu: [arm64] - os: [android] - - '@rollup/rollup-darwin-arm64@4.21.2': - resolution: {integrity: sha512-99AhQ3/ZMxU7jw34Sq8brzXqWH/bMnf7ZVhvLk9QU2cOepbQSVTns6qoErJmSiAvU3InRqC2RRZ5ovh1KN0d0Q==} - cpu: [arm64] - os: [darwin] - - '@rollup/rollup-darwin-x64@4.21.2': - resolution: {integrity: sha512-ZbRaUvw2iN/y37x6dY50D8m2BnDbBjlnMPotDi/qITMJ4sIxNY33HArjikDyakhSv0+ybdUxhWxE6kTI4oX26w==} - cpu: [x64] - os: [darwin] - - '@rollup/rollup-linux-arm-gnueabihf@4.21.2': - resolution: {integrity: sha512-ztRJJMiE8nnU1YFcdbd9BcH6bGWG1z+jP+IPW2oDUAPxPjo9dverIOyXz76m6IPA6udEL12reYeLojzW2cYL7w==} - cpu: [arm] - os: [linux] - - '@rollup/rollup-linux-arm-musleabihf@4.21.2': - resolution: {integrity: sha512-flOcGHDZajGKYpLV0JNc0VFH361M7rnV1ee+NTeC/BQQ1/0pllYcFmxpagltANYt8FYf9+kL6RSk80Ziwyhr7w==} - cpu: [arm] - os: [linux] - - '@rollup/rollup-linux-arm64-gnu@4.21.2': - resolution: {integrity: sha512-69CF19Kp3TdMopyteO/LJbWufOzqqXzkrv4L2sP8kfMaAQ6iwky7NoXTp7bD6/irKgknDKM0P9E/1l5XxVQAhw==} - cpu: [arm64] - os: [linux] - - '@rollup/rollup-linux-arm64-musl@4.21.2': - resolution: {integrity: sha512-48pD/fJkTiHAZTnZwR0VzHrao70/4MlzJrq0ZsILjLW/Ab/1XlVUStYyGt7tdyIiVSlGZbnliqmult/QGA2O2w==} - cpu: [arm64] - os: [linux] - - '@rollup/rollup-linux-powerpc64le-gnu@4.21.2': - resolution: {integrity: sha512-cZdyuInj0ofc7mAQpKcPR2a2iu4YM4FQfuUzCVA2u4HI95lCwzjoPtdWjdpDKyHxI0UO82bLDoOaLfpZ/wviyQ==} - cpu: [ppc64] - os: [linux] - - '@rollup/rollup-linux-riscv64-gnu@4.21.2': - resolution: {integrity: sha512-RL56JMT6NwQ0lXIQmMIWr1SW28z4E4pOhRRNqwWZeXpRlykRIlEpSWdsgNWJbYBEWD84eocjSGDu/XxbYeCmwg==} - cpu: [riscv64] - os: [linux] - - '@rollup/rollup-linux-s390x-gnu@4.21.2': - resolution: {integrity: sha512-PMxkrWS9z38bCr3rWvDFVGD6sFeZJw4iQlhrup7ReGmfn7Oukrr/zweLhYX6v2/8J6Cep9IEA/SmjXjCmSbrMQ==} - cpu: [s390x] - os: [linux] - - '@rollup/rollup-linux-x64-gnu@4.21.2': - resolution: {integrity: sha512-B90tYAUoLhU22olrafY3JQCFLnT3NglazdwkHyxNDYF/zAxJt5fJUB/yBoWFoIQ7SQj+KLe3iL4BhOMa9fzgpw==} - cpu: [x64] - os: [linux] - - '@rollup/rollup-linux-x64-musl@4.21.2': - resolution: {integrity: sha512-7twFizNXudESmC9oneLGIUmoHiiLppz/Xs5uJQ4ShvE6234K0VB1/aJYU3f/4g7PhssLGKBVCC37uRkkOi8wjg==} - cpu: [x64] - os: [linux] - - '@rollup/rollup-win32-arm64-msvc@4.21.2': - resolution: {integrity: sha512-9rRero0E7qTeYf6+rFh3AErTNU1VCQg2mn7CQcI44vNUWM9Ze7MSRS/9RFuSsox+vstRt97+x3sOhEey024FRQ==} - cpu: [arm64] - os: [win32] - - '@rollup/rollup-win32-ia32-msvc@4.21.2': - resolution: {integrity: sha512-5rA4vjlqgrpbFVVHX3qkrCo/fZTj1q0Xxpg+Z7yIo3J2AilW7t2+n6Q8Jrx+4MrYpAnjttTYF8rr7bP46BPzRw==} - cpu: [ia32] - os: [win32] - - '@rollup/rollup-win32-x64-msvc@4.21.2': - resolution: {integrity: sha512-6UUxd0+SKomjdzuAcp+HAmxw1FlGBnl1v2yEPSabtx4lBfdXHDVsW7+lQkgz9cNFJGY3AWR7+V8P5BqkD9L9nA==} - cpu: [x64] - os: [win32] - - '@sapphire/async-queue@1.5.3': - resolution: {integrity: sha512-x7zadcfJGxFka1Q3f8gCts1F0xMwCKbZweM85xECGI0hBTeIZJGGCrHgLggihBoprlQ/hBmDR5LKfIPqnmHM3w==} - engines: {node: '>=v14.0.0', npm: '>=7.0.0'} - - '@sapphire/shapeshift@4.0.0': - resolution: {integrity: sha512-d9dUmWVA7MMiKobL3VpLF8P2aeanRTu6ypG2OIaEv/ZHH/SUQ2iHOVyi5wAPjQ+HmnMuL0whK9ez8I/raWbtIg==} - engines: {node: '>=v16'} - - '@sapphire/snowflake@3.5.3': - resolution: {integrity: sha512-jjmJywLAFoWeBi1W7994zZyiNWPIiqRRNAmSERxyg93xRGzNYvGjlZ0gR6x0F4gPRi2+0O6S71kOZYyr3cxaIQ==} - engines: {node: '>=v14.0.0', npm: '>=7.0.0'} - - '@sec-ant/readable-stream@0.4.1': - resolution: {integrity: sha512-831qok9r2t8AlxLko40y2ebgSDhenenCatLVeW/uBtnHPyhHOvG0C7TvfgecV+wHzIm5KUICgzmVpWS+IMEAeg==} - - '@sindresorhus/merge-streams@4.0.0': - resolution: {integrity: sha512-tlqY9xq5ukxTUZBmoOp+m61cqwQD5pHJtFY3Mn8CA8ps6yghLH/Hw8UPdqg4OLmFW3IFlcXnQNmo/dh8HzXYIQ==} - engines: {node: '>=18'} - - '@sinonjs/commons@2.0.0': - resolution: {integrity: sha512-uLa0j859mMrg2slwQYdO/AkrOfmH+X6LTVmNTS9CqexuE2IvVORIkSpJLqePAbEnKJ77aMmCwr1NUZ57120Xcg==} - - '@sinonjs/commons@3.0.1': - resolution: {integrity: sha512-K3mCHKQ9sVh8o1C9cxkwxaOmXoAMlDxC1mYyHrjqOWEcBjYr76t96zL2zlj5dUGZ3HSw240X1qgH3Mjf1yJWpQ==} - - '@sinonjs/fake-timers@10.3.0': - resolution: {integrity: sha512-V4BG07kuYSUkTCSBHG8G8TNhM+F19jXFWnQtzj+we8DrkpSBCee9Z3Ms8yiGer/dlmhe35/Xdgyo3/0rQKg7YA==} - - '@sinonjs/fake-timers@11.3.1': - resolution: {integrity: sha512-EVJO7nW5M/F5Tur0Rf2z/QoMo+1Ia963RiMtapiQrEWvY0iBUvADo8Beegwjpnle5BHkyHuoxSTW3jF43H1XRA==} - - '@sinonjs/samsam@8.0.0': - resolution: {integrity: sha512-Bp8KUVlLp8ibJZrnvq2foVhP0IVX2CIprMJPK0vqGqgrDa0OHVKeZyBykqskkrdxV6yKBPmGasO8LVjAKR3Gew==} - - '@sinonjs/text-encoding@0.7.3': - resolution: {integrity: sha512-DE427ROAphMQzU4ENbliGYrBSYPXF+TtLg9S8vzeA+OF4ZKzoDdzfL8sxuMUGS/lgRhM6j1URSk9ghf7Xo1tyA==} - - '@smithy/abort-controller@3.1.1': - resolution: {integrity: sha512-MBJBiidoe+0cTFhyxT8g+9g7CeVccLM0IOKKUMCNQ1CNMJ/eIfoo0RTfVrXOONEI1UCN1W+zkiHSbzUNE9dZtQ==} - engines: {node: '>=16.0.0'} - - '@smithy/chunked-blob-reader-native@3.0.0': - resolution: {integrity: sha512-VDkpCYW+peSuM4zJip5WDfqvg2Mo/e8yxOv3VF1m11y7B8KKMKVFtmZWDe36Fvk8rGuWrPZHHXZ7rR7uM5yWyg==} - - '@smithy/chunked-blob-reader@3.0.0': - resolution: {integrity: sha512-sbnURCwjF0gSToGlsBiAmd1lRCmSn72nu9axfJu5lIx6RUEgHu6GwTMbqCdhQSi0Pumcm5vFxsi9XWXb2mTaoA==} - - '@smithy/config-resolver@3.0.5': - resolution: {integrity: sha512-SkW5LxfkSI1bUC74OtfBbdz+grQXYiPYolyu8VfpLIjEoN/sHVBlLeGXMQ1vX4ejkgfv6sxVbQJ32yF2cl1veA==} - engines: {node: '>=16.0.0'} - - '@smithy/core@2.4.0': - resolution: {integrity: sha512-cHXq+FneIF/KJbt4q4pjN186+Jf4ZB0ZOqEaZMBhT79srEyGDDBV31NqBRBjazz8ppQ1bJbDJMY9ba5wKFV36w==} - engines: {node: '>=16.0.0'} - - '@smithy/credential-provider-imds@3.2.0': - resolution: {integrity: sha512-0SCIzgd8LYZ9EJxUjLXBmEKSZR/P/w6l7Rz/pab9culE/RWuqelAKGJvn5qUOl8BgX8Yj5HWM50A5hiB/RzsgA==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-codec@3.1.2': - resolution: {integrity: sha512-0mBcu49JWt4MXhrhRAlxASNy0IjDRFU+aWNDRal9OtUJvJNiwDuyKMUONSOjLjSCeGwZaE0wOErdqULer8r7yw==} - - '@smithy/eventstream-serde-browser@3.0.6': - resolution: {integrity: sha512-2hM54UWQUOrki4BtsUI1WzmD13/SeaqT/AB3EUJKbcver/WgKNaiJ5y5F5XXuVe6UekffVzuUDrBZVAA3AWRpQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-config-resolver@3.0.3': - resolution: {integrity: sha512-NVTYjOuYpGfrN/VbRQgn31x73KDLfCXCsFdad8DiIc3IcdxL+dYA9zEQPyOP7Fy2QL8CPy2WE4WCUD+ZsLNfaQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-node@3.0.5': - resolution: {integrity: sha512-+upXvnHNyZP095s11jF5dhGw/Ihzqwl5G+/KtMnoQOpdfC3B5HYCcDVG9EmgkhJMXJlM64PyN5gjJl0uXFQehQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-universal@3.0.5': - resolution: {integrity: sha512-5u/nXbyoh1s4QxrvNre9V6vfyoLWuiVvvd5TlZjGThIikc3G+uNiG9uOTCWweSRjv1asdDIWK7nOmN7le4RYHQ==} - engines: {node: '>=16.0.0'} - - '@smithy/fetch-http-handler@3.2.4': - resolution: {integrity: sha512-kBprh5Gs5h7ug4nBWZi1FZthdqSM+T7zMmsZxx0IBvWUn7dK3diz2SHn7Bs4dQGFDk8plDv375gzenDoNwrXjg==} - - '@smithy/hash-blob-browser@3.1.2': - resolution: {integrity: sha512-hAbfqN2UbISltakCC2TP0kx4LqXBttEv2MqSPE98gVuDFMf05lU+TpC41QtqGP3Ff5A3GwZMPfKnEy0VmEUpmg==} - - '@smithy/hash-node@3.0.3': - resolution: {integrity: sha512-2ctBXpPMG+B3BtWSGNnKELJ7SH9e4TNefJS0cd2eSkOOROeBnnVBnAy9LtJ8tY4vUEoe55N4CNPxzbWvR39iBw==} - engines: {node: '>=16.0.0'} - - '@smithy/hash-stream-node@3.1.2': - resolution: {integrity: sha512-PBgDMeEdDzi6JxKwbfBtwQG9eT9cVwsf0dZzLXoJF4sHKHs5HEo/3lJWpn6jibfJwT34I1EBXpBnZE8AxAft6g==} - engines: {node: '>=16.0.0'} - - '@smithy/invalid-dependency@3.0.3': - resolution: {integrity: sha512-ID1eL/zpDULmHJbflb864k72/SNOZCADRc9i7Exq3RUNJw6raWUSlFEQ+3PX3EYs++bTxZB2dE9mEHTQLv61tw==} - - '@smithy/is-array-buffer@2.2.0': - resolution: {integrity: sha512-GGP3O9QFD24uGeAXYUjwSTXARoqpZykHadOmA8G5vfJPK0/DC67qa//0qvqrJzL1xc8WQWX7/yc7fwudjPHPhA==} - engines: {node: '>=14.0.0'} - - '@smithy/is-array-buffer@3.0.0': - resolution: {integrity: sha512-+Fsu6Q6C4RSJiy81Y8eApjEB5gVtM+oFKTffg+jSuwtvomJJrhUJBu2zS8wjXSgH/g1MKEWrzyChTBe6clb5FQ==} - engines: {node: '>=16.0.0'} - - '@smithy/md5-js@3.0.3': - resolution: {integrity: sha512-O/SAkGVwpWmelpj/8yDtsaVe6sINHLB1q8YE/+ZQbDxIw3SRLbTZuRaI10K12sVoENdnHqzPp5i3/H+BcZ3m3Q==} - - '@smithy/middleware-content-length@3.0.5': - resolution: {integrity: sha512-ILEzC2eyxx6ncej3zZSwMpB5RJ0zuqH7eMptxC4KN3f+v9bqT8ohssKbhNR78k/2tWW+KS5Spw+tbPF4Ejyqvw==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-endpoint@3.1.0': - resolution: {integrity: sha512-5y5aiKCEwg9TDPB4yFE7H6tYvGFf1OJHNczeY10/EFF8Ir8jZbNntQJxMWNfeQjC1mxPsaQ6mR9cvQbf+0YeMw==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-retry@3.0.15': - resolution: {integrity: sha512-iTMedvNt1ApdvkaoE8aSDuwaoc+BhvHqttbA/FO4Ty+y/S5hW6Ci/CTScG7vam4RYJWZxdTElc3MEfHRVH6cgQ==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-serde@3.0.3': - resolution: {integrity: sha512-puUbyJQBcg9eSErFXjKNiGILJGtiqmuuNKEYNYfUD57fUl4i9+mfmThtQhvFXU0hCVG0iEJhvQUipUf+/SsFdA==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-stack@3.0.3': - resolution: {integrity: sha512-r4klY9nFudB0r9UdSMaGSyjyQK5adUyPnQN/ZM6M75phTxOdnc/AhpvGD1fQUvgmqjQEBGCwpnPbDm8pH5PapA==} - engines: {node: '>=16.0.0'} - - '@smithy/node-config-provider@3.1.4': - resolution: {integrity: sha512-YvnElQy8HR4vDcAjoy7Xkx9YT8xZP4cBXcbJSgm/kxmiQu08DwUwj8rkGnyoJTpfl/3xYHH+d8zE+eHqoDCSdQ==} - engines: {node: '>=16.0.0'} - - '@smithy/node-http-handler@3.1.4': - resolution: {integrity: sha512-+UmxgixgOr/yLsUxcEKGH0fMNVteJFGkmRltYFHnBMlogyFdpzn2CwqWmxOrfJELhV34v0WSlaqG1UtE1uXlJg==} - engines: {node: '>=16.0.0'} - - '@smithy/property-provider@3.1.3': - resolution: {integrity: sha512-zahyOVR9Q4PEoguJ/NrFP4O7SMAfYO1HLhB18M+q+Z4KFd4V2obiMnlVoUFzFLSPeVt1POyNWneHHrZaTMoc/g==} - engines: {node: '>=16.0.0'} - - '@smithy/protocol-http@4.1.0': - resolution: {integrity: sha512-dPVoHYQ2wcHooGXg3LQisa1hH0e4y0pAddPMeeUPipI1tEOqL6A4N0/G7abeq+K8wrwSgjk4C0wnD1XZpJm5aA==} - engines: {node: '>=16.0.0'} - - '@smithy/querystring-builder@3.0.3': - resolution: {integrity: sha512-vyWckeUeesFKzCDaRwWLUA1Xym9McaA6XpFfAK5qI9DKJ4M33ooQGqvM4J+LalH4u/Dq9nFiC8U6Qn1qi0+9zw==} - engines: {node: '>=16.0.0'} - - '@smithy/querystring-parser@3.0.3': - resolution: {integrity: sha512-zahM1lQv2YjmznnfQsWbYojFe55l0SLG/988brlLv1i8z3dubloLF+75ATRsqPBboUXsW6I9CPGE5rQgLfY0vQ==} - engines: {node: '>=16.0.0'} - - '@smithy/service-error-classification@3.0.3': - resolution: {integrity: sha512-Jn39sSl8cim/VlkLsUhRFq/dKDnRUFlfRkvhOJaUbLBXUsLRLNf9WaxDv/z9BjuQ3A6k/qE8af1lsqcwm7+DaQ==} - engines: {node: '>=16.0.0'} - - '@smithy/shared-ini-file-loader@3.1.4': - resolution: {integrity: sha512-qMxS4hBGB8FY2GQqshcRUy1K6k8aBWP5vwm8qKkCT3A9K2dawUwOIJfqh9Yste/Bl0J2lzosVyrXDj68kLcHXQ==} - engines: {node: '>=16.0.0'} - - '@smithy/signature-v4@4.1.0': - resolution: {integrity: sha512-aRryp2XNZeRcOtuJoxjydO6QTaVhxx/vjaR+gx7ZjaFgrgPRyZ3HCTbfwqYj6ZWEBHkCSUfcaymKPURaByukag==} - engines: {node: '>=16.0.0'} - - '@smithy/smithy-client@3.2.0': - resolution: {integrity: sha512-pDbtxs8WOhJLJSeaF/eAbPgXg4VVYFlRcL/zoNYA5WbG3wBL06CHtBSg53ppkttDpAJ/hdiede+xApip1CwSLw==} - engines: {node: '>=16.0.0'} - - '@smithy/types@3.3.0': - resolution: {integrity: sha512-IxvBBCTFDHbVoK7zIxqA1ZOdc4QfM5HM7rGleCuHi7L1wnKv5Pn69xXJQ9hgxH60ZVygH9/JG0jRgtUncE3QUA==} - engines: {node: '>=16.0.0'} - - '@smithy/url-parser@3.0.3': - resolution: {integrity: sha512-pw3VtZtX2rg+s6HMs6/+u9+hu6oY6U7IohGhVNnjbgKy86wcIsSZwgHrFR+t67Uyxvp4Xz3p3kGXXIpTNisq8A==} - - '@smithy/util-base64@3.0.0': - resolution: {integrity: sha512-Kxvoh5Qtt0CDsfajiZOCpJxgtPHXOKwmM+Zy4waD43UoEMA+qPxxa98aE/7ZhdnBFZFXMOiBR5xbcaMhLtznQQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-body-length-browser@3.0.0': - resolution: {integrity: sha512-cbjJs2A1mLYmqmyVl80uoLTJhAcfzMOyPgjwAYusWKMdLeNtzmMz9YxNl3/jRLoxSS3wkqkf0jwNdtXWtyEBaQ==} - - '@smithy/util-body-length-node@3.0.0': - resolution: {integrity: sha512-Tj7pZ4bUloNUP6PzwhN7K386tmSmEET9QtQg0TgdNOnxhZvCssHji+oZTUIuzxECRfG8rdm2PMw2WCFs6eIYkA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-buffer-from@2.2.0': - resolution: {integrity: sha512-IJdWBbTcMQ6DA0gdNhh/BwrLkDR+ADW5Kr1aZmd4k3DIF6ezMV4R2NIAmT08wQJ3yUK82thHWmC/TnK/wpMMIA==} - engines: {node: '>=14.0.0'} - - '@smithy/util-buffer-from@3.0.0': - resolution: {integrity: sha512-aEOHCgq5RWFbP+UDPvPot26EJHjOC+bRgse5A8V3FSShqd5E5UN4qc7zkwsvJPPAVsf73QwYcHN1/gt/rtLwQA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-config-provider@3.0.0': - resolution: {integrity: sha512-pbjk4s0fwq3Di/ANL+rCvJMKM5bzAQdE5S/6RL5NXgMExFAi6UgQMPOm5yPaIWPpr+EOXKXRonJ3FoxKf4mCJQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-defaults-mode-browser@3.0.15': - resolution: {integrity: sha512-FZ4Psa3vjp8kOXcd3HJOiDPBCWtiilLl57r0cnNtq/Ga9RSDrM5ERL6xt+tO43+2af6Pn5Yp92x2n5vPuduNfg==} - engines: {node: '>= 10.0.0'} - - '@smithy/util-defaults-mode-node@3.0.15': - resolution: {integrity: sha512-KSyAAx2q6d0t6f/S4XB2+3+6aQacm3aLMhs9aLMqn18uYGUepbdssfogW5JQZpc6lXNBnp0tEnR5e9CEKmEd7A==} - engines: {node: '>= 10.0.0'} - - '@smithy/util-endpoints@2.0.5': - resolution: {integrity: sha512-ReQP0BWihIE68OAblC/WQmDD40Gx+QY1Ez8mTdFMXpmjfxSyz2fVQu3A4zXRfQU9sZXtewk3GmhfOHswvX+eNg==} - engines: {node: '>=16.0.0'} - - '@smithy/util-hex-encoding@3.0.0': - resolution: {integrity: sha512-eFndh1WEK5YMUYvy3lPlVmYY/fZcQE1D8oSf41Id2vCeIkKJXPcYDCZD+4+xViI6b1XSd7tE+s5AmXzz5ilabQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-middleware@3.0.3': - resolution: {integrity: sha512-l+StyYYK/eO3DlVPbU+4Bi06Jjal+PFLSMmlWM1BEwyLxZ3aKkf1ROnoIakfaA7mC6uw3ny7JBkau4Yc+5zfWw==} - engines: {node: '>=16.0.0'} - - '@smithy/util-retry@3.0.3': - resolution: {integrity: sha512-AFw+hjpbtVApzpNDhbjNG5NA3kyoMs7vx0gsgmlJF4s+yz1Zlepde7J58zpIRIsdjc+emhpAITxA88qLkPF26w==} - engines: {node: '>=16.0.0'} - - '@smithy/util-stream@3.1.3': - resolution: {integrity: sha512-FIv/bRhIlAxC0U7xM1BCnF2aDRPq0UaelqBHkM2lsCp26mcBbgI0tCVTv+jGdsQLUmAMybua/bjDsSu8RQHbmw==} - engines: {node: '>=16.0.0'} - - '@smithy/util-uri-escape@3.0.0': - resolution: {integrity: sha512-LqR7qYLgZTD7nWLBecUi4aqolw8Mhza9ArpNEQ881MJJIU2sE5iHCK6TdyqqzcDLy0OPe10IY4T8ctVdtynubg==} - engines: {node: '>=16.0.0'} - - '@smithy/util-utf8@2.3.0': - resolution: {integrity: sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A==} - engines: {node: '>=14.0.0'} - - '@smithy/util-utf8@3.0.0': - resolution: {integrity: sha512-rUeT12bxFnplYDe815GXbq/oixEGHfRFFtcTF3YdDi/JaENIM6aSYYLJydG83UNzLXeRI5K8abYd/8Sp/QM0kA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-waiter@3.1.2': - resolution: {integrity: sha512-4pP0EV3iTsexDx+8PPGAKCQpd/6hsQBaQhqWzU4hqKPHN5epPsxKbvUTIiYIHTxaKt6/kEaqPBpu/ufvfbrRzw==} - engines: {node: '>=16.0.0'} - - '@tsconfig/node10@1.0.11': - resolution: {integrity: sha512-DcRjDCujK/kCk/cUe8Xz8ZSpm8mS3mNNpta+jGCA6USEDfktlNvm1+IuZ9eTcDbNk41BHwpHHeW+N1lKCz4zOw==} - - '@tsconfig/node12@1.0.11': - resolution: {integrity: sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==} - - '@tsconfig/node14@1.0.3': - resolution: {integrity: sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==} - - '@tsconfig/node16@1.0.4': - resolution: {integrity: sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==} - - '@types/chai-as-promised@8.0.0': - resolution: {integrity: sha512-YbYaXFqJwSABp9OXQTVrPPmstZgNjkRieWVd/xAl5Yc/e5+F44bXLeQggpvm0sjsS1bg+2Y5cwU+rquwwD2dXA==} - - '@types/chai@4.3.19': - resolution: {integrity: sha512-2hHHvQBVE2FiSK4eN0Br6snX9MtolHaTo/batnLjlGRhoQzlCL61iVpxoqO7SfFyOw+P/pwv+0zNHzKoGWz9Cw==} - - '@types/debug@4.1.12': - resolution: {integrity: sha512-vIChWdVG3LG1SMxEvI/AK+FWJthlrqlTu7fbrlywTkkaONwk/UAGaULXRlf8vkzFBLVm0zkMdCquhL5aOjhXPQ==} - - '@types/estree@1.0.5': - resolution: {integrity: sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==} - - '@types/fluent-ffmpeg@2.1.26': - resolution: {integrity: sha512-0JVF3wdQG+pN0ImwWD0bNgJiKF2OHg/7CDBHw5UIbRTvlnkgGHK6V5doE54ltvhud4o31/dEiHm23CAlxFiUQg==} - - '@types/interpret@1.1.3': - resolution: {integrity: sha512-uBaBhj/BhilG58r64mtDb/BEdH51HIQLgP5bmWzc5qCtFMja8dCk/IOJmk36j0lbi9QHwI6sbtUNGuqXdKCAtQ==} - - '@types/mocha@10.0.7': - resolution: {integrity: sha512-GN8yJ1mNTcFcah/wKEFIJckJx9iJLoMSzWcfRRuxz/Jk+U6KQNnml+etbtxFK8lPjzOw3zp4Ha/kjSst9fsHYw==} - - '@types/ms@0.7.34': - resolution: {integrity: sha512-nG96G3Wp6acyAgJqGasjODb+acrI7KltPiRxzHPXnP3NgI28bpQDRv53olbqGXbfcgF5aiiHmO3xpwEpS5Ld9g==} - - '@types/node@20.16.3': - resolution: {integrity: sha512-/wdGiWRkMOm53gAsSyFMXFZHbVg7C6CbkrzHNpaHoYfsUWPg7m6ZRKtvQjgvQ9i8WT540a3ydRlRQbxjY30XxQ==} - - '@types/node@22.5.2': - resolution: {integrity: sha512-acJsPTEqYqulZS/Yp/S3GgeE6GZ0qYODUR8aVr/DkhHQ8l9nd4j5x1/ZJy9/gHrRlFMqkO6i0I3E27Alu4jjPg==} - - '@types/pg@8.11.8': - resolution: {integrity: sha512-IqpCf8/569txXN/HoP5i1LjXfKZWL76Yr2R77xgeIICUbAYHeoaEZFhYHo2uDftecLWrTJUq63JvQu8q3lnDyA==} - - '@types/qs@6.9.15': - resolution: {integrity: sha512-uXHQKES6DQKKCLh441Xv/dwxOq1TVS3JPUMlEqoEglvlhR6Mxnlew/Xq/LRVHpLyk7iK3zODe1qYHIMltO7XGg==} - - '@types/retry@0.12.2': - resolution: {integrity: sha512-XISRgDJ2Tc5q4TRqvgJtzsRkFYNJzZrhTdtMoGVBttwzzQJkPnS3WWTFc7kuDRoPtPakl+T+OfdEUjYJj7Jbow==} - - '@types/semver@7.5.8': - resolution: {integrity: sha512-I8EUhyrgfLrcTkzV3TSsGyl1tSuPrEDzr0yd5m90UgNxQkyDXULk3b6MlQqTCpZpNtWe1K0hzclnZkTcLBe2UQ==} - - '@types/sinon-chai@3.2.12': - resolution: {integrity: sha512-9y0Gflk3b0+NhQZ/oxGtaAJDvRywCa5sIyaVnounqLvmf93yBF4EgIRspePtkMs3Tr844nCclYMlcCNmLCvjuQ==} - - '@types/sinon@10.0.20': - resolution: {integrity: sha512-2APKKruFNCAZgx3daAyACGzWuJ028VVCUDk6o2rw/Z4PXT0ogwdV4KUegW0MwVs0Zu59auPXbbuBJHF12Sx1Eg==} - - '@types/sinon@17.0.3': - resolution: {integrity: sha512-j3uovdn8ewky9kRBG19bOwaZbexJu/XjtkHyjvUgt4xfPFz18dcORIMqnYh66Fx3Powhcr85NT5+er3+oViapw==} - - '@types/sinonjs__fake-timers@8.1.5': - resolution: {integrity: sha512-mQkU2jY8jJEF7YHjHvsQO8+3ughTL1mcnn96igfhONmR+fUPSKIkefQYpSe8bsly2Ep7oQbn/6VG5/9/0qcArQ==} - - '@types/triple-beam@1.3.5': - resolution: {integrity: sha512-6WaYesThRMCl19iryMYP7/x2OVgCtbIVflDGFpWnb9irXI3UjYE4AzmYuiUKY1AJstGijoY+MgUszMgRxIYTYw==} - - '@types/ws@8.5.12': - resolution: {integrity: sha512-3tPRkv1EtkDpzlgyKyI8pGsGZAGPEaXeu0DOj5DI25Ja91bdAYddYHbADRYVrZMRbfW+1l5YwXVDKohDJNQxkQ==} - - '@vladfrangu/async_event_emitter@2.4.6': - resolution: {integrity: sha512-RaI5qZo6D2CVS6sTHFKg1v5Ohq/+Bo2LZ5gzUEwZ/WkHhwtGTCB/sVLw8ijOkAUxasZ+WshN/Rzj4ywsABJ5ZA==} - engines: {node: '>=v14.0.0', npm: '>=7.0.0'} - - abort-controller@3.0.0: - resolution: {integrity: sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==} - engines: {node: '>=6.5'} - - abstract-logging@2.0.1: - resolution: {integrity: sha512-2BjRTZxTPvheOvGbBslFSYOUkr+SjPtOnrLP33f+VIWLzezQpZcqVg7ja3L4dBXmzzgwT+a029jRx5PCi3JuiA==} - - acorn-walk@8.3.3: - resolution: {integrity: sha512-MxXdReSRhGO7VlFe1bRG/oI7/mdLV9B9JJT0N8vZOhF7gFRR5l3M8W9G8JxmKV+JC5mGqJ0QvqfSOLsCPa4nUw==} - engines: {node: '>=0.4.0'} - - acorn@8.12.1: - resolution: {integrity: sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==} - engines: {node: '>=0.4.0'} - hasBin: true - - ajv-formats@2.1.1: - resolution: {integrity: sha512-Wx0Kx52hxE7C18hkMEggYlEifqWZtYaRgouJor+WMdPnQyEK13vgEWyVNup7SoeeoLMsr4kf5h6dOW11I15MUA==} - peerDependencies: - ajv: ^8.0.0 - peerDependenciesMeta: - ajv: - optional: true - - ajv-formats@3.0.1: - resolution: {integrity: sha512-8iUql50EUR+uUcdRQ3HDqa6EVyo3docL8g5WJ3FNcWmu62IbkGUue/pEyLBW8VGKKucTPgqeks4fIU1DA4yowQ==} - peerDependencies: - ajv: ^8.0.0 - peerDependenciesMeta: - ajv: - optional: true - - ajv@8.17.1: - resolution: {integrity: sha512-B/gBuNg5SiMTrPkC+A2+cW0RszwxYmn6VYxB/inlBStS5nx6xHIt/ehKRhIMhqusl7a8LjQoZnjCs5vhwxOQ1g==} - - ansi-colors@4.1.3: - resolution: {integrity: sha512-/6w/C21Pm1A7aZitlI5Ni/2J6FFQN8i1Cvz3kHABAAbw93v/NlvKdVOqz7CCWz/3iv/JplRSEEZ83XION15ovw==} - engines: {node: '>=6'} - - ansi-regex@5.0.1: - resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==} - engines: {node: '>=8'} - - ansi-regex@6.0.1: - resolution: {integrity: sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==} - engines: {node: '>=12'} - - ansi-styles@3.2.1: - resolution: {integrity: sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==} - engines: {node: '>=4'} - - ansi-styles@4.3.0: - resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} - engines: {node: '>=8'} - - ansi-styles@6.2.1: - resolution: {integrity: sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==} - engines: {node: '>=12'} - - any-promise@1.3.0: - resolution: {integrity: sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==} - - anymatch@3.1.3: - resolution: {integrity: sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==} - engines: {node: '>= 8'} - - arg@4.1.3: - resolution: {integrity: sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==} - - argparse@2.0.1: - resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==} - - array-union@2.1.0: - resolution: {integrity: sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==} - engines: {node: '>=8'} - - asap@2.0.6: - resolution: {integrity: sha512-BSHWgDSAiKs50o2Re8ppvp3seVHXSRM44cdSsT9FfNEUUZLOGWVCsiWaRPWM1Znn+mqZ1OfVZ3z3DWEzSp7hRA==} - - assertion-error@2.0.1: - resolution: {integrity: sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==} - engines: {node: '>=12'} - - async@0.2.10: - resolution: {integrity: sha512-eAkdoKxU6/LkKDBzLpT+t6Ff5EtfSF4wx1WfJiPEEV7WNLnDaRXk0oVysiEPm262roaachGexwUv94WhSgN5TQ==} - - async@3.2.6: - resolution: {integrity: sha512-htCUDlxyyCLMgaM3xXg0C0LW2xqfuQ6p05pCEIsXuyQ+a1koYKTuBMzRNwmybfLgvJDMd0r1LTn4+E0Ti6C2AA==} - - atomic-sleep@1.0.0: - resolution: {integrity: sha512-kNOjDqAh7px0XWNI+4QbzoiR/nTkHAWNud2uvnJquD1/x5a7EQZMJT0AczqK0Qn67oY/TTQ1LbUKajZpp3I9tQ==} - engines: {node: '>=8.0.0'} - - available-typed-arrays@1.0.7: - resolution: {integrity: sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==} - engines: {node: '>= 0.4'} - - avvio@8.4.0: - resolution: {integrity: sha512-CDSwaxINFy59iNwhYnkvALBwZiTydGkOecZyPkqBpABYR1KqGEsET0VOOYDwtleZSUIdeY36DC2bSZ24CO1igA==} - - aws-sdk-client-mock@4.0.1: - resolution: {integrity: sha512-yD2mmgy73Xce097G5hIpr1k7j50qzvJ49/+6osGZiCyk4m6cwhb+2x7kKFY1gEMwTzaS8+m8fXv9SB29SkRYyQ==} - - aws-sdk-mock@6.1.1: - resolution: {integrity: sha512-WqGodclZDv5tZmoC/1A4V4c4hSwageOoy90cJisrkdLUq/1jEhulajkNL4UkmD9Nc6zhQuLD6WXIbB/ROM+K2A==} - engines: {node: '>=18.0.0'} - - aws-sdk@2.1687.0: - resolution: {integrity: sha512-Pk7RbIxJ8yDmFJRKzaapiUsAvz5cTPKCz7soomU+lASx1jvO29Z9KAPB6KJR22m7rDDMO/HNNN9OJRzfdvh7xQ==} - engines: {node: '>= 10.0.0'} - - balanced-match@1.0.2: - resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} - - base64-js@1.5.1: - resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==} - - binary-extensions@2.3.0: - resolution: {integrity: sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==} - engines: {node: '>=8'} - - boolbase@1.0.0: - resolution: {integrity: sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==} - - bowser@2.11.0: - resolution: {integrity: sha512-AlcaJBi/pqqJBIQ8U9Mcpc9i8Aqxn88Skv5d+xBX006BY5u8N3mGLHa5Lgppa7L/HfwgwLgZ6NYs+Ag6uUmJRA==} - - brace-expansion@1.1.11: - resolution: {integrity: sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==} - - brace-expansion@2.0.1: - resolution: {integrity: sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==} - - braces@3.0.3: - resolution: {integrity: sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==} - engines: {node: '>=8'} - - browser-stdout@1.3.1: - resolution: {integrity: sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==} - - buffer@4.9.2: - resolution: {integrity: sha512-xq+q3SRMOxGivLhBNaUdC64hDTQwejJ+H0T/NB1XMtTVEwNTrfFF3gAxiyW0Bu/xWEGhjVKgUcMhCrUy2+uCWg==} - - buffer@5.6.0: - resolution: {integrity: sha512-/gDYp/UtU0eA1ys8bOs9J6a+E/KWIY+DZ+Q2WESNUA0jFRsJOc0SNUO6xJ5SGA1xueg3NL65W6s+NY5l9cunuw==} - - buffer@6.0.3: - resolution: {integrity: sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==} - - bundle-require@5.0.0: - resolution: {integrity: sha512-GuziW3fSSmopcx4KRymQEJVbZUfqlCqcq7dvs6TYwKRZiegK/2buMxQTPs6MGlNv50wms1699qYO54R8XfRX4w==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - peerDependencies: - esbuild: '>=0.18' - - cac@6.7.14: - resolution: {integrity: sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==} - engines: {node: '>=8'} - - call-bind@1.0.7: - resolution: {integrity: sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==} - engines: {node: '>= 0.4'} - - callsites@3.1.0: - resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==} - engines: {node: '>=6'} - - camelcase@6.3.0: - resolution: {integrity: sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==} - engines: {node: '>=10'} - - chai-as-promised@8.0.0: - resolution: {integrity: sha512-sMsGXTrS3FunP/wbqh/KxM8Kj/aLPXQGkNtvE5wPfSToq8wkkvBpTZo1LIiEVmC4BwkKpag+l5h/20lBMk6nUg==} - peerDependencies: - chai: '>= 2.1.2 < 6' - - chai@5.1.1: - resolution: {integrity: sha512-pT1ZgP8rPNqUgieVaEY+ryQr6Q4HXNg8Ei9UnLUrjN4IA7dvQC5JB+/kxVcPNDHyBcc/26CXPkbNzq3qwrOEKA==} - engines: {node: '>=12'} - - chalk@2.4.2: - resolution: {integrity: sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==} - engines: {node: '>=4'} - - chalk@4.1.2: - resolution: {integrity: sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==} - engines: {node: '>=10'} - - check-error@2.1.1: - resolution: {integrity: sha512-OAlb+T7V4Op9OwdkjmguYRqncdlx5JiofwOAUkmTF+jNdHwzTaTs4sRAGpzLF3oOz5xAyDGrPgeIDFQmDOTiJw==} - engines: {node: '>= 16'} - - cheerio-select@2.1.0: - resolution: {integrity: sha512-9v9kG0LvzrlcungtnJtpGNxY+fzECQKhK4EGJX2vByejiMX84MFNQw4UxPJl3bFbTMw+Dfs37XaIkCwTZfLh4g==} - - cheerio@1.0.0: - resolution: {integrity: sha512-quS9HgjQpdaXOvsZz82Oz7uxtXiy6UIsIQcpBj7HRw2M63Skasm9qlDocAM7jNuaxdhpPU7c4kJN+gA5MCu4ww==} - engines: {node: '>=18.17'} - - chokidar@3.6.0: - resolution: {integrity: sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==} - engines: {node: '>= 8.10.0'} - - cliui@7.0.4: - resolution: {integrity: sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==} - - cliui@8.0.1: - resolution: {integrity: sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==} - engines: {node: '>=12'} - - cluster-key-slot@1.1.2: - resolution: {integrity: sha512-RMr0FhtfXemyinomL4hrWcYJxmX6deFdCxpJzhDttxgO1+bcCnkk+9drydLVDmAMG7NE6aN/fl4F7ucU/90gAA==} - engines: {node: '>=0.10.0'} - - color-convert@1.9.3: - resolution: {integrity: sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==} - - color-convert@2.0.1: - resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} - engines: {node: '>=7.0.0'} - - color-name@1.1.3: - resolution: {integrity: sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==} - - color-name@1.1.4: - resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} - - color-string@1.9.1: - resolution: {integrity: sha512-shrVawQFojnZv6xM40anx4CkoDP+fZsw/ZerEMsW/pyzsRbElpsL/DBVW7q3ExxwusdNXI3lXpuhEZkzs8p5Eg==} - - color@3.2.1: - resolution: {integrity: sha512-aBl7dZI9ENN6fUGC7mWpMTPNHmWUSNan9tuWN6ahh5ZLNk9baLJOnSMlrQkHcrfFgz2/RigjUVAjdx36VcemKA==} - - colorette@2.0.20: - resolution: {integrity: sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w==} - - colorspace@1.1.4: - resolution: {integrity: sha512-BgvKJiuVu1igBUF2kEjRCZXol6wiiGbY5ipL/oVPwm0BL9sIpMIzM8IK7vwuxIIzOXMV3Ey5w+vxhm0rR/TN8w==} - - commander@4.1.1: - resolution: {integrity: sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==} - engines: {node: '>= 6'} - - concat-map@0.0.1: - resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} - - consola@3.2.3: - resolution: {integrity: sha512-I5qxpzLv+sJhTVEoLYNcTW+bThDCPsit0vLNKShZx6rLtpilNpmmeTPaeqJb9ZE9dV3DGaeby6Vuhrw38WjeyQ==} - engines: {node: ^14.18.0 || >=16.10.0} - - cookie@0.6.0: - resolution: {integrity: sha512-U71cyTamuh1CRNCfpGY6to28lxvNwPG4Guz/EVjgf3Jmzv0vlDp1atT9eS5dDjMYHucpHbWns6Lwf3BKz6svdw==} - engines: {node: '>= 0.6'} - - cosmiconfig@8.3.6: - resolution: {integrity: sha512-kcZ6+W5QzcJ3P1Mt+83OUv/oHFqZHIx8DuxG6eZ5RGMERoLqp4BuGjhHLYGK+Kf5XVkQvqBSmAy/nGWN3qDgEA==} - engines: {node: '>=14'} - peerDependencies: - typescript: '>=4.9.5' - peerDependenciesMeta: - typescript: - optional: true - - create-require@1.1.1: - resolution: {integrity: sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==} - - cron-parser@4.9.0: - resolution: {integrity: sha512-p0SaNjrHOnQeR8/VnfGbmg9te2kfyYSQ7Sc/j/6DtPL3JQvKxmjO9TSjNFpujqV3vEYYBvNNvXSxzyksBWAx1Q==} - engines: {node: '>=12.0.0'} - - cross-spawn@7.0.3: - resolution: {integrity: sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==} - engines: {node: '>= 8'} - - csprng@0.1.2: - resolution: {integrity: sha512-D3WAbvvgUVIqSxUfdvLeGjuotsB32bvfVPd+AaaTWMtyUeC9zgCnw5xs94no89yFLVsafvY9dMZEhTwsY/ZecA==} - engines: {node: '>=0.6.0'} - - css-select@5.1.0: - resolution: {integrity: sha512-nwoRF1rvRRnnCqqY7updORDsuqKzqYJ28+oSMaJMMgOauh3fvwHqMS7EZpIPqK8GL+g9mKxF1vP/ZjSeNjEVHg==} - - css-what@6.1.0: - resolution: {integrity: sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw==} - engines: {node: '>= 6'} - - data-uri-to-buffer@4.0.1: - resolution: {integrity: sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A==} - engines: {node: '>= 12'} - - date-fns@3.6.0: - resolution: {integrity: sha512-fRHTG8g/Gif+kSh50gaGEdToemgfj74aRX3swtiouboip5JDLAyDE9F11nHMIcvOaXeOC6D7SpNhi7uFyB7Uww==} - - dateformat@4.6.3: - resolution: {integrity: sha512-2P0p0pFGzHS5EMnhdxQi7aJN+iMheud0UhG4dlE1DLAlvL8JHjJJTX/CSm4JXwV0Ka5nGk3zC5mcb5bUQUxxMA==} - - debug@4.3.6: - resolution: {integrity: sha512-O/09Bd4Z1fBrU4VzkhFqVgpPzaGbw6Sm9FEkBT1A/YBXQFGuuSxa1dN2nxgxS34JmKXqYx8CZAwEVoJFImUXIg==} - engines: {node: '>=6.0'} - peerDependencies: - supports-color: '*' - peerDependenciesMeta: - supports-color: - optional: true - - decamelize@4.0.0: - resolution: {integrity: sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==} - engines: {node: '>=10'} - - deep-eql@5.0.2: - resolution: {integrity: sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==} - engines: {node: '>=6'} - - define-data-property@1.1.4: - resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==} - engines: {node: '>= 0.4'} - - denque@2.1.0: - resolution: {integrity: sha512-HVQE3AAb/pxF8fQAoiqpvg9i3evqug3hoiwakOyZAwJm+6vZehbkYXZ0l4JxS+I3QxM97v5aaRNhj8v5oBhekw==} - engines: {node: '>=0.10'} - - diff@4.0.2: - resolution: {integrity: sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==} - engines: {node: '>=0.3.1'} - - diff@5.2.0: - resolution: {integrity: sha512-uIFDxqpRZGZ6ThOk84hEfqWoHx2devRFvpTZcTHur85vImfaxUbTW9Ryh4CpCuDnToOP1CEtXKIgytHBPVff5A==} - engines: {node: '>=0.3.1'} - - dir-glob@3.0.1: - resolution: {integrity: sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==} - engines: {node: '>=8'} - - discord-api-types@0.37.83: - resolution: {integrity: sha512-urGGYeWtWNYMKnYlZnOnDHm8fVRffQs3U0SpE8RHeiuLKb/u92APS8HoQnPTFbnXmY1vVnXjXO4dOxcAn3J+DA==} - - discord-api-types@0.37.97: - resolution: {integrity: sha512-No1BXPcVkyVD4ZVmbNgDKaBoqgeQ+FJpzZ8wqHkfmBnTZig1FcH3iPPersiK1TUIAzgClh2IvOuVUYfcWLQAOA==} - - discord.js@14.16.1: - resolution: {integrity: sha512-/diX4shp3q1F3EySGQbQl10el+KIpffLSOJdtM35gGV7zw2ED7rKbASKJT7UIR9L/lTd0KtNenZ/h739TN7diA==} - engines: {node: '>=18'} - - diskusage@1.2.0: - resolution: {integrity: sha512-2u3OG3xuf5MFyzc4MctNRUKjjwK+UkovRYdD2ed/NZNZPrt0lqHnLKxGhlFVvAb4/oufIgQG3nWgwmeTbHOvXA==} - - dom-serializer@2.0.0: - resolution: {integrity: sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg==} - - domelementtype@2.3.0: - resolution: {integrity: sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==} - - domhandler@5.0.3: - resolution: {integrity: sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==} - engines: {node: '>= 4'} - - domutils@3.1.0: - resolution: {integrity: sha512-H78uMmQtI2AhgDJjWeQmHwJJ2bLPD3GMmO7Zja/ZZh84wkm+4ut+IUnUdRa8uCGX88DiVx1j6FRe1XfxEgjEZA==} - - dotenv@16.4.5: - resolution: {integrity: sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==} - engines: {node: '>=12'} - - eastasianwidth@0.2.0: - resolution: {integrity: sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==} - - emoji-regex@8.0.0: - resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==} - - emoji-regex@9.2.2: - resolution: {integrity: sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==} - - enabled@2.0.0: - resolution: {integrity: sha512-AKrN98kuwOzMIdAizXGI86UFBoo26CL21UM763y1h/GMSJ4/OHU9k2YlsmBpyScFo/wbLzWQJBMCW4+IO3/+OQ==} - - encoding-sniffer@0.2.0: - resolution: {integrity: sha512-ju7Wq1kg04I3HtiYIOrUrdfdDvkyO9s5XM8QAj/bN61Yo/Vb4vgJxy5vi4Yxk01gWHbrofpPtpxM8bKger9jhg==} - - end-of-stream@1.4.4: - resolution: {integrity: sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==} - - entities@4.5.0: - resolution: {integrity: sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==} - engines: {node: '>=0.12'} - - error-ex@1.3.2: - resolution: {integrity: sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==} - - es-define-property@1.0.0: - resolution: {integrity: sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==} - engines: {node: '>= 0.4'} - - es-errors@1.3.0: - resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==} - engines: {node: '>= 0.4'} - - es6-promise@4.2.8: - resolution: {integrity: sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==} - - esbuild@0.23.1: - resolution: {integrity: sha512-VVNz/9Sa0bs5SELtn3f7qhJCDPCF5oMEl5cO9/SSinpE9hbPVvxbd572HH5AKiP7WD8INO53GgfDDhRjkylHEg==} - engines: {node: '>=18'} - hasBin: true - - escalade@3.2.0: - resolution: {integrity: sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==} - engines: {node: '>=6'} - - escape-string-regexp@1.0.5: - resolution: {integrity: sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==} - engines: {node: '>=0.8.0'} - - escape-string-regexp@4.0.0: - resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} - engines: {node: '>=10'} - - event-target-shim@5.0.1: - resolution: {integrity: sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==} - engines: {node: '>=6'} - - events@1.1.1: - resolution: {integrity: sha512-kEcvvCBByWXGnZy6JUlgAp2gBIUjfCAV6P6TgT1/aaQKcmuAEC4OZTV1I4EWQLz2gxZw76atuVyvHhTxvi0Flw==} - engines: {node: '>=0.4.x'} - - events@3.3.0: - resolution: {integrity: sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==} - engines: {node: '>=0.8.x'} - - execa@5.1.1: - resolution: {integrity: sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==} - engines: {node: '>=10'} - - execa@9.3.1: - resolution: {integrity: sha512-gdhefCCNy/8tpH/2+ajP9IQc14vXchNdd0weyzSJEFURhRMGncQ+zKFxwjAufIewPEJm9BPOaJnvg2UtlH2gPQ==} - engines: {node: ^18.19.0 || >=20.5.0} - - exponential-backoff@3.1.1: - resolution: {integrity: sha512-dX7e/LHVJ6W3DE1MHWi9S1EYzDESENfLrYohG2G++ovZrYOkm4Knwa0mc1cn84xJOR4KEU0WSchhLbd0UklbHw==} - - fast-content-type-parse@1.1.0: - resolution: {integrity: sha512-fBHHqSTFLVnR61C+gltJuE5GkVQMV0S2nqUO8TJ+5Z3qAKG8vAx4FKai1s5jq/inV1+sREynIWSuQ6HgoSXpDQ==} - - fast-copy@3.0.2: - resolution: {integrity: sha512-dl0O9Vhju8IrcLndv2eU4ldt1ftXMqqfgN4H1cpmGV7P6jeB9FwpN9a2c8DPGE1Ys88rNUJVYDHq73CGAGOPfQ==} - - fast-decode-uri-component@1.0.1: - resolution: {integrity: sha512-WKgKWg5eUxvRZGwW8FvfbaH7AXSh2cL+3j5fMGzUMCxWBJ3dV3a7Wz8y2f/uQ0e3B6WmodD3oS54jTQ9HVTIIg==} - - fast-deep-equal@3.1.3: - resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==} - - fast-glob@3.3.2: - resolution: {integrity: sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==} - engines: {node: '>=8.6.0'} - - fast-json-stringify@5.16.1: - resolution: {integrity: sha512-KAdnLvy1yu/XrRtP+LJnxbBGrhN+xXu+gt3EUvZhYGKCr3lFHq/7UFJHHFgmJKoqlh6B40bZLEv7w46B0mqn1g==} - - fast-querystring@1.1.2: - resolution: {integrity: sha512-g6KuKWmFXc0fID8WWH0jit4g0AGBoJhCkJMb1RmbsSEUNvQ+ZC8D6CUZ+GtF8nMzSPXnhiePyyqqipzNNEnHjg==} - - fast-redact@3.5.0: - resolution: {integrity: sha512-dwsoQlS7h9hMeYUq1W++23NDcBLV4KqONnITDV9DjfS3q1SgDGVrBdvvTLUotWtPSD7asWDV9/CmsZPy8Hf70A==} - engines: {node: '>=6'} - - fast-safe-stringify@2.1.1: - resolution: {integrity: sha512-W+KJc2dmILlPplD/H4K9l9LcAHAfPtP6BY84uVLXQ6Evcz9Lcg33Y2z1IVblT6xdY54PXYVHEv+0Wpq8Io6zkA==} - - fast-uri@2.4.0: - resolution: {integrity: sha512-ypuAmmMKInk5q7XcepxlnUWDLWv4GFtaJqAzWKqn62IpQ3pejtr5dTVbt3vwqVaMKmkNR55sTT+CqUKIaT21BA==} - - fast-uri@3.0.1: - resolution: {integrity: sha512-MWipKbbYiYI0UC7cl8m/i/IWTqfC8YXsqjzybjddLsFjStroQzsHXkc73JutMvBiXmOvapk+axIl79ig5t55Bw==} - - fast-xml-parser@4.4.1: - resolution: {integrity: sha512-xkjOecfnKGkSsOwtZ5Pz7Us/T6mrbPQrq0nh+aCO5V9nk5NLWmasAHumTKjiPJPWANe+kAZ84Jc8ooJkzZ88Sw==} - hasBin: true - - fastify-plugin@4.5.1: - resolution: {integrity: sha512-stRHYGeuqpEZTL1Ef0Ovr2ltazUT9g844X5z/zEBFLG8RYlpDiOCIG+ATvYEp+/zmc7sN29mcIMp8gvYplYPIQ==} - - fastify@4.28.1: - resolution: {integrity: sha512-kFWUtpNr4i7t5vY2EJPCN2KgMVpuqfU4NjnJNCgiNB900oiDeYqaNDRcAfeBbOF5hGixixxcKnOU4KN9z6QncQ==} - - fastq@1.17.1: - resolution: {integrity: sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==} - - faye-websocket@0.11.4: - resolution: {integrity: sha512-CzbClwlXAuiRQAlUyfqPgvPoNKTckTPGfwZV4ZdAhVcP2lh9KUxJg2b5GkE7XbjKQ3YJnQ9z6D9ntLAlB+tP8g==} - engines: {node: '>=0.8.0'} - - faye@1.4.0: - resolution: {integrity: sha512-kRrIg4be8VNYhycS2PY//hpBJSzZPr/DBbcy9VWelhZMW3KhyLkQR0HL0k0MNpmVoNFF4EdfMFkNAWjTP65g6w==} - engines: {node: '>=0.8.0'} - - fecha@4.2.3: - resolution: {integrity: sha512-OP2IUU6HeYKJi3i0z4A19kHMQoLVs4Hc+DPqqxI2h/DPZHTm/vjsfC6P0b4jCMy14XizLBqvndQ+UilD7707Jw==} - - fetch-blob@3.2.0: - resolution: {integrity: sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ==} - engines: {node: ^12.20 || >= 14.13} - - figures@6.1.0: - resolution: {integrity: sha512-d+l3qxjSesT4V7v2fh+QnmFnUWv9lSpjarhShNTgBOfA0ttejbQUAlHLitbjkoRiDulW0OPoQPYIGhIC8ohejg==} - engines: {node: '>=18'} - - fill-range@7.1.1: - resolution: {integrity: sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==} - engines: {node: '>=8'} - - find-my-way@8.2.0: - resolution: {integrity: sha512-HdWXgFYc6b1BJcOBDBwjqWuHJj1WYiqrxSh25qtU4DabpMFdj/gSunNBQb83t+8Zt67D7CXEzJWTkxaShMTMOA==} - engines: {node: '>=14'} - - find-up@5.0.0: - resolution: {integrity: sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==} - engines: {node: '>=10'} - - flat@5.0.2: - resolution: {integrity: sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==} - hasBin: true - - fluent-ffmpeg@2.1.3: - resolution: {integrity: sha512-Be3narBNt2s6bsaqP6Jzq91heDgOEaDCJAXcE3qcma/EJBSy5FB4cvO31XBInuAuKBx8Kptf8dkhjK0IOru39Q==} - engines: {node: '>=18'} - - fn.name@1.1.0: - resolution: {integrity: sha512-GRnmB5gPyJpAhTQdSZTSp9uaPSvl09KoYcMQtsB9rQoOmzs9dH6ffeccH+Z+cv6P68Hu5bC6JjRh4Ah/mHSNRw==} - - for-each@0.3.3: - resolution: {integrity: sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==} - - foreground-child@3.3.0: - resolution: {integrity: sha512-Ld2g8rrAyMYFXBhEqMz8ZAHBi4J4uS1i/CxGMDnjyFWddMXLVcDp051DZfu+t7+ab7Wv6SMqpWmyFIj5UbfFvg==} - engines: {node: '>=14'} - - formdata-polyfill@4.0.10: - resolution: {integrity: sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g==} - engines: {node: '>=12.20.0'} - - forwarded@0.2.0: - resolution: {integrity: sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==} - engines: {node: '>= 0.6'} - - fs.realpath@1.0.0: - resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==} - - fsevents@2.3.3: - resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} - engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} - os: [darwin] - - function-bind@1.1.2: - resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} - - get-caller-file@2.0.5: - resolution: {integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==} - engines: {node: 6.* || 8.* || >= 10.*} - - get-func-name@2.0.2: - resolution: {integrity: sha512-8vXOvuE167CtIc3OyItco7N/dpRtBbYOsPsXCz7X/PMnlGjYjSGuZJgM1Y7mmew7BKf9BqvLX2tnOVy1BBUsxQ==} - - get-intrinsic@1.2.4: - resolution: {integrity: sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==} - engines: {node: '>= 0.4'} - - get-stream@6.0.1: - resolution: {integrity: sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==} - engines: {node: '>=10'} - - get-stream@9.0.1: - resolution: {integrity: sha512-kVCxPF3vQM/N0B1PmoqVUqgHP+EeVjmZSQn+1oCRPxd2P21P2F19lIgbR3HBosbB1PUhOAoctJnfEn2GbN2eZA==} - engines: {node: '>=18'} - - get-tsconfig@4.8.0: - resolution: {integrity: sha512-Pgba6TExTZ0FJAn1qkJAjIeKoDJ3CsI2ChuLohJnZl/tTU8MVrq3b+2t5UOPfRa4RMsorClBjJALkJUMjG1PAw==} - - glob-parent@5.1.2: - resolution: {integrity: sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==} - engines: {node: '>= 6'} - - glob@10.4.5: - resolution: {integrity: sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg==} - hasBin: true - - glob@8.1.0: - resolution: {integrity: sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==} - engines: {node: '>=12'} - deprecated: Glob versions prior to v9 are no longer supported - - globby@11.1.0: - resolution: {integrity: sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==} - engines: {node: '>=10'} - - gopd@1.0.1: - resolution: {integrity: sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==} - - graphile-config@0.0.1-beta.9: - resolution: {integrity: sha512-7vNxXZ24OAgXxDKXYi9JtgWPMuNbBL3057Yf32Ux+/rVP4+EePgySCc+NNnn0tORi8qwqVreN8bdWqGIcSwNXg==} - engines: {node: '>=16'} - - graphile-worker@0.16.6: - resolution: {integrity: sha512-e7gGYDmGqzju2l83MpzX8vNG/lOtVJiSzI3eZpAFubSxh/cxs7sRrRGBGjzBP1kNG0H+c95etPpNRNlH65PYhw==} - engines: {node: '>=14.0.0'} - hasBin: true - - has-flag@3.0.0: - resolution: {integrity: sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==} - engines: {node: '>=4'} - - has-flag@4.0.0: - resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} - engines: {node: '>=8'} - - has-property-descriptors@1.0.2: - resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==} - - has-proto@1.0.3: - resolution: {integrity: sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==} - engines: {node: '>= 0.4'} - - has-symbols@1.0.3: - resolution: {integrity: sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==} - engines: {node: '>= 0.4'} - - has-tostringtag@1.0.2: - resolution: {integrity: sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==} - engines: {node: '>= 0.4'} - - hasown@2.0.2: - resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} - engines: {node: '>= 0.4'} - - he@1.2.0: - resolution: {integrity: sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==} - hasBin: true - - help-me@5.0.0: - resolution: {integrity: sha512-7xgomUX6ADmcYzFik0HzAxh/73YlKR9bmFzf51CZwR+b6YtzU2m0u49hQCqV6SvlqIqsaxovfwdvbnsw3b/zpg==} - - htmlparser2@9.1.0: - resolution: {integrity: sha512-5zfg6mHUoaer/97TxnGpxmbR7zJtPwIYFMZ/H5ucTlPZhKvtum05yiPK3Mgai3a0DyVxv7qYqoweaEd2nrYQzQ==} - - http-parser-js@0.5.8: - resolution: {integrity: sha512-SGeBX54F94Wgu5RH3X5jsDtf4eHyRogWX1XGT3b4HuW3tQPM4AaBzoUji/4AAJNXCEOWZ5O0DgZmJw1947gD5Q==} - - https@1.0.0: - resolution: {integrity: sha512-4EC57ddXrkaF0x83Oj8sM6SLQHAWXw90Skqu2M4AEWENZ3F02dFJE/GARA8igO79tcgYqGrD7ae4f5L3um2lgg==} - - human-signals@2.1.0: - resolution: {integrity: sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==} - engines: {node: '>=10.17.0'} - - human-signals@8.0.0: - resolution: {integrity: sha512-/1/GPCpDUCCYwlERiYjxoczfP0zfvZMU/OWgQPMya9AbAE24vseigFdhAMObpc8Q4lc/kjutPfUddDYyAmejnA==} - engines: {node: '>=18.18.0'} - - iconv-lite@0.6.3: - resolution: {integrity: sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==} - engines: {node: '>=0.10.0'} - - ieee754@1.1.13: - resolution: {integrity: sha512-4vf7I2LYV/HaWerSo3XmlMkp5eZ83i+/CDluXi/IGTs/O1sejBNhTtnxzmRZfvOUqj7lZjqHkeTvpgSFDlWZTg==} - - ieee754@1.2.1: - resolution: {integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==} - - ignore-by-default@1.0.1: - resolution: {integrity: sha512-Ius2VYcGNk7T90CppJqcIkS5ooHUZyIQK+ClZfMfMNFEF9VSE73Fq+906u/CWu92x4gzZMWOwfFYckPObzdEbA==} - - ignore@5.3.2: - resolution: {integrity: sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==} - engines: {node: '>= 4'} - - import-fresh@3.3.0: - resolution: {integrity: sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==} - engines: {node: '>=6'} - - inflight@1.0.6: - resolution: {integrity: sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==} - deprecated: This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful. - - inherits@2.0.4: - resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} - - interpret@3.1.1: - resolution: {integrity: sha512-6xwYfHbajpoF0xLW+iwLkhwgvLoZDfjYfoFNu8ftMoXINzwuymNLd9u/KmwtdT2GbR+/Cz66otEGEVVUHX9QLQ==} - engines: {node: '>=10.13.0'} - - ioredis@5.4.1: - resolution: {integrity: sha512-2YZsvl7jopIa1gaePkeMtd9rAcSjOOjPtpcLlOeusyO+XH2SK5ZcT+UCrElPP+WVIInh2TzeI4XW9ENaSLVVHA==} - engines: {node: '>=12.22.0'} - - ipaddr.js@1.9.1: - resolution: {integrity: sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==} - engines: {node: '>= 0.10'} - - is-arguments@1.1.1: - resolution: {integrity: sha512-8Q7EARjzEnKpt/PCD7e1cgUS0a6X8u5tdSiMqXhojOdoV9TsMsiO+9VLC5vAmO8N7/GmXn7yjR8qnA6bVAEzfA==} - engines: {node: '>= 0.4'} - - is-arrayish@0.2.1: - resolution: {integrity: sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==} - - is-arrayish@0.3.2: - resolution: {integrity: sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==} - - is-binary-path@2.1.0: - resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==} - engines: {node: '>=8'} - - is-callable@1.2.7: - resolution: {integrity: sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==} - engines: {node: '>= 0.4'} - - is-extglob@2.1.1: - resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} - engines: {node: '>=0.10.0'} - - is-fullwidth-code-point@3.0.0: - resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==} - engines: {node: '>=8'} - - is-generator-function@1.0.10: - resolution: {integrity: sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==} - engines: {node: '>= 0.4'} - - is-glob@4.0.3: - resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==} - engines: {node: '>=0.10.0'} - - is-network-error@1.1.0: - resolution: {integrity: sha512-tUdRRAnhT+OtCZR/LxZelH/C7QtjtFrTu5tXCA8pl55eTUElUHT+GPYV8MBMBvea/j+NxQqVt3LbWMRir7Gx9g==} - engines: {node: '>=16'} - - is-number@7.0.0: - resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==} - engines: {node: '>=0.12.0'} - - is-plain-obj@2.1.0: - resolution: {integrity: sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==} - engines: {node: '>=8'} - - is-plain-obj@4.1.0: - resolution: {integrity: sha512-+Pgi+vMuUNkJyExiMBt5IlFoMyKnr5zhJ4Uspz58WOhBF5QoIZkFyNHIbBAtHwzVAgk5RtndVNsDRN61/mmDqg==} - engines: {node: '>=12'} - - is-stream@2.0.1: - resolution: {integrity: sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==} - engines: {node: '>=8'} - - is-stream@4.0.1: - resolution: {integrity: sha512-Dnz92NInDqYckGEUJv689RbRiTSEHCQ7wOVeALbkOz999YpqT46yMRIGtSNl2iCL1waAZSx40+h59NV/EwzV/A==} - engines: {node: '>=18'} - - is-typed-array@1.1.13: - resolution: {integrity: sha512-uZ25/bUAlUY5fR4OKT4rZQEBrzQWYV9ZJYGGsUmEJ6thodVJ1HX64ePQ6Z0qPWP+m+Uq6e9UugrE38jeYsDSMw==} - engines: {node: '>= 0.4'} - - is-unicode-supported@0.1.0: - resolution: {integrity: sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==} - engines: {node: '>=10'} - - is-unicode-supported@2.0.0: - resolution: {integrity: sha512-FRdAyx5lusK1iHG0TWpVtk9+1i+GjrzRffhDg4ovQ7mcidMQ6mj+MhKPmvh7Xwyv5gIS06ns49CA7Sqg7lC22Q==} - engines: {node: '>=18'} - - isarray@1.0.0: - resolution: {integrity: sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==} - - isexe@2.0.0: - resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} - - jackspeak@3.4.3: - resolution: {integrity: sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==} - - jmespath@0.16.0: - resolution: {integrity: sha512-9FzQjJ7MATs1tSpnco1K6ayiYE3figslrXA72G2HQ/n76RzvYlofyi5QM+iX4YRs/pu3yzxlVQSST23+dMDknw==} - engines: {node: '>= 0.6.0'} - - joycon@3.1.1: - resolution: {integrity: sha512-34wB/Y7MW7bzjKRjUKTa46I2Z7eV62Rkhva+KkopW7Qvv/OSWBqvkSY7vusOPrNuZcUG3tApvdVgNB8POj3SPw==} - engines: {node: '>=10'} - - js-tokens@4.0.0: - resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} - - js-yaml@4.1.0: - resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==} - hasBin: true - - json-parse-even-better-errors@2.3.1: - resolution: {integrity: sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==} - - json-schema-ref-resolver@1.0.1: - resolution: {integrity: sha512-EJAj1pgHc1hxF6vo2Z3s69fMjO1INq6eGHXZ8Z6wCQeldCuwxGK9Sxf4/cScGn3FZubCVUehfWtcDM/PLteCQw==} - - json-schema-traverse@1.0.0: - resolution: {integrity: sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==} - - json5@2.2.3: - resolution: {integrity: sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==} - engines: {node: '>=6'} - hasBin: true - - just-extend@6.2.0: - resolution: {integrity: sha512-cYofQu2Xpom82S6qD778jBDpwvvy39s1l/hrYij2u9AMdQcGRpaBu6kY4mVhuno5kJVi1DAz4aiphA2WI1/OAw==} - - kuler@2.0.0: - resolution: {integrity: sha512-Xq9nH7KlWZmXAtodXDDRE7vs6DU1gTU8zYDHDiWLSip45Egwq3plLHzPn27NgvzL2r1LMPC1vdqh98sQxtqj4A==} - - light-my-request@5.13.0: - resolution: {integrity: sha512-9IjUN9ZyCS9pTG+KqTDEQo68Sui2lHsYBrfMyVUTTZ3XhH8PMZq7xO94Kr+eP9dhi/kcKsx4N41p2IXEBil1pQ==} - - lilconfig@3.1.2: - resolution: {integrity: sha512-eop+wDAvpItUys0FWkHIKeC9ybYrTGbU41U5K7+bttZZeohvnY7M9dZ5kB21GNWiFT2q1OoPTvncPCgSOVO5ow==} - engines: {node: '>=14'} - - lines-and-columns@1.2.4: - resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} - - load-tsconfig@0.2.5: - resolution: {integrity: sha512-IXO6OCs9yg8tMKzfPZ1YmheJbZCiEsnBdcB03l0OcfK9prKnJb96siuHCr5Fl37/yo9DnKU+TLpxzTUspw9shg==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - locate-path@6.0.0: - resolution: {integrity: sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==} - engines: {node: '>=10'} - - lodash.defaults@4.2.0: - resolution: {integrity: sha512-qjxPLHd3r5DnsdGacqOMU6pb/avJzdh9tFX2ymgoZE27BmjXrNy/y4LoaiTeAb+O3gL8AfpJGtqfX/ae2leYYQ==} - - lodash.get@4.4.2: - resolution: {integrity: sha512-z+Uw/vLuy6gQe8cfaFWD7p0wVv8fJl3mbzXh33RS+0oW2wvUqiRXiQ69gLWSLpgB5/6sU+r6BlQR0MBILadqTQ==} - - lodash.isarguments@3.1.0: - resolution: {integrity: sha512-chi4NHZlZqZD18a0imDHnZPrDeBbTtVN7GXMwuGdRH9qotxAjYs3aVLKc7zNOG9eddR5Ksd8rvFEBc9SsggPpg==} - - lodash.snakecase@4.1.1: - resolution: {integrity: sha512-QZ1d4xoBHYUeuouhEq3lk3Uq7ldgyFXGBhg04+oRLnIz8o9T65Eh+8YdroUwn846zchkA9yDsDl5CVVaV2nqYw==} - - lodash.sortby@4.7.0: - resolution: {integrity: sha512-HDWXG8isMntAyRF5vZ7xKuEvOhT4AhlRt/3czTSjvGUxjYCBVRQY48ViDHyfYz9VIoBkW4TMGQNapx+l3RUwdA==} - - lodash@4.1.0: - resolution: {integrity: sha512-B9sgtKUlz0xe7lkYb80BcOpwwJJw5iOiz4HkBDzF0+i5nJLiwfBnL08m7bBkCOPBfi+0aqvrJDMdZDfAvs8vYg==} - - lodash@4.17.21: - resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==} - - log-symbols@4.1.0: - resolution: {integrity: sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==} - engines: {node: '>=10'} - - logform@2.6.1: - resolution: {integrity: sha512-CdaO738xRapbKIMVn2m4F6KTj4j7ooJ8POVnebSgKo3KBz5axNXRAL7ZdRjIV6NOr2Uf4vjtRkxrFETOioCqSA==} - engines: {node: '>= 12.0.0'} - - loupe@3.1.1: - resolution: {integrity: sha512-edNu/8D5MKVfGVFRhFf8aAxiTM6Wumfz5XsaatSxlD3w4R1d/WEKUTydCdPGbl9K7QG/Ca3GnDV2sIKIpXRQcw==} - - lru-cache@10.4.3: - resolution: {integrity: sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==} - - luxon@3.5.0: - resolution: {integrity: sha512-rh+Zjr6DNfUYR3bPwJEnuwDdqMbxZW7LOQfUN4B54+Cl+0o5zaU9RJ6bcidfDtC1cWCZXQ+nvX8bf6bAji37QQ==} - engines: {node: '>=12'} - - magic-bytes.js@1.10.0: - resolution: {integrity: sha512-/k20Lg2q8LE5xiaaSkMXk4sfvI+9EGEykFS4b0CHHGWqDYU0bGUFSwchNOMA56D7TCs9GwVTkqe9als1/ns8UQ==} - - make-error@1.3.6: - resolution: {integrity: sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==} - - merge-stream@2.0.0: - resolution: {integrity: sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==} - - merge2@1.4.1: - resolution: {integrity: sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==} - engines: {node: '>= 8'} - - micromatch@4.0.8: - resolution: {integrity: sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==} - engines: {node: '>=8.6'} - - mimic-fn@2.1.0: - resolution: {integrity: sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==} - engines: {node: '>=6'} - - minimatch@10.0.1: - resolution: {integrity: sha512-ethXTt3SGGR+95gudmqJ1eNhRO7eGEGIgYA9vnPatK4/etz2MEVDno5GMCibdMTuBMyElzIlgxMna3K94XDIDQ==} - engines: {node: 20 || >=22} - - minimatch@3.1.2: - resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==} - - minimatch@5.1.6: - resolution: {integrity: sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==} - engines: {node: '>=10'} - - minimatch@9.0.5: - resolution: {integrity: sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==} - engines: {node: '>=16 || 14 >=14.17'} - - minimist@1.2.8: - resolution: {integrity: sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==} - - minipass@7.1.2: - resolution: {integrity: sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==} - engines: {node: '>=16 || 14 >=14.17'} - - mocha@10.7.3: - resolution: {integrity: sha512-uQWxAu44wwiACGqjbPYmjo7Lg8sFrS3dQe7PP2FQI+woptP4vZXSMcfMyFL/e1yFEeEpV4RtyTpZROOKmxis+A==} - engines: {node: '>= 14.0.0'} - hasBin: true - - ms@2.1.2: - resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==} - - ms@2.1.3: - resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} - - multiformats@13.2.2: - resolution: {integrity: sha512-RWI+nyf0q64vyOxL8LbKtjJMki0sogRL/8axvklNtiTM0iFCVtHwME9w6+0P1/v4dQvsIg8A45oT3ka1t/M/+A==} - - mz@2.7.0: - resolution: {integrity: sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==} - - nan@2.20.0: - resolution: {integrity: sha512-bk3gXBZDGILuuo/6sKtr0DQmSThYHLtNCdSdXk9YkxD/jK6X2vmCyyXBBxyqZ4XcnzTyYEAThfX3DCEnLf6igw==} - - nanoid@5.0.7: - resolution: {integrity: sha512-oLxFY2gd2IqnjcYyOXD8XGCftpGtZP2AbHbOkthDkvRywH5ayNtPVy9YlOPcHckXzbLTCHpkb7FB+yuxKV13pQ==} - engines: {node: ^18 || >=20} - hasBin: true - - neotraverse@0.6.18: - resolution: {integrity: sha512-Z4SmBUweYa09+o6pG+eASabEpP6QkQ70yHj351pQoEXIs8uHbaU2DWVmzBANKgflPa47A50PtB2+NgRpQvr7vA==} - engines: {node: '>= 10'} - - nise@5.1.9: - resolution: {integrity: sha512-qOnoujW4SV6e40dYxJOb3uvuoPHtmLzIk4TFo+j0jPJoC+5Z9xja5qH5JZobEPsa8+YYphMrOSwnrshEhG2qww==} - - nise@6.0.0: - resolution: {integrity: sha512-K8ePqo9BFvN31HXwEtTNGzgrPpmvgciDsFz8aztFjt4LqKO/JeFD8tBOeuDiCMXrIl/m1YvfH8auSpxfaD09wg==} - - node-abort-controller@3.1.1: - resolution: {integrity: sha512-AGK2yQKIjRuqnc6VkX2Xj5d+QW8xZ87pa1UK6yA6ouUyuxfHuMP6umE5QK7UmTeOAymo+Zx1Fxiuw9rVx8taHQ==} - - node-domexception@1.0.0: - resolution: {integrity: sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==} - engines: {node: '>=10.5.0'} - - node-fetch@3.3.2: - resolution: {integrity: sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - nodemon@3.1.4: - resolution: {integrity: sha512-wjPBbFhtpJwmIeY2yP7QF+UKzPfltVGtfce1g/bB15/8vCGZj8uxD62b/b9M9/WVgme0NZudpownKN+c0plXlQ==} - engines: {node: '>=10'} - hasBin: true - - normalize-path@3.0.0: - resolution: {integrity: sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==} - engines: {node: '>=0.10.0'} - - npm-run-path@4.0.1: - resolution: {integrity: sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==} - engines: {node: '>=8'} - - npm-run-path@5.3.0: - resolution: {integrity: sha512-ppwTtiJZq0O/ai0z7yfudtBpWIoxM8yE6nHi1X47eFR2EWORqfbu6CnPlNsjeN683eT0qG6H/Pyf9fCcvjnnnQ==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - nth-check@2.1.1: - resolution: {integrity: sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==} - - object-assign@4.1.1: - resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} - engines: {node: '>=0.10.0'} - - object-inspect@1.13.2: - resolution: {integrity: sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==} - engines: {node: '>= 0.4'} - - obuf@1.1.2: - resolution: {integrity: sha512-PX1wu0AmAdPqOL1mWhqmlOd8kOIZQwGZw6rh7uby9fTc5lhaOWFLX3I6R1hrF9k3zUY40e6igsLGkDXK92LJNg==} - - on-exit-leak-free@2.1.2: - resolution: {integrity: sha512-0eJJY6hXLGf1udHwfNftBqH+g73EU4B504nZeKpz1sYRKafAghwxEJunB2O7rDZkL4PGfsMVnTXZ2EjibbqcsA==} - engines: {node: '>=14.0.0'} - - once@1.4.0: - resolution: {integrity: sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==} - - one-time@1.0.0: - resolution: {integrity: sha512-5DXOiRKwuSEcQ/l0kGCF6Q3jcADFv5tSmRaJck/OqkVFcOzutB134KRSfF0xDrL39MNnqxbHBbUUcjZIhTgb2g==} - - onetime@5.1.2: - resolution: {integrity: sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==} - engines: {node: '>=6'} - - p-limit@3.1.0: - resolution: {integrity: sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==} - engines: {node: '>=10'} - - p-locate@5.0.0: - resolution: {integrity: sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==} - engines: {node: '>=10'} - - p-retry@6.2.0: - resolution: {integrity: sha512-JA6nkq6hKyWLLasXQXUrO4z8BUZGUt/LjlJxx8Gb2+2ntodU/SS63YZ8b0LUTbQ8ZB9iwOfhEPhg4ykKnn2KsA==} - engines: {node: '>=16.17'} - - package-json-from-dist@1.0.0: - resolution: {integrity: sha512-dATvCeZN/8wQsGywez1mzHtTlP22H8OEfPrVMLNr4/eGa+ijtLn/6M5f0dY8UKNrC2O9UCU6SSoG3qRKnt7STw==} - - parent-module@1.0.1: - resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} - engines: {node: '>=6'} - - parse-json@5.2.0: - resolution: {integrity: sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==} - engines: {node: '>=8'} - - parse-ms@4.0.0: - resolution: {integrity: sha512-TXfryirbmq34y8QBwgqCVLi+8oA3oWx2eAnSn62ITyEhEYaWRlVZ2DvMM9eZbMs/RfxPu/PK/aBLyGj4IrqMHw==} - engines: {node: '>=18'} - - parse5-htmlparser2-tree-adapter@7.0.0: - resolution: {integrity: sha512-B77tOZrqqfUfnVcOrUvfdLbz4pu4RopLD/4vmu3HUPswwTA8OH0EMW9BlWR2B0RCoiZRAHEUu7IxeP1Pd1UU+g==} - - parse5-parser-stream@7.1.2: - resolution: {integrity: sha512-JyeQc9iwFLn5TbvvqACIF/VXG6abODeB3Fwmv/TGdLk2LfbWkaySGY72at4+Ty7EkPZj854u4CrICqNk2qIbow==} - - parse5@7.1.2: - resolution: {integrity: sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==} - - path-exists@4.0.0: - resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==} - engines: {node: '>=8'} - - path-key@3.1.1: - resolution: {integrity: sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==} - engines: {node: '>=8'} - - path-key@4.0.0: - resolution: {integrity: sha512-haREypq7xkM7ErfgIyA0z+Bj4AGKlMSdlQE2jvJo6huWD1EdkKYV+G/T4nq0YEF2vgTT8kqMFKo1uHn950r4SQ==} - engines: {node: '>=12'} - - path-scurry@1.11.1: - resolution: {integrity: sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==} - engines: {node: '>=16 || 14 >=14.18'} - - path-to-regexp@6.2.2: - resolution: {integrity: sha512-GQX3SSMokngb36+whdpRXE+3f9V8UzyAorlYvOGx87ufGHehNTn5lCxrKtLyZ4Yl/wEKnNnr98ZzOwwDZV5ogw==} - - path-type@4.0.0: - resolution: {integrity: sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==} - engines: {node: '>=8'} - - pathval@2.0.0: - resolution: {integrity: sha512-vE7JKRyES09KiunauX7nd2Q9/L7lhok4smP9RZTDeD4MVs72Dp2qNFVz39Nz5a0FVEW0BJR6C0DYrq6unoziZA==} - engines: {node: '>= 14.16'} - - pg-boss@10.1.1: - resolution: {integrity: sha512-2t7gz5nEUYFabj8czWWFRUSyPDQ5t+K/EF5l9Q5lHn2iwyPPKgIfwK+8LKgRfyHRUePTDQhogsGcwOlNczfZ5Q==} - engines: {node: '>=20'} - - pg-cloudflare@1.1.1: - resolution: {integrity: sha512-xWPagP/4B6BgFO+EKz3JONXv3YDgvkbVrGw2mTo3D6tVDQRh1e7cqVGvyR3BE+eQgAvx1XhW/iEASj4/jCWl3Q==} - - pg-connection-string@2.6.4: - resolution: {integrity: sha512-v+Z7W/0EO707aNMaAEfiGnGL9sxxumwLl2fJvCQtMn9Fxsg+lPpPkdcyBSv/KFgpGdYkMfn+EI1Or2EHjpgLCA==} - - pg-int8@1.0.1: - resolution: {integrity: sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw==} - engines: {node: '>=4.0.0'} - - pg-numeric@1.0.2: - resolution: {integrity: sha512-BM/Thnrw5jm2kKLE5uJkXqqExRUY/toLHda65XgFTBTFYZyopbKjBe29Ii3RbkvlsMoFwD+tHeGaCjjv0gHlyw==} - engines: {node: '>=4'} - - pg-pool@3.6.2: - resolution: {integrity: sha512-Htjbg8BlwXqSBQ9V8Vjtc+vzf/6fVUuak/3/XXKA9oxZprwW3IMDQTGHP+KDmVL7rtd+R1QjbnCFPuTHm3G4hg==} - peerDependencies: - pg: '>=8.0' - - pg-protocol@1.6.1: - resolution: {integrity: sha512-jPIlvgoD63hrEuihvIg+tJhoGjUsLPn6poJY9N5CnlPd91c2T18T/9zBtLxZSb1EhYxBRoZJtzScCaWlYLtktg==} - - pg-types@2.2.0: - resolution: {integrity: sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==} - engines: {node: '>=4'} - - pg-types@4.0.2: - resolution: {integrity: sha512-cRL3JpS3lKMGsKaWndugWQoLOCoP+Cic8oseVcbr0qhPzYD5DWXK+RZ9LY9wxRf7RQia4SCwQlXk0q6FCPrVng==} - engines: {node: '>=10'} - - pg@8.12.0: - resolution: {integrity: sha512-A+LHUSnwnxrnL/tZ+OLfqR1SxLN3c/pgDztZ47Rpbsd4jUytsTtwQo/TLPRzPJMp/1pbhYVhH9cuSZLAajNfjQ==} - engines: {node: '>= 8.0.0'} - peerDependencies: - pg-native: '>=3.0.1' - peerDependenciesMeta: - pg-native: - optional: true - - pgpass@1.0.5: - resolution: {integrity: sha512-FdW9r/jQZhSeohs1Z3sI1yxFQNFvMcnmfuj4WBMUTxOrAyLMaTcE1aAMBiTlbMNaXvBCQuVi0R7hd8udDSP7ug==} - - picocolors@1.1.0: - resolution: {integrity: sha512-TQ92mBOW0l3LeMeyLV6mzy/kWr8lkd/hp3mTg7wYK7zJhuBStmGMBG0BdeDZS/dZx1IukaX6Bk11zcln25o1Aw==} - - picomatch@2.3.1: - resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==} - engines: {node: '>=8.6'} - - pino-abstract-transport@1.2.0: - resolution: {integrity: sha512-Guhh8EZfPCfH+PMXAb6rKOjGQEoy0xlAIn+irODG5kgfYV+BQ0rGYYWTIel3P5mmyXqkYkPmdIkywsn6QKUR1Q==} - - pino-pretty@11.2.2: - resolution: {integrity: sha512-2FnyGir8nAJAqD3srROdrF1J5BIcMT4nwj7hHSc60El6Uxlym00UbCCd8pYIterstVBFlMyF1yFV8XdGIPbj4A==} - hasBin: true - - pino-std-serializers@7.0.0: - resolution: {integrity: sha512-e906FRY0+tV27iq4juKzSYPbUj2do2X2JX4EzSca1631EB2QJQUqGbDuERal7LCtOpxl6x3+nvo9NPZcmjkiFA==} - - pino@9.4.0: - resolution: {integrity: sha512-nbkQb5+9YPhQRz/BeQmrWpEknAaqjpAqRK8NwJpmrX/JHu7JuZC5G1CeAwJDJfGes4h+YihC6in3Q2nGb+Y09w==} - hasBin: true - - pirates@4.0.6: - resolution: {integrity: sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==} - engines: {node: '>= 6'} - - possible-typed-array-names@1.0.0: - resolution: {integrity: sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==} - engines: {node: '>= 0.4'} - - postcss-load-config@6.0.1: - resolution: {integrity: sha512-oPtTM4oerL+UXmx+93ytZVN82RrlY/wPUV8IeDxFrzIjXOLF1pN+EmKPLbubvKHT2HC20xXsCAH2Z+CKV6Oz/g==} - engines: {node: '>= 18'} - peerDependencies: - jiti: '>=1.21.0' - postcss: '>=8.0.9' - tsx: ^4.8.1 - yaml: ^2.4.2 - peerDependenciesMeta: - jiti: - optional: true - postcss: - optional: true - tsx: - optional: true - yaml: - optional: true - - postgres-array@2.0.0: - resolution: {integrity: sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA==} - engines: {node: '>=4'} - - postgres-array@3.0.2: - resolution: {integrity: sha512-6faShkdFugNQCLwucjPcY5ARoW1SlbnrZjmGl0IrrqewpvxvhSLHimCVzqeuULCbG0fQv7Dtk1yDbG3xv7Veog==} - engines: {node: '>=12'} - - postgres-bytea@1.0.0: - resolution: {integrity: sha512-xy3pmLuQqRBZBXDULy7KbaitYqLcmxigw14Q5sj8QBVLqEwXfeybIKVWiqAXTlcvdvb0+xkOtDbfQMOf4lST1w==} - engines: {node: '>=0.10.0'} - - postgres-bytea@3.0.0: - resolution: {integrity: sha512-CNd4jim9RFPkObHSjVHlVrxoVQXz7quwNFpz7RY1okNNme49+sVyiTvTRobiLV548Hx/hb1BG+iE7h9493WzFw==} - engines: {node: '>= 6'} - - postgres-date@1.0.7: - resolution: {integrity: sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q==} - engines: {node: '>=0.10.0'} - - postgres-date@2.1.0: - resolution: {integrity: sha512-K7Juri8gtgXVcDfZttFKVmhglp7epKb1K4pgrkLxehjqkrgPhfG6OO8LHLkfaqkbpjNRnra018XwAr1yQFWGcA==} - engines: {node: '>=12'} - - postgres-interval@1.2.0: - resolution: {integrity: sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==} - engines: {node: '>=0.10.0'} - - postgres-interval@3.0.0: - resolution: {integrity: sha512-BSNDnbyZCXSxgA+1f5UU2GmwhoI0aU5yMxRGO8CdFEcY2BQF9xm/7MqKnYoM1nJDk8nONNWDk9WeSmePFhQdlw==} - engines: {node: '>=12'} - - postgres-range@1.1.4: - resolution: {integrity: sha512-i/hbxIE9803Alj/6ytL7UHQxRvZkI9O4Sy+J3HGc4F4oo/2eQAjTSNJ0bfxyse3bH0nuVesCk+3IRLaMtG3H6w==} - - postgres@3.4.4: - resolution: {integrity: sha512-IbyN+9KslkqcXa8AO9fxpk97PA4pzewvpi2B3Dwy9u4zpV32QicaEdgmF3eSQUzdRk7ttDHQejNgAEr4XoeH4A==} - engines: {node: '>=12'} - - pretty-bytes@6.1.1: - resolution: {integrity: sha512-mQUvGU6aUFQ+rNvTIAcZuWGRT9a6f6Yrg9bHs4ImKF+HZCEK+plBvnAZYSIQztknZF2qnzNtr6F8s0+IuptdlQ==} - engines: {node: ^14.13.1 || >=16.0.0} - - pretty-ms@9.1.0: - resolution: {integrity: sha512-o1piW0n3tgKIKCwk2vpM/vOV13zjJzvP37Ioze54YlTHE06m4tjEbzg9WsKkvTuyYln2DHjo5pY4qrZGI0otpw==} - engines: {node: '>=18'} - - process-warning@3.0.0: - resolution: {integrity: sha512-mqn0kFRl0EoqhnL0GQ0veqFHyIN1yig9RHh/InzORTUiZHFRAur+aMtRkELNwGs9aNwKS6tg/An4NYBPGwvtzQ==} - - process-warning@4.0.0: - resolution: {integrity: sha512-/MyYDxttz7DfGMMHiysAsFE4qF+pQYAA8ziO/3NcRVrQ5fSk+Mns4QZA/oRPFzvcqNoVJXQNWNAsdwBXLUkQKw==} - - process@0.11.10: - resolution: {integrity: sha512-cdGef/drWFoydD1JsMzuFf8100nZl+GT+yacc2bEced5f9Rjk4z+WtFUTBu9PhOi9j/jfmBPu0mMEY4wIdAF8A==} - engines: {node: '>= 0.6.0'} - - proxy-addr@2.0.7: - resolution: {integrity: sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==} - engines: {node: '>= 0.10'} - - psl@1.9.0: - resolution: {integrity: sha512-E/ZsdU4HLs/68gYzgGTkMicWTLPdAftJLfJFlLUAAKZGkStNU72sZjT66SnMDVOfOWY/YAoiD7Jxa9iHvngcag==} - - pstree.remy@1.1.8: - resolution: {integrity: sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==} - - pump@3.0.0: - resolution: {integrity: sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==} - - punycode@1.3.2: - resolution: {integrity: sha512-RofWgt/7fL5wP1Y7fxE7/EmTLzQVnB0ycyibJ0OOHIlJqTNzglYFxVwETOcIoJqJmpDXJ9xImDv+Fq34F/d4Dw==} - - punycode@2.3.1: - resolution: {integrity: sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==} - engines: {node: '>=6'} - - qs@6.13.0: - resolution: {integrity: sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==} - engines: {node: '>=0.6'} - - querystring@0.2.0: - resolution: {integrity: sha512-X/xY82scca2tau62i9mDyU9K+I+djTMUsvwf7xnUX5GLvVzgJybOJf4Y6o9Zx3oJK/LSXg5tTZBjwzqVPaPO2g==} - engines: {node: '>=0.4.x'} - deprecated: The querystring API is considered Legacy. new code should use the URLSearchParams API instead. - - querystringify@2.2.0: - resolution: {integrity: sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==} - - queue-microtask@1.2.3: - resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==} - - quick-format-unescaped@4.0.4: - resolution: {integrity: sha512-tYC1Q1hgyRuHgloV/YXs2w15unPVh8qfu/qCTfhTYamaw7fyhumKa2yGpdSo87vY32rIclj+4fWYQXUMs9EHvg==} - - randombytes@2.1.0: - resolution: {integrity: sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==} - - readable-stream@3.6.2: - resolution: {integrity: sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==} - engines: {node: '>= 6'} - - readable-stream@4.5.2: - resolution: {integrity: sha512-yjavECdqeZ3GLXNgRXgeQEdz9fvDDkNKyHnbHRFtOr7/LcfgBcmct7t/ET+HaCTqfh06OzoAxrkN/IfjJBVe+g==} - engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} - - readdirp@3.6.0: - resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==} - engines: {node: '>=8.10.0'} - - real-require@0.2.0: - resolution: {integrity: sha512-57frrGM/OCTLqLOAh0mhVA9VBMHd+9U7Zb2THMGdBUoZVOtGbJzjxsYGDJ3A9AYYCP4hn6y1TVbaOfzWtm5GFg==} - engines: {node: '>= 12.13.0'} - - redis-errors@1.2.0: - resolution: {integrity: sha512-1qny3OExCf0UvUV/5wpYKf2YwPcOqXzkwKKSmKHiE6ZMQs5heeE/c8eXK+PNllPvmjgAbfnsbpkGZWy8cBpn9w==} - engines: {node: '>=4'} - - redis-parser@3.0.0: - resolution: {integrity: sha512-DJnGAeenTdpMEH6uAJRK/uiyEIH9WVsUmoLwzudwGJUwZPp80PDBWPHXSAGNPwNvIXAbe7MSUB1zQFugFml66A==} - engines: {node: '>=4'} - - require-directory@2.1.1: - resolution: {integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==} - engines: {node: '>=0.10.0'} - - require-from-string@2.0.2: - resolution: {integrity: sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==} - engines: {node: '>=0.10.0'} - - requires-port@1.0.0: - resolution: {integrity: sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==} - - resolve-from@4.0.0: - resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==} - engines: {node: '>=4'} - - resolve-from@5.0.0: - resolution: {integrity: sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==} - engines: {node: '>=8'} - - resolve-pkg-maps@1.0.0: - resolution: {integrity: sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==} - - ret@0.4.3: - resolution: {integrity: sha512-0f4Memo5QP7WQyUEAYUO3esD/XjOc3Zjjg5CPsAq1p8sIu0XPeMbHJemKA0BO7tV0X7+A0FoEpbmHXWxPyD3wQ==} - engines: {node: '>=10'} - - retry@0.13.1: - resolution: {integrity: sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg==} - engines: {node: '>= 4'} - - reusify@1.0.4: - resolution: {integrity: sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==} - engines: {iojs: '>=1.0.0', node: '>=0.10.0'} - - rfdc@1.4.1: - resolution: {integrity: sha512-q1b3N5QkRUWUl7iyylaaj3kOpIT0N2i9MqIEQXP73GVsN9cw3fdx8X63cEmWhJGi2PPCF23Ijp7ktmd39rawIA==} - - rollup@4.21.2: - resolution: {integrity: sha512-e3TapAgYf9xjdLvKQCkQTnbTKd4a6jwlpQSJJFokHGaX2IVjoEqkIIhiQfqsi0cdwlOD+tQGuOd5AJkc5RngBw==} - engines: {node: '>=18.0.0', npm: '>=8.0.0'} - hasBin: true - - run-parallel@1.2.0: - resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} - - rxjs@7.8.1: - resolution: {integrity: sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg==} - - safe-buffer@5.2.1: - resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} - - safe-regex2@3.1.0: - resolution: {integrity: sha512-RAAZAGbap2kBfbVhvmnTFv73NWLMvDGOITFYTZBAaY8eR+Ir4ef7Up/e7amo+y1+AH+3PtLkrt9mvcTsG9LXug==} - - safe-stable-stringify@2.5.0: - resolution: {integrity: sha512-b3rppTKm9T+PsVCBEOUR46GWI7fdOs00VKZ1+9c1EWDaDMvjQc6tUwuFyIprgGgTcWoVHSKrU8H31ZHA2e0RHA==} - engines: {node: '>=10'} - - safer-buffer@2.1.2: - resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==} - - sax@1.2.1: - resolution: {integrity: sha512-8I2a3LovHTOpm7NV5yOyO8IHqgVsfK4+UuySrXU8YXkSRX7k6hCV9b3HrkKCr3nMpgj+0bmocaJJWpvp1oc7ZA==} - - secure-json-parse@2.7.0: - resolution: {integrity: sha512-6aU+Rwsezw7VR8/nyvKTx8QpWH9FrcYiXXlqC4z5d5XQBDRqtbfsRjnwGyqbi3gddNtWHuEk9OANUotL26qKUw==} - - semver@7.6.3: - resolution: {integrity: sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==} - engines: {node: '>=10'} - hasBin: true - - sequin@0.1.1: - resolution: {integrity: sha512-hJWMZRwP75ocoBM+1/YaCsvS0j5MTPeBHJkS2/wruehl9xwtX30HlDF1Gt6UZ8HHHY8SJa2/IL+jo+JJCd59rA==} - engines: {node: '>=0.4.0'} - - serialize-error@8.1.0: - resolution: {integrity: sha512-3NnuWfM6vBYoy5gZFvHiYsVbafvI9vZv/+jlIigFn4oP4zjNPK3LhcY0xSCgeb1a5L8jO71Mit9LlNoi2UfDDQ==} - engines: {node: '>=10'} - - serialize-javascript@6.0.2: - resolution: {integrity: sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g==} - - set-cookie-parser@2.7.0: - resolution: {integrity: sha512-lXLOiqpkUumhRdFF3k1osNXCy9akgx/dyPZ5p8qAg9seJzXr5ZrlqZuWIMuY6ejOsVLE6flJ5/h3lsn57fQ/PQ==} - - set-function-length@1.2.2: - resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==} - engines: {node: '>= 0.4'} - - shebang-command@2.0.0: - resolution: {integrity: sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==} - engines: {node: '>=8'} - - shebang-regex@3.0.0: - resolution: {integrity: sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==} - engines: {node: '>=8'} - - side-channel@1.0.6: - resolution: {integrity: sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==} - engines: {node: '>= 0.4'} - - signal-exit@3.0.7: - resolution: {integrity: sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==} - - signal-exit@4.1.0: - resolution: {integrity: sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==} - engines: {node: '>=14'} - - simple-swizzle@0.2.2: - resolution: {integrity: sha512-JA//kQgZtbuY83m+xT+tXJkmJncGMTFT+C+g2h2R9uxkYIrE2yy9sgmcLhCnw57/WSD+Eh3J97FPEDFnbXnDUg==} - - simple-update-notifier@2.0.0: - resolution: {integrity: sha512-a2B9Y0KlNXl9u/vsW6sTIu9vGEpfKu2wRV6l1H3XEas/0gUIzGzBoP/IouTcUQbm9JWZLH3COxyn03TYlFax6w==} - engines: {node: '>=10'} - - sinon-chai@4.0.0: - resolution: {integrity: sha512-cWqO7O2I4XfJDWyWElAQ9D/dtdh5Mo0RHndsfiiYyjWnlPzBJdIvjCVURO4EjyYaC3BjV+ISNXCfTXPXTEIEWA==} - peerDependencies: - chai: ^5.0.0 - sinon: '>=4.0.0' - - sinon-test@3.1.6: - resolution: {integrity: sha512-3jBJGf61sS2EN3M+YuIiIbeutKrubP6SFolceTcJrubG+4s+zq3rey/y0huSEwU2ECKOcyCs7EkzANnwqHWPjA==} - peerDependencies: - sinon: '>= 2.x' - - sinon@16.1.3: - resolution: {integrity: sha512-mjnWWeyxcAf9nC0bXcPmiDut+oE8HYridTNzBbF98AYVLmWwGRp2ISEpyhYflG1ifILT+eNn3BmKUJPxjXUPlA==} - - sinon@18.0.0: - resolution: {integrity: sha512-+dXDXzD1sBO6HlmZDd7mXZCR/y5ECiEiGCBSGuFD/kZ0bDTofPYc6JaeGmPSF+1j1MejGUWkORbYOLDyvqCWpA==} - - slash@3.0.0: - resolution: {integrity: sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==} - engines: {node: '>=8'} - - sliced@0.0.5: - resolution: {integrity: sha512-9bYT917D6H3+q8GlQBJmLVz3bc4OeVGfZ2BB12wvLnluTGfG6/8UdOUbKJDW1EEx9SZMDbjnatkau5/XcUeyOw==} - - sonic-boom@4.1.0: - resolution: {integrity: sha512-NGipjjRicyJJ03rPiZCJYjwlsuP2d1/5QUviozRXC7S3WdVWNK5e3Ojieb9CCyfhq2UC+3+SRd9nG3I2lPRvUw==} - - source-map@0.8.0-beta.0: - resolution: {integrity: sha512-2ymg6oRBpebeZi9UUNsgQ89bhx01TcTkmNTGnNO88imTmbSgy4nfujrgVEFKWpMTEGA11EDkTt7mqObTPdigIA==} - engines: {node: '>= 8'} - - split2@4.2.0: - resolution: {integrity: sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==} - engines: {node: '>= 10.x'} - - sql@0.78.0: - resolution: {integrity: sha512-+If27aF3UQxmnuF5JgTJKd/X6TiELa1f58Zo5TX9ULbPnqs1fBDODwwbtU0WwM5H3lSiwx2En4m2lB1rCoIhvQ==} - - stack-trace@0.0.10: - resolution: {integrity: sha512-KGzahc7puUKkzyMt+IqAep+TVNbKP+k2Lmwhub39m1AsTSkaDutx56aDCo+HLDzf/D26BIHTJWNiTG1KAJiQCg==} - - standard-as-callback@2.1.0: - resolution: {integrity: sha512-qoRRSyROncaz1z0mvYqIE4lCd9p2R90i6GxW3uZv5ucSu8tU7B5HXUP1gG8pVZsYNVaXjk8ClXHPttLyxAL48A==} - - stream-browserify@3.0.0: - resolution: {integrity: sha512-H73RAHsVBapbim0tU2JwwOiXUj+fikfiaoYAKHF3VJfA0pe2BCzkhAHBlLG6REzE+2WNZcxOXjK7lkso+9euLA==} - - string-width@4.2.3: - resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} - engines: {node: '>=8'} - - string-width@5.1.2: - resolution: {integrity: sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==} - engines: {node: '>=12'} - - string_decoder@1.3.0: - resolution: {integrity: sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==} - - strip-ansi@6.0.1: - resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} - engines: {node: '>=8'} - - strip-ansi@7.1.0: - resolution: {integrity: sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==} - engines: {node: '>=12'} - - strip-final-newline@2.0.0: - resolution: {integrity: sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==} - engines: {node: '>=6'} - - strip-final-newline@4.0.0: - resolution: {integrity: sha512-aulFJcD6YK8V1G7iRB5tigAP4TsHBZZrOV8pjV++zdUwmeV8uzbY7yn6h9MswN62adStNZFuCIx4haBnRuMDaw==} - engines: {node: '>=18'} - - strip-json-comments@3.1.1: - resolution: {integrity: sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==} - engines: {node: '>=8'} - - strnum@1.0.5: - resolution: {integrity: sha512-J8bbNyKKXl5qYcR36TIO8W3mVGVHrmmxsd5PAItGkmyzwJvybiw2IVq5nqd0i4LSNSkB/sx9VHllbfFdr9k1JA==} - - sucrase@3.35.0: - resolution: {integrity: sha512-8EbVDiu9iN/nESwxeSxDKe0dunta1GOlHufmSSXxMD2z2/tMZpDMpvXQGsc+ajGo8y2uYUmixaSRUc/QPoQ0GA==} - engines: {node: '>=16 || 14 >=14.17'} - hasBin: true - - supports-color@5.5.0: - resolution: {integrity: sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==} - engines: {node: '>=4'} - - supports-color@7.2.0: - resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==} - engines: {node: '>=8'} - - supports-color@8.1.1: - resolution: {integrity: sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==} - engines: {node: '>=10'} - - text-hex@1.0.0: - resolution: {integrity: sha512-uuVGNWzgJ4yhRaNSiubPY7OjISw4sw4E5Uv0wbjp+OzcbmVU/rsT8ujgcXJhn9ypzsgr5vlzpPqP+MBBKcGvbg==} - - thenify-all@1.6.0: - resolution: {integrity: sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==} - engines: {node: '>=0.8'} - - thenify@3.3.1: - resolution: {integrity: sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==} - - thread-stream@3.1.0: - resolution: {integrity: sha512-OqyPZ9u96VohAyMfJykzmivOrY2wfMSf3C5TtFJVgN+Hm6aj+voFhlK+kZEIv2FBh1X6Xp3DlnCOfEQ3B2J86A==} - - to-regex-range@5.0.1: - resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} - engines: {node: '>=8.0'} - - toad-cache@3.7.0: - resolution: {integrity: sha512-/m8M+2BJUpoJdgAHoG+baCwBT+tf2VraSfkBgl0Y00qIWt41DJ8R5B8nsEw0I58YwF5IZH6z24/2TobDKnqSWw==} - engines: {node: '>=12'} - - touch@3.1.1: - resolution: {integrity: sha512-r0eojU4bI8MnHr8c5bNo7lJDdI2qXlWWJk6a9EAFG7vbhTjElYhBVS3/miuE0uOuoLdb8Mc/rVfsmm6eo5o9GA==} - hasBin: true - - tough-cookie@4.1.4: - resolution: {integrity: sha512-Loo5UUvLD9ScZ6jh8beX1T6sO1w2/MpCRpEP7V280GKMVUQ0Jzar2U3UJPsrdbziLEMMhu3Ujnq//rhiFuIeag==} - engines: {node: '>=6'} - - tr46@1.0.1: - resolution: {integrity: sha512-dTpowEjclQ7Kgx5SdBkqRzVhERQXov8/l9Ft9dVM9fmg0W0KQSVaXX9T4i6twCPNtYiZM53lpSSUAwJbFPOHxA==} - - tree-kill@1.2.2: - resolution: {integrity: sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==} - hasBin: true - - triple-beam@1.4.1: - resolution: {integrity: sha512-aZbgViZrg1QNcG+LULa7nhZpJTZSLm/mXnHXnbAbjmN5aSa0y7V+wvv6+4WaBtpISJzThKy+PIPxc1Nq1EJ9mg==} - engines: {node: '>= 14.0.0'} - - ts-interface-checker@0.1.13: - resolution: {integrity: sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==} - - ts-mixer@6.0.4: - resolution: {integrity: sha512-ufKpbmrugz5Aou4wcr5Wc1UUFWOLhq+Fm6qa6P0w0K5Qw2yhaUoiWszhCVuNQyNwrlGiscHOmqYoAox1PtvgjA==} - - ts-node@10.9.2: - resolution: {integrity: sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==} - hasBin: true - peerDependencies: - '@swc/core': '>=1.2.50' - '@swc/wasm': '>=1.2.50' - '@types/node': '*' - typescript: '>=2.7' - peerDependenciesMeta: - '@swc/core': - optional: true - '@swc/wasm': - optional: true - - tslib@2.7.0: - resolution: {integrity: sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA==} - - tsup@8.2.4: - resolution: {integrity: sha512-akpCPePnBnC/CXgRrcy72ZSntgIEUa1jN0oJbbvpALWKNOz1B7aM+UVDWGRGIO/T/PZugAESWDJUAb5FD48o8Q==} - engines: {node: '>=18'} - hasBin: true - peerDependencies: - '@microsoft/api-extractor': ^7.36.0 - '@swc/core': ^1 - postcss: ^8.4.12 - typescript: '>=4.5.0' - peerDependenciesMeta: - '@microsoft/api-extractor': - optional: true - '@swc/core': - optional: true - postcss: - optional: true - typescript: - optional: true - - tsx@4.19.0: - resolution: {integrity: sha512-bV30kM7bsLZKZIOCHeMNVMJ32/LuJzLVajkQI/qf92J2Qr08ueLQvW00PUZGiuLPP760UINwupgUj8qrSCPUKg==} - engines: {node: '>=18.0.0'} - hasBin: true - - tunnel-agent@0.6.0: - resolution: {integrity: sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==} - - type-detect@4.0.8: - resolution: {integrity: sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==} - engines: {node: '>=4'} - - type-detect@4.1.0: - resolution: {integrity: sha512-Acylog8/luQ8L7il+geoSxhEkazvkslg7PSNKOX59mbB9cOveP5aq9h74Y7YU8yDpJwetzQQrfIwtf4Wp4LKcw==} - engines: {node: '>=4'} - - type-fest@0.20.2: - resolution: {integrity: sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==} - engines: {node: '>=10'} - - typescript@5.5.4: - resolution: {integrity: sha512-Mtq29sKDAEYP7aljRgtPOpTvOfbwRWlS6dPRzwjdE+C0R4brX/GUyhHSecbHMFLNBLcJIPt9nl9yG5TZ1weH+Q==} - engines: {node: '>=14.17'} - hasBin: true - - undefsafe@2.0.5: - resolution: {integrity: sha512-WxONCrssBM8TSPRqN5EmsjVrsv4A8X12J4ArBiiayv3DyyG3ZlIg6yysuuSYdZsVz3TKcTg2fd//Ujd4CHV1iA==} - - undici-types@6.19.8: - resolution: {integrity: sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==} - - undici@6.19.8: - resolution: {integrity: sha512-U8uCCl2x9TK3WANvmBavymRzxbfFYG+tAu+fgx3zxQy3qdagQqBLwJVrdyO1TBfUXvfKveMKJZhpvUYoOjM+4g==} - engines: {node: '>=18.17'} - - universalify@0.2.0: - resolution: {integrity: sha512-CJ1QgKmNg3CwvAv/kOFmtnEN05f0D/cn9QntgNOQlQF9dgvVTHj3t+8JPdjqawCHk7V/KA+fbUqzZ9XWhcqPUg==} - engines: {node: '>= 4.0.0'} - - url-parse@1.5.10: - resolution: {integrity: sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ==} - - url@0.10.3: - resolution: {integrity: sha512-hzSUW2q06EqL1gKM/a+obYHLIO6ct2hwPuviqTTOcfFVc61UbfJ2Q32+uGL/HCPxKqrdGB5QUwIe7UqlDgwsOQ==} - - util-deprecate@1.0.2: - resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==} - - util@0.12.5: - resolution: {integrity: sha512-kZf/K6hEIrWHI6XqOFUiiMa+79wE/D8Q+NCNAWclkyg3b4d2k7s0QGepNjiABc+aR3N1PAyHL7p6UcLY6LmrnA==} - - uuid@8.0.0: - resolution: {integrity: sha512-jOXGuXZAWdsTH7eZLtyXMqUb9EcWMGZNbL9YcGBJl4MH4nrxHmZJhEHvyLFrkxo+28uLb/NYRcStH48fnD0Vzw==} - hasBin: true - - uuid@9.0.1: - resolution: {integrity: sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==} - hasBin: true - - v8-compile-cache-lib@3.0.1: - resolution: {integrity: sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==} - - web-streams-polyfill@3.3.3: - resolution: {integrity: sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw==} - engines: {node: '>= 8'} - - webidl-conversions@4.0.2: - resolution: {integrity: sha512-YQ+BmxuTgd6UXZW3+ICGfyqRyHXVlD5GtQr5+qjiNW7bF0cqrzX500HVXPBOvgXb5YnzDd+h0zqyv61KUD7+Sg==} - - websocket-driver@0.7.4: - resolution: {integrity: sha512-b17KeDIQVjvb0ssuSDF2cYXSg2iztliJ4B9WdsuB6J952qCPKmnVq4DyW5motImXHDC1cBT/1UezrJVsKw5zjg==} - engines: {node: '>=0.8.0'} - - websocket-extensions@0.1.4: - resolution: {integrity: sha512-OqedPIGOfsDlo31UNwYbCFMSaO9m9G/0faIHj5/dZFDMFqPTcx6UwqyOy3COEaEOg/9VsGIpdqn62W5KhoKSpg==} - engines: {node: '>=0.8.0'} - - whatwg-encoding@3.1.1: - resolution: {integrity: sha512-6qN4hJdMwfYBtE3YBTTHhoeuUrDBPZmbQaxWAqSALV/MeEnR5z1xd8UKud2RAkFoPkmB+hli1TZSnyi84xz1vQ==} - engines: {node: '>=18'} - - whatwg-mimetype@4.0.0: - resolution: {integrity: sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg==} - engines: {node: '>=18'} - - whatwg-url@7.1.0: - resolution: {integrity: sha512-WUu7Rg1DroM7oQvGWfOiAK21n74Gg+T4elXEQYkOhtyLeWiJFoOGLXPKI/9gzIie9CtwVLm8wtw6YJdKyxSjeg==} - - which-typed-array@1.1.15: - resolution: {integrity: sha512-oV0jmFtUky6CXfkqehVvBP/LSWJ2sy4vWMioiENyJLePrBO/yKyV9OyJySfAKosh+RYkIl5zJCNZ8/4JncrpdA==} - engines: {node: '>= 0.4'} - - which@1.3.1: - resolution: {integrity: sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==} - hasBin: true - - which@2.0.2: - resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==} - engines: {node: '>= 8'} - hasBin: true - - winston-transport@4.7.1: - resolution: {integrity: sha512-wQCXXVgfv/wUPOfb2x0ruxzwkcZfxcktz6JIMUaPLmcNhO4bZTwA/WtDWK74xV3F2dKu8YadrFv0qhwYjVEwhA==} - engines: {node: '>= 12.0.0'} - - winston@3.14.2: - resolution: {integrity: sha512-CO8cdpBB2yqzEf8v895L+GNKYJiEq8eKlHU38af3snQBQ+sdAIUepjMSguOIJC7ICbzm0ZI+Af2If4vIJrtmOg==} - engines: {node: '>= 12.0.0'} - - workerpool@6.5.1: - resolution: {integrity: sha512-Fs4dNYcsdpYSAfVxhnl1L5zTksjvOJxtC5hzMNl+1t9B8hTJTdKDyZ5ju7ztgPy+ft9tBFXoOlDNiOT9WUXZlA==} - - wrap-ansi@7.0.0: - resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} - engines: {node: '>=10'} - - wrap-ansi@8.1.0: - resolution: {integrity: sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==} - engines: {node: '>=12'} - - wrappy@1.0.2: - resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==} - - ws@8.18.0: - resolution: {integrity: sha512-8VbfWfHLbbwu3+N6OKsOMpBdT4kXPDDB9cJk2bJ6mh9ucxdlnNvH1e+roYkKmN9Nxw2yjz7VzeO9oOz2zJ04Pw==} - engines: {node: '>=10.0.0'} - peerDependencies: - bufferutil: ^4.0.1 - utf-8-validate: '>=5.0.2' - peerDependenciesMeta: - bufferutil: - optional: true - utf-8-validate: - optional: true - - xml2js@0.6.2: - resolution: {integrity: sha512-T4rieHaC1EXcES0Kxxj4JWgaUQHDk+qwHcYOCFHfiwKz7tOVPLq7Hjq9dM1WCMhylqMEfP7hMcOIChvotiZegA==} - engines: {node: '>=4.0.0'} - - xmlbuilder@11.0.1: - resolution: {integrity: sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==} - engines: {node: '>=4.0'} - - xtend@4.0.2: - resolution: {integrity: sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==} - engines: {node: '>=0.4'} - - y18n@5.0.8: - resolution: {integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==} - engines: {node: '>=10'} - - yargs-parser@20.2.9: - resolution: {integrity: sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==} - engines: {node: '>=10'} - - yargs-parser@21.1.1: - resolution: {integrity: sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==} - engines: {node: '>=12'} - - yargs-unparser@2.0.0: - resolution: {integrity: sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==} - engines: {node: '>=10'} - - yargs@16.2.0: - resolution: {integrity: sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==} - engines: {node: '>=10'} - - yargs@17.7.2: - resolution: {integrity: sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==} - engines: {node: '>=12'} - - yn@3.1.1: - resolution: {integrity: sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==} - engines: {node: '>=6'} - - yocto-queue@0.1.0: - resolution: {integrity: sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==} - engines: {node: '>=10'} - - yoctocolors@2.1.1: - resolution: {integrity: sha512-GQHQqAopRhwU8Kt1DDM8NjibDXHC8eoh1erhGAJPEyveY9qqVeXvVikNKrDz69sHowPMorbPUrH/mx8c50eiBQ==} - engines: {node: '>=18'} - - youtube-dl-wrap@https://codeload.github.com/insanity54/youtube-dl-wrap/tar.gz/b47e9388063b4ef516624112ef59b7d11f7775d9: - resolution: {tarball: https://codeload.github.com/insanity54/youtube-dl-wrap/tar.gz/b47e9388063b4ef516624112ef59b7d11f7775d9} - version: 2.1.1 - -snapshots: - - '@aws-crypto/crc32@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/crc32c@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/sha1-browser@5.2.0': - dependencies: - '@aws-crypto/supports-web-crypto': 5.2.0 - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-locate-window': 3.568.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-crypto/sha256-browser@5.2.0': - dependencies: - '@aws-crypto/sha256-js': 5.2.0 - '@aws-crypto/supports-web-crypto': 5.2.0 - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-locate-window': 3.568.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-crypto/sha256-js@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/supports-web-crypto@5.2.0': - dependencies: - tslib: 2.7.0 - - '@aws-crypto/util@5.2.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-sdk/client-s3@3.637.0': - dependencies: - '@aws-crypto/sha1-browser': 5.2.0 - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-bucket-endpoint': 3.620.0 - '@aws-sdk/middleware-expect-continue': 3.620.0 - '@aws-sdk/middleware-flexible-checksums': 3.620.0 - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-location-constraint': 3.609.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-sdk-s3': 3.635.0 - '@aws-sdk/middleware-ssec': 3.609.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/signature-v4-multi-region': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@aws-sdk/xml-builder': 3.609.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/eventstream-serde-browser': 3.0.6 - '@smithy/eventstream-serde-config-resolver': 3.0.3 - '@smithy/eventstream-serde-node': 3.0.5 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-blob-browser': 3.1.2 - '@smithy/hash-node': 3.0.3 - '@smithy/hash-stream-node': 3.1.2 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/md5-js': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-stream': 3.1.3 - '@smithy/util-utf8': 3.0.0 - '@smithy/util-waiter': 3.1.2 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sso@3.637.0': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sts@3.637.0': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/core@3.635.0': - dependencies: - '@smithy/core': 2.4.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-middleware': 3.0.3 - fast-xml-parser: 4.4.1 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-env@3.620.1': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-http@3.635.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-stream': 3.1.3 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-ini@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/credential-provider-env': 3.620.1 - '@aws-sdk/credential-provider-http': 3.635.0 - '@aws-sdk/credential-provider-process': 3.620.1 - '@aws-sdk/credential-provider-sso': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/credential-provider-web-identity': 3.621.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - aws-crt - - '@aws-sdk/credential-provider-node@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/credential-provider-env': 3.620.1 - '@aws-sdk/credential-provider-http': 3.635.0 - '@aws-sdk/credential-provider-ini': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/credential-provider-process': 3.620.1 - '@aws-sdk/credential-provider-sso': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/credential-provider-web-identity': 3.621.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - '@aws-sdk/client-sts' - - aws-crt - - '@aws-sdk/credential-provider-process@3.620.1': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-sso@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))': - dependencies: - '@aws-sdk/client-sso': 3.637.0 - '@aws-sdk/token-providers': 3.614.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - aws-crt - - '@aws-sdk/credential-provider-web-identity@3.621.0(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/lib-storage@3.637.0(@aws-sdk/client-s3@3.637.0)': - dependencies: - '@aws-sdk/client-s3': 3.637.0 - '@smithy/abort-controller': 3.1.1 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/smithy-client': 3.2.0 - buffer: 5.6.0 - events: 3.3.0 - stream-browserify: 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-bucket-endpoint@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-arn-parser': 3.568.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-expect-continue@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-flexible-checksums@3.620.0': - dependencies: - '@aws-crypto/crc32': 5.2.0 - '@aws-crypto/crc32c': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@smithy/is-array-buffer': 3.0.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-host-header@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-location-constraint@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-logger@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-recursion-detection@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-sdk-s3@3.635.0': - dependencies: - '@aws-sdk/core': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-arn-parser': 3.568.0 - '@smithy/core': 2.4.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-stream': 3.1.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-ssec@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-user-agent@3.637.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/region-config-resolver@3.614.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@aws-sdk/signature-v4-multi-region@3.635.0': - dependencies: - '@aws-sdk/middleware-sdk-s3': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/token-providers@3.614.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))': - dependencies: - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/types@3.609.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/util-arn-parser@3.568.0': - dependencies: - tslib: 2.7.0 - - '@aws-sdk/util-endpoints@3.637.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - '@smithy/util-endpoints': 2.0.5 - tslib: 2.7.0 - - '@aws-sdk/util-locate-window@3.568.0': - dependencies: - tslib: 2.7.0 - - '@aws-sdk/util-user-agent-browser@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - bowser: 2.11.0 - tslib: 2.7.0 - - '@aws-sdk/util-user-agent-node@3.614.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/xml-builder@3.609.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@babel/code-frame@7.24.7': - dependencies: - '@babel/highlight': 7.24.7 - picocolors: 1.1.0 - - '@babel/helper-validator-identifier@7.24.7': {} - - '@babel/highlight@7.24.7': - dependencies: - '@babel/helper-validator-identifier': 7.24.7 - chalk: 2.4.2 - js-tokens: 4.0.0 - picocolors: 1.1.0 - - '@colors/colors@1.6.0': {} - - '@cspotcode/source-map-support@0.8.1': - dependencies: - '@jridgewell/trace-mapping': 0.3.9 - - '@dabh/diagnostics@2.0.3': - dependencies: - colorspace: 1.1.4 - enabled: 2.0.0 - kuler: 2.0.0 - - '@discordjs/builders@1.9.0': - dependencies: - '@discordjs/formatters': 0.5.0 - '@discordjs/util': 1.1.1 - '@sapphire/shapeshift': 4.0.0 - discord-api-types: 0.37.97 - fast-deep-equal: 3.1.3 - ts-mixer: 6.0.4 - tslib: 2.7.0 - - '@discordjs/collection@1.5.3': {} - - '@discordjs/collection@2.1.1': {} - - '@discordjs/formatters@0.5.0': - dependencies: - discord-api-types: 0.37.97 - - '@discordjs/rest@2.4.0': - dependencies: - '@discordjs/collection': 2.1.1 - '@discordjs/util': 1.1.1 - '@sapphire/async-queue': 1.5.3 - '@sapphire/snowflake': 3.5.3 - '@vladfrangu/async_event_emitter': 2.4.6 - discord-api-types: 0.37.97 - magic-bytes.js: 1.10.0 - tslib: 2.7.0 - undici: 6.19.8 - - '@discordjs/util@1.1.1': {} - - '@discordjs/ws@1.1.1': - dependencies: - '@discordjs/collection': 2.1.1 - '@discordjs/rest': 2.4.0 - '@discordjs/util': 1.1.1 - '@sapphire/async-queue': 1.5.3 - '@types/ws': 8.5.12 - '@vladfrangu/async_event_emitter': 2.4.6 - discord-api-types: 0.37.83 - tslib: 2.7.0 - ws: 8.18.0 - transitivePeerDependencies: - - bufferutil - - utf-8-validate - - '@esbuild/aix-ppc64@0.23.1': - optional: true - - '@esbuild/android-arm64@0.23.1': - optional: true - - '@esbuild/android-arm@0.23.1': - optional: true - - '@esbuild/android-x64@0.23.1': - optional: true - - '@esbuild/darwin-arm64@0.23.1': - optional: true - - '@esbuild/darwin-x64@0.23.1': - optional: true - - '@esbuild/freebsd-arm64@0.23.1': - optional: true - - '@esbuild/freebsd-x64@0.23.1': - optional: true - - '@esbuild/linux-arm64@0.23.1': - optional: true - - '@esbuild/linux-arm@0.23.1': - optional: true - - '@esbuild/linux-ia32@0.23.1': - optional: true - - '@esbuild/linux-loong64@0.23.1': - optional: true - - '@esbuild/linux-mips64el@0.23.1': - optional: true - - '@esbuild/linux-ppc64@0.23.1': - optional: true - - '@esbuild/linux-riscv64@0.23.1': - optional: true - - '@esbuild/linux-s390x@0.23.1': - optional: true - - '@esbuild/linux-x64@0.23.1': - optional: true - - '@esbuild/netbsd-x64@0.23.1': - optional: true - - '@esbuild/openbsd-arm64@0.23.1': - optional: true - - '@esbuild/openbsd-x64@0.23.1': - optional: true - - '@esbuild/sunos-x64@0.23.1': - optional: true - - '@esbuild/win32-arm64@0.23.1': - optional: true - - '@esbuild/win32-ia32@0.23.1': - optional: true - - '@esbuild/win32-x64@0.23.1': - optional: true - - '@fastify/ajv-compiler@3.6.0': - dependencies: - ajv: 8.17.1 - ajv-formats: 2.1.1(ajv@8.17.1) - fast-uri: 2.4.0 - - '@fastify/error@3.4.1': {} - - '@fastify/fast-json-stringify-compiler@4.3.0': - dependencies: - fast-json-stringify: 5.16.1 - - '@fastify/merge-json-schemas@0.1.1': - dependencies: - fast-deep-equal: 3.1.3 - - '@graphile/logger@0.2.0': {} - - '@ioredis/commands@1.2.0': {} - - '@isaacs/cliui@8.0.2': - dependencies: - string-width: 5.1.2 - string-width-cjs: string-width@4.2.3 - strip-ansi: 7.1.0 - strip-ansi-cjs: strip-ansi@6.0.1 - wrap-ansi: 8.1.0 - wrap-ansi-cjs: wrap-ansi@7.0.0 - - '@jridgewell/gen-mapping@0.3.5': - dependencies: - '@jridgewell/set-array': 1.2.1 - '@jridgewell/sourcemap-codec': 1.5.0 - '@jridgewell/trace-mapping': 0.3.25 - - '@jridgewell/resolve-uri@3.1.2': {} - - '@jridgewell/set-array@1.2.1': {} - - '@jridgewell/sourcemap-codec@1.5.0': {} - - '@jridgewell/trace-mapping@0.3.25': - dependencies: - '@jridgewell/resolve-uri': 3.1.2 - '@jridgewell/sourcemap-codec': 1.5.0 - - '@jridgewell/trace-mapping@0.3.9': - dependencies: - '@jridgewell/resolve-uri': 3.1.2 - '@jridgewell/sourcemap-codec': 1.5.0 - - '@noble/hashes@1.5.0': {} - - '@nodelib/fs.scandir@2.1.5': - dependencies: - '@nodelib/fs.stat': 2.0.5 - run-parallel: 1.2.0 - - '@nodelib/fs.stat@2.0.5': {} - - '@nodelib/fs.walk@1.2.8': - dependencies: - '@nodelib/fs.scandir': 2.1.5 - fastq: 1.17.1 - - '@paralleldrive/cuid2@2.2.2': - dependencies: - '@noble/hashes': 1.5.0 - - '@pkgjs/parseargs@0.11.0': - optional: true - - '@rollup/rollup-android-arm-eabi@4.21.2': - optional: true - - '@rollup/rollup-android-arm64@4.21.2': - optional: true - - '@rollup/rollup-darwin-arm64@4.21.2': - optional: true - - '@rollup/rollup-darwin-x64@4.21.2': - optional: true - - '@rollup/rollup-linux-arm-gnueabihf@4.21.2': - optional: true - - '@rollup/rollup-linux-arm-musleabihf@4.21.2': - optional: true - - '@rollup/rollup-linux-arm64-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-arm64-musl@4.21.2': - optional: true - - '@rollup/rollup-linux-powerpc64le-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-riscv64-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-s390x-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-x64-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-x64-musl@4.21.2': - optional: true - - '@rollup/rollup-win32-arm64-msvc@4.21.2': - optional: true - - '@rollup/rollup-win32-ia32-msvc@4.21.2': - optional: true - - '@rollup/rollup-win32-x64-msvc@4.21.2': - optional: true - - '@sapphire/async-queue@1.5.3': {} - - '@sapphire/shapeshift@4.0.0': - dependencies: - fast-deep-equal: 3.1.3 - lodash: 4.17.21 - - '@sapphire/snowflake@3.5.3': {} - - '@sec-ant/readable-stream@0.4.1': {} - - '@sindresorhus/merge-streams@4.0.0': {} - - '@sinonjs/commons@2.0.0': - dependencies: - type-detect: 4.0.8 - - '@sinonjs/commons@3.0.1': - dependencies: - type-detect: 4.0.8 - - '@sinonjs/fake-timers@10.3.0': - dependencies: - '@sinonjs/commons': 3.0.1 - - '@sinonjs/fake-timers@11.3.1': - dependencies: - '@sinonjs/commons': 3.0.1 - - '@sinonjs/samsam@8.0.0': - dependencies: - '@sinonjs/commons': 2.0.0 - lodash.get: 4.4.2 - type-detect: 4.1.0 - - '@sinonjs/text-encoding@0.7.3': {} - - '@smithy/abort-controller@3.1.1': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/chunked-blob-reader-native@3.0.0': - dependencies: - '@smithy/util-base64': 3.0.0 - tslib: 2.7.0 - - '@smithy/chunked-blob-reader@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/config-resolver@3.0.5': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@smithy/core@2.4.0': - dependencies: - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/credential-provider-imds@3.2.0': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - tslib: 2.7.0 - - '@smithy/eventstream-codec@3.1.2': - dependencies: - '@aws-crypto/crc32': 5.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-hex-encoding': 3.0.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-browser@3.0.6': - dependencies: - '@smithy/eventstream-serde-universal': 3.0.5 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-config-resolver@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-node@3.0.5': - dependencies: - '@smithy/eventstream-serde-universal': 3.0.5 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-universal@3.0.5': - dependencies: - '@smithy/eventstream-codec': 3.1.2 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/fetch-http-handler@3.2.4': - dependencies: - '@smithy/protocol-http': 4.1.0 - '@smithy/querystring-builder': 3.0.3 - '@smithy/types': 3.3.0 - '@smithy/util-base64': 3.0.0 - tslib: 2.7.0 - - '@smithy/hash-blob-browser@3.1.2': - dependencies: - '@smithy/chunked-blob-reader': 3.0.0 - '@smithy/chunked-blob-reader-native': 3.0.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/hash-node@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/hash-stream-node@3.1.2': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/invalid-dependency@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/is-array-buffer@2.2.0': - dependencies: - tslib: 2.7.0 - - '@smithy/is-array-buffer@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/md5-js@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/middleware-content-length@3.0.5': - dependencies: - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/middleware-endpoint@3.1.0': - dependencies: - '@smithy/middleware-serde': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@smithy/middleware-retry@3.0.15': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/service-error-classification': 3.0.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - tslib: 2.7.0 - uuid: 9.0.1 - - '@smithy/middleware-serde@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/middleware-stack@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/node-config-provider@3.1.4': - dependencies: - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/node-http-handler@3.1.4': - dependencies: - '@smithy/abort-controller': 3.1.1 - '@smithy/protocol-http': 4.1.0 - '@smithy/querystring-builder': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/property-provider@3.1.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/protocol-http@4.1.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/querystring-builder@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-uri-escape': 3.0.0 - tslib: 2.7.0 - - '@smithy/querystring-parser@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/service-error-classification@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - - '@smithy/shared-ini-file-loader@3.1.4': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/signature-v4@4.1.0': - dependencies: - '@smithy/is-array-buffer': 3.0.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-hex-encoding': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-uri-escape': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/smithy-client@3.2.0': - dependencies: - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-stack': 3.0.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-stream': 3.1.3 - tslib: 2.7.0 - - '@smithy/types@3.3.0': - dependencies: - tslib: 2.7.0 - - '@smithy/url-parser@3.0.3': - dependencies: - '@smithy/querystring-parser': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-base64@3.0.0': - dependencies: - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-body-length-browser@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-body-length-node@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-buffer-from@2.2.0': - dependencies: - '@smithy/is-array-buffer': 2.2.0 - tslib: 2.7.0 - - '@smithy/util-buffer-from@3.0.0': - dependencies: - '@smithy/is-array-buffer': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-config-provider@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-defaults-mode-browser@3.0.15': - dependencies: - '@smithy/property-provider': 3.1.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - bowser: 2.11.0 - tslib: 2.7.0 - - '@smithy/util-defaults-mode-node@3.0.15': - dependencies: - '@smithy/config-resolver': 3.0.5 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-endpoints@2.0.5': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-hex-encoding@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-middleware@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-retry@3.0.3': - dependencies: - '@smithy/service-error-classification': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-stream@3.1.3': - dependencies: - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-base64': 3.0.0 - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-hex-encoding': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-uri-escape@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-utf8@2.3.0': - dependencies: - '@smithy/util-buffer-from': 2.2.0 - tslib: 2.7.0 - - '@smithy/util-utf8@3.0.0': - dependencies: - '@smithy/util-buffer-from': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-waiter@3.1.2': - dependencies: - '@smithy/abort-controller': 3.1.1 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@tsconfig/node10@1.0.11': {} - - '@tsconfig/node12@1.0.11': {} - - '@tsconfig/node14@1.0.3': {} - - '@tsconfig/node16@1.0.4': {} - - '@types/chai-as-promised@8.0.0': - dependencies: - '@types/chai': 4.3.19 - - '@types/chai@4.3.19': {} - - '@types/debug@4.1.12': - dependencies: - '@types/ms': 0.7.34 - - '@types/estree@1.0.5': {} - - '@types/fluent-ffmpeg@2.1.26': - dependencies: - '@types/node': 22.5.2 - - '@types/interpret@1.1.3': - dependencies: - '@types/node': 22.5.2 - - '@types/mocha@10.0.7': {} - - '@types/ms@0.7.34': {} - - '@types/node@20.16.3': - dependencies: - undici-types: 6.19.8 - - '@types/node@22.5.2': - dependencies: - undici-types: 6.19.8 - - '@types/pg@8.11.8': - dependencies: - '@types/node': 22.5.2 - pg-protocol: 1.6.1 - pg-types: 4.0.2 - - '@types/qs@6.9.15': {} - - '@types/retry@0.12.2': {} - - '@types/semver@7.5.8': {} - - '@types/sinon-chai@3.2.12': - dependencies: - '@types/chai': 4.3.19 - '@types/sinon': 17.0.3 - - '@types/sinon@10.0.20': - dependencies: - '@types/sinonjs__fake-timers': 8.1.5 - - '@types/sinon@17.0.3': - dependencies: - '@types/sinonjs__fake-timers': 8.1.5 - - '@types/sinonjs__fake-timers@8.1.5': {} - - '@types/triple-beam@1.3.5': {} - - '@types/ws@8.5.12': - dependencies: - '@types/node': 22.5.2 - - '@vladfrangu/async_event_emitter@2.4.6': {} - - abort-controller@3.0.0: - dependencies: - event-target-shim: 5.0.1 - - abstract-logging@2.0.1: {} - - acorn-walk@8.3.3: - dependencies: - acorn: 8.12.1 - - acorn@8.12.1: {} - - ajv-formats@2.1.1(ajv@8.17.1): - optionalDependencies: - ajv: 8.17.1 - - ajv-formats@3.0.1(ajv@8.17.1): - optionalDependencies: - ajv: 8.17.1 - - ajv@8.17.1: - dependencies: - fast-deep-equal: 3.1.3 - fast-uri: 3.0.1 - json-schema-traverse: 1.0.0 - require-from-string: 2.0.2 - - ansi-colors@4.1.3: {} - - ansi-regex@5.0.1: {} - - ansi-regex@6.0.1: {} - - ansi-styles@3.2.1: - dependencies: - color-convert: 1.9.3 - - ansi-styles@4.3.0: - dependencies: - color-convert: 2.0.1 - - ansi-styles@6.2.1: {} - - any-promise@1.3.0: {} - - anymatch@3.1.3: - dependencies: - normalize-path: 3.0.0 - picomatch: 2.3.1 - - arg@4.1.3: {} - - argparse@2.0.1: {} - - array-union@2.1.0: {} - - asap@2.0.6: {} - - assertion-error@2.0.1: {} - - async@0.2.10: {} - - async@3.2.6: {} - - atomic-sleep@1.0.0: {} - - available-typed-arrays@1.0.7: - dependencies: - possible-typed-array-names: 1.0.0 - - avvio@8.4.0: - dependencies: - '@fastify/error': 3.4.1 - fastq: 1.17.1 - - aws-sdk-client-mock@4.0.1: - dependencies: - '@types/sinon': 10.0.20 - sinon: 16.1.3 - tslib: 2.7.0 - - aws-sdk-mock@6.1.1: - dependencies: - aws-sdk: 2.1687.0 - neotraverse: 0.6.18 - sinon: 18.0.0 - - aws-sdk@2.1687.0: - dependencies: - buffer: 4.9.2 - events: 1.1.1 - ieee754: 1.1.13 - jmespath: 0.16.0 - querystring: 0.2.0 - sax: 1.2.1 - url: 0.10.3 - util: 0.12.5 - uuid: 8.0.0 - xml2js: 0.6.2 - - balanced-match@1.0.2: {} - - base64-js@1.5.1: {} - - binary-extensions@2.3.0: {} - - boolbase@1.0.0: {} - - bowser@2.11.0: {} - - brace-expansion@1.1.11: - dependencies: - balanced-match: 1.0.2 - concat-map: 0.0.1 - - brace-expansion@2.0.1: - dependencies: - balanced-match: 1.0.2 - - braces@3.0.3: - dependencies: - fill-range: 7.1.1 - - browser-stdout@1.3.1: {} - - buffer@4.9.2: - dependencies: - base64-js: 1.5.1 - ieee754: 1.1.13 - isarray: 1.0.0 - - buffer@5.6.0: - dependencies: - base64-js: 1.5.1 - ieee754: 1.2.1 - - buffer@6.0.3: - dependencies: - base64-js: 1.5.1 - ieee754: 1.2.1 - - bundle-require@5.0.0(esbuild@0.23.1): - dependencies: - esbuild: 0.23.1 - load-tsconfig: 0.2.5 - - cac@6.7.14: {} - - call-bind@1.0.7: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - set-function-length: 1.2.2 - - callsites@3.1.0: {} - - camelcase@6.3.0: {} - - chai-as-promised@8.0.0(chai@5.1.1): - dependencies: - chai: 5.1.1 - check-error: 2.1.1 - - chai@5.1.1: - dependencies: - assertion-error: 2.0.1 - check-error: 2.1.1 - deep-eql: 5.0.2 - loupe: 3.1.1 - pathval: 2.0.0 - - chalk@2.4.2: - dependencies: - ansi-styles: 3.2.1 - escape-string-regexp: 1.0.5 - supports-color: 5.5.0 - - chalk@4.1.2: - dependencies: - ansi-styles: 4.3.0 - supports-color: 7.2.0 - - check-error@2.1.1: {} - - cheerio-select@2.1.0: - dependencies: - boolbase: 1.0.0 - css-select: 5.1.0 - css-what: 6.1.0 - domelementtype: 2.3.0 - domhandler: 5.0.3 - domutils: 3.1.0 - - cheerio@1.0.0: - dependencies: - cheerio-select: 2.1.0 - dom-serializer: 2.0.0 - domhandler: 5.0.3 - domutils: 3.1.0 - encoding-sniffer: 0.2.0 - htmlparser2: 9.1.0 - parse5: 7.1.2 - parse5-htmlparser2-tree-adapter: 7.0.0 - parse5-parser-stream: 7.1.2 - undici: 6.19.8 - whatwg-mimetype: 4.0.0 - - chokidar@3.6.0: - dependencies: - anymatch: 3.1.3 - braces: 3.0.3 - glob-parent: 5.1.2 - is-binary-path: 2.1.0 - is-glob: 4.0.3 - normalize-path: 3.0.0 - readdirp: 3.6.0 - optionalDependencies: - fsevents: 2.3.3 - - cliui@7.0.4: - dependencies: - string-width: 4.2.3 - strip-ansi: 6.0.1 - wrap-ansi: 7.0.0 - - cliui@8.0.1: - dependencies: - string-width: 4.2.3 - strip-ansi: 6.0.1 - wrap-ansi: 7.0.0 - - cluster-key-slot@1.1.2: {} - - color-convert@1.9.3: - dependencies: - color-name: 1.1.3 - - color-convert@2.0.1: - dependencies: - color-name: 1.1.4 - - color-name@1.1.3: {} - - color-name@1.1.4: {} - - color-string@1.9.1: - dependencies: - color-name: 1.1.4 - simple-swizzle: 0.2.2 - - color@3.2.1: - dependencies: - color-convert: 1.9.3 - color-string: 1.9.1 - - colorette@2.0.20: {} - - colorspace@1.1.4: - dependencies: - color: 3.2.1 - text-hex: 1.0.0 - - commander@4.1.1: {} - - concat-map@0.0.1: {} - - consola@3.2.3: {} - - cookie@0.6.0: {} - - cosmiconfig@8.3.6(typescript@5.5.4): - dependencies: - import-fresh: 3.3.0 - js-yaml: 4.1.0 - parse-json: 5.2.0 - path-type: 4.0.0 - optionalDependencies: - typescript: 5.5.4 - - create-require@1.1.1: {} - - cron-parser@4.9.0: - dependencies: - luxon: 3.5.0 - - cross-spawn@7.0.3: - dependencies: - path-key: 3.1.1 - shebang-command: 2.0.0 - which: 2.0.2 - - csprng@0.1.2: - dependencies: - sequin: 0.1.1 - - css-select@5.1.0: - dependencies: - boolbase: 1.0.0 - css-what: 6.1.0 - domhandler: 5.0.3 - domutils: 3.1.0 - nth-check: 2.1.1 - - css-what@6.1.0: {} - - data-uri-to-buffer@4.0.1: {} - - date-fns@3.6.0: {} - - dateformat@4.6.3: {} - - debug@4.3.6: - dependencies: - ms: 2.1.2 - - debug@4.3.6(supports-color@5.5.0): - dependencies: - ms: 2.1.2 - optionalDependencies: - supports-color: 5.5.0 - - debug@4.3.6(supports-color@8.1.1): - dependencies: - ms: 2.1.2 - optionalDependencies: - supports-color: 8.1.1 - - decamelize@4.0.0: {} - - deep-eql@5.0.2: {} - - define-data-property@1.1.4: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - gopd: 1.0.1 - - denque@2.1.0: {} - - diff@4.0.2: {} - - diff@5.2.0: {} - - dir-glob@3.0.1: - dependencies: - path-type: 4.0.0 - - discord-api-types@0.37.83: {} - - discord-api-types@0.37.97: {} - - discord.js@14.16.1: - dependencies: - '@discordjs/builders': 1.9.0 - '@discordjs/collection': 1.5.3 - '@discordjs/formatters': 0.5.0 - '@discordjs/rest': 2.4.0 - '@discordjs/util': 1.1.1 - '@discordjs/ws': 1.1.1 - '@sapphire/snowflake': 3.5.3 - discord-api-types: 0.37.97 - fast-deep-equal: 3.1.3 - lodash.snakecase: 4.1.1 - tslib: 2.7.0 - undici: 6.19.8 - transitivePeerDependencies: - - bufferutil - - utf-8-validate - - diskusage@1.2.0: - dependencies: - es6-promise: 4.2.8 - nan: 2.20.0 - - dom-serializer@2.0.0: - dependencies: - domelementtype: 2.3.0 - domhandler: 5.0.3 - entities: 4.5.0 - - domelementtype@2.3.0: {} - - domhandler@5.0.3: - dependencies: - domelementtype: 2.3.0 - - domutils@3.1.0: - dependencies: - dom-serializer: 2.0.0 - domelementtype: 2.3.0 - domhandler: 5.0.3 - - dotenv@16.4.5: {} - - eastasianwidth@0.2.0: {} - - emoji-regex@8.0.0: {} - - emoji-regex@9.2.2: {} - - enabled@2.0.0: {} - - encoding-sniffer@0.2.0: - dependencies: - iconv-lite: 0.6.3 - whatwg-encoding: 3.1.1 - - end-of-stream@1.4.4: - dependencies: - once: 1.4.0 - - entities@4.5.0: {} - - error-ex@1.3.2: - dependencies: - is-arrayish: 0.2.1 - - es-define-property@1.0.0: - dependencies: - get-intrinsic: 1.2.4 - - es-errors@1.3.0: {} - - es6-promise@4.2.8: {} - - esbuild@0.23.1: - optionalDependencies: - '@esbuild/aix-ppc64': 0.23.1 - '@esbuild/android-arm': 0.23.1 - '@esbuild/android-arm64': 0.23.1 - '@esbuild/android-x64': 0.23.1 - '@esbuild/darwin-arm64': 0.23.1 - '@esbuild/darwin-x64': 0.23.1 - '@esbuild/freebsd-arm64': 0.23.1 - '@esbuild/freebsd-x64': 0.23.1 - '@esbuild/linux-arm': 0.23.1 - '@esbuild/linux-arm64': 0.23.1 - '@esbuild/linux-ia32': 0.23.1 - '@esbuild/linux-loong64': 0.23.1 - '@esbuild/linux-mips64el': 0.23.1 - '@esbuild/linux-ppc64': 0.23.1 - '@esbuild/linux-riscv64': 0.23.1 - '@esbuild/linux-s390x': 0.23.1 - '@esbuild/linux-x64': 0.23.1 - '@esbuild/netbsd-x64': 0.23.1 - '@esbuild/openbsd-arm64': 0.23.1 - '@esbuild/openbsd-x64': 0.23.1 - '@esbuild/sunos-x64': 0.23.1 - '@esbuild/win32-arm64': 0.23.1 - '@esbuild/win32-ia32': 0.23.1 - '@esbuild/win32-x64': 0.23.1 - - escalade@3.2.0: {} - - escape-string-regexp@1.0.5: {} - - escape-string-regexp@4.0.0: {} - - event-target-shim@5.0.1: {} - - events@1.1.1: {} - - events@3.3.0: {} - - execa@5.1.1: - dependencies: - cross-spawn: 7.0.3 - get-stream: 6.0.1 - human-signals: 2.1.0 - is-stream: 2.0.1 - merge-stream: 2.0.0 - npm-run-path: 4.0.1 - onetime: 5.1.2 - signal-exit: 3.0.7 - strip-final-newline: 2.0.0 - - execa@9.3.1: - dependencies: - '@sindresorhus/merge-streams': 4.0.0 - cross-spawn: 7.0.3 - figures: 6.1.0 - get-stream: 9.0.1 - human-signals: 8.0.0 - is-plain-obj: 4.1.0 - is-stream: 4.0.1 - npm-run-path: 5.3.0 - pretty-ms: 9.1.0 - signal-exit: 4.1.0 - strip-final-newline: 4.0.0 - yoctocolors: 2.1.1 - - exponential-backoff@3.1.1: {} - - fast-content-type-parse@1.1.0: {} - - fast-copy@3.0.2: {} - - fast-decode-uri-component@1.0.1: {} - - fast-deep-equal@3.1.3: {} - - fast-glob@3.3.2: - dependencies: - '@nodelib/fs.stat': 2.0.5 - '@nodelib/fs.walk': 1.2.8 - glob-parent: 5.1.2 - merge2: 1.4.1 - micromatch: 4.0.8 - - fast-json-stringify@5.16.1: - dependencies: - '@fastify/merge-json-schemas': 0.1.1 - ajv: 8.17.1 - ajv-formats: 3.0.1(ajv@8.17.1) - fast-deep-equal: 3.1.3 - fast-uri: 2.4.0 - json-schema-ref-resolver: 1.0.1 - rfdc: 1.4.1 - - fast-querystring@1.1.2: - dependencies: - fast-decode-uri-component: 1.0.1 - - fast-redact@3.5.0: {} - - fast-safe-stringify@2.1.1: {} - - fast-uri@2.4.0: {} - - fast-uri@3.0.1: {} - - fast-xml-parser@4.4.1: - dependencies: - strnum: 1.0.5 - - fastify-plugin@4.5.1: {} - - fastify@4.28.1: - dependencies: - '@fastify/ajv-compiler': 3.6.0 - '@fastify/error': 3.4.1 - '@fastify/fast-json-stringify-compiler': 4.3.0 - abstract-logging: 2.0.1 - avvio: 8.4.0 - fast-content-type-parse: 1.1.0 - fast-json-stringify: 5.16.1 - find-my-way: 8.2.0 - light-my-request: 5.13.0 - pino: 9.4.0 - process-warning: 3.0.0 - proxy-addr: 2.0.7 - rfdc: 1.4.1 - secure-json-parse: 2.7.0 - semver: 7.6.3 - toad-cache: 3.7.0 - - fastq@1.17.1: - dependencies: - reusify: 1.0.4 - - faye-websocket@0.11.4: - dependencies: - websocket-driver: 0.7.4 - - faye@1.4.0: - dependencies: - asap: 2.0.6 - csprng: 0.1.2 - faye-websocket: 0.11.4 - safe-buffer: 5.2.1 - tough-cookie: 4.1.4 - tunnel-agent: 0.6.0 - - fecha@4.2.3: {} - - fetch-blob@3.2.0: - dependencies: - node-domexception: 1.0.0 - web-streams-polyfill: 3.3.3 - - figures@6.1.0: - dependencies: - is-unicode-supported: 2.0.0 - - fill-range@7.1.1: - dependencies: - to-regex-range: 5.0.1 - - find-my-way@8.2.0: - dependencies: - fast-deep-equal: 3.1.3 - fast-querystring: 1.1.2 - safe-regex2: 3.1.0 - - find-up@5.0.0: - dependencies: - locate-path: 6.0.0 - path-exists: 4.0.0 - - flat@5.0.2: {} - - fluent-ffmpeg@2.1.3: - dependencies: - async: 0.2.10 - which: 1.3.1 - - fn.name@1.1.0: {} - - for-each@0.3.3: - dependencies: - is-callable: 1.2.7 - - foreground-child@3.3.0: - dependencies: - cross-spawn: 7.0.3 - signal-exit: 4.1.0 - - formdata-polyfill@4.0.10: - dependencies: - fetch-blob: 3.2.0 - - forwarded@0.2.0: {} - - fs.realpath@1.0.0: {} - - fsevents@2.3.3: - optional: true - - function-bind@1.1.2: {} - - get-caller-file@2.0.5: {} - - get-func-name@2.0.2: {} - - get-intrinsic@1.2.4: - dependencies: - es-errors: 1.3.0 - function-bind: 1.1.2 - has-proto: 1.0.3 - has-symbols: 1.0.3 - hasown: 2.0.2 - - get-stream@6.0.1: {} - - get-stream@9.0.1: - dependencies: - '@sec-ant/readable-stream': 0.4.1 - is-stream: 4.0.1 - - get-tsconfig@4.8.0: - dependencies: - resolve-pkg-maps: 1.0.0 - - glob-parent@5.1.2: - dependencies: - is-glob: 4.0.3 - - glob@10.4.5: - dependencies: - foreground-child: 3.3.0 - jackspeak: 3.4.3 - minimatch: 9.0.5 - minipass: 7.1.2 - package-json-from-dist: 1.0.0 - path-scurry: 1.11.1 - - glob@8.1.0: - dependencies: - fs.realpath: 1.0.0 - inflight: 1.0.6 - inherits: 2.0.4 - minimatch: 5.1.6 - once: 1.4.0 - - globby@11.1.0: - dependencies: - array-union: 2.1.0 - dir-glob: 3.0.1 - fast-glob: 3.3.2 - ignore: 5.3.2 - merge2: 1.4.1 - slash: 3.0.0 - - gopd@1.0.1: - dependencies: - get-intrinsic: 1.2.4 - - graphile-config@0.0.1-beta.9: - dependencies: - '@types/interpret': 1.1.3 - '@types/node': 20.16.3 - '@types/semver': 7.5.8 - chalk: 4.1.2 - debug: 4.3.6 - interpret: 3.1.1 - semver: 7.6.3 - tslib: 2.7.0 - yargs: 17.7.2 - transitivePeerDependencies: - - supports-color - - graphile-worker@0.16.6(typescript@5.5.4): - dependencies: - '@graphile/logger': 0.2.0 - '@types/debug': 4.1.12 - '@types/pg': 8.11.8 - cosmiconfig: 8.3.6(typescript@5.5.4) - graphile-config: 0.0.1-beta.9 - json5: 2.2.3 - pg: 8.12.0 - tslib: 2.7.0 - yargs: 17.7.2 - transitivePeerDependencies: - - pg-native - - supports-color - - typescript - - has-flag@3.0.0: {} - - has-flag@4.0.0: {} - - has-property-descriptors@1.0.2: - dependencies: - es-define-property: 1.0.0 - - has-proto@1.0.3: {} - - has-symbols@1.0.3: {} - - has-tostringtag@1.0.2: - dependencies: - has-symbols: 1.0.3 - - hasown@2.0.2: - dependencies: - function-bind: 1.1.2 - - he@1.2.0: {} - - help-me@5.0.0: {} - - htmlparser2@9.1.0: - dependencies: - domelementtype: 2.3.0 - domhandler: 5.0.3 - domutils: 3.1.0 - entities: 4.5.0 - - http-parser-js@0.5.8: {} - - https@1.0.0: {} - - human-signals@2.1.0: {} - - human-signals@8.0.0: {} - - iconv-lite@0.6.3: - dependencies: - safer-buffer: 2.1.2 - - ieee754@1.1.13: {} - - ieee754@1.2.1: {} - - ignore-by-default@1.0.1: {} - - ignore@5.3.2: {} - - import-fresh@3.3.0: - dependencies: - parent-module: 1.0.1 - resolve-from: 4.0.0 - - inflight@1.0.6: - dependencies: - once: 1.4.0 - wrappy: 1.0.2 - - inherits@2.0.4: {} - - interpret@3.1.1: {} - - ioredis@5.4.1: - dependencies: - '@ioredis/commands': 1.2.0 - cluster-key-slot: 1.1.2 - debug: 4.3.6 - denque: 2.1.0 - lodash.defaults: 4.2.0 - lodash.isarguments: 3.1.0 - redis-errors: 1.2.0 - redis-parser: 3.0.0 - standard-as-callback: 2.1.0 - transitivePeerDependencies: - - supports-color - - ipaddr.js@1.9.1: {} - - is-arguments@1.1.1: - dependencies: - call-bind: 1.0.7 - has-tostringtag: 1.0.2 - - is-arrayish@0.2.1: {} - - is-arrayish@0.3.2: {} - - is-binary-path@2.1.0: - dependencies: - binary-extensions: 2.3.0 - - is-callable@1.2.7: {} - - is-extglob@2.1.1: {} - - is-fullwidth-code-point@3.0.0: {} - - is-generator-function@1.0.10: - dependencies: - has-tostringtag: 1.0.2 - - is-glob@4.0.3: - dependencies: - is-extglob: 2.1.1 - - is-network-error@1.1.0: {} - - is-number@7.0.0: {} - - is-plain-obj@2.1.0: {} - - is-plain-obj@4.1.0: {} - - is-stream@2.0.1: {} - - is-stream@4.0.1: {} - - is-typed-array@1.1.13: - dependencies: - which-typed-array: 1.1.15 - - is-unicode-supported@0.1.0: {} - - is-unicode-supported@2.0.0: {} - - isarray@1.0.0: {} - - isexe@2.0.0: {} - - jackspeak@3.4.3: - dependencies: - '@isaacs/cliui': 8.0.2 - optionalDependencies: - '@pkgjs/parseargs': 0.11.0 - - jmespath@0.16.0: {} - - joycon@3.1.1: {} - - js-tokens@4.0.0: {} - - js-yaml@4.1.0: - dependencies: - argparse: 2.0.1 - - json-parse-even-better-errors@2.3.1: {} - - json-schema-ref-resolver@1.0.1: - dependencies: - fast-deep-equal: 3.1.3 - - json-schema-traverse@1.0.0: {} - - json5@2.2.3: {} - - just-extend@6.2.0: {} - - kuler@2.0.0: {} - - light-my-request@5.13.0: - dependencies: - cookie: 0.6.0 - process-warning: 3.0.0 - set-cookie-parser: 2.7.0 - - lilconfig@3.1.2: {} - - lines-and-columns@1.2.4: {} - - load-tsconfig@0.2.5: {} - - locate-path@6.0.0: - dependencies: - p-locate: 5.0.0 - - lodash.defaults@4.2.0: {} - - lodash.get@4.4.2: {} - - lodash.isarguments@3.1.0: {} - - lodash.snakecase@4.1.1: {} - - lodash.sortby@4.7.0: {} - - lodash@4.1.0: {} - - lodash@4.17.21: {} - - log-symbols@4.1.0: - dependencies: - chalk: 4.1.2 - is-unicode-supported: 0.1.0 - - logform@2.6.1: - dependencies: - '@colors/colors': 1.6.0 - '@types/triple-beam': 1.3.5 - fecha: 4.2.3 - ms: 2.1.3 - safe-stable-stringify: 2.5.0 - triple-beam: 1.4.1 - - loupe@3.1.1: - dependencies: - get-func-name: 2.0.2 - - lru-cache@10.4.3: {} - - luxon@3.5.0: {} - - magic-bytes.js@1.10.0: {} - - make-error@1.3.6: {} - - merge-stream@2.0.0: {} - - merge2@1.4.1: {} - - micromatch@4.0.8: - dependencies: - braces: 3.0.3 - picomatch: 2.3.1 - - mimic-fn@2.1.0: {} - - minimatch@10.0.1: - dependencies: - brace-expansion: 2.0.1 - - minimatch@3.1.2: - dependencies: - brace-expansion: 1.1.11 - - minimatch@5.1.6: - dependencies: - brace-expansion: 2.0.1 - - minimatch@9.0.5: - dependencies: - brace-expansion: 2.0.1 - - minimist@1.2.8: {} - - minipass@7.1.2: {} - - mocha@10.7.3: - dependencies: - ansi-colors: 4.1.3 - browser-stdout: 1.3.1 - chokidar: 3.6.0 - debug: 4.3.6(supports-color@8.1.1) - diff: 5.2.0 - escape-string-regexp: 4.0.0 - find-up: 5.0.0 - glob: 8.1.0 - he: 1.2.0 - js-yaml: 4.1.0 - log-symbols: 4.1.0 - minimatch: 5.1.6 - ms: 2.1.3 - serialize-javascript: 6.0.2 - strip-json-comments: 3.1.1 - supports-color: 8.1.1 - workerpool: 6.5.1 - yargs: 16.2.0 - yargs-parser: 20.2.9 - yargs-unparser: 2.0.0 - - ms@2.1.2: {} - - ms@2.1.3: {} - - multiformats@13.2.2: {} - - mz@2.7.0: - dependencies: - any-promise: 1.3.0 - object-assign: 4.1.1 - thenify-all: 1.6.0 - - nan@2.20.0: {} - - nanoid@5.0.7: {} - - neotraverse@0.6.18: {} - - nise@5.1.9: - dependencies: - '@sinonjs/commons': 3.0.1 - '@sinonjs/fake-timers': 11.3.1 - '@sinonjs/text-encoding': 0.7.3 - just-extend: 6.2.0 - path-to-regexp: 6.2.2 - - nise@6.0.0: - dependencies: - '@sinonjs/commons': 3.0.1 - '@sinonjs/fake-timers': 11.3.1 - '@sinonjs/text-encoding': 0.7.3 - just-extend: 6.2.0 - path-to-regexp: 6.2.2 - - node-abort-controller@3.1.1: {} - - node-domexception@1.0.0: {} - - node-fetch@3.3.2: - dependencies: - data-uri-to-buffer: 4.0.1 - fetch-blob: 3.2.0 - formdata-polyfill: 4.0.10 - - nodemon@3.1.4: - dependencies: - chokidar: 3.6.0 - debug: 4.3.6(supports-color@5.5.0) - ignore-by-default: 1.0.1 - minimatch: 3.1.2 - pstree.remy: 1.1.8 - semver: 7.6.3 - simple-update-notifier: 2.0.0 - supports-color: 5.5.0 - touch: 3.1.1 - undefsafe: 2.0.5 - - normalize-path@3.0.0: {} - - npm-run-path@4.0.1: - dependencies: - path-key: 3.1.1 - - npm-run-path@5.3.0: - dependencies: - path-key: 4.0.0 - - nth-check@2.1.1: - dependencies: - boolbase: 1.0.0 - - object-assign@4.1.1: {} - - object-inspect@1.13.2: {} - - obuf@1.1.2: {} - - on-exit-leak-free@2.1.2: {} - - once@1.4.0: - dependencies: - wrappy: 1.0.2 - - one-time@1.0.0: - dependencies: - fn.name: 1.1.0 - - onetime@5.1.2: - dependencies: - mimic-fn: 2.1.0 - - p-limit@3.1.0: - dependencies: - yocto-queue: 0.1.0 - - p-locate@5.0.0: - dependencies: - p-limit: 3.1.0 - - p-retry@6.2.0: - dependencies: - '@types/retry': 0.12.2 - is-network-error: 1.1.0 - retry: 0.13.1 - - package-json-from-dist@1.0.0: {} - - parent-module@1.0.1: - dependencies: - callsites: 3.1.0 - - parse-json@5.2.0: - dependencies: - '@babel/code-frame': 7.24.7 - error-ex: 1.3.2 - json-parse-even-better-errors: 2.3.1 - lines-and-columns: 1.2.4 - - parse-ms@4.0.0: {} - - parse5-htmlparser2-tree-adapter@7.0.0: - dependencies: - domhandler: 5.0.3 - parse5: 7.1.2 - - parse5-parser-stream@7.1.2: - dependencies: - parse5: 7.1.2 - - parse5@7.1.2: - dependencies: - entities: 4.5.0 - - path-exists@4.0.0: {} - - path-key@3.1.1: {} - - path-key@4.0.0: {} - - path-scurry@1.11.1: - dependencies: - lru-cache: 10.4.3 - minipass: 7.1.2 - - path-to-regexp@6.2.2: {} - - path-type@4.0.0: {} - - pathval@2.0.0: {} - - pg-boss@10.1.1: - dependencies: - cron-parser: 4.9.0 - pg: 8.12.0 - serialize-error: 8.1.0 - transitivePeerDependencies: - - pg-native - - pg-cloudflare@1.1.1: - optional: true - - pg-connection-string@2.6.4: {} - - pg-int8@1.0.1: {} - - pg-numeric@1.0.2: {} - - pg-pool@3.6.2(pg@8.12.0): - dependencies: - pg: 8.12.0 - - pg-protocol@1.6.1: {} - - pg-types@2.2.0: - dependencies: - pg-int8: 1.0.1 - postgres-array: 2.0.0 - postgres-bytea: 1.0.0 - postgres-date: 1.0.7 - postgres-interval: 1.2.0 - - pg-types@4.0.2: - dependencies: - pg-int8: 1.0.1 - pg-numeric: 1.0.2 - postgres-array: 3.0.2 - postgres-bytea: 3.0.0 - postgres-date: 2.1.0 - postgres-interval: 3.0.0 - postgres-range: 1.1.4 - - pg@8.12.0: - dependencies: - pg-connection-string: 2.6.4 - pg-pool: 3.6.2(pg@8.12.0) - pg-protocol: 1.6.1 - pg-types: 2.2.0 - pgpass: 1.0.5 - optionalDependencies: - pg-cloudflare: 1.1.1 - - pgpass@1.0.5: - dependencies: - split2: 4.2.0 - - picocolors@1.1.0: {} - - picomatch@2.3.1: {} - - pino-abstract-transport@1.2.0: - dependencies: - readable-stream: 4.5.2 - split2: 4.2.0 - - pino-pretty@11.2.2: - dependencies: - colorette: 2.0.20 - dateformat: 4.6.3 - fast-copy: 3.0.2 - fast-safe-stringify: 2.1.1 - help-me: 5.0.0 - joycon: 3.1.1 - minimist: 1.2.8 - on-exit-leak-free: 2.1.2 - pino-abstract-transport: 1.2.0 - pump: 3.0.0 - readable-stream: 4.5.2 - secure-json-parse: 2.7.0 - sonic-boom: 4.1.0 - strip-json-comments: 3.1.1 - - pino-std-serializers@7.0.0: {} - - pino@9.4.0: - dependencies: - atomic-sleep: 1.0.0 - fast-redact: 3.5.0 - on-exit-leak-free: 2.1.2 - pino-abstract-transport: 1.2.0 - pino-std-serializers: 7.0.0 - process-warning: 4.0.0 - quick-format-unescaped: 4.0.4 - real-require: 0.2.0 - safe-stable-stringify: 2.5.0 - sonic-boom: 4.1.0 - thread-stream: 3.1.0 - - pirates@4.0.6: {} - - possible-typed-array-names@1.0.0: {} - - postcss-load-config@6.0.1(tsx@4.19.0): - dependencies: - lilconfig: 3.1.2 - optionalDependencies: - tsx: 4.19.0 - - postgres-array@2.0.0: {} - - postgres-array@3.0.2: {} - - postgres-bytea@1.0.0: {} - - postgres-bytea@3.0.0: - dependencies: - obuf: 1.1.2 - - postgres-date@1.0.7: {} - - postgres-date@2.1.0: {} - - postgres-interval@1.2.0: - dependencies: - xtend: 4.0.2 - - postgres-interval@3.0.0: {} - - postgres-range@1.1.4: {} - - postgres@3.4.4: {} - - pretty-bytes@6.1.1: {} - - pretty-ms@9.1.0: - dependencies: - parse-ms: 4.0.0 - - process-warning@3.0.0: {} - - process-warning@4.0.0: {} - - process@0.11.10: {} - - proxy-addr@2.0.7: - dependencies: - forwarded: 0.2.0 - ipaddr.js: 1.9.1 - - psl@1.9.0: {} - - pstree.remy@1.1.8: {} - - pump@3.0.0: - dependencies: - end-of-stream: 1.4.4 - once: 1.4.0 - - punycode@1.3.2: {} - - punycode@2.3.1: {} - - qs@6.13.0: - dependencies: - side-channel: 1.0.6 - - querystring@0.2.0: {} - - querystringify@2.2.0: {} - - queue-microtask@1.2.3: {} - - quick-format-unescaped@4.0.4: {} - - randombytes@2.1.0: - dependencies: - safe-buffer: 5.2.1 - - readable-stream@3.6.2: - dependencies: - inherits: 2.0.4 - string_decoder: 1.3.0 - util-deprecate: 1.0.2 - - readable-stream@4.5.2: - dependencies: - abort-controller: 3.0.0 - buffer: 6.0.3 - events: 3.3.0 - process: 0.11.10 - string_decoder: 1.3.0 - - readdirp@3.6.0: - dependencies: - picomatch: 2.3.1 - - real-require@0.2.0: {} - - redis-errors@1.2.0: {} - - redis-parser@3.0.0: - dependencies: - redis-errors: 1.2.0 - - require-directory@2.1.1: {} - - require-from-string@2.0.2: {} - - requires-port@1.0.0: {} - - resolve-from@4.0.0: {} - - resolve-from@5.0.0: {} - - resolve-pkg-maps@1.0.0: {} - - ret@0.4.3: {} - - retry@0.13.1: {} - - reusify@1.0.4: {} - - rfdc@1.4.1: {} - - rollup@4.21.2: - dependencies: - '@types/estree': 1.0.5 - optionalDependencies: - '@rollup/rollup-android-arm-eabi': 4.21.2 - '@rollup/rollup-android-arm64': 4.21.2 - '@rollup/rollup-darwin-arm64': 4.21.2 - '@rollup/rollup-darwin-x64': 4.21.2 - '@rollup/rollup-linux-arm-gnueabihf': 4.21.2 - '@rollup/rollup-linux-arm-musleabihf': 4.21.2 - '@rollup/rollup-linux-arm64-gnu': 4.21.2 - '@rollup/rollup-linux-arm64-musl': 4.21.2 - '@rollup/rollup-linux-powerpc64le-gnu': 4.21.2 - '@rollup/rollup-linux-riscv64-gnu': 4.21.2 - '@rollup/rollup-linux-s390x-gnu': 4.21.2 - '@rollup/rollup-linux-x64-gnu': 4.21.2 - '@rollup/rollup-linux-x64-musl': 4.21.2 - '@rollup/rollup-win32-arm64-msvc': 4.21.2 - '@rollup/rollup-win32-ia32-msvc': 4.21.2 - '@rollup/rollup-win32-x64-msvc': 4.21.2 - fsevents: 2.3.3 - - run-parallel@1.2.0: - dependencies: - queue-microtask: 1.2.3 - - rxjs@7.8.1: - dependencies: - tslib: 2.7.0 - - safe-buffer@5.2.1: {} - - safe-regex2@3.1.0: - dependencies: - ret: 0.4.3 - - safe-stable-stringify@2.5.0: {} - - safer-buffer@2.1.2: {} - - sax@1.2.1: {} - - secure-json-parse@2.7.0: {} - - semver@7.6.3: {} - - sequin@0.1.1: {} - - serialize-error@8.1.0: - dependencies: - type-fest: 0.20.2 - - serialize-javascript@6.0.2: - dependencies: - randombytes: 2.1.0 - - set-cookie-parser@2.7.0: {} - - set-function-length@1.2.2: - dependencies: - define-data-property: 1.1.4 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - gopd: 1.0.1 - has-property-descriptors: 1.0.2 - - shebang-command@2.0.0: - dependencies: - shebang-regex: 3.0.0 - - shebang-regex@3.0.0: {} - - side-channel@1.0.6: - dependencies: - call-bind: 1.0.7 - es-errors: 1.3.0 - get-intrinsic: 1.2.4 - object-inspect: 1.13.2 - - signal-exit@3.0.7: {} - - signal-exit@4.1.0: {} - - simple-swizzle@0.2.2: - dependencies: - is-arrayish: 0.3.2 - - simple-update-notifier@2.0.0: - dependencies: - semver: 7.6.3 - - sinon-chai@4.0.0(chai@5.1.1)(sinon@18.0.0): - dependencies: - chai: 5.1.1 - sinon: 18.0.0 - - sinon-test@3.1.6(sinon@18.0.0): - dependencies: - sinon: 18.0.0 - - sinon@16.1.3: - dependencies: - '@sinonjs/commons': 3.0.1 - '@sinonjs/fake-timers': 10.3.0 - '@sinonjs/samsam': 8.0.0 - diff: 5.2.0 - nise: 5.1.9 - supports-color: 7.2.0 - - sinon@18.0.0: - dependencies: - '@sinonjs/commons': 3.0.1 - '@sinonjs/fake-timers': 11.3.1 - '@sinonjs/samsam': 8.0.0 - diff: 5.2.0 - nise: 6.0.0 - supports-color: 7.2.0 - - slash@3.0.0: {} - - sliced@0.0.5: {} - - sonic-boom@4.1.0: - dependencies: - atomic-sleep: 1.0.0 - - source-map@0.8.0-beta.0: - dependencies: - whatwg-url: 7.1.0 - - split2@4.2.0: {} - - sql@0.78.0: - dependencies: - lodash: 4.1.0 - sliced: 0.0.5 - - stack-trace@0.0.10: {} - - standard-as-callback@2.1.0: {} - - stream-browserify@3.0.0: - dependencies: - inherits: 2.0.4 - readable-stream: 3.6.2 - - string-width@4.2.3: - dependencies: - emoji-regex: 8.0.0 - is-fullwidth-code-point: 3.0.0 - strip-ansi: 6.0.1 - - string-width@5.1.2: - dependencies: - eastasianwidth: 0.2.0 - emoji-regex: 9.2.2 - strip-ansi: 7.1.0 - - string_decoder@1.3.0: - dependencies: - safe-buffer: 5.2.1 - - strip-ansi@6.0.1: - dependencies: - ansi-regex: 5.0.1 - - strip-ansi@7.1.0: - dependencies: - ansi-regex: 6.0.1 - - strip-final-newline@2.0.0: {} - - strip-final-newline@4.0.0: {} - - strip-json-comments@3.1.1: {} - - strnum@1.0.5: {} - - sucrase@3.35.0: - dependencies: - '@jridgewell/gen-mapping': 0.3.5 - commander: 4.1.1 - glob: 10.4.5 - lines-and-columns: 1.2.4 - mz: 2.7.0 - pirates: 4.0.6 - ts-interface-checker: 0.1.13 - - supports-color@5.5.0: - dependencies: - has-flag: 3.0.0 - - supports-color@7.2.0: - dependencies: - has-flag: 4.0.0 - - supports-color@8.1.1: - dependencies: - has-flag: 4.0.0 - - text-hex@1.0.0: {} - - thenify-all@1.6.0: - dependencies: - thenify: 3.3.1 - - thenify@3.3.1: - dependencies: - any-promise: 1.3.0 - - thread-stream@3.1.0: - dependencies: - real-require: 0.2.0 - - to-regex-range@5.0.1: - dependencies: - is-number: 7.0.0 - - toad-cache@3.7.0: {} - - touch@3.1.1: {} - - tough-cookie@4.1.4: - dependencies: - psl: 1.9.0 - punycode: 2.3.1 - universalify: 0.2.0 - url-parse: 1.5.10 - - tr46@1.0.1: - dependencies: - punycode: 2.3.1 - - tree-kill@1.2.2: {} - - triple-beam@1.4.1: {} - - ts-interface-checker@0.1.13: {} - - ts-mixer@6.0.4: {} - - ts-node@10.9.2(@types/node@22.5.2)(typescript@5.5.4): - dependencies: - '@cspotcode/source-map-support': 0.8.1 - '@tsconfig/node10': 1.0.11 - '@tsconfig/node12': 1.0.11 - '@tsconfig/node14': 1.0.3 - '@tsconfig/node16': 1.0.4 - '@types/node': 22.5.2 - acorn: 8.12.1 - acorn-walk: 8.3.3 - arg: 4.1.3 - create-require: 1.1.1 - diff: 4.0.2 - make-error: 1.3.6 - typescript: 5.5.4 - v8-compile-cache-lib: 3.0.1 - yn: 3.1.1 - - tslib@2.7.0: {} - - tsup@8.2.4(tsx@4.19.0)(typescript@5.5.4): - dependencies: - bundle-require: 5.0.0(esbuild@0.23.1) - cac: 6.7.14 - chokidar: 3.6.0 - consola: 3.2.3 - debug: 4.3.6 - esbuild: 0.23.1 - execa: 5.1.1 - globby: 11.1.0 - joycon: 3.1.1 - picocolors: 1.1.0 - postcss-load-config: 6.0.1(tsx@4.19.0) - resolve-from: 5.0.0 - rollup: 4.21.2 - source-map: 0.8.0-beta.0 - sucrase: 3.35.0 - tree-kill: 1.2.2 - optionalDependencies: - typescript: 5.5.4 - transitivePeerDependencies: - - jiti - - supports-color - - tsx - - yaml - - tsx@4.19.0: - dependencies: - esbuild: 0.23.1 - get-tsconfig: 4.8.0 - optionalDependencies: - fsevents: 2.3.3 - - tunnel-agent@0.6.0: - dependencies: - safe-buffer: 5.2.1 - - type-detect@4.0.8: {} - - type-detect@4.1.0: {} - - type-fest@0.20.2: {} - - typescript@5.5.4: {} - - undefsafe@2.0.5: {} - - undici-types@6.19.8: {} - - undici@6.19.8: {} - - universalify@0.2.0: {} - - url-parse@1.5.10: - dependencies: - querystringify: 2.2.0 - requires-port: 1.0.0 - - url@0.10.3: - dependencies: - punycode: 1.3.2 - querystring: 0.2.0 - - util-deprecate@1.0.2: {} - - util@0.12.5: - dependencies: - inherits: 2.0.4 - is-arguments: 1.1.1 - is-generator-function: 1.0.10 - is-typed-array: 1.1.13 - which-typed-array: 1.1.15 - - uuid@8.0.0: {} - - uuid@9.0.1: {} - - v8-compile-cache-lib@3.0.1: {} - - web-streams-polyfill@3.3.3: {} - - webidl-conversions@4.0.2: {} - - websocket-driver@0.7.4: - dependencies: - http-parser-js: 0.5.8 - safe-buffer: 5.2.1 - websocket-extensions: 0.1.4 - - websocket-extensions@0.1.4: {} - - whatwg-encoding@3.1.1: - dependencies: - iconv-lite: 0.6.3 - - whatwg-mimetype@4.0.0: {} - - whatwg-url@7.1.0: - dependencies: - lodash.sortby: 4.7.0 - tr46: 1.0.1 - webidl-conversions: 4.0.2 - - which-typed-array@1.1.15: - dependencies: - available-typed-arrays: 1.0.7 - call-bind: 1.0.7 - for-each: 0.3.3 - gopd: 1.0.1 - has-tostringtag: 1.0.2 - - which@1.3.1: - dependencies: - isexe: 2.0.0 - - which@2.0.2: - dependencies: - isexe: 2.0.0 - - winston-transport@4.7.1: - dependencies: - logform: 2.6.1 - readable-stream: 3.6.2 - triple-beam: 1.4.1 - - winston@3.14.2: - dependencies: - '@colors/colors': 1.6.0 - '@dabh/diagnostics': 2.0.3 - async: 3.2.6 - is-stream: 2.0.1 - logform: 2.6.1 - one-time: 1.0.0 - readable-stream: 3.6.2 - safe-stable-stringify: 2.5.0 - stack-trace: 0.0.10 - triple-beam: 1.4.1 - winston-transport: 4.7.1 - - workerpool@6.5.1: {} - - wrap-ansi@7.0.0: - dependencies: - ansi-styles: 4.3.0 - string-width: 4.2.3 - strip-ansi: 6.0.1 - - wrap-ansi@8.1.0: - dependencies: - ansi-styles: 6.2.1 - string-width: 5.1.2 - strip-ansi: 7.1.0 - - wrappy@1.0.2: {} - - ws@8.18.0: {} - - xml2js@0.6.2: - dependencies: - sax: 1.2.1 - xmlbuilder: 11.0.1 - - xmlbuilder@11.0.1: {} - - xtend@4.0.2: {} - - y18n@5.0.8: {} - - yargs-parser@20.2.9: {} - - yargs-parser@21.1.1: {} - - yargs-unparser@2.0.0: - dependencies: - camelcase: 6.3.0 - decamelize: 4.0.0 - flat: 5.0.2 - is-plain-obj: 2.1.0 - - yargs@16.2.0: - dependencies: - cliui: 7.0.4 - escalade: 3.2.0 - get-caller-file: 2.0.5 - require-directory: 2.1.1 - string-width: 4.2.3 - y18n: 5.0.8 - yargs-parser: 20.2.9 - - yargs@17.7.2: - dependencies: - cliui: 8.0.1 - escalade: 3.2.0 - get-caller-file: 2.0.5 - require-directory: 2.1.1 - string-width: 4.2.3 - y18n: 5.0.8 - yargs-parser: 21.1.1 - - yn@3.1.1: {} - - yocto-queue@0.1.0: {} - - yoctocolors@2.1.1: {} - - youtube-dl-wrap@https://codeload.github.com/insanity54/youtube-dl-wrap/tar.gz/b47e9388063b4ef516624112ef59b7d11f7775d9: {} diff --git a/services/capture/src/Capture.js b/services/capture/src/Capture.js deleted file mode 100644 index c0f9c80..0000000 --- a/services/capture/src/Capture.js +++ /dev/null @@ -1,121 +0,0 @@ - -import Voddo from './Voddo.js' - - -export default class Capture { - - constructor(opts) { - this.date = opts?.date - this.sql = opts.sql - this.ipfs = opts.ipfs - this.idleTimeout = opts.idleTimeout || 1000*60*15 - this.video = opts.video - this.voddo = opts.voddo - this.workerId = opts.workerId - return this - } - - - /** - * upload VOD to ipfs - * - * @return {Promise} - * @resolves {String} cid - */ - async upload (filename) { - const cid = await this.ipfs.upload(filename) - return cid - } - - - - /** - * save Vod data to db - */ - async save (cid, timestamp) { - console.log({ level: 'debug', message: `saving ${cid} \n w/ captureDate ${timestamp}` }) - this.date = timestamp - return await this.sql`INSERT INTO vod ( "videoSrcHash", "captureDate" ) values (${cid}, ${timestamp}) returning *` - } - - - /** - * advertise the vod segment(s) we captured. - * futureporn/commander uses this data to elect one worker to upload the VOD - */ - async advertise () { - const segments = await this.voddo.getRecordedSegments() - const streams = Voddo.groupStreamSegments(segments) - const workerId = this.workerId - logger.log({ level: 'debug', message: `Advertising our VOD streams(s) ${JSON.stringify({segments, streams, workerId})}` }) - this.sql.notify('capture/vod/advertisement', JSON.stringify({segments, streams, workerId})) - } - - - listen () { - this.sql.listen('scout/stream/stop', (data) => { - logger.log({ level: 'debug', message: 'Scout said the stream has stopped. I will advertise the vod segment(s) I have.' }) - this.advertise() - }) - - this.sql.listen('commander/vod/election', async (data) => { - if (data.workerId === this.workerId) { - logger.log({ level: 'debug', message: 'Commander elected me to process/upload' }) - this.process(await this.voddo.getFilenames()) - } else { - logger.log({ level: 'debug', message: `Commander elected ${data.workerId} to process/upload their vod segment(s)` }) - } - }) - - return this - } - - - /** - * process video(s) after end of stream - * - * @param {String[]} filenames - * @returns {void} - */ - async process (filenames) { - this.date = filenames[0].timestamp - - logger.log({ level: 'debug', message: 'concatenation in progress...' }) - const file = await this.video.concat(filenames) - - logger.log({ level: 'debug', message: `uploading ${file}` }) - const cid = await this.ipfs.upload(file) - - logger.log({ level: 'debug', message: 'db save in progress' }) - await this.save(cid, this.date) - - } - - - - /** - * download a livestream - * - * - initializes Voddo - * - invokes this.process() as side effect - * - * @return {void} - */ - async download () { - this.voddo.on('start', (data) => { - logger.log({ level: 'debug', message: 'voddo started' }) - logger.log({ level: 'debug', message: data }) - this.sql.notify('capture/file', JSON.stringify(data)) - }) - this.voddo.on('stop', (report) => { - logger.log({ level: 'debug', message: `Got a stop event from Voddo` }) - }) - logger.log({ level: 'debug', message: 'starting voddo' }) - this.voddo.start() - } - -} - - - - diff --git a/services/capture/src/Capture.spec.js b/services/capture/src/Capture.spec.js deleted file mode 100644 index 6b71bfa..0000000 --- a/services/capture/src/Capture.spec.js +++ /dev/null @@ -1,147 +0,0 @@ - -import Video from './Video.js' -import Capture from './Capture.js' -import Ipfs from './Ipfs.js' -import chai, { expect } from 'chai' -import { dirname } from 'path'; -import { fileURLToPath } from 'url'; -import path from 'node:path' -import sinon from 'sinon' -import sinonChai from 'sinon-chai' -import { CID } from 'multiformats/cid' -import Voddo from './Voddo.js' -import EventEmitter from 'node:events' -import postgres from 'postgres' - -chai.use(sinonChai) - -const Timer = setTimeout(()=>{},0).constructor -const fixtureDate = 1581117660000 -const cidFixture = 'bafybeid3mg5lzrvnmpfi5ftwhiupp7i5bgkmdo7dnlwrvklbv33telrrry' -const __dirname = dirname(fileURLToPath(import.meta.url)); - -describe('Capture', function () { - - let clock - - const sandbox = sinon.createSandbox() - - beforeEach(() => { - - clock = sandbox.useFakeTimers({ - toFake: ["setTimeout", "setInterval"], - shouldAdvanceTime: false - }); - - // // const sql = postgres({ - // // idle_timeout: 1 - // // }) - - // let pgStub = (opts) => { - // let sql = (args) => {} - // return sql - // } - const sqlRaw = postgres() - const sql = sandbox.stub(sqlRaw) - // sql.listen.resolves(fixtureDate) - // sql.notify.resolves(92834) - // sinon.stub(postgres, 'notify') - // sinon.createStubInstance(postgres) - // sql - // .withArgs('INSERT INTO vod ( videoSrcHash, captureDate ) values (bafybeid3mg5lzrvnmpfi5ftwhiupp7i5bgkmdo7dnlwrvklbv33telrrry, 1581117660000) returning *') - // .resolves({ msg: 'idk' }) - // sinon.stub(sql, 'notify').returns() - - - - - // const ipfs = sandbox.createStubInstance(Ipfs) - // ipfs.upload.withArgs('/tmp/mycoolfile.mp4').resolves(cidFixture) - // capture = new Capture({ - // sql, - // ipfs, - // video, - // voddo - // }) - // sandbox.stub(capture, 'process').resolves() - }) - - afterEach(() => { - sandbox.restore() - clock.restore() - }) - - - - describe('upload', function () { - it('should upload a video to ipfs', async function () { - - const sqlRaw = postgres() - const sql = sandbox.stub(sqlRaw) - - const video = sandbox.stub() - const voddo = sandbox.createStubInstance(Voddo) - voddo.on.callThrough() - voddo.emit.callThrough() - voddo.listeners.callThrough() - voddo.listenerCount.callThrough() - - - voddo.start.callsFake(() => { - voddo.emit('start', { file: '/tmp/burrito.mp4', timestamp: 1 }) - }) - - const ipfs = sandbox.createStubInstance(Ipfs) - ipfs.upload.withArgs('/tmp/mycoolfile.mp4').resolves(cidFixture) - const capture = new Capture({ - sql, - ipfs, - video, - voddo - }) - - const cid = await capture.upload('/tmp/mycoolfile.mp4') - expect(() => CID.parse(cid), `The IPFS CID '${cid}' is invalid.`).to.not.throw() - expect(capture.ipfs.upload).calledOnce - }) - }) - describe('save', function () { - it('should save to db', async function () { - - const sqlRaw = postgres() - const sql = sandbox.stub(sqlRaw) - - const video = sandbox.stub() - const voddo = sandbox.createStubInstance(Voddo) - voddo.on.callThrough() - voddo.emit.callThrough() - voddo.listeners.callThrough() - voddo.listenerCount.callThrough() - - - voddo.start.callsFake(() => { - voddo.emit('start', { file: '/tmp/burrito.mp4', timestamp: 1 }) - }) - - const ipfs = sandbox.createStubInstance(Ipfs) - ipfs.upload.withArgs('/tmp/mycoolfile.mp4').resolves(cidFixture) - const capture = new Capture({ - sql, - ipfs, - video, - voddo - }) - - // I can't stub sql`` because of that template string override so i'm just stubbing capture.save - // I think this is an evergreen test ¯\_(ツ)_/¯ - sandbox.stub(capture, 'save').resolves([ - { id: 1, cid: cidFixture, captureDate: fixtureDate } - ]) - const vod = await capture.save(cidFixture, fixtureDate) - }) - }) - - - - -}) \ No newline at end of file diff --git a/services/capture/src/Ipfs.js b/services/capture/src/Ipfs.js deleted file mode 100644 index 7d20638..0000000 --- a/services/capture/src/Ipfs.js +++ /dev/null @@ -1,57 +0,0 @@ - -import {execa} from 'execa' -import {loggerFactory} from 'common/logger' - -const logger = loggerFactory({ - service: 'futureporn/capture' -}) - -export default class Ipfs { - constructor(opts) { - this.multiaddr = opts?.IPFS_CLUSTER_HTTP_API_MULTIADDR - this.username = opts?.IPFS_CLUSTER_HTTP_API_USERNAME - this.password = opts?.IPFS_CLUSTER_HTTP_API_PASSWORD - this.ctlExecutable = opts?.ctlExecutable || '/usr/local/bin/ipfs-cluster-ctl' - this.ipfsExecutable = opts?.ipfsExecutable || '/usr/local/bin/ipfs' - } - getArgs () { - let args = [ - '--no-check-certificate', - '--host', this.multiaddr, - '--basic-auth', `${this.username}:${this.password}` - ] - return args - } - async upload (filename, expiryDuration = false) { - try { - let args = getArgs() - - args = args.concat([ - 'add', - '--quieter', - '--cid-version', 1 - ]) - - if (expiryDuration) { - args = args.concat(['--expire-in', expiryDuration]) - } - - args.push(filename) - - const { stdout } = await execa(this.ctlExecutable, args) - return stdout - } catch (e) { - logger.log({ level: 'error', message: 'Error while adding file to ipfs' }) - logger.log({ level: 'error', message: e }) - } - } - async hash (filename) { - try { - const { stdout } = await execa(this.ipfsExecutable, ['add', '--quiet', '--cid-version=1', '--only-hash', filename]) - return stdout - } catch (e) { - logger.log({ level: 'error', message: 'Error while hashing file' }) - logger.log({ level: 'error', message: e }) - } - } -} \ No newline at end of file diff --git a/services/capture/src/Record.spec.ts b/services/capture/src/Record.spec.ts deleted file mode 100644 index dc0aab5..0000000 --- a/services/capture/src/Record.spec.ts +++ /dev/null @@ -1,97 +0,0 @@ -import Record, { UploadStreamClosedError } from "./Record.js" -import * as chai from 'chai' -import { ChildProcess, spawn } from "child_process" -import { createReadStream, readFileSync, ReadStream } from "fs" -import AWSMock from 'aws-sdk-mock' -import sinon from 'sinon'; -import sinonChai from 'sinon-chai'; -import { PutObjectCommand, PutObjectCommandInput, S3Client, CreateMultipartUploadCommand, UploadPartCommand, UploadPartCommandInput } from "@aws-sdk/client-s3" -import { join, dirname } from "path" -import { fileURLToPath } from "url" -import { S3 } from 'aws-sdk'; -import { HeadObjectOutput } from 'aws-sdk/clients/s3'; -import { Readable } from 'stream'; -import { mockClient } from 'aws-sdk-client-mock'; -import { sdkStreamMixin } from '@smithy/util-stream' -import chaiAsPromised from 'chai-as-promised' -chai.use(chaiAsPromised) -const expect = chai.expect - -// "pay no attention to that man behind the curtain" - -// export function getObjectReadStream(s3Client: S3, Bucket: string, Key: string): Readable { -// return s3Client.getObject({ Bucket, Key }).createReadStream(); -// } - -// export async function waitForObjectExists( -// s3Client: S3Client, -// Bucket: string, -// Key: string -// ): Promise { -// return s3Client.waitFor('objectExists', { Bucket, Key }).promise(); -// } -const __dirname = dirname(fileURLToPath(import.meta.url)); - - - -const testStreamDir = '/tmp/record-test-stream.m3u8' - -describe('Record', function () { - // mocking @aws-sdk/lib-storage Upload() has some nuances. - // @see https://github.com/m-radzikowski/aws-sdk-client-mock?tab=readme-ov-file#lib-storage-upload - - - it('should accept a {ReadStream} as input', async function () { - const inputStream = createReadStream(join(__dirname, './fixtures/mock-stream0.mp4')) // 192627 bytes - const s3ClientMock = mockClient(S3Client) - s3ClientMock.on(CreateMultipartUploadCommand).resolves({UploadId: '1'}); - s3ClientMock.on(UploadPartCommand).resolves({ETag: '1'}); - const s3Client = new S3Client({ region: 'us-west-000' }) - const jobId = 'test-job-1234' - const record = new Record({ inputStream, s3Client, bucket: 'test', jobId }) - await record.start() - expect(record).to.have.property('counter', 192627) - expect(record).to.have.property('bucket', 'test') - }) - - xit('should be abortable', async function () { - const inputStream = createReadStream(join(__dirname, './fixtures/mock-stream0.mp4')) // 192627 bytes - const s3ClientMock = mockClient(S3Client) - const s3Client = new S3Client({ region: 'us-west-000' }) - s3ClientMock.on(CreateMultipartUploadCommand).resolves({UploadId: '1'}); - s3ClientMock.on(UploadPartCommand).resolves({ETag: '1'}); - const jobId = 'test-job-3456' - const record = new Record({ inputStream, s3Client, jobId, bucket: 'test' }) - await record.start() - expect(record).to.have.property('abortController') - await record.abort() - }) - - xit('should throw if the upload stream closes before the download stream closes', async function () { - - const s3Mock = mockClient(S3Client) - // const inputStream = createReadStream(join(__dirname, './fixtures/mock-stream0.mp4')) - const inputStream = createReadStream('/dev/random') // forever random - // const s3Client = new S3Client({ region: 'us-west-000' }) - // s3ClientMock.on() - s3Mock.on(PutObjectCommand).resolvesOnce({}).resolvesOnce({}).rejects({}) - const s3 = new S3Client({ region: 'us-west-000' }) - - return expect(s3.send(new PutObjectCommand({ Body: inputStream, Bucket: 'taco', Key: 'my-cool-taco.mp4' }))).to.be.rejectedWith(UploadStreamClosedError) - - }) - - xit('should restart if a EPIPE is encountered', async function () { - // @todo IDK how to implement this. - const inputStream = createReadStream(join(__dirname, './fixtures/mock-stream0.mp4')) - const s3ClientMock = mockClient(S3Client) - s3ClientMock.on(CreateMultipartUploadCommand).resolves({UploadId: '1'}) - s3ClientMock.on(UploadPartCommand).rejectsOnce({cause: 'simulated network interruption'}).resolves({ ETag: '1' }) // this rejection is probably not specific enough to simulate EPIPE - const s3Client = new S3Client({ region: 'us-west-000' }) - const record = new Record({ inputStream, s3Client, channel: 'coolguy_69', bucket: 'test' }) - await record.start() - expect(record).to.have.property('counter', 192627) - }) - - -}) \ No newline at end of file diff --git a/services/capture/src/Record.ts b/services/capture/src/Record.ts deleted file mode 100644 index b6df091..0000000 --- a/services/capture/src/Record.ts +++ /dev/null @@ -1,269 +0,0 @@ -import { spawn } from 'child_process'; -import { PassThrough, pipeline, Readable } from 'stream'; -import EventEmitter from 'events'; -import { Upload } from "@aws-sdk/lib-storage"; -import { CompleteMultipartUploadCommand, S3Client } from "@aws-sdk/client-s3"; -import 'dotenv/config' -import { promisify } from 'util'; - -// @see https://nodejs.org/api/events.html#capture-rejections-of-promises -EventEmitter.captureRejections = true; - -const pipelinePromise = promisify(pipeline) -const ua0 = 'Mozilla/5.0 (X11; Linux x86_64; rv:105.0) Gecko/20100101 Firefox/105.0' - -export class UploadStreamClosedError extends Error { - constructor(message: string) { - super(message) - Object.setPrototypeOf(this, UploadStreamClosedError.prototype) - } -} - -export interface RecordArgs { - filename?: string; - s3Client: S3Client; - bucket: string; - date?: string; - inputStream: Readable; - jobId: string; - abortSignal: AbortSignal; - onProgress: (fileSize: number) => void; -} - -interface MakeS3ClientOptions { - accessKeyId: string; - secretAccessKey: string; - region: string; - endpoint: string -} - -interface getFFmpegOptions { - url: string; -} - -export default class Record { - private s3Client: S3Client; - private uploadStream: PassThrough; - inputStream: Readable; - counter: number; - bucket: string; - keyName: string; - datestamp: string; - filename?: string; - jobId: string; - date?: string; - abortSignal: AbortSignal; - onProgress: Function; - upload?: Upload; - - constructor({ inputStream, s3Client, bucket, jobId, abortSignal, onProgress }: RecordArgs) { - if (!inputStream) throw new Error('Record constructor was missing inputStream.'); - if (!bucket) throw new Error('Record constructor was missing bucket.'); - if (!jobId) throw new Error('Record constructer was missing jobId!'); - if (!s3Client) throw new Error('Record constructer was missing s3Client'); - if (!abortSignal) throw new Error('Record constructer was missing abortSignal'); - this.inputStream = inputStream - this.onProgress = onProgress - this.s3Client = s3Client - this.bucket = bucket - this.jobId = jobId - this.counter = 0 - this.datestamp = new Date().toISOString() - this.keyName = `${this.datestamp}-${jobId}.ts` - this.uploadStream = new PassThrough() - this.abortSignal = abortSignal - this.abortSignal.addEventListener("abort", this.abortEventListener.bind(this)) - this.upload - } - - - - static makeS3Client({ - accessKeyId, - secretAccessKey, - region, - endpoint - }: MakeS3ClientOptions): S3Client { - const client = new S3Client({ - endpoint, - region, - credentials: { - accessKeyId, - secretAccessKey - } - }) - return client - } - - static getFFmpegStream({ url }: getFFmpegOptions): Readable { - console.log(`getFFmpegStream using url=${url}`) - const ffmpegProc = spawn('ffmpeg', [ - '-headers', `"User-Agent: ${ua0}"`, - '-i', url, - '-c:v', 'copy', - '-c:a', 'copy', - '-movflags', 'faststart', - '-y', - '-f', 'mpegts', - '-loglevel', 'quiet', - 'pipe:1' - ], { - // ignoring stderr is important because if not, ffmpeg will fill that buffer and node will hang - stdio: ['pipe', 'pipe', 'ignore'] - }) - return ffmpegProc.stdout - } - - - - abortEventListener() { - console.log(`abortEventListener has been invoked. this.abortSignal is as follows`) - console.log(this.abortSignal) - console.log(JSON.stringify(this.abortSignal, null, 2)) - const reason = this.abortSignal.reason - console.log(`aborted the stream download with reason=${reason}`) - this.inputStream.destroy(new Error(reason)) - } - - async uploadToS3() { - const target = { - Bucket: this.bucket, - Key: this.keyName, - Body: this.uploadStream - } - - // greets https://stackoverflow.com/a/70159394/1004931 - try { - this.upload = new Upload({ - client: this.s3Client, - partSize: 1024 * 1024 * 5, - queueSize: 1, - leavePartsOnError: false, - params: target, - }); - - this.upload.on("httpUploadProgress", (progress) => { - if (progress?.loaded) { - // console.log(progress) - if (this.onProgress) this.onProgress(this.counter); - // console.log(`uploaded ${progress.loaded} bytes (${prettyBytes(progress.loaded)})`); - } else { - console.log(`httpUploadProgress ${JSON.stringify(progress, null, 2)}`) - } - }); - - console.log(`Uploading to bucket=${this.bucket}. Waiting for parallelUploads3 to finish...`) - // @todo there is a problem that results in COMPLETE LOSS OF SEGMENT DATA. - // when the download stream closes before the upload stream, I think the upload stream gets cut off. - // this means the upload isn't allowed to save and that means no data whatsoever gets put to S3. - // is that right? IDK what's happening, but we don't get any segment data on S3 at all?? - // Ok I just checked the Backblaze dashboard and we are uploading. Backblaze says the bytes are at 0 but - // it also shows a partial upload of 550MB which matches what capture-worker is showing has been captured so far. - // So I think what is happening is the upload is happening, but it's not finishing. - // It looks like the finish is only allowed to happen under completely normal circumstances. - // However, the segment upload may fail in production, and we need to let the upload finish even then. - // - // I think I need to call CompleteMultipartUpload. https://docs.aws.amazon.com/AmazonS3/latest/API/API_CompleteMultipartUpload.html - // Yes, that's right. Apparently parallelUploads3.done() returns a Promise which will resolve to CompleteMultipartUploadCommandOutput. - // But because of the catch, that promise will never resolve? - // What happens to an in-progress Promise when an error is thrown? - - // await this.upload.done(); - // console.log('Upload is complete.') - // return this.uploadStream - - - } catch (e) { - // if we got an abort error, e.name is not AbortError as expected. Instead, e.name is Error. - // so in order to catch AbortError, we don't even look there. instead, we check if our abortcontroller was aborted. - // in other words, `(e.name === 'AbortError')` will never be true. - if (this.abortSignal.aborted) { - console.error('While uploading, the upload was aborted.') - setTimeout(async () => { - await this.upload?.abort() - }, 1000) - // if (this.upload) { - // const command = new CompleteMultipartUploadCommand() - // } - return; - } - - if (e instanceof Error) { - console.error(`We were uploading a file to S3 but then we encountered an exception!`) - console.error(e) - throw e - } else { - throw new Error(`error of some sort ${JSON.stringify(e, null, 2)}`) - } - } - } - - - - async start() { - - - - // streams setup - - this.uploadStream.on('data', (data) => { - this.counter += data.length - }) - this.uploadStream.on('close', () => { - // if uploadStream closes before inputStream, throw an error. - if (!this.inputStream.closed) { - const msg = 'upload stream closed before download stream, which suggests the S3 upload failed.' - console.error(msg) - throw new UploadStreamClosedError(msg); - } else { - console.log('upload stream has closed. In this instance it is OK since the input stream is also closed.') - } - }) - this.uploadStream.on('error', (e) => { - console.error('there was an error on the uploadStream. error as follows') - console.error(e) - }) - - // input stream event handlers - this.inputStream.on('close', () => { - console.log('[!!!] input stream has closed.') - }) - this.inputStream.on('error', (e) => { - console.error('there was an error on the inputStream. error as follows') - console.error(e) - }) - this.inputStream.on('drain', () => { - console.info('[vvv] drain on inputStream.') - }) - - - // pipe the ffmpeg stream to the S3 upload stream - // this has the effect of uploading the stream to S3 at the same time we're recording it. - console.log(`>>> awaiting pipeline promise`) - const streamPromise = pipelinePromise( - this.inputStream, - this.uploadStream - ) - - this.uploadToS3() - await Promise.all([streamPromise, this.upload?.done()]) - - // console.log('awaiting uploadToS3()...') - // await this.uploadToS3() - // console.log('uploadToS3() is complete.') - - console.log(`streamPromise completed with jobId=${this.jobId}, keyName=${this.keyName}`) - - return { - jobId: this.jobId, - keyName: this.keyName - } - } - - async stop() { - throw new Error('@todo please implement') - } -} - - - diff --git a/services/capture/src/RecordNextGeneration.spec.ts b/services/capture/src/RecordNextGeneration.spec.ts deleted file mode 100644 index 8aed063..0000000 --- a/services/capture/src/RecordNextGeneration.spec.ts +++ /dev/null @@ -1,15 +0,0 @@ -import RecordNextGeneration from './RecordNextGeneration.ts' -import getVod from '@futureporn/fetchers/getVod.ts' -import createVod from '@futureporn/fetchers/createVod.ts' - -describe('RecordNextGeneration', function () { - describe('integration', function () { - it('should stream to S3', async function () { - this.timeout(30000) - const cv = await createVod() - if (!cv) throw new Error('failed to get vod from createVod()'); - const recordNG = new RecordNextGeneration({ url: 'https://futureporn-b2.b-cdn.net/projektmelody-chaturbate-2024-06-15.mp4', vodId: cv.id }) - const uv = await getVod(cv.id) - }) - }) -}) \ No newline at end of file diff --git a/services/capture/src/RecordNextGeneration.ts b/services/capture/src/RecordNextGeneration.ts deleted file mode 100644 index d9380bb..0000000 --- a/services/capture/src/RecordNextGeneration.ts +++ /dev/null @@ -1,570 +0,0 @@ -/** - * RecordNextGeneration.ts - */ - -import { VodResponse } from "@futureporn/types" -import { PassThrough, Readable, type Writable } from "stream" -import { join } from 'node:path' -import { ua0 } from '@futureporn/utils/name.ts' -import { ChildProcessByStdio, spawn } from 'child_process' -import { pipeline } from "stream/promises" -import { configs } from "./config" -import { nanoid } from 'nanoid' -import { Upload, type Progress } from "@aws-sdk/lib-storage" -import { S3Client } from '@aws-sdk/client-s3' -import prettyBytes from "pretty-bytes" -import updateSegmentInDatabase from "@futureporn/fetchers/updateSegmentInDatabase.ts" -import { getRecordingRelatedToVod } from "@futureporn/fetchers/getRecording.ts" -import createSegment from "@futureporn/fetchers/createSegment.ts" -import { createReadStream, createWriteStream } from "fs" -import pRetry from 'p-retry' -import { type SegmentResponse } from '@futureporn/types' -import getPlaylistUrl from "@futureporn/fetchers/getPlaylistUrl.ts" -import { isBefore, isAfter, sub } from 'date-fns' -import { setTimeout } from 'node:timers/promises'; -import { - RoomOfflineError, - PlaylistFailedError, - AdminAbortedError, - ExhaustedRetriesError, -} from '@futureporn/utils/error.ts' - -export interface RecordNextGenerationArguments { - vodId: string; - url: string; -} - - - - - -/** - * # RecordNextGeneration - * - * The function which records VODs in a Futureporn specific, fault-tolerant way. - * - * ## Issues/TODO list - * - * @done [x] onProgress stops firing - * @done [x] OOMKilled seen via development environment - * @todo [ ] undefined behavior during CB private shows - * @todo [ ] does not handle CB Hidden Camera ticket shows - * @done [x] Upload segments in a way that does not interrupt downloading new segments. - * There is an issue where a segment download ends, and the segment upload immediately begins. - * At first glance this looks like good behavior, but what is happening during the segment upload is that the livestream - * is continuing, but we aren't recording it anymore. We are using Backblaze, thus uploads are slow. - * We miss a lot of the stream because the upload takes many minutes. - * Instead of this behavior of immediately uploading after a segment download completes, we should upload once the livestream is finished, - * OR we should upload while concurrently downloading the next segment. - * @done [x] Move retrying from the {Task} `record` context to the class `RecordNextGeneration` context. - * There is an issue where the `record` task needs to retry after a temporary failure, but it cannot because there aren't any available workers. - * The solution is to not exit the `record` task at all, and instead keep the `record` task running, but suspended while a exponential backoff timer elapses. - * This way, the worker stays focused on the recording and retries until the stream has been offline for n minutes, at which point `record` is complete. - * @done [x] Abort process results in corrupted .ts files that ffmpeg/vlc/kdenlive cannot read. Granted, aborted vods are those which are NOT desirable to save, so maybe we ignore this issue? - * @done [x] RecordNextGeneration gives up immediately in response to RoomOffline. It must retry until 5 minutes have elapsed. - * @done [x] .bytes and .bytes_uploaded do not match at the end of the upload. .bytes_uploaded is curiously larger than .bytes! - * @todo [ ] Two jobs can occur concurrently (technically env var WORKER_CONCURRENCY allows for >1 so this isn't a bug) - * I set it to 1. Now we just need to test that only 1 job can happen at a time. - * - * ``` - * updateSegmentBytes() Segment 0 -- segment.id=2bf39450-2911-4f26-aca6-41cf092fd5e6 .bytes=6017952380 .bytes_uploaded=4571791360 .s3_key=MufqGsVF2hY6sAtN5rmrX.ts .vod.id=cbc80caf-73e5-41e5-8bbb-2d03ca234aca -updateSegmentBytes() Segment 0 -- segment.id=2bf39450-2911-4f26-aca6-41cf092fd5e6 .bytes=6017952380 .bytes_uploaded=4592762880 .s3_key=MufqGsVF2hY6sAtN5rmrX.ts .vod.id=cbc80caf-73e5-41e5-8bbb-2d03ca234aca -updateSegmentBytes() Segment 0 -- segment.id=2bf39450-2911-4f26-aca6-41cf092fd5e6 .bytes=6017952380 .bytes_uploaded=4613734400 .s3_key=MufqGsVF2hY6sAtN5rmrX.ts .vod.id=cbc80caf-73e5-41e5-8bbb-2d03ca234aca -updateSegmentBytes() Segment 0 -- segment.id=2bf39450-2911-4f26-aca6-41cf092fd5e6 .bytes=6017952380 .bytes_uploaded=4624220160 .s3_key=MufqGsVF2hY6sAtN5rmrX.ts .vod.id=cbc80caf-73e5-41e5-8bbb-2d03ca234aca -updateSegmentBytes() Segment 0 -- segment.id=2bf39450-2911-4f26-aca6-41cf092fd5e6 .bytes=6017952380 .bytes_uploaded=4645191680 .s3_key=MufqGsVF2hY6sAtN5rmrX.ts .vod.id=cbc80caf-73e5-41e5-8bbb-2d03ca234aca -during startProgressReports(), we encountered the following error. -TypeError: fetch failed - at node:internal/deps/undici/undici:13178:13 - at process.processTicksAndRejections (node:internal/process/task_queues:95:5) - at async updateSegmentInDatabase (/app/packages/fetchers/src/updateSegmentInDatabase.ts:29:15) - at async RecordNextGeneration.updateSegmentBytes (/app/services/capture/src/RecordNextGeneration.ts:342:21) - at async RecordNextGeneration.updateDatabaseRecords (/app/services/capture/src/RecordNextGeneration.ts:326:5) - at async Timeout._onTimeout (/app/services/capture/src/RecordNextGeneration.ts:352:9) { - [cause]: Error: getaddrinfo EAI_AGAIN postgrest.futureporn.svc.cluster.local - at GetAddrInfoReqWrap.onlookupall [as oncomplete] (node:dns:120:26) { - errno: -3001, - code: 'EAI_AGAIN', - syscall: 'getaddrinfo', - hostname: 'postgrest.futureporn.svc.cluster.local' - } -} -``` - * - * - */ -export default class RecordNextGeneration { - - public vodId: string; - public url: string; - public s3Key?: string; - public s3Bucket?: string; - public s3Client?: S3Client; - public tmpDiskPath?: string; - private vod?: VodResponse | null; - private downloadStream?: Readable; - private downloadProcess?: ChildProcessByStdio; - private uploadStream?: PassThrough; - private uploadInstance?: Upload; - private diskStream?: Writable; - private uploadCounter: number; - private downloadCounter: number; - private databaseUpdateTimer?: NodeJS.Timeout; - private updateTimeout: number; - private abortController: AbortController; - private segments: SegmentResponse[]; - private retries: number; - - - - constructor({ vodId, url }: RecordNextGenerationArguments) { - this.vodId = vodId - this.url = url - this.uploadCounter = 0 - this.downloadCounter = 0 - this.updateTimeout = 30*1000 - this.abortController = new AbortController() - this.abortController.signal.addEventListener("abort", this.abortEventListener.bind(this)) - this.retries = 0 - this.segments = [] - - } - - async withRetry(fn: any, retries = 3) { - return pRetry(fn, { - onFailedAttempt: (e) => { - console.error(`Error during attempt:`, e); - }, - retries - }); - } - - abortEventListener() { - console.log(`abortEventListener has been invoked. this.abortSignal is as follows`) - console.log(this.abortController.signal) - // console.log(JSON.stringify(this.abortController.signal, null, 2)) - // const reason = this.abortController.signal.reason - if (this.downloadProcess) { - // console.log(`aborted the stream process with reason=${reason}`) - // we want to send SIGINT to ffmpeg rather than forcefully destroying the stream. - // this prevents the downloaded .ts file from being corrupted. - // this.downloadStream.destroy(new AdminAbortedError()) - this.downloadProcess.kill('SIGINT'); - if (this.downloadStream) { - this.downloadStream.emit('error', new AdminAbortedError()); - } - } else { - console.warn(`downloadProcess does not exist. Perhaps it has already been aborted?`) - } - } - - - // getMultipartUpload({ - // client, - // bucket, - // key, - // body, - // }: { - // client: S3Client, - // bucket: string, - // key: string, - // body: Readable, - // }) { - // const params = { - // Bucket: bucket, - // Key: key, - // Body: body - // } - // const upload = new Upload({ - // client, - // partSize: 1024 * 1024 * 5, - // queueSize: 1, - // // @see https://github.com/aws/aws-sdk-js-v3/issues/2311 - // // tl;dr: the variable name, 'leavePartsOnError' is not representative of the behavior. - // // It should instead be interpreted as, 'throwOnPartsError' - // leavePartsOnError: true, - // params - // }) - - // /** - // * aws client docs recommend against using async onProgress handlers. - // * therefore, I'm only setting this.uploadCounter inside the syncronous handler and we call async updateSegmentInDatabase() elsewhere. - // */ - // const onProgress = (progress: Progress) => { - // if (progress?.loaded) { - // console.log(`Upload progress! ${progress.loaded} bytes loaded (${prettyBytes(progress.loaded)}).`) - // this.reportMemoryUsage() - // this.uploadCounter = progress.loaded - // } - // } - // upload.on("httpUploadProgress", onProgress); - - // return upload - // } - - - // @todo there is a problem that results in COMPLETE LOSS OF SEGMENT DATA. - // when the download stream closes before the upload stream, I think the upload stream gets cut off. - // this means the upload isn't allowed to save and that means no data whatsoever gets put to S3. - // is that right? IDK what's happening, but we don't get any segment data on S3 at all?? - // Ok I just checked the Backblaze dashboard and we are uploading. Backblaze says the bytes are at 0 but - // it also shows a partial upload of 550MB which matches what capture-worker is showing has been captured so far. - // So I think what is happening is the upload is happening, but it's not finishing. - // It looks like the finish is only allowed to happen under completely normal circumstances. - // However, the segment upload may fail in production, and we need to let the upload finish even then. - // - // I think I need to call CompleteMultipartUpload. https://docs.aws.amazon.com/AmazonS3/latest/API/API_CompleteMultipartUpload.html - // Yes, that's right. Apparently parallelUploads3.done() returns a Promise which will resolve to CompleteMultipartUploadCommandOutput. - // But because of the catch, that promise will never resolve? - // What happens to an in-progress Promise when an error is thrown? - // - // maybe @see https://github.com/aws/aws-sdk-js-v3/issues/2694 - - - static getDiskStream(s3Key?: string) { - const tmpDiskPath = join(configs.cacheDir, s3Key || `${nanoid()}.ts`) - console.log(`getDiskStream() tmpDiskPath=${tmpDiskPath}`) - return createWriteStream(tmpDiskPath, { encoding: 'utf-8' }) - } - - static getFFmpeg({ playlistUrl }: { playlistUrl: string }): ChildProcessByStdio { - console.log(`getFFmpegStream using playlistUrl=${playlistUrl}`) - const ffmpegProc = spawn('ffmpeg', [ - '-headers', `"User-Agent: ${ua0}"`, - '-i', playlistUrl, - '-c:v', 'copy', - '-c:a', 'copy', - '-movflags', 'faststart', - '-y', - '-f', 'mpegts', - '-loglevel', 'quiet', - 'pipe:1' - ], { - // ignoring stderr is important because if not, ffmpeg will fill that buffer and node will hang - stdio: ['pipe', 'pipe', 'ignore'] - }) - return ffmpegProc - } - - - - onUploadProgress(segment: SegmentResponse, progress: Progress) { - // console.log('onUploadProgress() running now. progress as follows') - // console.log(progress) - // find the matching segment and update it's bytes_uploaded - const matchingSegment = this.segments.find((s) => s.id === segment.id) - if (progress?.loaded && matchingSegment) { - matchingSegment.bytes_uploaded = progress.loaded - this.reportMemoryUsage() - } - } - - - formatMemoryStats(stats: NodeJS.MemoryUsage): Record { - const formattedStats: Record = {}; - for (const key of Object.keys(stats) as Array) { - const value = stats[key]; - formattedStats[key] = `${value} (${prettyBytes(value)})`; - } - return formattedStats; - } - - reportMemoryUsage() { - const mem = process.memoryUsage() - if (mem.rss > 256000000) console.warn(`High memory usage! ${JSON.stringify(this.formatMemoryStats(mem))}`); - } - - - getS3Client() { - const clientOptions: any = { - endpoint: configs.s3Endpoint, - region: configs.s3Region, - credentials: { - accessKeyId: configs.s3AccessKeyId, - secretAccessKey: configs.s3SecretAccessKey, - }, - } - this.s3Client = new S3Client(clientOptions) - this.s3Bucket = configs.s3UscBucket - - - if (!this.s3Client) throw new Error('getS3Client() failed to get S3Client'); - if (!this.s3Bucket) throw new Error('getS3Client() failed to get s3Bucket'); - - return this.s3Client - } - - getNames() { - const s3Key = `${nanoid()}.ts` - const tmpDiskPath = join(configs.cacheDir, s3Key) - console.log(`tmpDiskPath=${tmpDiskPath}`) - return { tmpDiskPath, s3Key } - } - - - static async _dl(url: string, s3Key: string) { - const { data, error } = (await getPlaylistUrl(url)) - if (!data) throw new PlaylistFailedError(); - if (error) { - if (error === 'PlaylistFailedError') throw new PlaylistFailedError(); - if (error === 'RoomOfflineError') throw new RoomOfflineError(); - } - console.log(`_dl playlistUrl=${data.playlistUrl}`) - const ffmpegProcess = RecordNextGeneration.getFFmpeg({ playlistUrl: data.playlistUrl }) - const ffmpegStream = ffmpegProcess.stdout - const diskStream = RecordNextGeneration.getDiskStream(s3Key) - const streamPipeline = pipeline(ffmpegStream, diskStream) - return { - pipeline: streamPipeline, - ffmpegStream, - ffmpegProcess, - diskStream, - } - } - - static async _ul(client: S3Client, diskPath: string, key: string) { - const diskStream = createReadStream(diskPath) - - const params = { - Bucket: configs.s3UscBucket, - Key: key, - Body: diskStream - } - const uploadInstance = new Upload({ - client, - partSize: 1024 * 1024 * 10, - queueSize: 1, - // @see https://github.com/aws/aws-sdk-js-v3/issues/2311 - // tl;dr: the variable name, 'leavePartsOnError' is not representative of the behavior. - // It should instead be interpreted as, 'throwOnPartsError' - leavePartsOnError: true, - params - }) - - return { - uploadInstance, - diskStream, - } - } - - - async updateDatabaseRecords() { - await this.updateSegmentBytes() - await this.checkForAborted() - } - - async checkForAborted() { - const recording = await getRecordingRelatedToVod(this.vodId) - if (!recording) throw new Error(`failed to get recording related to vodId=${this.vodId}`); - if (recording.is_aborted) { - this.abortController.abort() - } - } - - async updateSegmentBytes() { - for (const [index, segment] of this.segments.entries()) { - if (segment.id) { - console.log(`updateSegmentBytes() Segment ${index} -- segment.id=${segment.id} .bytes=${segment.bytes} .bytes_uploaded=${segment.bytes_uploaded} .s3_key=${segment.s3_key} .vod.id=${segment?.vod?.id}`) - const seg = await updateSegmentInDatabase({ segment_id: segment.id, bytes: segment.bytes, bytes_uploaded: segment.bytes_uploaded }) - let matchingSegment = this.segments.find((s) => s.id === seg.id) - if (matchingSegment) matchingSegment = seg // update the locally cached segment so update_at is accurate (used during isTryingDownload()) - } - } - } - - startProgressReports() { - this.databaseUpdateTimer = setInterval(async () => { - try { - await this.updateDatabaseRecords() - } catch (e) { - console.error(`during startProgressReports(), we encountered the following error.`) - console.error(e) - } - }, this.updateTimeout) - } - - stopProgressReports() { - clearInterval(this.databaseUpdateTimer) - } - - /** - * isTryingDownload - * - * There are always more tries unless the stream has been offline for greater than 5 minutes. - */ - isTryingDownload() { - - const isSegmentPresent = (!!this.segments && this.segments?.length > 0) - // console.log(`isTryingDownload() this.segments=${this.segments}, this.segments.length=${this.segments?.length}, isSegmentPresent=${isSegmentPresent} `); - if (!isSegmentPresent) return true; - const latestSegment = this.segments.at(-1) - const hasUpdatedTimestamp = latestSegment?.updated_at - if (!hasUpdatedTimestamp) throw new Error('latestSegment does not have an updated_at property'); - const fiveMinsAgo = sub(new Date(), { minutes: 5 }) - const lastUpdatedAt = latestSegment.updated_at - console.log(`isTryingDownload fiveMinsAgo=${fiveMinsAgo.toISOString()}, lastUpdatedAt=${lastUpdatedAt}`) - return isAfter(lastUpdatedAt, fiveMinsAgo) - } - - - - /** - * done() - * - * Repeatedly download segments until there is no more stream. - * Stream is considered no-more once it has been offline for >5 minutes. - * When downloading is done, upload segments to S3. - * - * input: stream URL, such as 'https://chaturbate.com/projektmelody' - * ouptut: list of S3Files, such as [{ key: 'example1.ts' }, { key: 'example2.ts' }] - */ - async done() { - this.startProgressReports(); - try { - console.log(`>> downloadSegments() begin.`) - await this.downloadSegments(); - console.log(`>> downloadsSegments() finished. ${this.segments.length} downloaded.`) - console.log(`waiting 30 seconds for downloadSegments() to close ffmpeg ...`) // @todo potential speed optimization is to proceed as soon as file is done being written to - await setTimeout(30*1000) - console.log(`>> uploadSegments() begin.`) - await this.uploadSegments(); - console.log(`>> uploadSegments() finished.`) - } catch (e) { - console.error(`An error was encountered during done() function. This should not happen under nominal scenarios. This may be a bug; please investigate.`) - throw e - } finally { - this.stopProgressReports(); - } - } - - /** - * downloadSegment - * - * Download a single segment. - * * Creates segment in the database - * * Pushes the segment to this.segments - */ - async downloadSegment() { - const s3_key = `${nanoid()}.ts` - const segment = await createSegment(s3_key, this.vodId) - if (!segment.id) { - throw new Error(`Failed to createSegment(). segment.id was missing.`); - } - console.log(`New segment created. @see http://localhost:9000/segments?id=eq.${segment.id}&select=*,vod:vods(*,recordings(*))`) - this.segments.push(segment) - const { pipeline, ffmpegStream, ffmpegProcess } = (await RecordNextGeneration._dl(this.url, s3_key)) - if (this.downloadStream) throw new Error(`If you see this error, there is a bug in your code. downloadSegment() tried to use this.downloadStream but it was already being used by some other part of the code. Please refactor so this.downloadStream is not used by more than one function at any given time.`); - this.downloadStream = ffmpegStream - this.downloadProcess = ffmpegProcess - console.log(`setting downloadProcess as follows. ${ffmpegProcess}`) - ffmpegStream.on('data', (data) => { - let mySegment = this.segments.find((s) => s.id === segment.id) - if (mySegment) { - mySegment.bytes += data.length; - } - }) - await pipeline - delete this.downloadStream // cleanup so another iteration can use - } - - /** - * downloadSegments - * - * Fault-tolerant segment downloader. - * * Creates segments in the database. - * * Handles common errors - * * Retries until the stream has been offline for >5 minutes. - * * Recursively called - */ - async downloadSegments(): Promise { - try { - while (this.isTryingDownload()) { - // Backoff timer, in case of followup segments. - // if most recent segment was created greater than 1 minute ago, there is no timer. (immediate retry) - // else the retry timer is 30 seconds (wait before retrying) - const mostRecentSegmentCreationTimestamp = this.segments.at(-1)?.created_at - if (mostRecentSegmentCreationTimestamp) { - const thirtySecondsAgo = sub(new Date(), { seconds: 30 }) - const mostRecentSegmentCreation = new Date(mostRecentSegmentCreationTimestamp) - const isMostRecentSegmentCreatedGreaterThanOneMinuteAgo = isBefore(mostRecentSegmentCreation, thirtySecondsAgo) - if (isMostRecentSegmentCreatedGreaterThanOneMinuteAgo) console.log(`Waiting 30 seconds before next downloadSegment().`); - await setTimeout((isMostRecentSegmentCreatedGreaterThanOneMinuteAgo) ? 1000*30 : 0) - } - await this.downloadSegment() - } - } catch (e) { - if (e instanceof Error && e.name === 'RoomOfflineError') { - // If the room is offline, then we want to retry immediately. - // We do this because the offline room might be a temporary situation. - // e.g. streamer's computer bluescreened and they're coming back after they reboot. - // If the room has been offline for >5 minutes, then we consider the stream concluded and we return. - console.warn('Room is offline! ~~ lets try again, if appropriate.') - console.log(`isTryingDownload()=${this.isTryingDownload()}`) - if (this.isTryingDownload()) { - return this.downloadSegments() - } else { - return - } - - } else if (e instanceof Error && e.name === 'PlaylistFailedError') { - console.error(`sometimes @futureporn/scout fails to get the playlist URL. We want to immediately try again.`) - return this.downloadSegments() - - } else if (e instanceof Error && e.name === 'AdminAbortedError') { - // An admin aborted the recording which means we don't want to retry recording. - // we return which causes the 'record' Task to be marked as successful. - console.log(`Clear as day, that is an AdminAbortedError! ❤️`) - return - - } else if (e instanceof Error && e.name === 'DownloadFailedError') { - console.error(`We encountered a DownloadFailedError. I'm unsure why this happens. I guess we will retry.`) - return this.downloadSegments() - } - } - } - - /** - * uploadSegments - * - * Fault-tolerant segment uploader. - * * Uploads local segment files to S3 - * * Handles common errors - * * Retries each segment up to 9 times - */ - async uploadSegments() { - try { - for (const segment of this.segments.filter((s) => s.bytes > 0)) { - await this.uploadSegment(segment) - } - } catch (e) { - console.error('error during uploadSegments(). error as follows.') - console.error(e) - throw e - } - } - - async uploadSegment(segment: SegmentResponse) { - const diskPath = join(configs.cacheDir, segment.s3_key) - const key = segment.s3_key - const client = this.getS3Client() - - await pRetry(async (attemptCount: number) => { - console.log(`uploadSegment() attempt ${attemptCount}`) - if (!this.s3Client) throw new Error('S3Client') - const { uploadInstance } = (await RecordNextGeneration._ul(client, diskPath, key)) - uploadInstance.on('httpUploadProgress', (progress: Progress) => this.onUploadProgress(segment, progress)) - return uploadInstance.done() - }, { - onFailedAttempt: (e) => { - console.error(`failed to uploadSegment() with the following error. Retrying with ${e.retriesLeft} retries left.`) - console.error(e) - }, - retries: 9 - }) - - const matchingSegment = this.segments.find((s) => s.id === segment.id) - if (matchingSegment?.bytes_uploaded && matchingSegment?.bytes) { - matchingSegment.bytes_uploaded = matchingSegment.bytes - } - - } -} diff --git a/services/capture/src/Video.js b/services/capture/src/Video.js deleted file mode 100644 index 253ae28..0000000 --- a/services/capture/src/Video.js +++ /dev/null @@ -1,68 +0,0 @@ - -import { execa } from 'execa' -import { tmpdir } from 'os' -import path from 'node:path' -import fs from 'node:fs' -import os from 'node:os' - -export class VideoConcatError extends Error { - constructor (msg) { - super(msg || 'Failed to concatenate video') - this.name = 'VideoConcatError' - } -} - - - -export default class Video { - constructor (opts) { - if (typeof opts.filePaths === 'undefined') throw new Error('Video must be called with opts.filePaths'); - if (typeof opts.cwd === 'undefined') throw new Error('Video must be called with opts.cwd'); - this.filePaths = opts.filePaths - this.cwd = opts.cwd - this.room = opts.room || 'projektmelody' - this.execa = opts.execa || execa - } - - - - getFilesTxt () { - return this.filePaths - .sort((a, b) => a.timestamp - b.timestamp) - .map((d) => `file '${d.file}'`) - .join('\n') - .concat('\n') - } - - - getFilesFile () { - const p = path.join(this.cwd, 'files.txt') - fs.writeFileSync( - p, - this.getFilesTxt(this.filePaths), - { encoding: 'utf-8' } - ) - return p - } - - async concat () { - const target = path.join(this.cwd, `${this.room}-chaturbate-${new Date().valueOf()}.mp4`) - - const { exitCode, killed, stdout, stderr } = await this.execa('ffmpeg', [ - '-y', - '-f', 'concat', - '-safe', '0', - '-i', this.getFilesFile(this.filePaths), - '-c', 'copy', - target - ], { - cwd: this.cwd - }); - - if (exitCode !== 0 || killed !== false) { - throw new VideoConcatError(`exitCode:${exitCode}, killed:${killed}, stdout:${stdout}, stderr:${stderr}`); - } - - return target - } -} diff --git a/services/capture/src/Video.spec.js b/services/capture/src/Video.spec.js deleted file mode 100644 index 8b54db8..0000000 --- a/services/capture/src/Video.spec.js +++ /dev/null @@ -1,86 +0,0 @@ - -import 'dotenv/config' -import Video from './Video.js' -import { dirname } from 'path'; -import { fileURLToPath } from 'url'; -import path from 'node:path' -import os from 'node:os' -import fs from 'node:fs' -import sinon from 'sinon' -import sinonChai from 'sinon-chai' -import chai, { expect } from 'chai' - -chai.use(sinonChai); - -const __dirname = dirname(fileURLToPath(import.meta.url)); - - - -const dataFixture = [ - { - timestamp: 1, - file: 'mock-stream0.mp4' - }, { - timestamp: 2, - file: 'mock-stream1.mp4' - }, { - timestamp: 3, - file: 'mock-stream2.mp4' - } -] - -describe('Video', function () { - - let video - - before(() => { - // copy files to /tmp so we dont clutter the fixtures dir - // and simulate cwd being process.env.FUTUREPORN_TMP - dataFixture.forEach((d) => { - fs.copyFileSync( - path.join(__dirname, 'fixtures', d.file), - path.join(os.tmpdir(), d.file) - ) - }) - }) - - - - beforeEach(() => { - video = new Video({ - cwd: os.tmpdir(), - filePaths: dataFixture, - execa: sinon.fake.resolves({ exitCode: 0, killed: false, stdout: "i am so horni rn", stderr: null }) - }) - }) - - afterEach(function() { - console.log('>> sinon.restore! (afterEach)') - sinon.restore(); - }) - - - describe('getFilesTxt', function () { - it('should generate contents suitable for input to `ffmpeg -f concat`', function () { - const txt = video.getFilesTxt() - expect(txt).to.deep.equal("file 'mock-stream0.mp4'\nfile 'mock-stream1.mp4'\nfile 'mock-stream2.mp4'\n") - }) - }) - - describe('concat', function () { - it('should join multiple videos into one', async function () { - const file = await video.concat() - expect(typeof file === 'string').to.be.true - expect(video.execa).calledOnce - expect(file).to.match(/\.mp4$/) - }) - }) - - describe('getFilesFile', function () { - it('should create a files.txt and return the path', async function () { - const file = await video.getFilesFile() - expect(typeof file === 'string').to.be.true - expect(file).to.equal(path.join(os.tmpdir(), 'files.txt')) - }) - }) -}) \ No newline at end of file diff --git a/services/capture/src/Voddo.js b/services/capture/src/Voddo.js deleted file mode 100644 index 6e1a75e..0000000 --- a/services/capture/src/Voddo.js +++ /dev/null @@ -1,243 +0,0 @@ -import 'dotenv/config' -import YoutubeDlWrap from "youtube-dl-wrap"; -import { EventEmitter } from 'node:events'; -import { AbortController } from "node-abort-controller"; -import { readdir, stat } from 'node:fs/promises'; -import { join } from 'node:path' -import ffmpeg from 'fluent-ffmpeg' -import { loggerFactory } from 'common/logger' - -const logger = loggerFactory({ - service: 'futureporn/capture' -}) -const defaultStats = {segments:[],lastUpdatedAt:null} - -export default class Voddo extends EventEmitter { - constructor(opts) { - super() - this.courtesyTimer = setTimeout(() => {}, 0); - this.retryCount = 0; - this.url = opts.url; - this.format = opts.format || 'best'; - this.cwd = opts.cwd; - this.ytdlee; // event emitter for ytdlwrap - this.stats = Object.assign({}, defaultStats); - this.abortController = new AbortController(); - this.ytdl = opts.ytdl || new YoutubeDlWrap(); - if (process.env.YOUTUBE_DL_BINARY) this.ytdl.setBinaryPath(process.env.YOUTUBE_DL_BINARY); - } - - static async getVideoLength (filePath) { - return new Promise((resolve, reject) => { - ffmpeg.ffprobe(filePath, function(err, metadata) { - if (err) reject(err) - resolve(Math.floor(metadata.format.duration*1000)) - }); - }) - } - - // greets ChatGPT - static groupStreamSegments(segments, threshold = 1000*60*60) { - segments.sort((a, b) => a.startTime - b.startTime); - const streams = []; - let currentStream = []; - - for (let i = 0; i < segments.length; i++) { - const currentSegment = segments[i]; - const previousSegment = currentStream[currentStream.length - 1]; - - if (!previousSegment || currentSegment.startTime - previousSegment.endTime <= threshold) { - currentStream.push(currentSegment); - } else { - streams.push(currentStream); - currentStream = [currentSegment]; - } - } - - streams.push(currentStream); - return streams; - } - - - - - - - - /** - * getRecordedStreams - * - * get the metadata of the videos captured - */ - async getRecordedSegments() { - let f = [] - const files = await readdir(this.cwd).then((raw) => raw.filter((f) => /\.mp4$/.test(f) )) - for (const file of files) { - const filePath = join(this.cwd, file) - const s = await stat(filePath) - const videoDuration = await Voddo.getVideoLength(filePath) - const startTime = parseInt(s.ctimeMs) - const endTime = startTime+videoDuration - const size = s.size - f.push({ - startTime, - endTime, - file, - size - }) - } - this.stats.segments = f - - - return this.stats.segments - } - - isDownloading() { - // if there are event emitter listeners for the progress event, - // we are probably downloading. - return ( - this.ytdlee?.listeners('progress').length !== undefined - ) - } - - delayedStart() { - // only for testing - this.retryCount = 500 - this.courtesyTimer = this.getCourtesyTimer(() => this.download()) - } - - - start() { - // if download is in progress, do nothing - if (this.isDownloading()) { - logger.log({ level: 'debug', message: 'Doing nothing because a download is in progress.' }) - return; - } - - // if download is not in progress, start download immediately - // reset the retryCount so the backoff timer resets to 1s between attempts - this.retryCount = 0 - clearTimeout(this.courtesyTimer) - - // create new abort controller - //this.abortController = new AbortController() // @todo do i need this? Can't I reuse the existing this.abortController? - - this.download() - } - - stop() { - logger.log({ level: 'info', message: 'Received stop(). Stopping.' }) - clearTimeout(this.courtesyTimer) - this.abortController.abort() - } - - /** generate a report **/ - getReport(errorMessage) { - let report = {} - report.stats = Object.assign({}, this.stats) - report.error = errorMessage - report.reason = (() => { - if (errorMessage) return 'error'; - else if (this.abortController.signal.aborted) return 'aborted'; - else return 'close'; - })() - // clear stats to prepare for next run - this.stats = Object.assign({}, defaultStats) - return report - } - - emitReport(report) { - logger.log({ level: 'debug', message: 'EMITTING REPORT' }) - this.emit('stop', report) - } - - getCourtesyTimer(callback) { - // 600000ms = 10m - const waitTime = Math.min(600000, (Math.pow(2, this.retryCount) * 1000)); - this.retryCount += 1; - logger.log({ level: 'debug', message: `courtesyWait for ${waitTime/1000} seconds. (retryCount: ${this.retryCount})` }) - return setTimeout(callback, waitTime) - } - - download() { - const handleProgress = (progress) => { - logger.log({ level: 'debug', message:` [*] progress event` }) - this.stats.lastUpdatedAt = Date.now(), - this.stats.totalSize = progress.totalSize - } - - const handleError = (error) => { - if (error?.message !== undefined && error.message.includes('Room is currently offline')) { - logger.log({ level: 'debug', message: 'Handled an expected \'Room is offline\' error' }) - - } else { - logger.log({ level: 'error', message: 'ytdl error' }) - logger.log({ level: 'error', message: error.message }) - } - this.ytdlee.off('progress', handleProgress) - this.ytdlee.off('handleYtdlEvent', handleYtdlEvent) - - // restart the download after the courtesyTimeout - this.courtesyTimer = this.getCourtesyTimer(() => this.download()) - this.emitReport(this.getReport(error.message)) - } - - - const handleYtdlEvent = (type, data) => { - logger.log({ level: 'debug', message: `handleYtdlEvent type: ${type}, data: ${data}` }) - logger.log({ level: 'debug', message: `handleYtdlEvent type: ${type}, data: ${data}` }) - if (type === 'download' && data.includes('Destination:')) { - let filePath = /Destination:\s(.*)$/.exec(data)[1] - logger.log({ level: 'debug', message: `Destination file detected: ${filePath}` }) - let datum = { file: filePath, timestamp: new Date().valueOf() } - let segments = this.stats.segments - segments.push(datum) && segments.length > 64 && segments.shift(); // limit the size of the segments array - this.emit('start', datum) - } else if (type === 'ffmpeg' && data.includes('bytes')) { - const bytes = /(\d*)\sbytes/.exec(data)[1] - logger.log({ level: 'debug', message: `ffmpeg reports ${bytes}`}) - let mostRecentFile = this.stats.segments[this.stats.segments.length-1] - mostRecentFile['size'] = bytes - logger.log({ level: 'debug', message: mostRecentFile }) - } - } - - const handleClose = () => { - logger.log({ level: 'debug', message: 'got a close event. handling!' }); - - this.ytdlee.off('progress', handleProgress) - this.ytdlee.off('handleYtdlEvent', handleYtdlEvent) - - // restart Voddo only if the close was not due to stop() - if (!this.abortController.signal.aborted) { - // restart the download after the courtesyTimeout - this.courtesyTimer = this.getCourtesyTimer(() => this.download()) - } - - this.emitReport(this.getReport()) - } - - logger.log({ level: 'debug', message: `Downloading url:${this.url} format:${this.format}` }) - logger.log({ level: 'debug', message: JSON.stringify(this.ytdl) }) - - // sanity check. ensure cwd exists - stat(this.cwd, (err) => { - if (err) logger.log({ level: 'error', message: `Error while getting cwd stats of ${this.cwd} Does it exist?` }) - }) - - this.ytdlee = this.ytdl.exec( - [this.url, '-f', this.format], - { - cwd: this.cwd - }, - this.abortController.signal - ); - this.ytdlee.on('progress', handleProgress); - this.ytdlee.on('youtubeDlEvent', handleYtdlEvent); - this.ytdlee.once('error', handleError); - this.ytdlee.once('close', handleClose); - } - - - -} \ No newline at end of file diff --git a/services/capture/src/Voddo.spec.js b/services/capture/src/Voddo.spec.js deleted file mode 100644 index 155925c..0000000 --- a/services/capture/src/Voddo.spec.js +++ /dev/null @@ -1,490 +0,0 @@ -import 'dotenv/config' -import Voddo from './Voddo.js' -import chai, { expect } from 'chai' -import sinon from 'sinon' -import YoutubeDlWrap from 'youtube-dl-wrap' -import { - AbortController -} from "node-abort-controller"; -import { - EventEmitter -} from 'events' -import debugFactory from 'debug' -import { join, dirname } from 'path'; -import { fileURLToPath } from 'url'; -import sinonChai from 'sinon-chai' -import sinonTest from "sinon-test"; - -chai.use(sinonChai); - -const test = sinonTest(sinon, { - toFake: ["setTimeout", "setInterval"], - shouldAdvanceTime: false -}); -const debug = debugFactory('voddo') -const __dirname = dirname(fileURLToPath(import.meta.url)); - - - - - -describe('Voddo', function() { - - - describe('groupStreamSegments', function () { - it('should separate two stream data objects', function () { - const fixture = [{ - "startTime": 1675386000000, - "file": "projektmelody 2023-02-02 17_00-projektmelody.mp4", - "size": 550799038, - "endTime": 1675391400000, - }, { - "startTime": 1675391405000, - "file": "projektmelody 2023-02-02 18_30-projektmelody.mp4", - "size": 6556534941, - "endTime": 1675396800000 - }, { - "startTime": 1675368000000, - "file": "projektmelody 2023-02-02 12_00-projektmelody.mp4", - "size": 6556534941, - "endTime": 1675378800000 - }] - - const streams = Voddo.groupStreamSegments(fixture) - expect(streams).to.deep.equal([ - [ - { - "startTime": 1675368000000, - "file": "projektmelody 2023-02-02 12_00-projektmelody.mp4", - "size": 6556534941, - "endTime": 1675378800000 - } - ], - [ - { - "startTime": 1675386000000, - "file": "projektmelody 2023-02-02 17_00-projektmelody.mp4", - "size": 550799038, - "endTime": 1675391400000, - }, { - "startTime": 1675391405000, - "file": "projektmelody 2023-02-02 18_30-projektmelody.mp4", - "size": 6556534941, - "endTime": 1675396800000 - } - ] - ]) - }) - }) - - - // let clock; - - // beforeEach(function() { - // clock = sinon.useFakeTimers({ - // toFake: ["setTimeout", "setInterval"], - // shouldAdvanceTime: false - // }); - // }) - - // afterEach(() => { - // sinon.restore() - // }) - - - - // Something faulty with Voddo or sinon or mocha, not sure. - // When running by itself, test succeeds. When running with 'should start and stop stream download', - // voddo.stats gets set to whatever that test sets it to. So bizarre, it's like the same Voddo class instance - // exists in two different tests even though they are named differently. - // Even though they are not in global scope. Even though each was called with `new Voddo(...)` - // Doesn't matter if I wrap both in sinon-test. Same leaky problem. - // Doesn't matter if I sinon.restore() afterEach. Same leaky problem. - // Doesn't matter if I manually set up a sinon sandbox. Same leaky problem. - // Fuck event emitters. I love their utility but I don't know how the fuck they are supposed to be tested. - // Solution might just call for a rewrite of Voddo, or perhaps deleting Voddo in favor of Capture - // For now, I'm moving forward because Voddo works even though this test does not. - describe('getRecordedSegments', function() { - xit('should populate it\'s log if log is empty', async function () { - const voddo = new Voddo({ - url: 'https://example.com', - cwd: join(__dirname, 'fixtures') - }) - const streams = await voddo.getRecordedSegments() - console.log(streams) - expect(streams.length).to.equal(3) - expect(streams[0]).to.have.property('startTime') - expect(streams[0]).to.have.property('file') - expect(streams[0]).to.have.property('size') - }) - xit('should use Voddo\'s stats history to get filenames of only the most recent stream', async function() { - const sb = sinon.createSandbox() - const viddo = new Voddo({ - url: 'https://example.com', - cwd: '~/Downloads' - }) - sb.stub(viddo, 'stats').value({ - segments: [{ - startTime: 1674147647000, - size: 192627, - file: 'projektmelody 2023-01-19 17_00-projektmelody.mp4' - }, { - startTime: 1674151247000, - size: 192627, - file: 'projektmelody 2023-01-19 18_00-projektmelody.mp4' - }, { - startTime: 1674154847000, - size: 192627, - file: 'projektmelody 2023-01-19 19_00-projektmelody.mp4' - }, { - file: 'projektmelody 2023-01-20 20_10-projektmelody.mp4', - size: 192627, - startTime: 1674245400000, - }, { - file: 'projektmelody 2023-01-20 21_10-projektmelody.mp4', - size: 192627, - startTime: 1674249000000, - }, { - file: 'projektmelody 2023-01-20 22_10-projektmelody.mp4', - size: 192627, - startTime: 1674252600000, - }] - }) - - const filenames = await viddo.getRecordedSegments() - sb.restore() - expect(filenames).to.have.lengthOf(3) - expect(filenames).to.deep.equal([{ - file: 'projektmelody 2023-01-20 20_10-projektmelody.mp4', - size: 192627, - startTime: 1674245400000, - }, { - file: 'projektmelody 2023-01-20 21_10-projektmelody.mp4', - size: 192627, - startTime: 1674249000000, - }, { - file: 'projektmelody 2023-01-20 22_10-projektmelody.mp4', - size: 192627, - startTime: 1674252600000, - }]) - }) - }) - - - xit('should keep a log of the files downloaded', function(done) { - const ee = new EventEmitter() - - - const ytdl = sinon.createStubInstance(YoutubeDlWrap) - ytdl.exec.returns(ee) - - - const times = [ - 1000, // start - 1000 * 60 * 60 * 1, // stop - 1000 * 60 * 60 * 1 + 1, // start - 1000 * 60 * 60 * 2, // stop - 1000 * 60 * 60 * 3 + 1, // start - 1000 * 60 * 60 * 4 // stop - ] - - clock.setTimeout(() => { - ee.emit('youtubeDlEvent', 'download', ' Destination: projektmelody 2023-01-18 21_10-projektmelody.mp4') - }, times[0]) - - clock.setTimeout(() => { - ee.emit('close') - }, times[1]) - - clock.setTimeout(() => { - ee.emit('youtubeDlEvent', 'download', ' Destination: projektmelody 2023-01-18 22_10-projektmelody.mp4') - }, times[2]) - - clock.setTimeout(() => { - ee.emit('close') - }, times[3]) - - clock.setTimeout(() => { - ee.emit('youtubeDlEvent', 'download', ' Destination: projektmelody 2023-01-18 23_10-projektmelody.mp4') - }, times[4]) - - clock.setTimeout(() => { - ee.emit('close') - }, times[5]) - - - let url = `https://chaturbate.com/projektmelody` - let cwd = process.env.FUTUREPORN_WORKDIR || '/tmp' - const voddo = new Voddo({ - url: url, - format: 'best', - cwd: cwd, - ytdl - }) - - voddo.once('start', (data) => { - expect(data).to.have.property('file') - expect(data).to.have.property('timestamp') - - voddo.once('start', (data) => { - expect(data).to.have.property('file') - expect(data).to.have.property('timestamp') - - voddo.once('start', (data) => { - expect(data).to.have.property('file') - expect(data).to.have.property('timestamp') - - voddo.once('stop', function(report) { - debug(report) - expect(report).to.have.property('stats') - expect(report.stats).to.have.property('files') - expect(report.stats.files).to.have.lengthOf(3) - debug(report.stats.files) - expect(report.stats.files[0]).to.include({ - file: 'projektmelody 2023-01-18 21_10-projektmelody.mp4' - }) - - expect(ytdl.exec).calledThrice - - console.log('>>WE ARE DONE') - expect(this.clock.countTimers()).to.equal(0) - done() - }) - clock.tick(times[5]) // stop - - }) - clock.tick(times[3]) // stop - clock.tick(times[4]) // start - - }) - clock.tick(times[1]) // stop - clock.tick(times[2]) // start - - }) - - - voddo.start() - expect(ytdl.exec).calledOnce - - clock.tick(times[0]) - - - - - }) - - xit('should keep a log of the files downloaded', function(done) { - this.timeout(5000) - // https://github.com/insanity54/futureporn/issues/13 - const ytdlStub = sinon.createStubInstance(YoutubeDlWrap) - ytdlStub.exec - .onCall(0) - .callsFake(function(args, opts, aborter) { - let ee = new EventEmitter() - clock.setTimeout(() => { - ee.emit('youtubeDlEvent', 'download', ' Destination: projektmelody 2023-01-18 21_10-projektmelody.mp4') - }, 50) - clock.setTimeout(() => { - ee.emit('close') - }, 100) - return ee - }) - .onCall(1) - .callsFake(function(args, opts, aborter) { - let ee = new EventEmitter() - clock.setTimeout(() => { - ee.emit('youtubeDlEvent', 'download', ' Destination: projektmelody 2023-01-18 22_10-projektmelody.mp4') - }, 50) - clock.setTimeout(() => { - ee.emit('close') - }, 100) - return ee - }) - .onCall(2) - .callsFake(function(args, opts, aborter) { - let ee = new EventEmitter() - clock.setTimeout(() => { - ee.emit('youtubeDlEvent', 'download', ' Destination: projektmelody 2023-01-18 23_10-projektmelody.mp4') - }, 50) - clock.setTimeout(() => { - ee.emit('close') - }, 100) - return ee - }) - let url = `https://chaturbate.com/projektmelody` - let cwd = process.env.FUTUREPORN_WORKDIR || '/tmp' - - const voddo = new Voddo({ - url: url, - format: 'best', - cwd: cwd, - ytdl: ytdlStub - }) - - // expect(clock.countTimers()).to.equal(0) - voddo.once('start', function(data) { - expect(data).to.have.property('file') - expect(data).to.have.property('timestamp') - - clock.next() - clock.next() - voddo.once('start', function(data) { - expect(data).to.have.property('file') - expect(data).to.have.property('timestamp') - - voddo.once('start', function(data) { - debug('fake start?') - expect(data).to.have.property('file') - expect(data).to.have.property('timestamp') - - voddo.once('stop', function(report) { - debug(report) - expect(report).to.have.property('stats') - expect(report.stats).to.have.property('files') - expect(report.stats.files).to.have.lengthOf(3) - debug(report.stats.files) - expect(report.stats.files[0]).to.include({ - file: 'projektmelody 2023-01-18 21_10-projektmelody.mp4' - }) - - sinon.assert.calledThrice(ytdlStub.exec) - expect(this.clock.countTimers()).to.equal(0) - done() - }) - - - }) - }) - }) - - voddo.start() - }) - - - it('should start and stop stream download', test(function(done) { - - const sandbox = this - - const ee = new EventEmitter() - - const ytdl = this.createStubInstance(YoutubeDlWrap); - ytdl.exec.returns(ee) - - - const url = 'https://chaturbate.com/projektmelody' - const format = 'best' - const cwd = '/tmp' - const v = new Voddo({ - url, - format, - cwd, - ytdl - }) - console.log(v.stats) - - v.once('stop', function(data) { - console.log('ffffff') - console.log(this) - expect(this.abortController.signal.aborted, 'abortController did not abort').to.be.true - expect(sandbox.clock.countTimers()).to.equal(0) - done() - }) - v.once('start', function(data) { - console.log('STARRRRRT') - expect(data).to.have.property('file') - expect(data).to.have.property('timestamp') - expect(this).to.have.property('abortController') - console.log('ey cool, voddo started') - }) - v.start() - - const times = [ - 500, - 1000, - 2000 - ] - - this.clock.setTimeout(() => { - ee.emit('youtubeDlEvent', 'download', ' Destination: projektmelody 2023-01-18 21_10-projektmelody.mp4') - }, times[0]) - - this.clock.setTimeout(() => { - v.stop() - }, times[1]) - - this.clock.setTimeout(() => { - ee.emit('close') - }, times[2]) - - this.clock.tick(times[0]) // start - this.clock.tick(times[1]) // stop - this.clock.tick(times[2]) // close - - })) - - - xit('should retry when a stream closes', function(done) { - - const ytdlStub = sinon.createStubInstance(YoutubeDlWrap); - ytdlStub.exec - .onCall(0) - .callsFake(function(args, opts, aborter) { - debug(' [test] callsFake 0') - let ee = new EventEmitter() - setTimeout(() => { - console.log('should retry when a stream closes -- emission') - ee.emit('youtubeDlEvent', 'download', ' Destination: projektmelody 2023-01-17 19_39-projektmelody.mp4') - }, 100) - setTimeout(() => { - console.log('should retry when a stream closes -- emission') - // this simulates youtube-dl closing - // (NOT Voddo closing) - ee.emit('close') - }, 550) - return ee - }) - .onCall(1) - .callsFake(function(args, opts, aborter) { - debug(' [test] callsFake 1') - let ee = new EventEmitter() - setTimeout(() => { - ee.emit('youtubeDlEvent', 'download', ' Destination: projektmelody 2023-01-17 19_45-projektmelody.mp4') - }, 100) - return ee - }) - let url = `https://chaturbate.com/projektmelody` - let cwd = process.env.FUTUREPORN_WORKDIR || '/tmp' - let abortController = new AbortController() - - const voddo = new Voddo({ - url: url, - format: 'best', - cwd: cwd, - ytdl: ytdlStub - }) - - voddo.once('start', function(data) { - debug(' [test] voddo <<<<<-----') - expect(data).to.have.property('file') - expect(data).to.have.property('timestamp') - - voddo.once('start', function(data) { - debug(' [test] restarted after dl close! (expected) <<<<<-----') - - sinon.assert.calledTwice(ytdlStub.exec) - expect(this.clock.countTimers()).to.equal(0) - done() - }) - }) - - voddo.start() - - clock.next() - clock.next() - clock.next() - clock.next() - clock.next() - - }) - -}) \ No newline at end of file diff --git a/services/capture/src/app.spec.ts b/services/capture/src/app.spec.ts deleted file mode 100644 index 2efd5f5..0000000 --- a/services/capture/src/app.spec.ts +++ /dev/null @@ -1,89 +0,0 @@ -'use strict' - -import { build } from './app.ts' -import { use, expect } from "chai" -import sinonChai from 'sinon-chai' -use(sinonChai) - -describe('app', function () { - const app = build({}, 'postgres://') - describe('/', function () { - it('GET', async function () { - - const response = await app.inject({ - method: 'GET', - url: '/' - }) - expect(response.statusCode).to.equal(200) - expect(JSON.parse(response.body)).to.have.property('version') - }) - }) - xdescribe('/api/records', function () { - it('GET -- list the records', async function () { - const response = await app.inject({ - method: 'GET', - url: '/api/records' - }) - expect(response.statusCode).to.equal(200) - const body = JSON.parse(response.body) - expect(body).to.have.property('data') - expect(body.data).to.be.an.instanceof(Array); - }) - it('DELETE -- delete all records', async function () { - const response = await app.inject({ - method: 'DELETE', - url: '/api/records' - }) - expect(response.statusCode).to.equal(200) - const body = JSON.parse(response.body) - expect(body).to.have.property('data') - expect(body.data).to.be.lengthOf(0); - }) - }) - describe('/api/record', function () { - describe('POST', function () { - it('should create', async function () { - let url = 'https://example.com/my-cool-stream' - const response = await app.inject({ - method: 'POST', - url: '/api/record', - body: { - url - } - }) - expect(response.statusCode).to.equal(200) - const body = JSON.parse(response.body) - expect(body).to.have.property('id') - expect(body).to.have.property('url', url) - }) - it('should return 400 if url is missing', async function () { - const response = await app.inject({ - method: 'POST', - url: '/api/record', - }) - expect(response.statusCode).to.equal(400) - }) - }) - xit('GET -- list a record', async function () { - const response = await app.inject({ - method: 'GET', - url: '/api/record' - }) - expect(response.statusCode).to.equal(200) - expect(JSON.parse(response.body)).to.have.property('id') - expect(JSON.parse(response.body)).to.have.property('sourceUrl') - expect(JSON.parse(response.body)).to.have.property('fileSize') - expect(JSON.parse(response.body)).to.have.property('outputUrl') - }) - it('DELETE -- delete a record', async function () { - const response = await app.inject({ - method: 'DELETE', - url: '/api/record' - }) - expect(response.statusCode).to.equal(200) - expect(JSON.parse(response.body)).to.have.property('id') - }) - }) - - -}) diff --git a/services/capture/src/app.ts b/services/capture/src/app.ts deleted file mode 100644 index 6b9d043..0000000 --- a/services/capture/src/app.ts +++ /dev/null @@ -1,59 +0,0 @@ -'use strict' - -import fastify, { type FastifyRequest } from 'fastify' -import { getPackageVersion } from '@futureporn/utils/file.ts' -import fastifyGraphileWorkerPlugin, { type ExtendedFastifyInstance } from './fastify-graphile-worker-plugin.ts' -import { join, dirname } from 'node:path' -import { fileURLToPath } from 'node:url' - -const __dirname = dirname(fileURLToPath(import.meta.url)); - -const version = getPackageVersion(join(__dirname, '../package.json')) -interface RecordBodyType { - url: string; - discordMessageId: string; -} -interface MessageBodyType { - state: 'pending' | 'recording' | 'aborted' | 'ended'; - discordMessageId: string; -} - -const build = function (opts: Record={}, connectionString: string) { - const app: ExtendedFastifyInstance = fastify(opts) - app.register(fastifyGraphileWorkerPlugin, { connectionString }) - - app.get('/', async function (request, reply) { - return { app: '@futureporn/capture', version } - }) - app.put('/api/message', async function (request: FastifyRequest<{ Body: MessageBodyType }>, reply) { - const { state, discordMessageId } = request.body - if (app?.graphile) { - const jobId = await app.graphile.addJob('update_discord_message', { - discordMessageId, - state - }, { maxAttempts: 3 }) - } else { - console.error('app.graphile was missing') - } - }) - app.post('/api/record', async function (request: FastifyRequest<{ Body: RecordBodyType }>, reply) { - const { url, discordMessageId } = request.body - console.log(`POST /api/record with url=${url}`) - - if (app?.graphile) { - const jobId = await app.graphile.addJob('start_recording', { - url, - discordMessageId - }, { maxAttempts: 3 }) - return { jobId } - } else { - console.error(`app.graphile was missing! Is the graphile worker plugin registered to the fastify instance?`) - } - return { 'idk': true } - }) - return app -} - -export { - build -} \ No newline at end of file diff --git a/services/capture/src/blah.ts b/services/capture/src/blah.ts deleted file mode 100644 index 8ea3b26..0000000 --- a/services/capture/src/blah.ts +++ /dev/null @@ -1,31 +0,0 @@ -import PgBoss from 'pg-boss'; - -async function readme() { - const boss = new PgBoss('postgres://william:mysecretpassword@localhost:5435/william'); - - boss.on('error', (error: Error) => console.error(error)); - - await boss.start(); - - const queue = 'some-queue'; - - let jobId = await boss.send(queue, { param1: 'foo' }) - - console.log(`created job in queue ${queue}: ${jobId}`); - - await boss.work(queue, someAsyncJobHandler); -} - -async function someAsyncJobHandler(job: any) { - console.log(`job ${job.id} received with data:`); - console.log(JSON.stringify(job.data)); - - await new Promise((resolve, reject) => { - console.log('waiting 3s') - setTimeout(() => { - resolve(job.data) - }, 3000) - }); -} - -readme() \ No newline at end of file diff --git a/services/capture/src/cb.js b/services/capture/src/cb.js deleted file mode 100644 index 3195731..0000000 --- a/services/capture/src/cb.js +++ /dev/null @@ -1,18 +0,0 @@ -import { load } from 'cheerio' -import fetch from 'node-fetch' -import scrapingFetch from '@futureporn/scout/scrapingFetch.ts' - -export async function getRandomRoom () { - const res = await scrapingFetch('https://chaturbate.com/') - const body = await res.text() - const $ = load(body) - let roomsRaw = $('a[data-room]') - let rooms = [] - $(roomsRaw).each((_, e) => { - rooms.push($(e).attr('href')) - }) - - // greets https://stackoverflow.com/a/4435017/1004931 - var randomIndex = Math.floor(Math.random() * rooms.length); - return rooms[randomIndex].replaceAll('/', '') -} \ No newline at end of file diff --git a/services/capture/src/config.ts b/services/capture/src/config.ts deleted file mode 100644 index f8a1fb5..0000000 --- a/services/capture/src/config.ts +++ /dev/null @@ -1,49 +0,0 @@ -import 'dotenv/config' - -const requiredEnvVars = [ - 'SCOUT_URL', - 'S3_ACCESS_KEY_ID', - 'S3_SECRET_ACCESS_KEY', - 'S3_REGION', - 'S3_ENDPOINT', - 'S3_USC_BUCKET', - 'POSTGREST_URL', - 'AUTOMATION_USER_JWT', - 'CACHE_DIR', - 'WORKER_CONCURRENCY', -] as const; - -const getEnvVar = (key: typeof requiredEnvVars[number]): string => { - const value = process.env[key]; - if (!value) { - throw new Error(`Missing ${key} env var`); - } - return value; -}; - -export interface Config { - scoutUrl: string; - postgrestUrl: string; - automationUserJwt: string; - s3AccessKeyId: string; - s3SecretAccessKey: string; - s3Region: string; - s3UscBucket: string; - s3Endpoint: string; - cacheDir: string; - workerConcurrency: number; -} - - -export const configs: Config = { - scoutUrl: getEnvVar('SCOUT_URL'), - postgrestUrl: getEnvVar('POSTGREST_URL'), - automationUserJwt: getEnvVar('AUTOMATION_USER_JWT'), - s3AccessKeyId: getEnvVar('S3_ACCESS_KEY_ID'), - s3SecretAccessKey: getEnvVar('S3_SECRET_ACCESS_KEY'), - s3Region: getEnvVar('S3_REGION'), - s3UscBucket: getEnvVar('S3_USC_BUCKET'), - s3Endpoint: getEnvVar('S3_ENDPOINT'), - cacheDir: getEnvVar('CACHE_DIR'), - workerConcurrency: parseInt(getEnvVar('WORKER_CONCURRENCY')), -} \ No newline at end of file diff --git a/services/capture/src/disk.js b/services/capture/src/disk.js deleted file mode 100644 index 2d18c5b..0000000 --- a/services/capture/src/disk.js +++ /dev/null @@ -1,33 +0,0 @@ -import disk from 'diskusage'; - - -export function verifyStorage (appContext) { - const mountPath = appContext.env.FUTUREPORN_WORKDIR - disk.check(mountPath, (err, info) => { - if (err) { - appContext.logger.log({ level: 'error', message: `Error retrieving disk usage for ${mountPath}: ${err}` }); - return; - } - - const totalSize = info.total; - const availableSize = info.available; - const freeSize = info.free; - - appContext.logger.log({ 'level': 'info', message: `${mountPath} Disk Usage:` }); - appContext.logger.log({ 'level': 'info', message: `Total: ${bytesToSize(totalSize)}` }); - appContext.logger.log({ 'level': 'info', message: `Free: ${bytesToSize(freeSize)}` }); - appContext.logger.log({ 'level': 'info', message: `Available: ${bytesToSize(availableSize)}` }); - - if (availableSize < 85899345920) appContext.logger.log({ 'level': 'warn', message: `⚠️ Available disk is getting low! ${bytesToSize(availableSize)}` }); - else if (availableSize < 42949672960) appContext.logger.log({ 'level': 'error', message: `⚠️☠️ AVAILABLE DISK IS TOO LOW! ${bytesToSize(availableSize)}` }); - }); -} - - -// Helper function to convert bytes to human-readable format -export function bytesToSize(bytes) { - const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB']; - if (bytes === 0) return '0 Bytes'; - const i = Math.floor(Math.log2(bytes) / 10); - return `${(bytes / Math.pow(1024, i)).toFixed(2)} ${sizes[i]}`; -} \ No newline at end of file diff --git a/services/capture/src/fastify-graphile-worker-plugin.ts b/services/capture/src/fastify-graphile-worker-plugin.ts deleted file mode 100644 index 4f9b608..0000000 --- a/services/capture/src/fastify-graphile-worker-plugin.ts +++ /dev/null @@ -1,22 +0,0 @@ -import { type FastifyInstance } from 'fastify' -import fp from 'fastify-plugin' -import { type WorkerUtils, makeWorkerUtils } from 'graphile-worker' - -type Options = { - connectionString: string; -} - - -export interface ExtendedFastifyInstance extends FastifyInstance { - graphile?: WorkerUtils -} - -async function graphileWorkerPlugin (fastify: ExtendedFastifyInstance, opts: Options) { - if (!fastify.graphile) { - if (!opts.connectionString) throw new Error('graphileWorkerPlugin requires connectionString passed in options argument, but it was missing'); - const workerUtils = await makeWorkerUtils({ connectionString: opts.connectionString }) - fastify.decorate('graphile', workerUtils) - } -} - -export default fp(graphileWorkerPlugin) \ No newline at end of file diff --git a/services/capture/src/fastify-pgboss-plugin.ts b/services/capture/src/fastify-pgboss-plugin.ts deleted file mode 100644 index 983e2c9..0000000 --- a/services/capture/src/fastify-pgboss-plugin.ts +++ /dev/null @@ -1,23 +0,0 @@ -import { type FastifyInstance } from 'fastify' -import fp from 'fastify-plugin' -import PgBoss from 'pg-boss' - -type Options = { - boss: PgBoss; -} - - -export interface ExtendedFastifyInstance extends FastifyInstance { - boss?: PgBoss -} - -async function pgbossPlugin (fastify: ExtendedFastifyInstance, opts: Options) { - - if (!fastify.boss) { - if (!opts.boss) throw new Error('pgbossPlugin requires boss passed in options argument, but it was missing'); - const boss = opts.boss - fastify.decorate('boss', boss) - } -} - -export default fp(pgbossPlugin) \ No newline at end of file diff --git a/services/capture/src/fixtures/just-a-text-file.txt b/services/capture/src/fixtures/just-a-text-file.txt deleted file mode 100644 index e69de29..0000000 diff --git a/services/capture/src/fixtures/mime.types b/services/capture/src/fixtures/mime.types deleted file mode 100644 index a8dbef1..0000000 --- a/services/capture/src/fixtures/mime.types +++ /dev/null @@ -1,2 +0,0 @@ -application/vnd.apple.mpegurl mp4 -text/x-abc abc \ No newline at end of file diff --git a/services/capture/src/fixtures/mock-stream0.mp4 b/services/capture/src/fixtures/mock-stream0.mp4 deleted file mode 100644 index d85732364cc2dbc18c9795bc3a7dcde7110737ed..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 192627 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

oT4?hzU<6wFkGA-MZ(x)|$P@o|GgE!+j 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

*|yz?O&Bx-V3wI<4-X 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

rg}~mgjnuAE zzAZ+1(ouxJYoW5_`po@y27(Sw8U#J{`{1Av`<5{=^YischO{_xg#o@@!bwWO!3-T_ zj;ykgKLjE{S2U!l5O1EFC6z~MXPW7vakfpEFJydGhRF(M7m8a5sr=>K#Xhc`G~2Qq z{y9W?mUz(+;cHi1W9%nSKGdn71*7^Eb0C;{`cM2C(2pEpVC$s!Gzm9%dobmxtDY4O z9OeD3yTn}s>1+XVANK;zCnH}W%0)~K5dsQ|#G-gsRK2aOQi(ogfa zTrBgVq$Lz%TDrYSe8zJ@DdRT^%rfuKw7v(0SYI30|K_emGcJ4gWviiRug6%ZW#nwj zvOS#b^E>oRab9&$?anQi#K{_AljGTH^<~2ngt=MlmvAp6jZp(!s)sMgg;UgLgyj8O zzNRVdpIJE(f=oPSC7HXGBk~!-{^Am1WN|FVLO?4AW2W=xm}vgoc70AjX|3DTR$&X8A4mD zPK@Pt$4T6{JxCK+<`eH#qqPL-NjHv;hes$kWmnX)za>`P4Wf|-&fq~SuiK@~1w!7sh? z!PIu4Kpplea=jQEr@6lGY0TjTl41Og{UvZ|;iY#+_v2NFZuM<}8pP_$ISqk@nSFC4 zIw$;2>XNc$fjZn!36SHPMDPs63i;~2l!kLs-?iOP(`RY7uVIJI3~*G#A*mmA=LVhR zPG|cGD2_$Z`H)lmh6oKa8(oThB0DLbxmkJDXO5ExHi&|(*eA<{ARe@*Mjz5jL9Z;Q zk>0qwdIAV3{p{CjDl^Y$_QLfG2~R@+k-SMxSMpVU&V6 ze~OHCT2(y+bVB{+lXKv(w>)3;xQ7{F#atEEH39%GkW-)A&V?Oa>^ag_e z=6bpo(O=*tGQu!JGz3NDash(fn&0sIhQ*xmy`Tw{kxILj)QDovDzFz(&5@@Z4iIIw;HNg|}pEamU^^zZ6=9nHDG^Gj&i~ zoGlK7+o;khE(jc1E0N0MQQof;Lt0<}CTY#XxEOK8$%LFZ<`v$CB(FafzM)`nHPgPN z&}NHWIy*UQ1*mUKEhg!6Y`y>k5uAWIp^oRYi#(a~*3T@Co$SUNY$h(&Cr_}`hHw*4 z5x>ETsNZZM7A3r?o>Z^)=z+bV?ctJj(9(#i>AA(+(U~ZIzkJUFI?Y| zF8iunPw$qhsKq+r;s-H~N271wccI-j%?rhl4?Q;>J51$(yG(c(^CTPpZ>J=iJ=Zu_W#C!33oU!Yt0W z1~3fGq|?1=?87NPQ~~|s;*fp3=e0LA*`=r?`lc7Qal*)tcf5VAXE=@Bn)$}iD6Kj1 z8yT^VsT0CM$P|aHr!B4L9e{BwPqNxH)eQR<77ZTHYN6Ue0r-hZ zCQ#>b;!p@Z8zcqQPP~;HN)j|UGLd>r-aEU6(96?(qd)2*g?|c$xp)BB=u#=XS0^4fvFTi>eBj_VRXxH=r`lOBCL7vj`*=TukC#P8fk~G-TJ~9 zBBg*z)!$q%@!+z6EXe3#ympP8;jqh0lAia4#I5paQZcoex2RTVsVcqb(3&wNvZeO- zLFI%Yv$?Ww&pnlVAaLk=NNV@+sK*QEz zUY7Z#sI&EKCKmeKV&3vOr;Q;d1SktXCf-xQG2{!#s=X3!oX{8F{1KlMXmSSrj z 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

|)W%9E&rV7EGXjnXi~E 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

WJAGnA8$# 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

XqF_)*4$m7@ zwqd#%Mmb?Q}NP7 zG$X|=yIcy*36UAY6NcB@$N*zS{n1t6Z_6t?5D+`_cK{p^@ITP{ z!C&ZRXx^<$5R?=S=@>ZT)pnINZ6=r}(3L5H8Q z_SM@d&~@j;?S8jwo3@E6?f5_BDl7Foxeqho4y4#yvSD|?4VKznm)u694WCAOPc`)lRDH7zey3u0g?7-*lKt*l;i`m7l~-s*3q~$BIqZ#gj`)t&BZo z_Wta>SC5H^-42we7_&Sfv_NWD<+3A#TD&<#_8LlQqBz&JJ3hx7pnQ>RCnJ(N%o-3K ze9StUxS}5enz)II3_gp8-5oSEisl=#Cwvn7h~UM4w+LYW-_(0(zQiU{W1=cq*wG-# z%ARN^yM3SzN3Yb*xUd$Dy{HZs`&(db*kUkJZ1R^3%~(jhVoY1TRg;MA)X&*SxHE)* z>?isc4%s~I+>6(kVaSHD==+1_{5yLAasHcm56#6` zmiBsU4<=AQ3gM3dYUwp9T>*T2X6s~5QkY(Fa92}%YXCZT0OF=IH7| zwer>QXWKx%D!J20=2-gj(6x^yj|Wvb)pKkPJCx)gqSvP1I6Es(*{9i_M7k6+XYJfJ z@K*0vb!RRV{E}Nmq2E<>Fn2sCYvY#?ep6h8<5tj*XU+0r6K{&uGB)VpaMjN%IWBq7 zS>*^w+LgPwE^6_wmqxux+0B3e6}R+U0kr=M=n0%6m}&kWa4Ix6Yt3OfJ2L&V7CQ(b zNa4%W?CKV8S8B1d+T{+OdM6{h=4%K4)dT`frwtw1ui~SWLoNlx93(4%N5?Ol0|5@h z3(l4Bdxx`zWZvG3>H#jn@`D47a%Uu&n%e#nB~5ee?ze;evntDK6jGgohzChs4sWRV z#uP@_bx%p>5R~;Ehui6^gyMf$-?>PXXFd*sE>-+cL1xFISx&M`r!L1A*T1+ly zoa_V6x@1qtQ$FD@9!Nlad};5|d$IH=k}of&3|WlUM9>TQ-vC>e1dDuB zZm1rl_BXXug9f~@dJ-0n_pw1|uy*6xA+!+YPn8#j!>Fqo>D?2Of@X4zp*-0Y2pc4g z7QELNW-$4~_rJ`_;(mG7h!+hnie=kqHcwu?&SPKgxJO-Tp(@2#_Pvh&Xf%g|JBA7b zv=q+-TuWit^xz;RYmr^z(T%>K=K2}hK58q1Po8*@fhveS70o6jgN;~b!$TqmvO0%m zt7L_xs{E%K99D3o53;A|>A4qa5F4w)7RbB^0Fe}sLPI}e&A^pZu(u-Wfy@w5{} zdlpHt$y&6+o6(|gmHYw8OrNk`yHXYilM$sZpt-%Mi9O73%H;n)agHXK z8~QIuKO}wonZ~EY6Fiqch>^2~QY!qG_xG(3G5K^^^aBf=jN0I3?f`8Gxpaqq@9=oKe#z&a1OFza6JX0$T{Qo~wi zf7{3x-co4A{o1wwiKaR@{;iv9C;Z2Bmg6q!{nstcdm`yaFD(@5cja2SOZ~mDRLoHKIY%uP&YraLvI+ingPH; ze^32DOhCCMe_^Vj1F94WFeg!c$QQPlw3-0xRNq0T0*@AvLMQ|X_Sx8eeJtxRobehq zC>O;@;Z2ni{y^)m5lpMmCyt|TxT`=Pu3$T$U#PHf>23SV8W4>4a~3?uj)Y$CYpgsMgH5j}HLnjKO+ z_lBxq_>~6Pr45XyU6>)>0sOes5NB^_3;4nkH=1kDnO7A56(~BqX!Q2-(5G zABO&TbqzxqXK&Cc1j5>?wg8DWljCiMAvZ1c z*l=`ZZUCsHm{_ilwFA3pgN#;vZp*i^-jrkuMDCgT1EAdh$HDl3a?Acg>q3vQ+{{-u zU-|q30<-BkdxS2mm$k149J8Vn^AG+Dg*5IyFMRffC$1&#P=jAH>$Bm`Pisbde)DJd?zHdIV!#Iz2LxJb z@&Wko7ka-}aC%_i>lxiGkVZ`K_XCNXtqV|^<8hTlTlu*l-2^DFr*oD8e;vPCKvjSZ#10>W*8b?<+9qT1ht3ZQ*yN@DVNU#jxNL;ykN zCHJ16p@Ps;YwHzGtRrF#9*i~+5=DlC@0zD!6d%@pMNzdONK66UX8M}4%;+Ce$QX8^ z3EWh}`6IMuWqfi{C3Wd987tus=*0S8;MpK=v0>(rRrW(x{guAJI^yi&irEertC)xo7k1?r{qtNmbhEE!x;4df zDO#T&nVZ!HYa2g^8LlL4Q^5n0_JN(;Q0k{9b_e0ut zm>B*>YBi%I1Pk6*y@VQKLn%rmI3tlKTtaLSJ+)7FMFZVm&f$5>^;gq|Hpj)Oo9ZM z@@g&Dmg7Rfzbi2Fv_)N!AI!ZC*!$PB;NKh)62g!%=?kLLo&9VkqXU8q^J0#46WnBuCq?}W(fEy zp+=cUok;5@44@fG2vOqExwFPYIGc22Su=G3-k1OYq~_lv6L9oTKt9yx`}NlvffA^V zHavf+*jZni18I4|)3S4ds5GA)?Mg@sT^!dnWYyJfaSV^t)bfl+U@q!s<;c?O$V%*P zf=tMTBsqS7e_@jy?vFmdA^P9v!CHI*n@Q$XdC%32V9!dV$4VXz{dqz_KT~4M${Z_- z3W-bjsiPT|$SuS3&p>50kLvlGoynX(El4nnU1BOGE(<5(6>Cj{HMcm4SK;7W-sKdr z$mU)3--O_X?uszSK34WFiG!8*IL?1f8ppV%15pv}Q^CSF|5G_I{gYJx z4_z!6owg<1usg1Bppv$Zkai~8uwauZT3Fpr@lWwQCRG4DiZ}u1L&6fM9f|C%IGO_m zgK@N_Tj;g+ezjJ9xUr{~lS>rjJFf9>9{mnm4!{VH_(Ambz zg+5l1ouz#@3cpn|J(x@CX~GZr<1{$POpRitdJK5V6QaDhdHceGZXkanW%rzy$0=Pi z-YlfN=iCl>fXMf*u}4kV<%cQIQ>h+O7_c(SDDcgJycvx5(<0WK^MliLrq8@-bcM7z z&sFcJRJxYAlopgvc40n=y;=a2hLq+n06_M?k~!e+-_-ef?Stu^uYN%FNNcR~wrKub zw{4Jr`87r1Z!O+ji0SJIzGq$>m#C=++)~k));dV4QO*STr=^O$uu3BQM)PyMO`PlK z)T3BT;nTEXiEH`iq^`d^!2eWOAo%=B^NFeRq|LmsWBX%@dS_P+_eLYmoM1y<+S3V0 zY@bs|W!Cb;{-w`u6V_0<N_sihGH`FXQ@MH_B*gCF7bBFa!sGW^}UDy=+%Gc8R<2X z@L&Mq=l@T{JdjK8F7`}5a0avCHBwxh5#k*1mZSRFl%Fc3(C&!Xs$IKV6ynRd0@sof z#TaWqK8_=RhU%k$U(D5TO`~607}iW#hUZq41^uswTQVR>^{CfD)x0i0RDuv{1@t%t zQ47F6eh4xF+^$E+*~8fU0TK+N+@sIp&&7)jiU{W4*Q4+6^76Ga2rC`o4*D)dEm(YIGp~kzzF-tQE}<&Z*lTJxA<>gh*4>$(cdEN!cuME z$8<*Gomih*<7;%dvCHKUkgSZUp*~O=!S#n390j|0k35~++|mX`&%YVC^CRFq?xzsL z6=x4BEjcc}7=DO6i7Xs|Qyx-?AeCMJ z{(fao9miJ+$$53em&7)`IX-|jIL&X7_zYg1WoG2lP3YP~TiU`}`?wTSda_>i5KCK! z!Da-CpjI-0-w83eGJdd^WMenjp*N21`lHs!!3B=^JMl>G4_G>|v=TZ{hn16+=lm)u z)yRs6anpsDIy=_64zQ>1O9lvwL-L&hKGy|szLk^Fb1u*TXqHFsEqXs7-XCgFu$TT% zE71O>pb3tGvT(GmQ)#8g)?FTCe6BaiV2Ubi=h?nZ#6P0*M2rLIT?NFyVDg+7MfZ1o zH;akZ37SiA+#6sCtwe6gqNk3>zaxH8pWqBjvVdI808OaJ+$6IM+f1!)^y)vipA@CKCz|>DVQ*A_FxVPA zW_Ti*JCup*&?W)eq~lf<+zNIAAUusaut6^+m<3K?Rwc{kAB!2TzS zXaW)4{5h};+`QdtLTVpph+QcLg1EVHmgR&@5P7Nx`~G`_LZOFl3tQi{6&{LDXMIi4 z%ZGpmvBkTvKfhk2;U;Byl4#G$lZ?=%w&_+d$R#R<60vxab7?CV0L)9$#vU1~o%S(r4yMpwJo&s#Uf_K;#`^;yGjx3v@xYb? z3mb^>-X9MZv%PNM<1%jOLIql+%#lm6)aQs5r^7aQdwtl6Kc|7|vm*YAUhn1VskkG*71bQR?E&8i>yKgiEenLZgdzl^zG*cHBhaIX#`+zxM5?d}h^D zpA8|Ythnd&u5yJ?Yhc!~p2S3F-mQbHbA3$pu;40M3O1jgz{2NtY(w&C#>JMRpPij= zZdr)7pVnG0+HX2}hY6|1iG&k;tyMmO!4qh>x`{4-v)91ko$rx7>7fbF?9T9TdrA>0 zH({Lcwz6h!RW9exc%cvOpWjxQotP|4@J$Anto#6#O>bBeE7zfd_0n(uxT0jU>tm{= zHNUNF8FV7Waz^2IHR zv*nmrTNZ94i-H~=kxe29uhV%vtz7&)Ex$WFzoEZrQiH-u#;1TcvM_!Ki6bG`M`;lY zihu%{mUWw;xOg#^B5~>vIdv7b4E%SJN$k`yiU2=Apetj~Fr~p-c}Hm;mIEk8^kB?d zXbAg2uMZPr>YG=Hf9tcfHRcj?2+Sm2I>V-1h0lWFE7++8INuk?!I}w@a~= zc}Ln!IU_kJjjyg>?yf&{3ISpUN1;o%w}TN{JJVx5%OfRV=dTu)?q#Z+|BJJGh!SNB)&yL(ZQHhO z+qP{Rr)=A{ZQHg_*;Re+TW{2(UOmid40i6!i1_}qL_zH8x5KyQGyoc*Z-xm)FrtLl zg#tW~s#Hn2k@7M!_=L~e*$KCp{&tQWm@^!Z=t(_b*xlXa)w1MadDJ`GtbG)xHegA) z%ZxA-a=+H;^ro786QcqU-S+w7{s&ahp1HlozX}b{wBjNhz61x4vwxVE{&Hm zhD)+N_$=p`b;Y;Glri4Xs%|KF{+^nfy>0`T#yNkiT<+nizWo%*_%S|fI@oHlwy)@=5|u?cPBn`+seKSauIr$oP*wI zQOOlr8n!Labi8m3h(~xa*2-3MXl}LwTd{9|NoQX#bf7KnR}snyZL6lL7xIyX78BFIztYRGvdzMRxFi0!nZh} z>P(u+P!o|5$Y%k1&fr0F)edoKdsH+gPd|i2jctj}YZi93I7YGkMWI?)t1YYFTjO=w z+R_4USv^ff&XF-DSQNB4;87dg+BH*QiG*SOk$c)os`zliBbZGu!o(5&m;cFBHI+!3g9ZG7f!Q`?@m4=(C!?DT|8XrW=bzrRTp`{bo z#AsOXRs;+W)w6_vZLJHTUwUDDoP?Y;VEI~X@(}TG1jb`sqfJNWcgl_8lrhe4E>x>< z&q=YEGk}1nx_5E8PCG|wmhW6eQ_Y=NHd4YtQ-O+6ag77bZwQwoXro(+o(e+G)wm7} zuz6>^&-TL65P7+4YsF|e-z@T>5j5?o`SWw&+syL)?l9#FoM2}n(sa}N&=$l^?bedD zxUtkYVA5yrx%bDO$P9=mu$(dG%z|7L9GGWGkd!w!7fi4wi@j%7$azGKO((LwS|zH^ z5B-BboK6l~1X;WhyXa#mLDdLAy8;9RBz+ER&qy2HothNl22ED|ODh+J%dcQ1rP(mS z;;PxZCrxy+Npx^`MNEQRm;6YYr5_Ww}FS;_R1hgf0{q1IP{wp}Y}t z;(g%{k!K!9@1**ch9~>80Ef7uS8@7Z$9hg}uE3<}^e&>IEBpw&BL!37Gg6U6WYaB2 zN!|K}BDb*nBIMwxMbF52k`>0kn?;>^=mTr=XvlxxV;EiU^{^-KsK*&We(x_A2d}0f znxe9#|5DJEfoIcX+ECksxb;9D|W^oit20Wx3y$bx<9tad_-rR za5gJAo{7$>e-Wj>)f=-8xZ1-c4ygU}lvR+hey*S$Z{j!)SL*)Ap#vU`xF&Ze-m66; z(Jy!2wV>?dMnR_6qd}W{ev-K3=y@M+*{}|N0+xkp_K_}B!2&NU#CW^tKfxIiu9R5>%g-h55>B;fM&9q=a?U*wYec)*0uH+ z0!+e@;>3s!WuLHNJ6XK#ydE!~AP5v%0hiK?jbm3i56>_Cc$r|%+^hwLTZFFI=J4F;SDYa+~ zXP$GHVH38OZTj=kOFbYG)33KL##HbsiU?Nc!Owr-^y1U`bF!SbSv!{aY1}LJONPk%T!b8OQ$?nn>09@eJ9&0?o{>2sKEH0O9Dy`J7gTL!+?pOC&i~;Nd8!LG z!M2D>2P<~}uG)uy>~!|=%x5H*I+=hO!b}mO0V>g$uQ4;!D-0buCbqeG?)*ioKcbh{ z&4p`-OTw(Dl;|)^fDkaFcbeA~XSrNXlhZI7xnt<)Wosu+RpXDdqfXV;P7ey3xiNIG zqT_*4KE-p_!CoJrU2mrNG^p6`K_)9{F&uH?Q5~(NrQC^GND`j#ET1e?dA4sj|!T{ z$@viZN7{(p(Hm!K0Lppzap5ZmBW~xj(xM8Cixle7GnKAv9P~wuH0~UJgf^j_jhd4w zFx)-U8^t7Bmb+@ghwCGfWg?`n@1Zp{;~xJkXS4?!Lb4!zR(O6zFV~2b=@zKGPAHKi zzK$J1_kC3iw;jsb0Q!chq`{*|bKI1lmrU)QOPQ=j{YnN-srvf3pGy~rB<*Fn(O$A? zW$rSJXA%gRkOes<$vTd!?;uU(d0U6(Okpu#&GkZk^%10exQXMCzG*_+6Q5N}hR&VY zU>+GeFTv;Y35zAV5Z~%90F7<$*+@d1E(tuA0=`z*w&R_>^Tk@3kjI9@}%3qM_ zrvNlZ$ZK8)Iqs+3b|7$SS`7|Z5dsKm5H}5i%z9GJ<7P>9C`JIM)_ zl{(AW{#HbahiIl2oeD5EUJm7v$Ifs`Xx-TWaV zk%ddlg%a<-{)pvDa;eV)w}{+ZYLks8fNVfS1C*E&%zyY+p58-Uf5R6pm|kUi*a8q&Xd(3Q#S~un(%kod8*wSFSae#$l~pEa)#Z}@d7rw zAGWjco`#xEPXt8PiAVO&K*EPUAQ_FKsWuHI|A+>A{*2psB%mpwjUroN5 zi07Rj=KIM{L$=#>m_jIZ%)qSo82pZ`=BlWZgzJj#^i0q!4&*kkJ;Vkx@=;=M^+{uU zxFU)=AKk??2;CO*;7k0=EG2_Gbkk0o6DF&&kf~*lNj~|Q6k$-#H*_yyuU{(&Yx5EK29v-dkp8uLOfJ*uzS7fBlg|DqgWt^x_d$=R zwbkc}ylBpSDE+W3)6D=Eq5{E_>vha?0w;--)-4pVp}*b3vg*H@96^3`WS3pLDlS-R zM%xsTeEL!rW=o=sYEyuC9hSGLoU`Ux!b!W*Of!`Lzw!HqXv!nhI(N1oFOd)il&jUX zH`ry_8j$foB2-?#0i8$D$V<7Xh*Ra*?=+>PSX=w0L=BbL62~lgY3nn2le1FFHf*_i zrx|zMm#YCxMf0pfa_w7(!FGd9hPN4`n~4f|Rw$*A*01H}4S3xJdjl#lUY%wqdD@0r zwb}-~=px5QyY1<6!$a!=n6XtNN3sKXG9{$!QAF`sj-#9am;vPvu@m#S=G zXjb9^0IEgz{wyi2VsID-0D}EPA1kv7wRMj`8#lK`yT&mzRJrp+IXB2{c;%A5$ReXn zk&9$#N+Pyg+$1?@7A>bzuEf>iN?%!v70VPl2xih{dEiD^|MK0LPWF;6ouI?{E>JT- z9uS6Xo3%L6-~gkUKLLZC$(363q(y-MBe>fd%^2AZz}7#FNx#jT!H|tWosCW>@SdK0 z^fcKf-`Q*oq47H9690H&`LgbK!gb@7TXe6wGzSmL`^l|p1l>Wcag7k0I=~I>ym_)4 ze5(3ri~47_A*MLpGu?{5q8IS4f(wp71DS&65>$#arq&Hua&|~Fs>>u|kV12`;$k0@ zXL!gWlR3IrCSO1!q#PzQy5gf`HCks&5XqG(w4nzkN0UG?e#%1->sOP~T`4M5hjMNv zXFF8(0b%ihY|5{OC`?vlHT0}NvezWy{!Ee_{}(=@-l;C!R;I}m|At5K&U8tB?2uJL z<5(xDbq?h(V%zYqc-S$ose~b#R+qrc==*bo!S2;&2Hm47fgf$>A^%v~8)GvllAM16 zkV9~)rJ5ip~5dNm8| zC3jp7pyVG5Bc!DA!c;G*d9gc9WNO^Dzj2dr8ER-XCRHx{-06<*+yXbK88F1{S3Rb0Zo3(B^AHINb*gF)e zMv6CK@if||s${H+ljNt|@=t9~!&~9Gt>%lS`Ng~9G}&&UIn^yqO51iecH9uHy@mC7 z=eC&lky3*^dGgM6vphC(4*Kd*C~Gx|xrOBs`Oka`hOu>0zRGjT zFwl)+E?!<$VTm@|6WSb#u4g}PsamLZS~@_DBfLbeZ6P4n(f9OqUj<8n>$*m?<{hn) z(q)`QsebGH!*1|ygJ0G{7cVihfkGnxiaJ{qBobAha$J_ zQE_OuGmjp-SI7)9h(3fzF=Y$qT50mdj@C3`JzCo;X6n%&0vYy>t1*4*iIwDv4%`Q$ zh#fr9_V=F{e>8K}V*(0e+M?b5Sz2-cK&RCxK_0Ji>-Hwi@Vb{hlQ6-Ze#x}TQ3 z$<-LNbwuK$Rjaz(*cpgoXhP;HcgO&Yj|RA_j=a-%x30Mr6x5NFwBX+(SadQpFxr-pU32~9h z<+c5vSik5MOKKGIRMOON&!l#c9juG&=3y-wFL1PL z8W9>eE`}K4rq4P}s;W82U7Q&BS??ZkL4zAtZS)u?VhQDOfV_@7T0;3kb_2&*ZB{@w zYep2T>*7exrqigAIkH?iMEF>2pqx6Cqbwdkjc(_;gVAJ}=;t{FjSXRYZ9I$$sBag5)+pe~p3;51;6Ru`WMv@8a0w z+y=Wx)vqEJewN+rBE0yqsGRAK}LBIP$E9sQT zQ#JQKN~=Nv0cLcT(MNL{2wj|Dc4>xzI`4F{DynW4@^rNcu%0wqFZ?CYwUA^6LKgLz z*|^r*FZw*?Fxtj&Rv{_ZK%3;62rgK6p;Xff&bk4^gh{bz$z{q1ZV`L^3V@8Zv{$DnS_bj@ zQWcY-QXblv_d%X({K|f|UbFA6#8%gLJCoRWK!*}#Dnn9VzoYVO|iA{_D6PFzxrIe@vlSfO$VHBY-UjAaYz+{V8H z4(t4#XdOS*m4?JJhz!~b{F&O8;mE6Gsna4b*JI zFW8Un-TM@2A-F)EUog_N1>riO!qiY98LCh6Vn!~o1&%>rYm2p9?Z{)5KVawsiFWvt5(7ys^~=x4ahOXail z;C^qdiu&?$)bKavYo-{fYI)b-&`~;?MBV7_KQsS)cN!)NY6ayQyCnt9b|E>u4B?Bp zlEgQ^Ox9nqyN-FHLAOa%XnnJhLUPW4mO2eOsKNSf!ZxB`hxlR0n%|_@6?27Uhg#P+ z8M(Pz3;Tcw8OKb&PqL}2<;)|ozSZN=nxOe^7W^YlafMqzES$7y58?+=vkTUc3=EI^F)y+6WNK6B~9uiX- zmj#KOwUz{T#%U?bnKtYfE*dt@Z}9~DPbfd5>av0BQ^4!wUSE7Df>6M zES95)!~eig2_${@8@`yyf`2dAxSxT(yI_Z4dNG}H(Lk4z)0r6yIgFTd9?teab$2?= zPwNAaU*7t$QasKuZV08^=TFfWX8m-Bo66r|cBt>V%Yv>_>IejAu6cH)6TQO@>cmA$ zlTH!m#we?xUQFu04Up+4$wo6U*1|`ogrn7Gi?1~Y)snhX4((eFns`VTF}uJj=)6`_ zK_aSN$>5B!arp04#F9)sy&+g|z>fjCr!<8(>>0po8c$gFE*U3eF^5M|nq9#H%LFL` zS7rKx5SdN(D-Rh=STGT~mEj1k@3$?ulvBSCdAE4G1>;Qd>dc!6V)q6BVA+LayGOOQ zQcm@$0C?ro^igXN!J|TAU{Fv8MkcPEdJSD{3ehvTQ@II$#d!m=V&3zmXkoQGOjU

*>Ke~B}Vpt!Q2inq3L0ahoMCaF>4Q3l3A*qSw|dVMWcblZrU9K z)sad?awdGGJzW3n4J|&vHx`}^V?1ES@4S%zN7ggz^NamQ)(cN}J|Qi`lf{%Et} zpFzX;4%)1CrT1fNXa{|)x;1saMn}-u(HpctECV= zV;Vq!)?+%hxkHZakb3IXjDfP4=Z4mshwQfcj7&T7g#a%$9-|MUdcWbLx(R74Gq{UD z4Rec$@h#c4O$Qr&IoT{Fi;z52o2o8ghs3`oUCyl{u6CY2LM5vO*YTUZBBP)Jx+SZO zAnH>0SL-)pm@!a~$+BJ79GYwN2-v{MT%&gKCT82r>gOWsQ`Y(MORTCOA#ANVzrnF{ zrog-92A!3>nGDVUX@VeEk>xww>d0CpyI}R4{0qAol2X8~$b6_-cr04<17^}*WfB>4 zXk5Atqqy&mTx-u97LSTP0qriZ2h>*oPWbRae|=pMSKyJ}tI%>bWGEDkUJemjgl}k^ zVgi62wU)z1!@mgMgsF2c074p!8$5+eE=u(#J@|77<0Qa#k?QV_xNU1N+Q`@tIl25D z2!2L_l+^&=v=lU(g$N;SZS;W7VG@}acSeIdRS!#vhWi3xhK7qIag3_3>3t0LkIt3L8Y=>;d?_btWNu2jVR z)>`y#Htz+0LTZYnJ{fZi^lvCQuHybF4L*Nhs&duRls4M(}a9=XI*#^0cB)N^92z|h6 zoxc3_WcD^pZg3ALBN)(fy(;T=k|fsN0PfDjqY)HB^3Hy46NK}^uGb;Vups&r`?{_zhg_z4s zssrq@B{L?7b`M9lo*iw-(5KO_hiF9*ZRV7K|n3|IR_o~oSp%pPL&MUvKj2!I;;a{C)_7B7sX=cf!5e#4&=a& z)$x-qYMml%H13#Jwlx-pmG^-E=ScQeoR+{t`iFIAoL|A3rbRG}zJI<(&g0&Ru#c}R zpjf7gGx-rY&Zvjj!+oyt54=<`IvD6LD>)YSO{ZV)f*L*b~dJ0vM5H5OXrq@sLoZS5KO7) z#@yl=`*NwA#4K_9a*+C;Q1=3f6tjBk}04qV>(j1I;J^UXqfuW ze_@Wk`I7J4_PYncy38Fi-bB16{*b^_t#L`8SiX`^``${02`_GFv`R8g83x_8jeFPMzwn89bFI zD-XTt5RSM)?{=Fg)L9L)xwu;H{o%gnORh{0zA39Uq_^!}k93}YK^yY$sIMzXFa~)q z1L2p)VvLCK#xJ{9>J}?foW+)O8rh z$SWfGlJ5!*>waCA^%II?whc>t0UX)MI#GYm_t*Ca!&y;U>PfN&K|uaR&L$Hv@!W6x zv4>JP(q%^AhqCkMZEWb$8H9Xp$+0Pk-PJhIS#J8B{Y3+hKQl@Yfj!Nte`H_YOG52U=`<@kCb^MoDAUvazz|>8pEfo z=5eAk6mwbf{MWy&R{3@4pZJmP=8sugpV4K!GxdXgo;!_Y}8HY)& z%ohPJf$M1Th)1ir8YSQ*!0Z>;zf_gWCBL-UegWc5Xct1Q(-a0(#`_4`_R+H6T7r|m z0geAtTnM3^RH7@*8}dL&+?Pk8%L_e-;FO|4wI`9iA$P^PfMJO78wU{#{@Jj^cqvt0 zPwZTl;UB6(^*5w$`?Y0X+7jA@@dMbd9&d{~B6~u@7_d0&RX+rgB-kER`g9%)@iY4Q z*LK-mw6V_-I)EB+3HsCA<5a}xKwr--&CaRIv61i^L8E?3QJCMeOp(cBM?wiy)BrxE zwT!Dqiz~1`pg%MDGUCgFe*02)PtBNZ*shYjC3`R6hqX5kv^hV@w`SW$8!;t0Zm*S;*;U$r_jY2vx9rq@SUBh6nx8np;Zr}Qq`iy^j-;n zTLjweZ#*z6QX+~JO0C5Vk|Fk&$aEprz2*y;C1aG?9$G)zW8=iz<@#4^tnq5}z7dN* z{3-60n$MN^_d}Zpb{b5bZtY};%U7}g{YLu^R0*kEuTF$v7_Nml4szhF8(1}$mejW4 z>WR41#J7t?Eb9zP&Z%sV*sE9f8aJDr3}u%MrF^+okMacYWF*3>+qO>yMfyC_)ob19 zcL%70bK&bVtIx&eq(=1s~5W*M&G&9pYiC z(HfO!T=% z#D-jZlTWQZZXJi%`8K8_I)lZ~JQL!}3m*`ELV#_>$KBVa3Ge$Vh`G(!slzByjPT@Q z;WV-dOw)%V3T2VkGAP-82ME6h@vG8YWSvQhNMWlEy%np$s)ITb-!SKMk__^t(;6R0 z4myKEmo_RfpZl{ySo($X3%6QWNG^Y;bj@D0eWyqC;p2@{df4u1a6p$C0CqvU?`kW@aI{Pd;MDPE|tX0Gw z`>}Zi`cL#820_HS-(C>bU|(rE!e=Uxcy}3#*ui=0~lYZfA#A*lq za{ei>fd(D}I$sy24UNu>OoQp0lonpySM3V>SL5L2F+svFeqoEC!% zmeZP@#|Z*Dxn^?(HME;g&QKaHe9Qs<$w007(`DUf+l4aMxEUkwE|w*Lpw2aPH)~zB z!}h7OQC7Sy<(zW(||K)I|k_I(Z~xbx3q4ymVK&h(K;LTu416tf91f_ilhr0X`9tZ`MFAERqvb~ z3v?QB`_YjI)!}0CCWIOX#KY|@+>JuyCIXLw#HUgwQrVppUN{d?l;S<*%ALQR`)Y<% zsww*LMIlSq<|_!dInxb6Eh6VLN~Kmq>Gb9L38xM(-^=rzoA#T((s-#TrvFW)oi_|f z_ekLfxs+Pzg_JVM~B0dv#5X>!_|2ZSk7O_;GiPXaEizc96hJ zXa@r+W{B9jQbIsyV7pt;@7(2KkZ959W_7{i)wAQqZXf7ODf+ zzE_i_g2c2n)ErOFHkm>9qPB(vBXF3+x7)4%Am<>;KmC%4$ep#{;5Cf|fd(cd>&&A2 zQ<-iNbBPv2*MP}HLW)_I*fSC(1oJB25gkJ3L3kT@G$oBF!qk;e_wz_(qwIVECcMK2 z3_uwmA^ki#i_2(!I{uj?3JIx&yJSUxmX)hLCt(?NzcDu0Jy{mZo*EHQ$ltMAgNsN5 zVpz2Zl0w@-CwW2fZwgv;Xk|GIFx`e*yxp?$v4dsXCHf6yU&|JItos8Y|H9}aV5|bX zXa9!t8+@1_?H93Ilk>ajvrq>6^6PFG z4kJd{Z@5qlu-sVV?N_@kVuG7Z8j*BnEDViaVU#V(hNR7AwYZY9k6?X5(t==rvr27p zQ1u9tD?R}w9i20aVR(&nep(ii5^kZRpQ5{qnad%K-epa~9`$Q!+LJ%LVuOKABwL`M zk8>|xk>9GPQ=4Dx`2pE~N4t4hfbxWqE_w?LLhh~EfFe3A66!Ng7n@oVK7WQ;mmVnZ z!{o~IB4?TWRF9T0SmnTFD=>(%*8lAV=#pOPDQ-0Gz?Cau)NQ$EshCp*Par@`DDV2| zPd$-uscP{xIKlo`h~w(~S9?9O(cjfvfwF9R{%ivk+UBij6B+}$Y~rvFyzP66tzkOE z&ea!X`I;y*CoAu7IJX9{{j^r_yKeec+*5cY_ z5{!oHhB~e^8=wZ)Z$eVbXK|Bnc29 z@_8?d0*lf%tvpme4z7lY~Dvc9wI|A*q!Y#v=u$89bfPJ^khj z3o(W^S!bCg4d6*H>nZz6&`ZlHkve2z?Xm-(O4)<>T$5t5p8{0C!A>Q@t^&)GcNklf zQ1drs?MZ3R9LwT>rC;{|4jINEqmrADscNr3gxTEyr$8Xctzzm849j2RKw?O;mlPjH zM>E3^pC1(<8y#D*Gr@w6+Wx28r~k%BI(x#Nr%TdgF_F==L0Wkwq(R30}`ltRkiHIE-RVFSzQcha)>>2$#79}q}(!mFkyDX%C#<`yqJIybq-5{ zUG977ZUrjuUMm%)U-f5LW~KiqAG}KeAaLTV&78L=UpdbgbE;o*58RnO?L(j1biI;# zxHF=xbo{Nr@?TX^2Z=+N#^A2M1Ls)galOlX4}gvk2n)tgnuXXl3d7-HpmB@YRNKrP zxiuNDnGq>u6^LGT53TI~xU5<59p6uc?sAIQXh8CcA8|SkI^~vAV&6x2-qZm;H;>YJ zZ>h(Xsr#pE%1sxW9div2q1eN`Hx(aY~UIWe{pq>-UQQ z=t@RsWH9C()qoJAvQg~?rXNTtr=hmNamNxRK-U+CQ;Oa zOHFYe>fuff_iJR7!{yn}(^@KdXX=SzxBEU~5@@-b2Rp38a z+@rrbjkauakk|&Za5>hJ`w(0-JFV|LlbCx^+yz4nqlE$Oz8fUTbMedF7}amRu=R($nbBVJR9y?(>bHizh<;Rz*_rfDIQ1 zwNn=96Hl=x`0(8EcKP&rX!v84C zNxoK=cv~V?C8z(v;D%N$pQR^l5hhBE+ z5Mr&63O+<&#;w;q<%owMl;{Y)EADRmAMV}`ekIlunR=qH_3D@v@IpvY_CN0atJZJv z3;5Vc#l46SLb`x^=(#QLun+Ir#4y%>>M4kp=d#R}E4zcCgVJN|Eg(Qe#U%b*G3c zU+)NQZl>wy(+G~Hjw;gYWH%N=Y&Dy>LnF@G_d;Yl=^f=@4`R;^4N$UG=``gglI;qJ zW{mdL)KDB8fgGjbMMsgd1rf~Saa1y{A=Ddw zc_<8qq#7?ZmyZlb)L^RTT63Ts&h`BUK`t~Ql@|${k4a|~b~Ktd`oGnMJ&dXap3Ije zx8EHUR0W$BO2ShZhcfV&0*qOb?(`ksB@9Nopk3i5Rq3mhYjFPSs3bIRh3D=P26Yeu z0!)TxgImN#Uc@qnzZqH}nl(+Gq2hSpn-}(oUFpzaM|H4@NhR*N#GZSoy1}7fzh7D@ zfRbqxXxjcrvLl-Q2Vw>C2Ui{#QROafr__yDnZ`5!R=pt=JKpc#xtIkwW+SP-4=@#H z8hOF9v5r3uZfUOUe^>bUD=>XH?d#-A|Nb$Oxxsf|VTeo!fTEV|oqlO%cx7ipJcwF_ z1Z0@CX}z62`i#~V)R_TS!O>u5M?vWka;w`UILkZqxaXSm-iYSB`k6ST>_&+u(0UGbIk zvphk#^85|Vys%^YPJ%WT&#=WwsL`tp<}^oJ4S%H^;nNgupCJOduEN3f^QI>=Dm-cIIBV+c5dwgvdXpNC%t8KoN? z)JE<;1>_&o;%5wP?foaX z@;ZS^&Ng}hUD7}^kkroN-;Kpf&ijM(0k&0A$8((60D-%|oUpfD^G{Ktgk^Z5H0U@* zL9wpErb=Pp-#%(M8XaZ6(fVIJEh|vMFMOZFCxPE6M^3o4$tZ9y*ONIrN}sNTYb!a` zWLS&Igd0c-6htTpb;jp_p5NJt_d3dyRkPip6k_IDSGgbBXXTq)HVh&+h7I*tYwBrr zGC$rmp!=x%*(x3%Yh7R%B4L&SQ2^I1m7vQlVtxNdKJxMJP`xE>va;&`YNZ8Dw4=u8 zRtYlLf`=HS`EEcLy_Xr44rX4F0=}KSll~|{Z05e$O}_62TJEFbQ1PD*f&Pf@L?Y#XkNRhtOUtG(HXZWMb7AH^KA{!>_51#^}f_JlP!7cMGiY zndy&sYx;7TK^^?y^V0*?K57yjahipI@Aq@%EH-Vn(=&tvdz80;P z5q9WZZ3t^Mj_>wEt7Tg=D(&pIGcDcdS}=V$VyLS9E0t+wvf*Z7D~pi2$6*Z=?k zBt&Qsk*6N84@Lo7>5V5Em7qX-qq{8_cVYHIJ9+~UcMO+0Q+8Ntr!wZAhl=qcSvGY9 zO4Hiwu~EOPhtxyT?iQJ-)9x=d3{wJ75_(W9n#YB?D9;jt2ICBs>9%OGm@wn~-@5K0 zCq|jD10&C%%`3CE#Hdl=@4zx<`t&_T24z!q>l`Qz-YUv%I6b{S8#Z;=q78`y&wd7)HHk1&~9#nWYqMHCljWg5Qd)L$3(*)p| zU-xXmkVQ-PHzqzjg~|6hjh*EO~&~5xbBM~fLzZJEomG8h!#{q zfb1ukoe!+Jb*HMCTC>Mo#P#{z<+>KLkPM8DYJnF9tlp;Y;1Woq=CumhETP}N(l-DA zg1!@|P0=%+&V)qev)|C?m=_DA^xblqB3@C~9+CTVg8XCVP!Z}VB74N*!PCQF>TBHs zl*af^85Ly9i+>duQ!W;?> zf?aBw?*ZhYA&1!pWPzZY4+?0}2z#uosr~)JEYxj`VSx1aVbC$Q!)rxC?>_DP7&TUP z?V(HPmnX`^axNgcoK>dF`arutXnNJl2t>;-o8pRaRWcHE&INuh^sWLO^=uTF-7?n=ZDS}uL~_c-mi*vFOnh{ zvy+c>Bmm+P0EA#r0Ra!hD-#@UB`W{`n5pjh$44l>Ys+vPm-~;l#w->&6#11C!;uVa zfFQmEp%ZD9^6K_L}ibd-=L`5Hv6|>AKlp zb*czpjlXFn2HP8wmT<59bpkMKPEB8pQ<_sHZ{TIk5)k?!FB_q57qsMMxc?ZK>V#qV zHRTq?g}S(GZ@&AQ;FZ}GQ5Eo8Si?Ng@V;1kt1k|9T!O=!EGprbTvqW^9*Zi5$dG4=wu%08t5S|0Hn}OYhf16}{ciXQcxfuJ z(2y#ik|UIN6&VmT?j90QdH${e(|R(Wk#xUhXqYzR0DFP6ml#vOxbh*HIKm+>%aKYD&P>l%0>|GS2;NEdc> z6PFT{exmc_PxFKXf)O>_>27gvXYO|T)&Q}#O3~h;CvQZs_39{gEFraNAS7YEIyw)? z6e6x;Fq(tnb0D~60=iM>Nv*O%H=vR zMUAk7l(vFSuY;o&3(YUk;(48NUQEtEa97$k@>XoYj2HPDs8Cm^n>)5uI9nTo?MqxD z^Vf@i3yJ~U>m36CM=}R^Agv?gkoNnhsvh3{^CQDDoeca7cG~CkWlkZ5=}qL?UI1=c(orZ%7Oyy;1sx&f@p-Bn zm)Pi8$hs8Rmv&Y|6#DnIJ$fH5wiA>A>P%bEx2SbEcZYWOI2qieAH^!Xye!g&F$1nRrJ z?h|Rw*lVd2C2dA~ClPX*;eM%7FsPJ@{)B{n;lEcX^Yol!5nE(h0nP{IbCLD65K{T9 zR6+C3ebYh9taCMY0Ac&>f068P$I%4UYNJ2e9VdW(^yceUMX6fWHlkC!Cgz} zDLjHHyixet2Ke8fH0qtyUQ5zdx)ml_3H%~0`ABjl>44e_2jvaI8NCG9^&Nqrwi8r2 zIf9F@6&^U0-mDr1WW8ioQMSJ`okflo2z*?JjQw=Ct zRF5!Fl@MqcHJyT0ux*pD%Sr?Ee4j$@hGw$S2z~gzF+~?x%DC8Kk@pED_nW~HB{9sZ zVRD0&+8Qru()5YBrGBuGGuDN*f7ld`z_3~#bEJrKa4@}m4(9jOVp-ktMjn&PB*VMNYciy?sORTdLc#L&_<||(>!7@sDK=M^ z{s~K$UF-j0?Vq9~+qMNzJJPmoXI84xwr$&XrES}`tx7v9ZQHg{d26kG_CEhQ?cc54 zhx-%}kE74thhofcXemt)SNN^jN-h`L^b0A>DP_UH4@5agyHmO_5!IhJF_tkakaZko1E8 zFnv~!X#5h_1FoTt0EI>YSLrr=sXqG}JCj~{Bc{&I* zE`B!zDM=ZKzG&ZJgA~D8Xe*8}risc!`b!;5LL!!MAMge`Vc7d`y$uK;kR$LT*!R!6 zls>>z)XWNB&ka9l1YYY{sJQ)52{1D$3u@VH{0N~%JC1jhqKs&8tj zD7v0=G5Yfd=_mJ|Zhwcb&=9h|lkcB-90(u;i*BETaDIUc)zS}kjdPChpMfML?=R{a zM}WyQK9UlJ8lJD~T49PguqoV7tE*dI>I0L(PB6P$5AE6ysXoc%|I%aGtBpZwoU2W3 z7*|>GPSI>;!ll#i<-FUd1^Y8IbgO@Hz;1R!I4Oy-age!vuB~8wN{uh4N21ea0`-gy zFpsJ)A!7;V_c>!CxNuGGc!#_hjE2akN$%AU_&+I529#s?H**Vf-<4CX_TB6^t%}V> zv~0FynX?REZy`5Qcb8Sc!Jv1F{Gl(>TbdvKAIO&KX>QPK+T zE}JZn_OqP-pAG-d(we@bsXi?;clllch-@${uT&u$F-6jmS^khfU+&E`fSoSvu!h~- z5B`$-tyCZQ374{4pGQgkqJykJoZSJBHAXS_3zopgwzt?Y4)<8R$L<2HTfu_U<7aGl zr=)HHuGmc2E^~wqo7jZZW*gS8I{yu4^$c>F=35-e#$6XT2UqmQxoZbqJkH#D0eT#4 zUWYm`Wo{J93HeTXP^gV=bI)yAbAw61`F3M&Y7^RR{W->H6mX1u;Ue~}KKaf+g9c0> zkYV<}HabiWZJHtg!rQrF=veOa>7%d(mjRL+Yr^y9SnB)^emS=L*xzl3VeDcd^@mK; zuw)sPkV}g%GaA6TSvPSw&d4g=D#cSzkx@_j0hq^EVO6%$c<=H%0Rv$W%vO7T!%|Cms z%UyAa*o6k-2bd;>gf(Jttu7G_Wt>ztEa|2{$-ay(DGSXL(#I*T?SmD&;Fk|yQDqRH z$E?!W z;q#sqfl$jyhUSDrOZhg>()b&9)j%Slyllwzdtj}YAzBC~T;7RSf^IODiY|pf+&~L- z6s))|He)x`2TE0+k+7hkB63aRV@{xP9}*a-kdaOQB7vPCds=+gf*>aum3riYWW_wL3eq{DK#kk7LB1r&lc!rCkk(LcxI zvnk4A4~i{?bvhfCxx~So<3r?0c$mVD?#27zjT+arRb!DXqAg*r`mx7+SSN4^#4dOL z`R7YyzMM4I9ygB;#DXm2A!YG$Swdt_pQwNerG7Los=}X@pFZf?K&v z33*ba-^o>E2_IYv{~lj7&vZNdzf~^n9_*NhF zLwfo0PQMSO?+m)3PZhP2q~s#wGe^|k4E*Ale2DT$X7H~sExL48ZB;L87Z_;-tdB7_ zsZRUl)vsvJ-uS;|mG$#1zGqz>h8`ytRU_*ejmi9mF^W&2Qe8t>{=?dT89X#)%k2f; z@Ob6u%9;3eI&U`dvdS1bC4rI}b6W#|P1Aj&I?s(p0U$^JEK*_gPmx(@ z1axTE;qKHB*kge3N#N`L1%2q`Vs}d(Tphc?88L-6DbVMQdBBdoib>IjOp@nWL`G$S zKL)kglHq7zt0GeMV-8J;(FuU1@Slvr0m>=)yT68IAFzWgh@SD3O&5F!xA8z<$n`v` z($!17*bzVkmAN2Gf1J9foe1k>%S&Rje#@(l7V?Hf=8gyD`1Mx(e>#SL{{$0g4^v;V zdTzHq1CFwC-R332>}?OqCv>vjt7VPtitX$Ax)lbvSsX8%39TH=1NYNYxA)V&UI%7$ zTsq+!eH>^0Lb(x7DGq6yM2IryqP7px%u9{5`td+y`ALz@Xf9*md%uN7oTNandBMr3 z>n`#dUyEw`=v4)p%tl2c<*esSuLN5q1M6|(HdiA`#g zkd6w*-RMF>o>J0sZ27x{q(lC+NJ+Uz;S2zx_@4;?-184}9F{-7g!Qi*EMCO)@sS8Y z%D;FsWE~s7cpw^*zTeIo&@w@H#)YEow~8G0#U%ZCKi&Ph!w8!pV&ya(E{%4wV$Gj2 zWzto0xZ3yq)#+KTi-@Z_=$WC@8<6q`O?@P??1iHNfKWyIQZtB2v^Z_Hnji<~Dz`M? zTm`z5l?PW;BPg6eJC;dw<=V&yW!iF|nsvr;Beby~|4;1Gjz^eJ*e4Qb_!v=_@G30 ze>gQi+qWk6xj1D##^i|B7==ALpJ-qruNl`Z`t~Bux{4?HGH;ACY<1^sCgJ}0=BU&B z=Z5k3ZtNX;(7`KHiSgUk%Tt+BkzEdC)zD-#X!<3j_ zMQ4r6T=}8LVWSd^Y^}8bwEy=0h_hV?Av)Yf@ix)GjJ}wYk~sS{m+8Q%Nxq2oD0=4} z)e%x_<#ZZk!xZZr$+O@G`|H@z*M!f5yrQrC(~2$K`i$|QOqCMwF?lZf7e)>buK2R! zvQ3Qm0f=V7i$)cpjJvXe>rE*1$eT)FJ;L!6Ftq*Tq6V%CYM%Okad)Uk_j9d3 zd*P>w0%b*8R=6lSK^Zy-nW`59?&7*C_buVPsd15X zb??=DFh6p$y%jrXlPiiUETe$UMoP;6oqtqP$yC2T2#uerdjFYqf*GMak5a3kGyOMx zCVPDykf{9_LsNvll0sjxE0HIG=}jt~!`tFF9kbRjQhI9M8zln) zlUo8O04DsOT>#(&DCgu~V?g2{CcI4i%F_^71m}r3AhNn$vB%4OStvR@OhBqcc>RO8 zK~M2}TnW|19YKs1jXY~=fazGT)!h-xv8nD%R#slqo=kis?{Wf<-=QkuKST7NVSr0_ z2Bhy&b#=uvlPSxfRGC7C58kEu@%YX3Eu_d41b3mqbY=j`e6FIb_cWKoPmsLQA)^(^ zSHtUR*|$IB+)Op|&*2j`;4p3!iSkRrm*`>ez?ppVIOtHJ_7JgB#>{=4iv7OK^e7t5 zK0Ir1>mFC5@@yhQ0Y6~g|K!S_`TW_x#)tiGRou`w4@{S*tYkv4Le+6}fssetPv?`& zRd|kccW?>jsg4}J&ALId&qR3Mvo_rpw#Tx;M?3$eje?!gX-zG|8KC!X z+~H1^GZA6v6p@|@y5S#X zj#g=={F}!UKOPW=bA}lV{1%-Cx`pX}a_!m|TC!pdR43c&^ zzLn88D{)2fV2sV`HRS|mE0F_k?|Xjic;)*;Idu$rN+Z<1Id#2~dz0c6iGE?R_>}Ra z^NcnlHtQ0ix*Shms%wc)RsTwJJY56W?ehZwRD}O0P2T=#;7_#!BU5v6C7*c~0d&Bv zS{jZd7@?Vj_Z1?=hCguHrYr6zKU3Nkt1s<~kihO`z)ww93BCG3rlm4nG`e?Dt{(e! zomfEB+c&?rpk*l@8=um_=Wngn5u0Fc;BX*kFlv*wkwF?K0F zAC9@_6qSgR;mi@Bt^)|#gDOMcc*0`r_6H^(I5M|^5Wb!sOa&j`M+AEnTLjFEOA82* zAqI@Y?vxNqAoW@ry3s!3ApV!e;nay`GZiv&8E{uH)J-D*oxaV}k^7V%JW z{Ql?3f&d8h|5FP5|7Pmp=(GM@2tgYl2=`akdLcCfMJ{>q&7CHUjRx&i!;&}Pe+izM|D6cj{BbcT zzA{EF>UBUugg&7ZLyG z)64F!lmMy1mW$mrXQ!G#IsBG!>q-c|SwBUIc%E*?Pz%04x+77{jaGAnmFffRk zDfFhD8-TFT8`tLTJFF}K(Jm$u!$~tz+vF$odo}S~zYj5&o9cit<5=+4?O!#-_?dkc zc}-*dcm9!8?fjQK{Lhf|=L>}607%iXTW;Y}m53mOSIct)%Ccuawi`m=aqx{c71S7AxDb8Y>b@BS7fdRX4+3)ZS3(X48K|`jpcauN(rdJ?ySZ^!lL>9 z)pNZ8!(G@!VKi!F_#vD*i?|~cNESlH2hRn>|vN_~(5W_(98=8g&(GoQx{H#9;eV#+Q&8B4Kt;|3f08rI1 z7_`VX<5wA8XZf{ovecMfczg=L#eM>?{f^=TL5a%YX%=wAot(TSHv6gVmc;g{w`qC&eO%iVPu{3e-D?~JTAy>b zbas7iuQx(mDeo-5;cKOe7v^5qgQ>usAR&tu`*#`qFRMO1$6Hp)+4S(eE=TDbg$-8G zndIeVMM59ZYc0oz7MMNF z25P|a08@*Phpd_xb@@AF9c{VE-bA}927n5SM9=7MwqqYy?2d{zbsRbRMiyuF#)cAy zRISqWaFizfvuMsZ$3h{ozfhdB(vNBM<_OA%k`*_{I51Sn`8N`r?x7(n+TC@+6}$i2 zhY0sVbE~x90Xgi0Bk!iK0x!{u80b|H;9K=JC`qnN{=IdyK1eZ8bTBW7`k5wo>2-pl zfAV@Oi;ngu3y9={RzI#*`DASn+X0SOGn%50tC7w#gzM6h2%SA zKG^|^W+nA%Mw+Yr-g~w%L#L~6zP(BTT<&FXClD0lfIoCQI*SOpkz##ELWTg)0HVd| z*MwZR>N9uw+{9uwxAV)iED?|(U*?_<$^@82pjZ@&Tkw|AuSEDGSt(=NK99Sr>Ur~? zGmyscE1p8RL3x2dLmDrfxN2D(N~qyKH#8yjeKXAZ*2HoVly(OmJRHPwGqeVYKqW98(Jdh-sZ5r5YXp!TGv`Q-$v`IA?v=Y{LCJUWIA zhC-z~3Z(oT`KnTCozC?J6s`{Je5YDW^(Egd1G;1S_x=4aata696d& zfN48tq`aCqrD+pZOF>$xgRW3VAd_M_K2xgfEMIUcruJNT;k#2388f-^5KC+d>Pn9KDYJD zY8JIfuM`1zB2OT_4N^dV59@irF#4J$p}l4~?XC|%e_Cn*^<5zUJpiz)E!}*$d^l**t^n+Z{Ln<=N6laKR%1&Qu=k|}IUu=1C_v^R% z|2p~(-v4&=Jrs5R<>-G06ZAVv-oJM|1ud1N3f~n@``gjC^`kzQLlSkkPFZE8x+%9v z=moSE+#Rwem%#}s_>$g-MdoIsHy#LM7D_9h%U&E!6Z@Bq!l6#F@twOt5&QcD#~sn3 zz%ohl_I!z*;MHRRjXPbI#ae7I5b9VXV2vQi4SvrE@-*cv^n`$fd}2#hyXKy5Bb%jw zjyWNK98*SJB7@AlWSUD^qz2;0$evOxq>rfIAF)k-@a^&ohe0PW1uZ_^KD1)m;2wo2 zPfZP4$Qn;Pz&o&B3qmAZ^uN6CuQk9gKI~rqG(3iit=m!0j-1=a?{SKoO}d>v7}`a| z4q{Pa3LQbunD;YWci0hAn zz12(B5|8v+xKXF5+7-sR*MjY8%eP$QZ+kx~kr}(p!iV=eIgByrgBHp2Zde3TVYUh-4XAhOf+$~4I z#fxgM>~Bbm!gRXulNQnOXk9$E_#29YrBWn!cDVu2KR7NB{q9=9H!a% zt-lqEdR7;;y}yAD8&V}J_0Hr9#4^wo8qp^Vn@!m_E2D0B!`H^Mw23{Mw9<1Og|Uv{ zdsS88S~zBgC&JP>>MTyvAHGye8lmNQ=NV~gjMnCmHlUNZM!n4!bfo)Kadz8fc(G<9 zsu)|G9U>moT`J2Cd9II7! z&dSr66mKwXpB_qDvR~1yV~GFU{JWOj4jO3p4QgJ(UUNwbwF-XX(m%BuVmupNW)GB` zA=nU4h?ez+DI8(?M~(>rd4W6x-0VG6X&EB&eG826M)h)KinT+b59zf6(^jN?U&>Frhhv^$vVoQ zMpI{R3eXp(VW5hIk^yis(!Tb zG}DuYy0}s=W`?N@lA1GvzlY3zERM&q>y06c65%X!Q{}phX{MDXR6$9)PqL?dW4*82 z^sS}2&;_EFL*pKVPF*RSUHT)q$E5ShG!+t9(0)zH7WwhwwrsvzV|$kF{6G>V2f=ea zVMh7hd#tNnLs?{sGJiKZelyx(n1`45}y6zayaZ=SJ4G(-!tnXnr z-`|#YMMhpize+8Z=t|LP*1}attIet*1S?}Nvx>~8>>*JobH-lG7rS~a8`ks>t2qo1 z_Y|opKnfq?mSWH#3XGZeji6-+w2 z;7L-X3UhHW;>J$`^WCGqXvwt0;B7mkT<1J)UkYY-jGt>gsJq?jnaO3-M?e=bru@hE z+_haEGJSm1Po-$7ZhqV477nV_nhIJ?Gu2g`pZbQdAO0G!4zSi#%$Czugs$U<&Ls5n zOZXgc@!Ea5J2lSR44AB#ms2Z=5#%dl&Wj!lk0MW&@y3jBik#ZmR@5yQV5P!JJ2CQ> z!T|gIoZrE)f#CKt*u;J7Lhu;r0O81~-T8@#gP@y022TPA0#ZY`iJ_^dNe0+6r;7Lf z7g5RDcQO<*RQ^4`Q6rIRYou4@?m|2oyyX6OXhwTGU;<5&6Z{Cc`UFF@FZnX4A9@fv z-0HIEzm|?OYm!wk+Y3ilAy!-xXX_=hISxO<450T7rZYm&l@_U6R^5=!5 zL-|3e**?1sgu6qK=@ueEI|Fp_xLZ`=!Xm2zk7iM`Qwx4^^=Fsnr9GbKKB+yD((goP zCN4Z^n68f)PN0RU_hjfp`9n;oP5vJRqz8=V09uK-n_HTFf-UWS=lsOl zL+fThyBKaODYc_|q<3S77^DwcM@{g4+luQrtJhYb>l!pFOHKun^%i7{za7Qzb3R56U`{O2&lEmUX1jQ}Y-M;Zo=YR!pvv|_lo zG0pi@4a zU#PtKmr1iJZ$KzEci`f_q0oell+fA(WGT~SiYG6Bn(g>~@g9h4g$3{F7r9^Ma{5vD zgFd#Odbjb-n>`WP=F7pNAH>N}!(($s$a3)?{}^y}PhP+tS!9|zC^C6_`^}gP{zMtZ z;`Hk+;cK(;a{8J&pLN#ud+a>X87smzl~q?;_nVayEQSfW``}K**8=8LQ2w;HOhYdT zej4n`n;iEp8djr!SlPgn@HrvW&=XnS^Z?ei1^l;p$ecGMs1w&@6l&NWHSei==Ef#i z={-_Py{9u!C}V8_m0Eg_@p>&9u9BgBcYcg`@#HWB{pQRKcjTp`X<)H`ar#U)9~Hvt z!4izQ%Kqb51u)_LXKR{fDd<%C!g>Ta=Gg|>q|CbWNDeZ&Yt$3lruaW&ER`u}vszJO zy%fa1xG)o|RY|fo)XzMg?L_!lC|r!`7ito-n(zvHdZzhX^<+qnfD3(ZCb1LdAdr4J zET=Rzl{z*`F7JfC?mAde(Ja-e10N=Sv~K;y>91Ao^|6Ye1|`WMK>?$>W80osl7^=O zal?ob;bJ|~;E=|e3H`jP{?)N4U=pJn#pGM=C7{R`SZ5C8gf5;k8VXLDwN;_D<{uL- z*y&RkuWt87(#iizityzbA9yQwe<8c?GP>=qMEc0ot7kM5j@00dxqOwl-&H&h3IjF> zP)~rjo3jT(TL}P@b?mRC>)R!Q?*}!A!V>q2cNA+P)+a;M1UIcFaC)4mj{(|MB$EcU93si~^a7{GEyFDRZl++L1<8lZOrdV97rP=0O-Sb5Lh|7s* z={?=ukqeM3R7-2c%m~T~L&z$rfW*xGN)dFv&DrM-;mUs6xmm|N%#}y9rRDJgiN()1 zazMr6OzV%bNuj>9(Jry!2RMDpMuIJyUy((J`;g)W7 zv}IlHDGew&)h3>YelTn3(4_pKDws($=3q`lDi}((Vm{0O!o)Q$?kkq{~1euC=Uy+*3VPI}Y*n`>H<_mUQ z0NsPkvoagU-<*=D=jD*W+tCXroJ8Wl!#P&8JY zSg7srt40*xwuRD@@Rm#roP3?iQrvZy;hZ}vuDKe!f3AVjLQBA+nO{*9Japs)R5@%c1r1d3D$;$?9GQdGg&ldSiJ!BGP{h$uLbqO z5dpeM`b3j-|9AjhmA>v>?aWa%_;a7QXDE|3NWsd(`f57*6TC{q!WcfNs6--$XHDJL z&NdwtVc1A-Oq58Gb#^!zqMiVEtV~{+Q2u?>mniuII0;NJAe`5g3Z%@-ocDq8sVr%LQucxFANVWoXiix#ukv5 z_hE(6YWMr2m__`6T;KC89H2R@{#Nd#vxjNA@AWpwVCQlPTkBpC$A`X&gs?yUfhh^VaV|K+(@1)cno6v#KYW5Ph$sWtI1P zh*||H{GR~=!%D#ogQw1Tk3FDj&wSEWIZ z`>|KTV|7Ue@8le_cP7Pc*=lxmk~kzXEXr94Czfq#U!HG1uUJVSVxj;aewjI_t!iv6 zn@$|JtiuOed(rQLP_+UyPPiSxD2%o(KnkX`YO`QI#wd5!op{)vY!K6^;CP)``MFCU zz&pL51FyaA*}TmYP%k6OI*XZ(JSzNqTX)L8h4gHl36~Q4z6ZLE0hkvaP?wuUWxcvDLHROr3Q{HojPq%r#tiKuXp=@eC`ej_$Llj)V2uUHKfi`e$2 zT?mbU5S0MTrP0A4lJOnNUAgXw`(a_}ycV0R+S_Tn5C^(BZ?iE2otcy_BX>52^Tcqo z=3=Z!K5o|tTT$T|jCxq^I3ld@?)NEZ-P}kIsWKd%h!fS2)3~KiC`&T;n>?iHqsJ!? z={MO!K^vsAR=N!Fkoq664OeAK15`2o$}`Uj`UK)>F8%_!yNpg90n@(%em&$uxS6@E zh*V#~xto$?fAwO@AM0a7rt0oZI;Ag?e-(;p?ep90>r#aL?uqL{`-L|B!XEy`bqM$T zgKXfCk1}*n20F3I-FcmRrWV{HB^A3U^SrHMp;H+204;{-6fIP-q{r7&dCz-SECvR9 z@Z`4|47C$fOj!8X^xqYD6iqGkCOI8UL8tn_JWs2mZfpy*kGBBRHVg}aMM&nf_~Hy^at^kKWna@3#Y0* zU`Gl-3#q7=e>;kpLe6*jojpXJ%G)2~N5pm6Gk&Ff$g(HslYBJ&O~Xk{zU5pU z-^ERVt$9QY`W0P+>rCF)#AXY3{$6Ct+NiRrT%kjy^{`IGy0T$j-72JLw@ZbDcTJJT z8hWB9L6k163jB_~V|^j99^1H*LsUbfo0!FE8!-EE^de}J`#?MOSix$Yo8AB!a zIHVkeRANNBS28F$HHdpUt;0AMSACbQiSlGH<+rrUG0A{$kc3ZX+wpsBq6=bn27cuN zpoF}<>4l}$z4P;m*$VK`st2K|6E6;o9%XTAKvHmAMBQ>wt*tOts#o8oliSQtMwKQ^ zg5bc=FLLP6U#;Qc&;4_n zzi{F-D!5pg?^ozqaYY%mk3n6~+gI1uPS@BDICcVdU4_^bEOlHyZtZ0T51jl3Rzjno zD^-OgnllkgZA8uUaZZ3D{&w{3MoW&fgBh*ou57Sm4+Lmrs*gV1;aTfV* zf@1r!tHkCl!>?ue0HFGkoEQbx5uD2f2E&>f(U3DMibIEZ?>OOKQdqnczI)nM<|x9y zG&~?HL1=@nQ+qdN0Rd=6uRcCwl5d}Q70QG(Q0VjAJJvY5hM+^@ry0U!9$)8 z>w`k|OkZEv_Uux-4a%@GP;cViH~VQcRHCJ5pksZ|G)@`1Ik%K(f@_s&^T}b9;$T?) zus^nNOm``p$t5Zu%esB;hqn=g-=^lG@mtMjvcIgDFoyH_299QS)B;=m&}8VWJY;86 z9rQYh>&E>2E~8*52swaQ0*}Hy-y7_OcxjxaZR3H$q26s7cJK)e6DNQUw@`^N>_1EG zPji+Aa(-xINNaYwP{>zCAmzFj8LDVTcGB{4_!~!j*qqCa)$o6$;D41?y ziF-I83~&~RtwB34KjMtt(`KTre^;i9t|Si1dIFwnw{Nqb|KPR>NLC8+QMcV00>fn* zC|em*{rOIj#20)<(JxHSKGEZC=$&XabxUSoeS5SdV6mNr@CX(r(rA6hdT^UeZa3`^ zfykez-NSc^iLV^~DVerG0aXJ(7-)1;N5tS~5spsb)vF#guRalhB>vrT?BAfizS`fQ zKBC52^S%00lg-Oe+;j+ak~&&@(u_+*DP>Q@+yYs61#hOn@H0h8k2RPIvQ^RzgG>KZ z1n_E?`_>!O|;%~gSco1g&4+-juc$BgEnorq6vqH9T=Vce_$ zwVz!=dAAUkYY{My^77LQ)XbL;EK#Fy>Yw8t`ZUn*9YL7FaD7_j=PTPr4G*B0i=Dqe zRTs=J^8=qQ@4v8-RDRuz_Tyz>HQqUnJvTqVV}>XXKQ{E*BG(ACiX?gy(gZKjTarNn zJ4unPa8~D-ET99%qJp2t@L0n(91@QsRl%T*Yg3DVif#V6oggqc66qT>Q~L`C>HIsl z;4utc4f1AZQT@@j-aim%7Wx=R_&I$D`s?};(N=GeCL&MHjw%V?j4DyK1e zek#k)f4|p0{D@2c`sKzWcx1BCp`c-@U7dF9$YVo`9(cbfhi}a}eT$g#!CuL(! zc&~LZec~TfA6ls~N4u_+(T}yR=x#O$qh0oIR6l(5@?L*oupQE>lM|Jrq)$cPz8Mkc zxb#9BTB1>3K&bl=iXya&8#GC``E_NKfO(pSBp~KmpV#p4N3OvBnH9L7% zOh1RF0L&0CpH&4_mB6^^Y#D1MTcc#HfdWjGiTWgF)#9Re=R|N@0$G&GbybB!0pj86 zHez=1a*WzAEX$UO<7Z8a8Mk%(yiAXJ7gdyu=%=jsk%X=hm+lAiYNx60Yi83EA;KqA zphsj;Q|oc?n4=M~2TV?kSXR$dw5W`0#GG`QPhgrq97>&=aku%6 ztRFd!t8@2Etjj^T=T}tq@vmz4=zS1xR%ZbP5nRYsA;DYT{ z!;$>L8Ea0a?-dQhX=xh-VZ>9_tuM~B0_IEh6+i^;Vx}HwRgP}{{?&S zUY7NFa7b_}LSjO?NB|KVwWVaMfv5dGKQqYQY4ha$yDc#&Ra|y-(;qpRKM}>D{}hHM zt9=uNg430y5^Rgx!o0~r*hXnl_Qd#13l!|kHZNxbE8qfQuX-Ke=1IS`!0nF?Y(t== z$0p?p62y#;GROIjMT?t=wFc1vSJlYD2lD*^P^N_VLpT;5QvvgNS?yi}wa3cN70PD3 z5}{%x!<+L8I%u1KL;RE{azVZu8Jme z`AD>8aHlhvwc}LCD-c8Q2AZd|D={Mub}IP6lz*Q)A>Q^38NBLf`1A>~qd$dk6p8YN zdo>yofsOJxHc)y)JZ2WI8q3#)Qv{hq!EU13>x1hBLjivPt{A&?aX*Emzd;~nOf0s0 z=!DeDEIgCXO-DiisZ49Yw1zKLb_tpsjQ)-o3axtpPDmW%F5?q9;35eQKH(i5)slh%G_Q@3}egOn6XM zzVU8RavBz5@u(fw6*s?REk5RR)C#x9Yh+f#ep8_6xIXk8kVA#!gf;Nn^fikSr{C>R zmjyBMLWRNWN<>?g6-7C($(v^h>WZOyuo@03p~+sm0>P6XgW(rD<5O3s3we_lrZ*)S(7|oR%^4S% zTGI9{s@Tf!oLo6NeX}58tOL25o-B5xzHYgs*4DeI-yvu-lDsm-csJk1dJs_aa&jLC zw%Z=&T!osRRRlz9cv33T#Jyc6Dl={g>$R4nxX+;In^CV2luxsEb~AH&#TwWk{|Ngb zhb|8}=IXD@)9dq4=L;pCOY7`osp!+`%{o#!e~|t0=hn;weIoaqEEPxGl^xNa5K<_^ zLEFYSr($0nZqn0L#V9RaMm;dGkQG^yVS4S?Zd0QQR|@KgkF0m~>t+u4i*orvK*+#7 zIwb?TAeEO{@^=C!pOdv&d4o><0a7T3D_DfCKJEZsGI;hJu$d z;8s|GJmN1{g9(+9nU}RJohl{^%Wxne@)k;!#a2Md)W!IeN5j7yMaz*Gnb@IXs|xTa z4su$J8d-kNv=u~TD>rjR2ADMuTwt-=TCb=*En6Fg;_!6bd@Qc5hf@`?g{Pm(of+7Ce@k*2a3#QVD068N6!eg8fuqlCjG`VjnLB3=@$2UVH+W`i$Bv{I`K0> z-PmHF9lwe{oaXmJXfV?0iJ}sNH-;ooV%@~)x72n+vMDsu(h8MP5WQg8 z(0UQtEs6y^;1@#@{A42X#lNzbq%lu@jEPt1L9D)Ujekjn^KM2G*Y?L=7W(9be^lwi zHA*#tObxCjmR4Yi)0eF|6mQjP?UpAVE-rvl=rFSc)Smr^$tYca9K(SZHid2c7NNY}ILjTr=d1(?v&Wb?AZOf;; zB3-As$iu49)M?wI%=7irr_}bXbq!u&o61Zo-$0U7h-Bfq?Ir zOY0f}Bw6$PDSZ(Jkew(i`~7cc|Hy9plaeU;KhFMLqJsS2&VJY5&i?u=2iDmCkF(#U z8x=A`S5H9K*xI_s`+>xCqyGQ!_Kx43EnWNXH@4YvCmq|iI!-#a-LY-kwryJ-+qP{d zCwt$|eV*UmL+3eG2^>-pmkkQ|2NMz5 zgdUsob&VcNZY^{}UKNWC7I{QHU8A-!{1tf_*+~CQrNrCOHT{Ae2|3n5nD0l*6EZe> zzE?|NT3|cs^mog$ls*x@pYjQSS~0K8All?pBP3slnz@T?gU9gyaQ2b@|KRMa|ASpd3BFu8e^Lw=hs*m~`v!60k9Iz*-(@P+^^DIa17XEdo$W5LZ0^qu z4@xsZ2z*s@wl9$u`?z}Dk(29&pdC~G&s!HD_F3wdd+aN1|mrJV26aLj8rd7cQURMxQ#i`{e>XhmxZ6 z-XCvLlq_nJv5JM}7GS&FK0r-V9@zkVb0B8`lqu0+qhDx-er$ff9(VZGfFI~L=Liub z@(!hwv4`3NQrmh+I9AwDjT!y8Umm*Te5xLE@xz2X3+JTufB*&~-%8Dx%zAU6k;d_f zK#`_At_@Z%x>~D#H%@uc@aFQ7N()R)cJUzQKd>pQbiY2a@~t~llp)`PTMMm#5<2=$lk7{pwGlIn z&=wF2m^!qhm^dz2Z1JlzB=@^c#^(@^UN4@M=x$&jxRc?veH=-S2wa(Z-F{l5QQ``} z#?)1PK6U?D0P?4B@{%aTBb*%;ddJ86hH=48V#$I}KjwtEx-&h^h+p}^j}FMZa_i9< z-87HE-bLfjLGN`GUus*ykhgM!MK)TiDT*lZ{6qUFjkWv{S`WPwxvys2``{5-vbjePBf5moi{@wh6^=jHZW?K`UJ&^W|OEYAq`uwQo9r0=q-CAZ$t>fMRUhR+lhDM zn+!W~;=fG$5d2z{Z+@^qkIcl@I2`(eHUNb89le(x?Xf|33wIAWaNqZTaRBGk zzu2U-C>~q`8uT9Jw8&)7;kuH08|^x71SHa;pzVfd<}6U7#VqFs&|VBN6H!U})E_kywARwquTyjk=PKT4{fId`EQ`yJtVD$~5i$sB( z_5Srw)E8?1PGa&4wQqfeljkZ$t}l#;@~7qX0E5`kLg3&p)c(rvH+73m5hm+z?w?1E zqcn45U!Z--1kEqd{%yvnI#P3$n^0HbGlDjMf6Ho@D{3+df$>?x&MP9#k>}$q^GKo^ z>t+sqRxO3BAQyWnGwOS0x# zp+0tBllkfQ0&?gv^Di9G1~I5WSXpzEopHf3a=&Ole+WOd)>_tvE+I<`1!}l_5Tk)rdgegxeCu#syiJbV z2NdSY)Tg2j>#wQ?48AeaI!fmIvAsW#Cpe+w@TZt^DHNQtj~m9KRM%lJUX&%lMpg`f zTz0;730n?=C0Uhb%GRU8&xSplrBuR7hu2i^!lb3N|8FrzrkB|1mp#8X=O- zNH}YP3`LL7OOZEv6XtZ3OU?f@9K#1x^YiS`J1Yg<=aMjYA@yFbo8o9A8%M-f$^l;$GmfKgww%E3qx1@CFw~aVoCC4#PwTKzr1>CbEj`r5j-v9O!<|t~`<#lvXp$x&_irs1> zaCAmP_l6nn#~M|KW-Q`Pb|~q`8w8e#KtI%<3Y=vXgrx`X>1q{Dv0-&()|9Axv>4>J zCLR!Gu%He>lXG1f=z|cZ-=PC5?~Pj@0&vnJ^x)t0dBhuZ^?=O0 zn}nBmOYCWI`GQ*C8ftev@R#n->|VBX({JHEPGmf`bTRc1l1V`{2*vq7BiBlpDJoOp z-joF_LI#RkP1wUK*3$rhaFM-qV<@Of#Lx@YPft1l+oWeD`GAE)7 zA^FhkGXv(v%DS717E85l4ie9w7!X!Fw}5WrD?6wSYSG%m0XNJakqmDsG=CB~@rZRD zW;a?#W!NQmS-stDYppL~xR?L1dRdTBLS6zz445|4EEn=;*m2i}&Wd;!x$qWW@~ni0 z&}V9syT-+NXvuodFX1y!N*-RhxjBW<%RHrGv3)fcO(ka>GluU$7OjdxAm5nxjY^6fh_x!smp8ScO`@%_6`{$Mm7b`#~w4<|3LP6fnZ-}!1uLJ?>k>d zAsq{zcxV|$r69KrGdT3rww#g#b)i0)ZpCdo^^~^4EcKA#+s@pT$-yA7E^m9S3(g+u z_*Pmyc}=7#;Y>ZiRkoTYvFm@hD- zpJKi)?vBuI7Lx5!>=Y+;D(*keOrTC`_Y&5xKZ=lSCDng_6N7m@BWd7rmRjhk) zURI#K8*ZK>58K`;A?{=~<@%)pV6Hrb*ex!rW8Tq;U^V$8fa6wO%khrJ#YeO6KzXRdTZOvWksA1{!;HdXmJa5PaeD-3r zQ2S=qXr2!ik3F{V)p_+HNUh?MW@@Zk-7Q)%8}cGf001tYtS!57d5k-XTc+6+hzYC@ zuzoaHBe<-DxDR5~&d^9Ur8St)&r#Do0wg0p?v69H%%G3T=IEcT$KFzTq-{Yp8eGmk*~WW1i!FOLqOxm`ODFL~r1 z;A}K&X+B0-yDyXrP->CewmDYtyTVJDDho{m+*zKe2gY#3ol=&6Vl6!W&Ft#+cR^@)oPfq)1ZoH1OfOhAC7jkYQB@XDYW-V7vMncPQ)yA>mcU0^rIrY?QY;$ zZ=Palc1*|A8dD)FMYX+E95?*U6;=Gd`o%HKrvNb9H*JUx$Y+4UE$+CD)ZuA;Zx-X{g!4rhBO}BmX6t$WKY@YmZpJwYrlPsbtm4m&hKti07|O$$ zu^}arEO_o9AcI$RunIgBw7J4_=RYZ%;T7!tUm~yG@SY0ax!`4XWCV7}aLFlQ|jAa^kf*aq9t)s?5PKvcI-Tf9E7B}oHpsj9evCs&Ir_pmlMkdgp5x=J@ zbOQw(1(`8S{nca-Jo+KKHq$Ft@-c9BBc@m9LDUR`_@nkHZ|d=FnB=D@qccu-y+9M# ztIY(bpjRaF;d}frcDe``^lOFn3UYJm>g#O&a|6->)>v{>I&8|fD#oW;Hzr?ZI?(4; zInP6{*wi$<-x@wfBkJsa`uR1?OQZChr8Sx8I8<8;8e<>q-;0cwn{SJ|_xp~$?0(M> zIyThMy*v$Pvkn+LiWN#J>GEsI26DIa@J4uIr-%AvZos!zY#nCqeWb&L!MawXvPO^K z@{wuB^RRn4X>Y{jvV+B!MVM{UR{GjwChqu0R&SCDAi<&AxDl}1jF9a`+^$eeIDfGz zAZ0Lg8$3_%B5u$Tu0^(4NA4kbfY18#qQcAai0{~2*NZ#vFd-;57xBlDr3^bni$++h zcLo9^AYt@v7{q8_WVfzjr4apnZ8m)%&oUK<4l`0gkr{{U+LIFp3K4WP0$=U>E4N4N zx*w|yxQ;j*Ju%TJDfn~wvf#@b!~K5^jp3-BUPb#gTkNV++|19F?%{xV;2M6m)Pd$e zV$kg(EW~W}F}Z7@KDwCOnnm?v-g|uoy32}4VEt!yKLwBez$jE#?Q)e30zU|lN12kXSA{xigEHlPz2{~V77ij>)b^)2p=yl>5Xp-O3rQK0o$8JfQ|2Y+BT`mogj0Q=~`=MX-1gPD;* znjBgwRz@`UeOndxLQBViQcpJtbkBImL1iA++DK5*CfO?Kv>XCj!%Ou7Z(x0s_@(Ai&jjW#B@U zRqarsbUFde3ep<@yf-u_(aA&6^KMXYz`-=|y&GXJIX0w)gmdYLz-y0NmHLdgeX4E} zIyrXxtT{B$x@mKtOBaGXBPL^glIaz@XV`WZJ}4tj-1(Yf z(I~4ma^IwADBpXqxmbaW)&sYOdq3#g4XEHlg>xg-f)1mNv{})U6d6J5nBdpP8DR#k80T0z3 znjn9zU~`UhzN##;8>|pi4ZhXL_G(B%Uq+-r1m| zPRGyJvB!jt3t_>eaMZ4|B0{t&S8o`^Nv1m7RCTD}fk?DCmSI`mBVmp9qmOO?pfOfK zfsRvu&9eoVb6|7G+b16N_Q#2_3g?n3UkNKD7UI|l57om3nd0t3%gF*hvG^HrY zURqWn{jZ8D4RQ7NXin7^M`&ljLb!jkrJa!OaIBM=SWxk^a{mlktAjG5C!dC$`t8!d z$dGCGLn2(6_%CSAw^RU)D+CucAWE~fHUTuHczsgcvzdMn$u-`p|4~TuvVWiG0%YRoc2%i(ZqMeH$4g>hPr?o zo=`HJ05L&e9?A#@+2S7atoD*hfzp?8R8W^7>jFOqv% z!DO;0_^d;z%i7?>OHF9s-XblAdoED}B0&OqzE{2|DMJF``1jey?xc3@NZ}uXBJTbX zf?`o3i)r~Po^mrWP9C)Ne;_l-w!6}qZ1OZ1A_{3Ke*PQ~drFfJ;1i)e*mI&)WU@Xu z8t)H&%>GNblJB_LoZHF@E5g!0k_5o*k#x|q`_vlKDrV|GztHKwt&epZlB3DyWA2{a z9(5HNDF;-X*9|NlWdg~^oa@CvfxP^n^%MfvASBr=%JAQO>= zX?&9YJC1%gHCwWc%-vcda|NyJU!Mol2E)wYX;~`bseA}cEj%u!Om1UQ5I<9A-o;1X zsTxf{!`li7BlPcqQVG!D7yeVjM8@Hbb2K@Y8jhV;(UuYhyTvzE!e2NXsBfgx(Vc7v zCWl-7(lBn-2C3<)QwDZm?OvLMI%1nTY2EPfX?<{5Ss+mXAn+A&JPcww?kZYdj0-eU zn;jZ_0r1aZ*-ooQBsB(^yrvL))_SXuuKoNWnU<`LV=J`kABPX4s+C?H&m>pf@b>a* zOy{opEjdAOrgVW^4e5M9(Vrwa*jXtl;*IMqiu`;0=L%ZIs&n|S7`dN9s0C0mX?h4c zzv*CoMZ3Z6E2Yzta{6YZPR>58ws&p<%+(l@$Y*da3hXxW&d2(|6H2L^!JA|Or)oGB zI45{})=^>&J0%rMD>5s}z&3@du$=Qh57dl^>~`9Wq$QEA&UIxKh?>w<>84DqC)$xk z{PG+24``BZ)c4Wu)0?0X8#U=xA5tTn0wl6S%YwCX)TlIk&T{_>KL8F1GVAtfmjHaM zonAgGieLiKF?5&y-e_{bF3g*L}P zjkBg4qSa*P?lMTMwP~4eP`hV^0V2e%;8d$%FTbQQ5KPT`!$bVrwG(g0$Y*m?1RS+< zS5%^eMCjLV;Qm^annws+x$vl;HUSkzHFG}SkNB;>-|Am~6_8N>#km0&@u&XyFN`0K z$+MsIa8?aKMkKv&)mr@2X0Fy^5sJ|mm~XFixs9vbLC324+Qxe|hCtb2A?6mxdnPMw zCfj3dW0@6_8p%!U4&+{WR|f62 zaS7^x=)lUeN<9%K7FE9*HqLbuyXoyFjyUVuOQ^~`O zzWtdVf!UmiQ#jdu<&&%+^z8ShxI}<<(JXD7`EvxJICMEuOiJuCZ~bp2_<|6&hdNou z{Wiz_NbbYd`G-7!=KmbmTs~5`?;O@deN2rX`%WbdJe=V$vh|$hPKXFl@1n|^ckcBaR*9#RK7>-_>&2jt{xoEP}2!I+s|YzzC6t;JiBUit~4qJY)C*|T+OWj zW*%j7rE@vXpyWTT`W8=n*H+R^(Wy58MB|3;0WdKo&h0cffT#w=5Ya{%Sj7plWTH*( zoNsnSvR6!RF4Rd+X3}c`Tu+k>(EtEA+JAE&2plL=_iv6XoZpbMHxQT<&*)rLtD#_7x~(4|zX^Ied8zW*Ya_}xgo}3S{tje6 zG$V`9P+i<6F7h{a#MeFw1i%XnKt)VXRr&W61QhEcX1re;144t8bS1~GO}{lE_PzYT z7YO7T%jqkd%w~oJ)?(yT2K~U?S6M7AAN!!M8N2XhTxJjr<6xi)@|XvD{dCCdI(AL? z@u>#e>TibDaYQ3beA>7ha$N)K%HLn0s)ErqgAt?E>hk9Ljmv?rGfMU!yQ6K*B*MtC zWxp@UZeHI6DftWBTmYaj|0S&-K$-e~vtHpQzW|pN4VTx3XthNVA%py{%8J^eUL~2% zc<9X;h}9=T&h(#!3id2s<-~oVETq$s#F&2RladK!C2)L~CZ_%&P@TsG|05FE<+*>z z8#9u z`@8+|Pv_f2`{=IG+6j{J_ewwwVs6A19z+LRnqPfwGiv_@u>@lHhjjy-TiK9Q_Rb^Y zhz}yrSeGOABguAV0PNe#BG}^3%^J@Puc;3-ky(9u2UfY7R(^KtwRwnqg*cij#^+ff zqLnJsgEk$(V$XU$GUZAo?%^o8qtWr<7{X^vx5e1sJ9%l|e@9Z~a#QAE!DoWvn1u0x zy8k8`5NxPqpU(ekbE{VRR~hF2c#s@|J*=aKhvEy>B7>Crfud(;x|xP~hqEBbzs~w^ z6e|YJTI@W+H9F;w!Jxg*1>LpjRH1&gw(y1Io4ysF99t(tU*8pIBn9qws~q>K9RK#> z3*Z%#B<@4DZbw<_H|D5T+5NVH>YV)zd7IGcN*dR1-!)i%(ANOba-{j}9aT+L*jgJC zZE4p&AU~156b#Y+e;h*q0KnA$x*5Qrxf$xMAT?PEsdYh4yqhZAZ3@qovPtf9uq~CA&X;1r_?Ok6}M-QVPnpYzz9<&g645Rm4Q~d%@I=&0b zxy;Z0PZ)qAfd93eK?V+G%G%Ex<=lnvYf1I^O_#Gv8!+dRP#@&-h6d9~e`9Q`Z`co? zn$SQ0p`$b*Pr^;njA=HKzd_lpkK6TZ+Av32TM4GVDcO9Y?PTg)6**1eyeI?!a{5<3 z(f*gM4^J{FQ4OR4Fc0%G7ph@vy&=r`6ABksOCfM!ke^#bvoQRYpHfD#sHD6qgIo+X z@Ge^K?%f-#+UADn#h5leW&+TP&Lf2J5PeWvdJ zW6-|@@I820r)X1_!T=3lY4;(F+HwALrh-!!I%%ExIh?g(Z_E@A%;~GcK%FJ}_p+$+ z{SF_pH_rsh8FapUUqV>mHz2>9df^`_0&92o-ms65YFB-Ikibn{(`T(#ALe_~DE67P z`BVz&+jocFqo2)j3>5h*?@u$c<-9Q)6pk&<=U7zFr=v?!VJBu=+np`|t$Zu_}@qM?gA?YZXm# zZZSe=##3*FTqm$g$erv` zc;;*BneXD^SvOJ$ts;H*Ldq`N4!X^*MF=PZkE6!gehP%zll#lNOJIy>a-3jHR>EYG!xcYAy#?$P(boQsUJ zIqUEE;a)S3Yg^cx+YzdHNHDAPuYjHPx69SpWbyn87Oor3NC>EA>~s_wE!y~Kn4wzK zC5xYwovive2w#42Y2;Xu`nl9%mk$tbc>~P^m2+%s1kC8L$9}_$al~!ddw(`45&BLa z5|FhuWFJ-e8x7iu_#5!<&rbN|Dlx@99uLL^j;S5S8-8r6d$zYp2Tor}7X0s@KXIE4 zKk~o3nU-M2!p$y#lbxVaQ?-2D&lBm(f3j*KVQm=oVvdJhQ2}Yaq%7wtDSe2a{*Wh! zKY22h7=I=2(d1k(C!w2=81mHw)2@ zQON7gejwwRlQU+VOn2J2cT?{S9jMwh60V3$3QdnJVkeu~sDQ^U+v63dRE$=&E1e`s#2w;%^Wp#5D?m20meyW@Zn6W&koix|!#a;CN{-G{Q7K`}8#} z+9wMtjB#yCyH4}q>_@byvD);N%(imZgt$VGD!~;W;PbzE3wZu79v|NC3y&`?JHrG3 zg0Z`Y^iKAp^{FZl~$H zCd-x$5WRuI{-r9F|5EeeSHE1aIIEo006(=JD1zLWvtGVA7LS|NMfj9Nhu_g^{_e2r z##gj(NE~jJduo~{0Cz+yD*s4aI}P)UBs6{v(;7>s(&Uz=(|~6dZm_1r5{)d=3 zxWm-@I(>6#Ha+|j<(5jW%lM`ErSa;PT_?TK`S}2fhan9_n-Bgx8%z|zN&v8|XDSH- zlF~FZQrO^B6KxE*qq^6y{>@=ghuVE;WtNE4!Z)|7uqT0)C=JboSNSdKc!H}%+#KCA z*Y%FN9rzGKD%lL8(~9S9pX-2>?RfN3!(G8|cc!Z_<9u4Dlnbim5R`9CMRM8_u=ecy zU65f=@WaX8G*a=}r49>45^|bJQ$YiOWb>~$Q~U$85B#mCa4j3SiEx8`)>k1wJi;^o zGp1$Y8q7w5C5Uhlz<04#^O?W8mCT@?A#4XaB+fn}w`$s3<5DaTqDcC%{?B*;fzkbQ;6MPb^Vh#h zY7=NKpX(WK1t##WTiOZ1j-)zS8t(;+$C_)?YdaM|8Q76a!?`@oFH-Z?LMJt&rtSx~5nhwZl{`(WPX6S(1Zj`f!j>CVy$1Msr(^+sa z^J^8QqmK#VfB$B}%%!*KpPBD}Cb9Jat?zHPSe4y-V8>I_oeA=tjP-_hcOq`TE?HLV zr>%S>@z-7l@%mp#o7=_P9bMN0Mr;?5lmvGRN1VWH)BJU~ zS@F++DlTYPI=b=OIfPw#86q1UD!Ttcm z&;K`B{4gH#i+AKI090OJ1{lXYN!gCrVkOf?2#%6*>;xX$`$KhiJgV)jn*Zlo-~3>589!D?u+EhZmGK z$J^7p!t$mY7!{U|k~)GNNCSc%Y6^CWMSmX<0lc7%2${@8`3~^*%SqjW;S*+W1vG#v zrV{`FHf$VUAbWj|FC}diIRgOFGLn^Cla|f}qQegXErl&$Sy*gz$-{clx=n4McC=_9 zxr!t~>7QgS6Sap!kob)aS1DdWv9TOwo=zi~iRLLCs(QDhsRMJ9jm7UIpEv%Yy!QN_ zv}o7D7D>sA$wStIB?3DqnDB+Xf9Hil9Xt$`gZI}VC+Nt(6XpLwZy2eTp~|ufvjB zsf`m}&&&qK1LG92%H*0x1*T18TV;(|#}pO7L=`eWBwl!!;X;DG9s0cKSLSAI`F;u@ z=FVV-Su@9;&CShOBY+P??;#(3r-#omMy11x6q!MGbd{k4D(_$eT&Ig;vUC^j%*%DM ziM&ovvNk*lAt)7$QGveAV>-9&E%wy+i*tSJ677j(X@aMR599dBW5Dr(&;0^h&=PZw z%zfO>i2-^No!8T{Eond9qk%~_r0boW;of*BrUg%`xkohjE5{jyLf4Hv18;$VK4K z%`^{4e#VJmv<#i9RGM04_-0rsP*uX(w(nHO`jPdi6ky>@01}5?>s1bLny)JXx;>L3 z#R#(8-FRqEKfexm=JJs6FJT?mSuT>@2<##Fk1K>G(ba*x;ng=ggW>;_?_UuzMr1`F zba=qQ$rT{Bs4ZTAK>G0p%a$)frH9#wn->4ctfOG=QWcpr+AZl8=jL^A#2~E(Yvbyw zzFYtEz!k$jDo(A(Ws>4H7<_`9p{O0Ia>@q)Pr<(;Dd{|7Yh~Fw{3sG>NbgJBuMgB3 zbaR&3JX6=XjvEv^LH72;VUAfHC=*XyoY)!e6(1n`MOwG8$)LeJBJmw`^~aM^c>g)P zRQz|HgQr)>oUW6Pml``14&=U)Yj<;m6uoUpZN89u(N;gLB_>-!JL2dfl89ZkGMm~B z1sS3n`2bhUHtlfo99m>7O>toVVw@xRCEW*D7LfsQy{S(D1END0Q`(uosQcqMClG}^ zhZ(zCsb)b{4&>n!9riWEC+RO2(QjCPKKymF3>tUmxm~Zbi*4kuw^PW8E{1U0 z+{EU^+f5e+74w|hbE5X-acSdyf8rqpgB^WQ_l<-U<+3C0rLwylPzaG4Q@rhcN}EC8 za*VXac3u%q$7TM^+X5dAFaM%8_sew*o>~G7@gk{AQ`E}YhH5jJEk3boFGlOz$M$gM z0AQFG%OQ6!Au8#6z_R50ZFIK|jo3p7g%)d>TQG5?jB^lGKJ0`&JD!PFfV1^0bWXK> zD5H9bs~p$M5Cv#&Pw}YSk1S~*xr|)J3`5hWff;jF8k7{XYv|6l2Oc?s_MA671-8nB zuz6Qudb-I5z49wB>w3<@0}BYXOZF|$&r~y|K9i$j8Z=tu3o?4 zY`zisYOFZpnwT#`k?q03L_nI47i<>b^+0EqJ5H~iVP7b_pLn%xIWanTvc@b7@CMky z>-|PE5*_KhAQiNY_}pvprPjpbBvO>V;V(b8&O^k}mQ0z)dw#l)hf5T628LdmP=3Ri zf7XK}JzH~*m*H+6CjXndFIuJ0?L?gk?(!>p;RoljSyTv0d?IO9&$gU2waRIYg8j`c zQ&acTYST7eD-iccTfxcMYWh~ZOHsGv#y8=vJ?>YL`-Zrhg2y;aM?C>CdbImCF(C>^ z?X9fl9eGQ@{RnP|&h#3jg8=5^W-c8BwE^vSGN*-k!H<`BU-n?q5Vazr^Ey3naxM_oLK4CTgT*KF zi1|dvv_@nP)cUh=OL~K`7`szF2pT*{GPZozcW(&oPT93jY1D=t~%$ek|Ab5{A-Z=-- z6XB4kYqj1$QpO5%Abe}y~qQbUn z8+Jq#v!k~MzQ;5Y)>E7AA?;!@hHuUQ9NHK=>;7Z-A!@K!q0BQIWRA#IS&hzKM8sH= zM-KgA?AmKv%Lwne(?(+U8jd6fZ#lAfqo)^`J7T}#b&?s=8&sE6ZoU|rZ9Re#pd1F} zOHFPe4~cT_vbIlR`&5R<<#1NtmoV4+3sRAy&qGfU*-Lr*J&bjngm2^diASHj-+7LITwl zy4L5*S~HCd&jEr~GQKkAio(1Bqp1?9owoAnfgBh)(Ab?(ewjGVu4+Jx_n z4&!b=-`Vz}9q1?{Q*OR;XLvCaBmbLCqja}r!zkmCB%@kb`H~tk{UJ@BS$xeqWRqy) zYU;Str0i6Usi;4!Ig=1H+{!>}6Q+vSLf>95YEr_#+kmjGwMo^Z4eO!G;Qha<*SurH3@Av@J40qc>=xEH!Uh>{)VIp@}s|cs{%+XUE07M#aziNRg2WslSZ)kK*_aLQ@Pm1fRG(A60m*9*b4k8G zXEI!kJ1@JXLK_i{Y|$`jGWW|MR*sBzTz>`|@nQsjqIX?RD9p6uDl&qLYk|ORY{W0j zb`g%p^i%Yu*ao9Sx5EQnnKT_os155SBpyXnGdy-b+cONe^95(zVVE2&;8?#+9yf_T zt!w(L#Hv6Ue+ON000!1_>@i78XuY=`foC$)e?0jty!M;CfZLkMq6nyTx-g(4(miM> z^#{Ei#eYwW!mSbK8oTqfL+nzW^dWcO&3a_He5CGJ!j1uAN#GJrlNf{2!M_e9V^&IF znj)8{joBUG<7Z4|d(U|RvY?hFH1WT>&W|fGQIwbHoQfE6ha5F8RwJ_&rQkbQksj@_Oq?Kr^2w-Cz8-#S(*%I@n zO2&^>Ufr5htxWs2Pe@Rm?oe(;!eW!4vA)??z|LgB#V&9o`{{;^+<-&cwu&rRo+@3B zybWV?Zak0S?1_-%UqBvoE1>3_hAGa*SII(XffBZ#7OwLk2OWBk6Jp6=`9ssXVhc*s zH^SNsZ@9I=;(ik7c!|1hzw{5Gl+nn6%B{OnD4c%OISPb%!$QJShODv>0`2U!sTMsn zM{b>?8~WT&$#w^;>4g1}uTF9~x9Gk(1R@&hE+a3TvFJv7>5_By8Snf;Y=IJc)RKuj zp)cvi>)v`3dFs+3WK*?{)DJ-zI>74N;c0;jo&T&Xv?9aTp`;lJHYH< z6xy9P?eORt?Q}Uh^>(bBqV+|4JCfbjdX*pzJ-$eO5!JUG5x8)OfZ#`$^}05jM|-O( z8LLj6S^ez%aqcGV3GAuq2BdkOcu{M(F#CXUVh_nAg{<_L`T+;#=C6PruZ|iCBo4+Z>>t zvKcTl^|lc*%Qod=RU=2XJpwSBS$z;HknRwK!Iy^VDL;!9K-+2kDl@4_VGK^~#uvp) z!!@Q~Bw)@=GMjI9c+OMjfrf;uIT0IJ$pq6*k&rN6&@~3dg(c?p8LouoF{VUuSEXPYX$H*x7ycc$GUcr`Fyc@B3aj z^S5dloUAJ$8%zjlf^n1pr_1EJS6JRxPx8bzSRdM*9CtM$=v}nwrJGG2FF!5dJ80DF zerxiFoBkFZV!OUecF!ykFQZgKt#KYyV(w+7)TRNU^)!qFs_>b=4jVZr;Ctys6zT22 z*tsvzB9v>i7>HDUJ;yEr>?Dg$=EN?fnNTLz&&4JFG=!v+XldK2%JGT48dI5Ir4w!; z@4NZ?V!9VU!16T5`GN(vc07wUd%e*&0M5A7U!B#LI@774 z*zSh;Tk~6;_m1EL7(#XYIn$?&&zpeNk9I^ccz7eQk@>P#e0_ynZF#UtjMQ(}BB|oY zqZck6f)IR;2A`3+D3J4})j5yd<-)y1tUH{s!4L1~f>WS~d^u(y#s1xLqM*S6l5p`9 zTE>>0vUZW;@e0nN4N9D%Ob44tSEbsj_Zs{-Br6|J1eJt$kl}rF!zXB-J-mLkYm>cB z5u9w(G|GP%)dzHJzN0H!!-`yf6md?r-QbEwaF9i?R0N3h)31 zukAp!V7-Pmn0Et_0H^^J^7@vJCdoHM{KK^`OC!WKNo2e5MYgB@tI5tLZrUb@7c&8X z*ltuGSj!LOqrSk%+_=ao_6@rAOhX6ontRE*7V>YNon7IE{2if$a2hwB`CX5{U+rE; zilyq+BZ%&XTqFD*N7OEDXggik`oNaOGHuuhyY9gmbI9yA<$qU04iAT;7fJHKhyK6Z8vg2EO8jq%><1FVXH zsAxXAQG2E|B7H(k?01{pXOPP>LAU}DsR5;*K z4y=FM!GfA`6b7}LkTxo!Ub=cB_FI){aC710Ihv7G9fGDPGX+Ybw0fv$Cz-!jM!1Ml z8PbF$l_|GdQ*OxEnXa`N4HuRC(4ooYxDeV88!Lj*6E=z$Cm+mG&kpdx-yl!}CxlYaMvkn+*w?kQ z;rc_mv0cZ&W8#eo)KHgeL3;KT)SBT6oUyJm7>rsRsD?0Rn<4oiOIA$s!`^rbTIoM% z6SJY>?Qz)VJ59+0re5d+4mkRuBdfAl-Ng~2(88p?a8s7s%~=tq8@|w4Z0van5W@n7 zQJ43XPGF~%OYbzlXLP@~`>0ci(iS#)^$W%iNKlIv6iyk8Hi6R;4dH||Lk9@Vv$c1h zNw6YV(4pNGo)P!bngzD|o}DA&VsSU+5Wo<>(`xk81i$K4Up2@_rrS={#{z{tmm%o% z!~!Yd2a7+S6Q;CYyZ)G-jXg4l;Ec@~!M;m8x*x`2!G3oW&R+TgIT&oGXuGtiM$W#6 z<8=pCZ=#*>GKJTP(WDh%OhNaaeHn-iODPnz@vs^4$`qV?!J-B+AB8cPw>3F(pUuna zN6xMmG4Sx-HuWNAC{%A?npm_CTYQ|DiB}zyzu@l-@z|Ct^2{U77ja-6qhl^o25{Jp z$s56ify2sxD#k{juhJBQ9G@3Z`D@lng`ldUNjg|OR#6 z4M@4wMjl527%ST#NIjHNxrJXO7$ z^8#GI$!2O_=IOEVuY!0z66H=u8K}h>mXM_kJcs@-_TDL2l&;;=~Go-b@WBWiRzo~h>Xa~%p5u9J9Ey+_Zd7^P$!ZRzX@`$I}!dw z1LvKo+Yl9Q>l^U@>6LX4i-UoFtS1;X;{8Y|;mVy;QAqkB;pF6{r;s75zd#0(=p<#( z6oxehbT*;{tS2m`*Mfc#i|yT!@E;p7pBKixA}P2gs(YskF_j*?C$^YFLqCXfgdsmQ zOYJ-R@{KSr*K8exLjZZ^<&nTT)QV@EdWB@=>#7YC@%)aXT*vtA%MQCpQzg>2s{0bt zJLY()fMV;_hjdSf&>_T*qV54ftqwz>wh;rX6aJBR?t6Ed5`7*VY91}Up!mAq5R_LF zE2AGYK~QHQQ^%)fDfr@6_R*v#9}-CRt>M2Y05^~@rWxNE@WckUnr=lZg8L~Ulm;zb zAM+&X-(hF1Tg?xt zI!jNJ*>_TiwZ?czDb*P!XS3F`&3P}45i@r*@8Ka>oqCTvYno$u!LuyY&ACE52f>r; zam=+a`jkBTcwKz2!9t5Kd^O`vKpw|#D&J8nd}h>7|^v)Mottv+k)34{p5LInq83_Gf#` zNWxo}dP?DmW4F2Fu-{C$ef(Up3Wo^*APl+R zVre?Bx)%{>?ei-U%Zt#tW++oAXN$fCuT`?}Kx26!a*qs6VXIn7n&lUnNf)V9C~^Pq zYo{!SCA}ruH+nfD$xA-Io+2?ggY2!k?Wb`w&Oq%Txz9$AP4?m>*IkT8!KA=a1S(9Q zF7h6n>JX>TaNVXf7OcV=<|`Dw2g`HU8wlN$nYCYl^xat3HOoJ%Wtjc6+?cE+nv+d{ z6Q&zayyE+nrP(=9-p_6ogR4$z^=nYHv_WoY2dtCb;8T^y>l37k_sfDFl;PTG6I3Iv z4!V5lpBidyVX1Ik&NObok~4#vvFsV)P(Ai}GJ^!WSLI>zEavDEnJ57lkqVfs=*kz1 zSDAWvPnPabEh5d0g4WQ|+~H+ny(P(2=q6!4inLRfktrPf>hfys)j`JSgr z_A-s(WAymVF}6lDIF)Doe*gYS2Hs)&+w^VHggrtqv)?*b53y z#3ub&X1(8IfRfI3%YjzIMYZ!22p>;~3UX&71nr_Uf$t%~G3C%Ek&6oqktVPyK~87q z-h{0^kYbM%MMxB^45?)H@PGB%yhXtQH?<(k~iNuf0zXe7s&cbG}jx& ziUr(w=5R4NHi8YqlL`130fwk}V#1mQ)thuVKSX6b)i$MYPA_mrm($!;{K@SskVi$w z#5k{F#d9qmCFnYcv3-3HGBHMo{+! zNkEl1ZB;C!80MdNmrsXM4da>?o$RrzeF^D-_~2J`;dHjWUiEo~8fL}V6frW9CPe=5 z_2vRwORlsTbI7xv`853o!u$5T+=>}0^3$QdWhxy1;20z?AC%m(U22+1;qMfSyEyKN z=G<09!r~F8OBCJ29ec}cq)%pkS9K6f`nyk>!l-x;D0r)`&5nlAh+mhUDh+QuD=TaF z7eAo2y2|^`SXZe9yUCev%p|m4{)KT+6GppnJyjtJ35Uj5OqT1QwPo_{Q5MBdf1Mx= zOXGbnDlosK&h$&2r)9;$SyOqo7$D4Ey>@Xy=$ z-C?!jeFAcVDo1#UDy~#%?~jezntOyt2bC3a@%uJSj;;0)BTg3 zGtmM_b_viN3Bm-j!-1U%KBF!rIa?^`2Q{u;RrW@7GVw9g-&>zeI)k@M=u0%6)P1$RoMz=;~?!dsQv6x72JzCi@rm5K- z-14%0h@(7qWuyh_9*CD|rWfU`BjJMW$4=_HG4TxREj|WPN-#wwN*15W=I&q=eP0(t z5Ed(D4@S(~zVQ>Cax1a9S6Zc}b>cO_NZC^XF}FVL2!ZzF6jEMo{FzoeG{KdPz%|dN zGG$byOQUv5Du@n(;H_$(e{m&j1>D{+pi&TofOn76xV&~?Iaw*^5e`bNBs@LlgOSn8 zB0+6X8kXy@_a3s{ZQ5F`js+&wgB&?RImS#FS_<&7p49^=;)EQcjO#wDe5tDD5Z6gk z5ODq5#6@*(d^ICRyog1V2O)w6?pO{(4RJN*%e_jzw!fxC!8%U&LCD%o>RA&7Rl|gj zpSwzF!?{Z7tTmbC>80TMG=a@eo_KAB9XQ|d|E!DPpM&iU4VlDZ9rAR$zUl^yw(PID zj1s9Hi-?%va?BhW?&RI;R`)rnut&r1rd@j$7CP&|o+1wti|Ki*Y|MmN z9da*F1k8eF(^DG(2?z8gj@<;Cc03392;H>BGnX-;rON-7;a8XU zRtI2MN>~n6;&U)m357WFx0ye+3OATa%|fo1RZ{FsXg#%0Z?o`dsDy0``~%QiqkzN1C;DitFUa4!ICd>2H<(w~%$9*!vJR&LsYxV` zAE-|~YzG+gb4+YzA~{ksHrDgphv8>`NjN}sUo6C_SmUh?n>QA`K*!fWcz%#vB(Tes z#`~g?@(SGo5}JiI3}BX_-y0VfgW*K0)8fMo0^M`!ob_^RuUZRVq7ZH@@`n9n0e?Tl zQya1hxZ=aWgh^_+Q0jm1jpt9t8kr1rfU;PIfcse;sV*c&4fhI4n?fN0W^}yPQ%%|r zT{ve>S)zeD>r9F&s%|!Hx3Uh9oDod3J(-=twSFzTx~w{Gl+ShVOI+-;P4qa%oP$b z?@_=jaKHsES0v_9>K*rjIgr6fZ^5<~Xs=M4O(O3NCd(OrA*d}0=C(1M)dNkEF-2Dp zN|2B_!Lk5?6b#Tdq39{levlBnyKEKpelrFV?ssOT3C|K918>KmV|nWkpuScSpK4%T zA1KWljohZt^|jq&a%Fa1AO;GSR?q$8p;(6hJjxzBOj7DhXM@~LWSCq>2m)*@%cE0d z3&)OKelV)56AQbtCO>sd^kop4l*afS_Ycy6wm=pc_GF=`2BrzQPY>8}1PDSU{^Y~6 zlKSfDl~QQ{?wK*5Stvl~Lg+4DIDLV3(HF;IeA6|TiYhxVG=I4auR-7MQFF^N62T{u zBgdKDiq}cGpZ7Cx#HZ`gi4r@bZ-6N>Qqtud9WK(4ji{5iGutg#7l`pH_(`rbB-(yt zu*O1HV}VxYNzgyzb=8i>vtFyccO+k~WAJ}X8iMrVT?W6#;igJV9n22UOoe*A(@WWh2;+s=0=`Gt<&oo)9 zSue=9vDHK&IjKWSoeUie%iTM$L8M4mNjCc5G7ccrYw3}*@#1g#0=COATg0~8Z;23?MNqOo%U-p z)21}S+^82oDUky0&0|TmerLkx_j@jNF1i?;Y5w^pTe$JjZRDqezzM|AD(NTKNy|e((Q{yHD@= zFYbP9CdKCxbxsrY|0~>mPv8Gt?*7mJS?>Pme}%hm_+RJl8~$%{_m$2rNZDsY+{Tp= z-8sv=Z;+Jl7OVA~I&1nrk~*?unb51e^*xr}0@`Quy%!SK{QNUwLAQ^x@|i%P}pl1;h7yG@LBaVe=QfoMhZd5F*ry0k+GJNqHT zzk2PuZ}e`4Xm-AqreDffvpH$kjyE>=c~3VA z5DiojZSX4?r6ad$FC8tbuq4eY9537#YGNdX!Ya+eO{A*3;29+);w;zz2VDw+AyUIN z6}qAMq+=w*xn26~&htX2%MAwE@UQQQC`vxs{P_#oF#R6g`6y*wvb_P=65B z-3u@0+aeH3gxuH`DI~6dWMoI7i)d<`PF!Suovp1@p?lmF7t9YMHp{DW=O>rjAfT6b z1bI7{LD%s|wb}swx$N~;xChaIZ1R=$`9r$~hcM+KFL7~cQs*RNZh?2n)-ZMW39~o{ zlrJN=hs48SC}O;$t3Pd`^i|!xM%qeO#aC%$`FPPrh3w;VL|pW&JrfkT{4JGjB4(;^ z&dtu_W(rIVdtv`ZFR( z)V^*|VFC2Bq*o-by9l0TZz>nwo@9*$t&CxGor;8(7Zt@Db_4fFN?20s z-uYNG)0>*=)+OmLu4TCyXK3@n;YWSISe=InH9)!Zg~D>{xRDB^E>wo%TC7tfKe^4H zF9xV^*m2gDg5ps^7RCbJ=g}E0m-HXCSoc6|vrv>*>mTVAPf{24gl|Hq%(!%)vNKSM zXA6^@%xUfs}B8a4F(w|L(wE)Ti7a2l(}?36%oS7yBPsThZ!p!Q4*uF#bA*`q=$~; zg=RFOj4IbJ1_=9hQV4*SFivpGU<5*el|HpS&x4E|l5v>j%;)mB8P%^8eM{p`xv$F3 zFCNEQA)^5ns)R2ZBBsuongXqf_o}{%N{_hT#KB=P(Fdr??n_CB+4LiUSor|2w;>UM zX;SwF>Ax!-Mpu%$1JQL00B=uU`M_12Yy5T*OJ(!aP%&nWTFHmAU3)o0{NxtCaF z&YzUO24Rwe@vS#Um8`}4!xw@S@Z_;-;R@?iJ!O9oMSSDW&K~bqz&u0*b3}fi-v$G5 zun*s)z5OX>PYrcJ3J*6FcYrP{Z||fa&wd2IKH_DuO}OP8xa`mha!;|2}Z*9Cyz=-oukks?( z1}1@`eM++du9%d2~E>{*iIBdkEsXy`gR{0UY-FUK3SPy!Vs=5=i;<*M4?3akt8 zf|YU=@eSY#4RL;wiO78+u}VZnCJxAWpCDZDle}=_X4b!gj-n`EmR4?SBl9N23J;+= zSzueG7ww9uURFDCP6qdiF|kx+yX0vWAxFJzex?)jyFr&EKYeu`HI=nT4Vl!=;xslj zPq`LwYOg-9)3{j6ZfdD3Tm@hYhqbo8J3$rUb)Frud9LYa?d?}cZ(5C$SaFNDw(QtwB<~YjJ_#o2~_?0OeDV~Cro;NMvmm!sX#jI zLr&gS?6qI2u}B8Z8Jgkc6Z2u7%ObYKq&H?%F0>R^M0)zJfCm9p+pUN!#gL~Y^En}9 zhF=KDUOjH3v4LsR4I$NZCz`g>CN6e((1x~kkwAq;>B(~F1QfO$^zn=ElJ}?&&4BF~rH`o=MX<&#e$O%6%ZDDi(>`TRiUhrHY@S~6j{qFh% zq@X3zU*ilF`wnUc+!8<1VRcgxC3Fn2+J%2H??zPzafH*2A<{$Zo@5wV6ZG?|;|?M1 zVrutt!xHN1`+0bNV;r_#HZx|!4!Z#eJo6iO8-Af#|Bw#Hs><}GIgnQj+T;6YFeXwe ziWEw%$qdT;2qDBGpws<(4J7pKiUct)jG7~ybdT2tqcFpXJ@H=1qZ$Wb;=d~Ikl1G-8ziGwH_DGFZ}4fhqrkZ)epR0A?}jWO~(+= z9vPAos)#Kv^~vs#4%0_%&+~Q~+N+T$PXW)X5Ug5k`(;p~&5;m&*)*Q(>{yi@zaRUh z?jN*?-ALl_mCL?<&F;4Ty?_&lc|hvhft)jQS<$UMW`6=q69Y3uWrM?4``)w~$`doP zg)a{;hF=cx7UsESK*N+wob!35+@XCK-)s~*D>bjj%>oT5?zwf~RITZ?{H;(2xz2KU zAnLmpNkC`z^|KI!sU|{aIB#*h++t73iV7a2q^fHDM+~yQ91P&ucxpzO+Fr>UVxRD`>VL&RIn<#F z$V*=CZUrc-=>$$L{G6RO!%JB4#Ml)dat%QwTzO4F5V0EPtaka1yWN_wHBE`cvn?QJ z+m$|ZUAy*6QBrnbR$5z>F44okL~uv>Jo(EN*PLo!_J^*w@`A)RKY@}6KZg&3Vc*0P zU4&<@iX8;n{;bunbCUTz!R=m9-bMWNYk`~wDG4>0Mz2TnDX>w?SOX2lo(vFoYu@7P7pAyl%&mr2isiAUnac}{%_##Oq{H^ zPU~LVPLu;ii|HVn6{Z$~8rSfhoTSwb+vlP;sh{ti0Imf$O)cI82shR)(FX5K^w^Ri>MmS7Y@#RW~~4Csv=%vd8XB(PZD@%`D9y&`~p0W{aWADdkT) z29vo>q4bWFGy(LR%#C3@wYEcox-!?ser-#nb|wM(znaCy;248fPLznXgP1$`&0~ExxI_CX=$9 z)4)s9B8M0Hg~%2N@jI_cvR&ziZd|4z`b<(GOV=J`1GhQb13_)71RbyOh(glo)AhYV z!##9x2MWCArvsTOp$kj@^JfcGYE;qh(59F8qrhi`?CRs?FL|H!zewKq9l!ankoVtM zImgmitt(mTO$AnRJ1+FXX)K=^jen|wIPnDR4$L}@Gzfn`-fshEOp%%=pq`Gj#yI2h z=C0tHZbPR|OCU>>8q)Hk3uYf`DL*GA9_I$Bj0!>S6GI7T-y=|@w2>BPC zBLc9IiNF30fQv}|c${w8;rf{&`1HPn41C}U*NTTuhiS#8Czhq!ElMUp5Pm86$)7Of z-rpphEb!n@3cW$R*>zp<`TFfGAdP2wTO;Q&(y~hcF?F+E$)jjQwi>|59?u1!b&^bd z8^ep1t6Y*BroG7SGZ`gIDOn1`2X~}Dmas-mEhRO)nuCC}1c;s{>p_RO0c`wd&IUOy zJO0>|(oS_0 z^|Le)+|Q?lcm{q$tVtYbt|A< zwpQ^V364CL-B@OIRjXry$<-$%0rV?Q%F$uJR@P8cwILE=TMdEgWN5WngK|q}lv=^8 z!n-xMP2lDk$74x}fFYs$+XgNABA?U!G*zyRVR*vk!TF!I4hvzD|`4WCC*= zR--QkI`cnvWLBv(Eq44u{5EBEpWyk*gSVLp1}^}%HA-3kW)e-~zZRu!|32?E<&7)c z=jbXozv{(VV=ZLGB?}OED9SR$mQlFnoRfu>eiq6IRJcnKa(k3~fJQrq|;s3Fi1c!%Q{%gAvTMBvvWi=A2ax zLT+eA$G*!>dx^_!dA=_@>8I7}02Mj-YJ==Sfg6tg=z1)|`c*O$Wvb;%pk?FxW)^M( zYRFXOxJ6lBypCmyVJ2F(K!Rk1uc9iReu{QqD57Qjm>d*93Y56NZjm2|=6DZPv=;TM zG(dH+mw>`B9ERcl05u+ zEAa`el|gf-Ab*V1)H4yMWj;1^G5Bk zv>I+aGMq;a%k`UsLAZ3+HG%bZg*Eiev#f!~}PAi)b(2o+R zhlIHSDAq06J0~rz&^wdqbGqi0Ad?(!h##U(kV6*9%i(8;259g|3iYP7X5>Z!`XavD z^Lu)oryAbIp3r{Gt$Rtw@QLLYWrwC$HVEOUAd39#1RTghXasUB7iLofe&Eevu|b1d za#C9zKQKN9H4v%0o%EUI)xDJ2;#QIN>&%nICTeu*mxt^nL{?MvLZ(q1poBVZs7I<} zC7Xj1SY=8PBOQ_`?&8v76%0Bg1HPaFac^tokWCu9t#5G-F8 zBN(JHA6KiQ5ujB-JM9F2q6-}?LwM0oYtgYU{J`su^Vfl4&N1x$BYD#WLkuJM0J3Wx zA}PUtx0y=Tthy+8nB}xFmd26qa1igm@OA*;#qb5SU9|fr8>p&jeFFH*I3xbXKhY#2 zJd&;G*e+63UIlfDv32hO=Su$}-%s`$12ptNn%yf9crGLl(M>CBV1_{o<6<|qRfAlx zor19uaC$7OTXB8+K#$SR)6F!d^_=$^hXYOy#?q!MC1=z3v9Xzym5?M4i&nZhA|Xsi zK6+32uD9&Z9HchKBnBlV}>6HVlg*m_E_`q#rPY6VciJ?yhu z>ylymNMGT(EU0L(uQE+qR?i7mr5JYkxiGe&Q8f3Q!8gT7&_pIsZcE0bk;Oy@)MJA! z7l*8XXAIiZCBzJ#Ux*(9iwlk@D=7dV4*_cEIqH<%-&8*p{kJFV<%eeGkkNPSw4%3B zwJ}%8xS)(mHVWr#6KA$2pdA>Qho%h_+Kg zf~&sqQAY%M2fE)Lawy$7D;C)5@}pNMk8_!hmk0}P7ed8Q&)6<pacP?T_4L#P z7FwAMK4@7C$hDAj5@6i}XFEs~9jfc1ACJR0raG!fFB4t8T$%%!7q22}**&L2TS(6# z{E>wIz9BDUnB!%Bh$4jz0=v9uA7(L@OA0g>FuX~&*D2~V;OZssPdH;h?d?w86bye7 zbMs!H$n3dNR9{(Ck)>Mv<*}(kdotOvLK%H{R7qM#uE?XL0l=EKc0XNF-ORdV5KFEw z#`=#~%ptb;6|y2w=0!!12RfVpyz%m^lLJ+X3ZbYw+BvQKUeuxY8L)IujXFZ#zrNct zuJmBx9K)hBlY>3Ba$cw2dTrt*9u{EJiXJXk=#SSHg1)dQ+u1+x9*?hfn4I}DjmTlu z-s(xS7oMP`sjt4*x)T7>@7xtoFo&r=_BM=tO;n(75AxA3iG_<$)Dy82OXOK`uzp`c z3(H^-M|;@za!)YyzJ1p|))B}TAaoUS>bnVWhb|o)A3Xs@EhVszoj`Pn8kv*j0RVGQ zcF*Gc#>_>k1(pT12q`<8tN(L?Hh0#Ja8oz8smZ<1+rgH&RQBD+MX%6}xme=qSw>w@ ztI%OO$o$~oMn@owVtC#s0yFaHSavQ>!x`SOz^^Wy+ngrMfm1WEIZH(z7qqDqvHJ!7 zxA`0x0J#1-o%98Zo7s$=HtVvk*Y;ULB9?*1A2o)!rj#!Tn#&xS9$JJuF#GYzSD3~Fx}4O z5{B}P|5Iw>EXuuN_YWAV$|ye z5A<4h4bf7`JGsYiKinW-k_h6fdh9N$0!mcw?~!&7@Lv_94`35~ASdEh1p(eKV?K}L zbJM3_S(bcLmc~(rk2WTva(oHN_T_jp_fpEb*9Hfm_eoYLE?tYEZAg48t@wb45^&jJ zO4|sdS-FVL+Z#M39i&9H-xajO;x&R7ziw26@mvpFgYjCbglHx<3ItLWvjTuF-gcOX z=m2hA75MH3*NQTa$Wx}J%R|~ctBflxJl4q&*ieNse=>@!GZnOT1K7_NvL$s!!h4ni zfGL40|Gh}Y;bb!8vnbR7ru3j73VUUkwo8nI&4bseAgXFdq~zCQ1m|4ci9(i2X{_Z> zIt{AUjD0U)vY>5KFFt@F@QKpm*0@CR#nU7!+wO^u?x2|lV$(SEuZ11Nv%$HbydRTD z^G_Y&K27Ogu@%cV=++7^QcurbLf*q1F^`BHt3d{d1=(|{UvQz7Dw+Er*d!u+2v^|x z14N@dqI?$wT08f94<(KDz`aiJ5hBewiVwLfp7-XHTc!+xUy3g17lZz#zU#+!fcgE58(*yMk!Xy>wM+Ji ze!Jh>Z;VaJbt_=Zm_MxinPBLfA9=w<`Y%LWiK@MfST(OEvjQ=(dEyKJ01)|hb@qbWTIg`?I7ykyx7pB~+Y_4J9zKVQJ);c@%I<|DPO`~S zUZt^x4Wek&^ec@oB5HR8X-*$9l4i#P==bHaRjFm`@|Td~l3ft3gb&c9?g?cuocp=U z+V*>Rw;AjQMq}be+8Y`1F<1?N1NEeW1J@1>5A=UICDX8%*cD~q&JYf zr!DH}BPF5~XrRG{%Kd#L>c*@5+dnUR$nWL)#h|szi1Zf*3izHTM2=1ZX#EqLHOV)( zqU%^09(4$qgQ^39G=6%)eXqa22Ul&xW;_(T5*i4ixG>Kaf;z&1;t&Z>`#LGgfyvF` z)h-`d{hrT#!k=m?uKh%h*%h#>*$r??C3 z|K;{!`Jb{cS=FS=hR0s##2F1)d+cbzfRY>7<>{cWXTW&d4)4C>8@Yn$*%>isw?qH<~b;TUhj5N@zBn2(>K3@N-4ZR~nhVIxbct*18BbG&W-Sqy;^3 zMj#{iT^_JKZglF+)a>~Hfeb-i(Z~HE`theX*15EXzP_0rbHZlyC9usk(292&rD^G* z_XO|LTqL~KU)PQW|Z59-=r>+^u@L^>rC z#K&Xd6z?R~Sw}CK`MrDEI7%6Z)b#k0q2^^K>a65Kr0JL#+1bulWh2CTCe8S-%fJl7 zCOwGDfc&`QT26>4T^S@5r$wTIa8wG0WsAq|LF42wWw04d$3`>|fi!`K-G*bTsO=95 z0D#VjbBeEt%9@Kj(j`x0v0AZ4m@U3QjyGe1sq;fB>KUDWs+2QeFVzV9`j!~>{TLkx zP3H*ij9iKE5$sRTU)dxQ003&j;PHb!x-JkG=(D?A3@B4hHK@F-Xj=6SHOJXP+ZsBP zbuGB~KU%J03U`{7&Z7I1LRB5MI2Kuaf2mWdI+RuFV$Y5nAE#(gEzPdG>1Zv#1_8Wz z#M2Fc-=PV*Wf$FHV5l%?slRF}sF+m~Er~*Jg~(}kspEk~NLwCjF9@$4e@Lfy_YyTC?NX6+j#*{(4q4XWJ@8$t?olDfsomdYJN?0T2AJ_fM^PUtuftXCtQe!?bn^E33wdi#lVed zyQ^$N9GUK30JN4fXAu|0G&NGaU02blvpwATCRjD6@GtvI!sHp?p_|4Vfa3?U-HHV= z=K3G%JZyIcfQlD8TH$c-N>sAF`7ymL=CxiEH(KoQv9MB4kQIW=GJq5@~pmHjI$5te1;n%5>BD?JBHBdIxn+te~__Ba8?^^<78<*=lPNnCL2|M%O zRsK05CEYKA*K8#rSpa#cXe97jp;PieQ~#C=EUQ0)L=G!O*>Dj8gm4Pv_jP>&|Ml2> ztdONJ&sPNcUj7dFT^r0%w7OO?Tyki#-(lJkIKXSI7o>|t>i%wp>zVLZ27sxbyR{p)F{s|R7nqj_ zi@F-5CEk04-q9^gUO%6Ncq%>m7MR{lt72FskPOl(oKg2#LesU!HNJcD zb_E#7eZhH3u_AFCMa>j-Uj44%+8tta487QHoGmz0{U!Ws86?wX4k$Yq7PfxYJf2N{ zjRd5yN9%5o3MS2qTtOJ(V3_l3HkP&>H8~ABXU*aVL8Wl#KlGB|Z_RjKvK3<>K0KkhebT*2e@#=Wcw_#s@tX@7t?T0m!$ z02AbqKSyop=~RM@L~Rsmyd1?&>OSAdxP!YLxS|xj7MpiI!03zo>F=^mU4ZUgW(1ANFSeY0*VD{3UaCMK@{i6XYC} z?XCs)uNMGNE|uq(OYyghkQyy_WnCBx#jxHICnrCoJf>KMX-$GDNqGR8@c|ZS$?V*d)y5Nr&F&kl2*~0*l}J zr_S`sJ4n}2e)KRNqrSC`rNASWV0Q@#HXTwt+38rco0U%ALh_gKp0Q|zp*cI3!ayZr zeJ>U?ARn-)pr$K1Q?Dj;D9w&ToTWW~$I0g90vf-)Ns?AyLTm0#EW~p;uvT@#0O9J; z<@=Z6lqI1xihGJr_U<6UjwLGI52shkZ;%Ge1ydNk7V^n{m4cd#)3!O=md(n=_z9>! zvx8{~-_JzF4Itr##Scc6hp$_QV-T!&c#gu3->YG9%c@HnkLz;n$u(^VQi95&y)^Yd%U5T3BC|B|7&`(Fkw zPY{iNTc!3d-PtKUnoEM*w$-**Z(qdk+D1i;7pYziKy^nIY9QVhR(bjm+S8gqVIc~9 z!)5Vj#LmUTY|>W<{gwlm13#c-gL=;DEpq%1p925@()}l#A11}DXn57g4J9K0 z^=L6t*fZh%?`q%|51*fbA^d7cmOtDnJbYzCB(Sp0BN8taXVlgbs|K@exx!?H@M5>ck^$$Xu8*Ynk~ zO=8;j7r6%vNdM3J0EqgZRDSr-o6P>e0$hQMF9H0j>=M$M{2Sp#N8utf(^OU!obsX4 zDZ*SWR`Oh>Bu_J#qIHn^xd_g(kNu7S4#-U2DbK_3^#aJ}G82DZ>n11X&^Y}1|BwY} z(3YSo=kM3i|5T{}8Ha=T{LVlEVW}flt|d<N<{~BwkgJvC2u1^d zJdq8@k=S4sP$E+0z4xsZU?Bqwm6;X%LHmFX4MD;hMV!(%S% z+{`x7n;Kt9d{=rvsNg?TRm7BeU2OM@P#Y)6+0_)8FKUlZ;)qmxq3ZDpXX#2D=)<$WOZp)aS z;s`z#qrx0-Cpm)$}YfpUuX#UB>l;3pikNJ9y%gJlJ{ z%a!{Kstb7NG`L-^aUhSD z-R3-+TE7%}O;8YmMlOc1_&aK%AvQ|&^xt;fevl@sVFH#N0N36PgRLt%mMb+@rfdI8 zUC4I?|zGfcx`A z*NfWQ)Bf^-*~Paf@=Or?iN>#yESv<_52eGANxP_|GV2iviHEcSz6t!Y`!P<^o3JS5 zZ=>>W?Lno`4F2JJ0Ce8bZ(}Bz(1K%0mDtymIYD|?IUk{%44R)?P4&LN!5avZjf0aA zlOTf7Dmw`$0XjZYU)g@a-Bbj-8wK)?H4&s_eZ4gEO^7-jhm4{JGIBB?FglMSi^F}O zf$EI-#Ot=O2ndRbmw1)iq4gfIq3!0+y6(Ar3#?>ZQSN4vLdktd>JQ|v2~WDiJt5Zh zC}s>{Aj_b_*^ct~F}cdbm^j&`);HS11spfKp{V(fqV4AuRYXE@V|9w3CO++dFcOnXczILjX{lM4@+eNij4RIHO zT`35gD`OglL(L(ERBWA(^(xDs4AplGzYl~mp#A%-iA1_YeynwFSmY77=x-zo$-}|_ zC?KAR|BH^$j-|Bisrs#-MedlPN*dBdw&EQ|I$*}=5rtz#0uV#~@*kTqXP8Ktnaw-= zr$D#gR)94_@lI;DqtQA z+kRBCyn-5zza<>StfqEd^vyO+oNt-Xl>lft(2RXm`)D~)fLw*l{lag4V<&+MVN#?m ziT~8OvTt&JQ1_&IAlLoVc>ryIIi>$jL_tu*%=8g_%74L;3{&VP($)(%DcyeCKuX zStpqJH@%niem<`3A_9A9FWK^?^CSUn(Apz2IXDF7Yc*z-w`0^zg+Nz6N>TSZw$?7pz(_R%$8mf-<4ps)@?>5{2GCX`kyBaG!N)+am7T)wMgSFcw*=-lZg|#&r#}3M`~e?8k1~f! z-l^qa6*3tYI+sl;014w4Dwv1c@2^`uY;~^wC;y`le4DJAOeynv07+J9u~XXIO0W8GOVk%@@-Sj|Z6$xk4evygk?RYowVmm`~|KeyRY zaR(!YPckf`j&zcOLI95jP%+vqnZ5mua-;lWS=Nx3Jq(ZfI$Ot4xiD&fLaX{vr8+zm zqeXdZli8S@Qq;~XF!9$Lm$EJn-tx9v$43Tvv7tLPT3@H(FhHbhLjC_8bJ3}d#{ zhq3z8huS)p$r)q>SHh28bsu$P8mP3tMaNCj%V@cuRS5P!#~`4yV9tan>c6W$Y;f-1 zga(OM`d=Z;A=hmc$7R15KtqYh9hyLGL{MrnP$GOFHEZ=PD&qRsZ?m`5kXcn~89A2G z!01z5Sxg=@25^y&$ zagbuzXdg=g|7kd20D5v)@K1)Uy=k8T9n?tI(XYm(Zl8JiB{^?FN8rqsd%3C9%4PVc82v$PNEVm5eVI%`_v* za$iGr8XFA!<5g5jH@@kblyK>Xd`$nL_n3#K?LiX#F{(Sy0NBHg0Ad_Sg|fX)FPn+p zB`Ipoqvc%qnl1tb&T}xk2c2ndpF3^F|CD7h$37&}_I_FZl(F`Naes8vvbnVXzA0(a zHxyLmnFhWe|Goo>jah!gK~G8N&|iRDMM*gJH`)~bFC^ppg+>H0>*SwpVIkv|k1?Y! z4M5_Uv)oe4-YS&eoW;@W>i&nR8ANe*42sxi1LolezO<}bXAv7Y)yEAz+b%E$Ddo8y07)WRVnT`$h?EC~G!tP{u=FY!|v zlol~?pJ&A6JV}UQMOmlKf94I58=~Cn(E{?81Qd$IO6w%d62myMTb?|`zs3n(VLj0E zyufn;%Do~Powo4yoagtatQ;FteO4dW5Q1C@4tlNDv_cB?abuEWPgGYyV>y+ z2lV&+LHYW^|K|-6ybZ-sewvRTPvO=+Nsmw;z2 z?68JYQRe3{jQFrqQ$qmAp9qjvh#m(#1g9cOb4G2E$fQ)S9ko{4yK(OfA}X6ddxd^W z#m}AKH67Lh<v`CB!m(JUGUp2neAlX|z3~asQ3IOow|Bs6o2VqcnTd-%YZ%=<3E?OCfzB~j@a{)5=u+!wQoiZ56=BTK@~99XSPYxW zB%$XZeKZb^BAjC(Q+6?7j1wl$6bwI&{CFC?J0`N>4v&>9-W-`8o1yjJQ-T8f!lL=< z);p8cBUV$p8uRTL*VoUZBzdRHyo^b ziDqLtsOy)P8^yG&Vr^ZfyEM-AREs(*_V1L1T|p6qIcboTpQBaODe24IRm#?|HQmAP zC3nctM2Sq9jv@Vq6DL5mfv%BLHfA6SvOr86+yCB-0hs}_VgLE(H>!Dm*si9Xbn>MK zg#$n*7l*0Ehh#(jl(nkq;oxm#@aT_%Mm;}C^l2YJAV;0;uk7Vcjoq!_fk|+@+D1=iFHVGApYN112J-9Xn8X|iUD<@+I`?Jw?GVI>Rh z=vte-#%zA}{zRSC!R@u_Iy7oORk~Gmi&Vq&RiUDqeTh)y1^Rmd*0mMJD|sRjK@kwE zhbS~13}HpOE)D>Mf1R!TdkSR$Ou_;HK=}WELIud2t7&H2`e3(ZeU=A|fmSr|ls-_f z73qB020gM_?$+u$23rYH>`NH6Wx61-S{KV{Z{L^s6z*Ze=aCN z&-d5rRq6CDwBO3$ob)8jCDPb1U?~yVa{vG^{6kPXe!@)DEC2xNVM&0xW&)51OYzON zKz@C}bTx(n5wO(w3cv@DCm#cxlLq_y!6~IIO8_j=_{9jo>;Z(M01PqtW_AOB^2I&_ z0My)q0|4Ojrx<0UDBI@G@JZqUAqzf%Q~Mr9PNL{8v|U|NTYylgvW2IdaD!o6UBLkgzZz<^%wFfuaj#uC$7?c}wBZ!>(J zikY*-xsxzC{?4x*SKPQ$La)_~T(EuLBtGvI3rf40olN2IYvf}uAGac?*_g{UXlrWY z=!ZBRy=57(qlto&CtX(y1Fd2yIw{-t!JL=&Y-PqNETy z;t23>(*e_}q=KdI>?N-g(=METxu?E( z4v5c@ej>^%Mx_JG-m|YsAxkD8H%x(IEyUO}ur&9gpui^L`bdA<1hz=-0gR6<$Q$Ld zBT>kWDSrl$(iq%byglD~OrRkH@ONp>K)q(17oC`PlPxnMxoD^G6g zZ({4F8ygaq4Ivl(=muhNs6Q|Xdo2PN-LEUOl3KprT$Re+RXe^P^=S*ob6I82x%2a3 z)4CqGSpOV+rx*7#J#yc3e)jVN`yJg#qwp^iMNx4BsbK;tg)%qSqQ=)qp@g45G+idb z8k7(S5!qQETJ~CtHDLQY&THSNI1zr92d4Vv?_s5-k|pMyK@;mcgP#@J@e@17 z%q~hYq3mQFy6nNkA#)zNEhRZ!^^(RJ0|O1L7qJn-NqQ7AO($uAD$;7A0b4euI(+ds zQ=9za_SjES^=B;w2cDdDR;t3XDcv#OxY0s>K}8WKt!dUMhCMgLmICuT<}AVqXePeg zdJV0CSQDpFmlVFQjx-RP`^zS{T&@!vGsTVh?tD7+XN8RZFhN!%yRD4Ku~Mf{7!a$F zACip|a%YToBX=SIbTD$RL4Ex`J}|cYBK<8LtFjPFbh2itWGJG2qg-M+CA1ahGH&wv zv>$5@6_i!Cl&_vr$q$j#v4&PG2)1V*q{M%h+uW>Bi+RGDMCGWGns!sJ)sOw?S&nYP z$mYJ4YTu|G+@lu8=GzE6{TbD1A9K%gHM4@DeS5fB9D7%XrY*7d-n`FRb}*VrEjvEd zScWwk2pLk6U7IInXG<5P)cIAbMBOm^xi&z*Noc)2xw$ZEC*-x z(FlNijcSO6JgVyj@9_-ehZvq~4?To;)a;#`Nx1tx=rS{r+svhnt)L5OF$9{yK985B zYnt_Tg8kE$Xi0#)dDW;xC*3h}HJ0TWD!q@Q={>Z|e6t?vsV*vc$4eESkA$eR$$}zS zZrlg?2}jDN8~3(n8V)GvR;$_OUCN^Y9}&lVhOvaqzQjAt8OX~Fh-)Z_RAoNhAO`*f zE3Az=X}QvP)DjnZDCv>?I0cqNq<3Mhs(t(I+=W{H%+D4*e~6waU7Z)~ARxb3f>R3+ zfq{iJB;;j1F)n$f6DA58ucR_sxi_uBe4~gE2Y_zxPv6InCC?rEN88A`|4kDZ!jgV{ z4U*@DP14QZ844Okx*umR8OSs)3*Ia7#KHX!n!w0&zlC}13x-!v(%R*}(F6m3(*&g_ z!cpkvc7m*Npa$7_cr<&Y;RPaDw)gm-A##i&OXm}NAhZ`1KY?>hmtB|0qG&;CG=C|8L%;pFcLVrp--P{*PQ4PL|P-Ili5K|NapPDrw<&Mqti_+&mJRx zF$@@kn`70_tYCZ?c%+ID#t_*RbH9*cF(&cKkf@&4i39V6r`iTq%KwonqqQ#?&ZcpG zdC(8+2rj(-O$~JK8D>KzWmN${z0%Vuf4%z6v!dV1lZCas%$RhF@L95Sz|3DxTHjJ& zMfvr}>=}K;7{An3Ro9%KRGAFe^~Sm}#&h!Y4p9rAt4(dlrX(?WUyy?1fN@jFY4aMTNO3r-V zAE4MQD^F#k&2^0~nl!t|!7J0#LzZ$TlMizbuQWgq2$lm69VKi&OC+XyH+=~#KYCYh zpQRg@?csIBI3WKnL+Pi7@S7(%Cfp%2UW@fd5JQpI(}w&QOgSD!Kqxfz1#=v}i6u}v zLn%Yg>Bz60Y5gfYi9MILE@)KrvvFJyF_yewG>4ndfZ$p-8U5ta)aNWDic_t_G?PX{5h?(~zvzSy$n+nYPpGU`Lgh?gb?( zTVL>KC3ohbMYl^Nwz2Br?PU0rMFX=+#gHxTE$7EeU_-u?mqc6mE`h}0qv*wT z>)qobyBNj$eX|UhfM9vKbMjhEmF44B>nv-%&e)3z1~L$8=dDZW@E%JR69DxmwH!D> zjQ9p>!P>@Xw5H(`z>;}|ta(TdE?BW(d3x$z86Xo=$OryfbUieZ08J84#RWX7aeN>* zw9vZZ%6Ho34or=Xz-EfSz|R@&Iyl|}a1dYTc{ozce@aKdHw55ZblXS3L zlt%f}By!=O+zCFGn5@Ta91V8?n?&KT2uq+x*>X+nzAX}_JKzd}27T%&& zrB0KxCsxGoQ}t?#!SlJj2}^mKdj?PRlsxxch)_H<%O6V#ejW3I zusXuSE&tuE-T0*cQv_sHhNGvKK$fWuHkI-Y#`UZjU zJmkK?c)6^D!G}DtP#Ti35*biTPnth0lp8Rz%w!bhtr!9wWH&?69gb~oj98^{=BcZy z7e#fl4HY0W$}(M29nG3cjkK4wMi&TgfN^&N-GEEhlmqF9-hx?mI&93NB7z1SXBwL_ z)_YXVZCxQ!4q9q%4#B%vRzze)o{P-C8%aVnSu)K*d%-4Fr^@KA-|^`orIg{fgX*8y z7!vIKwXMZh$uY;+F}_)IRMb;Wq;=zUiPr74ZlpF)tnGQ;-_Dk6XIa2=)mmm_s!<|o z6N%$Be>EIo4dJhn!7W);koU!&aHj|n-tfZ7Ko+lvn!X;=mc*sHoo?k*k0qR##OFN` zhL+%oX>0L~zqqFoT5sVhuR2NwA;8!~aaAi7b44T9z|H3?g(^*4vwd9i@~LZ5R8M;v zD29gVx?{jTS0-MBweo6DtK72BEO;D;7B`)I&DEcw&4tI2C#sTEm{ovXmd9RX6qrxi zK_F9PjXs+%baYwPuIe9Da2f#YDp8UH=ikRI#-Nrh86TZaZ#DL?Akn;+*?=veDBxbT zMW|@>KVkWwdyXfn3AE1F+3 zMNBR=f#CxSU|e>nd@Y?*RD~niD$#9tYlyU59#-ok;ubPGUAifV+ouPt^aax zX^RlfmPS}u*L3Y6mlE}C=q+Svh2NcBjBw|OCI6~*{T}l(io`bGwg8+}-D zc)87TP6?Bfm=blx$pRJDOSSK$R8?h>)C7%sf&Kz)lDui@?cb$X+xdHyZyap>lhfM4@nnU9Nz(M3Gnmv46kG&Si z&fc5rpPV0=w##_pXVB_gqGWB^-8FnPoc2*`T`QyimzH8gWjrY`1z|kf{013kH`)g|nj7t0z>YnsebF$L-m|+!-KV!0>cN zt64x%f)eBrA%Rfrxqcp7l7uG%@<6gXixNH&i~%Rw;yG___}#6CfCC+BdGyF&VD<{>vtq zv`!;Y3O^t6N_H8}%+W9(HX@Dag({(=n1%+-PRQjf=@(QIN4pvC%nzt4c0;_gW*oQY zBGb5}S41*bW*(6U+KNufFv6pTiL+;+KUB}|N|AZG_ccIO68!8)=LZ&s7&rex;I{CQ zs%@Vq5_lQm&H!u=Jr@*wGY`lV6Bvqln`U#}2L^}HKBVEVS1l#VX7d2U>hGj~U^T0* zlrMc0%Od`EvMX5V_d+34^)U=cY^N1XuG|c&(QGFF49Y?}n^D`O6}|Tc5p=egC_4EV zuykCwI*5pF%yu(#f8Or*Zpg@_I`6-xCoYg)GlR~kYNt72KikiQ>XhwIm_2Wg!Sh#D z?|0qECGU|}-IY}7@;#jiz3(inD<>E^hM{$|%_xO@f#jY15jgkHKdCm$g9e}G12$-nC|Lvs50fgXlaYJilL0bhE!RiHl_dlrrfs-0rgq+j@rxmz%xU~6)S07oVr zIVJY*+cbNyZ_u0`_N}|Z$PO;1^*76VdQPvY7*genxL?DK>J*Mq1V*!qZQwN>2}f}- zg*a=lViqUGtjsGYE<9n-ey9)+^d?d>&;lXWd_g#VaGKz?x9ra6)B2MviqZrQ=2SYR zgwIm?oYlZCS2M@kWc5a$?iqsstlclUnQwP@^JSnSO5`daZ1it{ib9Iw<^TRY%b(n9 zg;&F%vYRrsO2V+nfX+q~hmC~g@Q~;0WWM%sANn3;>W#v6%NrA9^j!OAoc2X~rp3+h5^QT^DB%W3DXb+CJgML<)azFE>1N2(r$u zU0aH@rW$(6^u##Sp`bYM`YhcZB(#IHuBeMeP+`rGt7XW*8bCnwmIXeSWcZy?Pg=>V zsVRPyGz;ZX%PQ!D(;d)~%hdd$S%`kJn!Z4e^M>fn@aOV;JCPZBe%3nA2YGm@OUYDT4n%9YnB3H3}NpWPmcPRb)BrHPcb97BGn5;R~esvw*< z_#^9y6&HroOr|~S2su7j(sGJZ%6VML(X{f%e39k3KF<}J=U&g?_*6+&dXg`dZBd%@ zO6APY9KrpwBh1GT`q8XBCAblBSHhgTbw>{CKGC_UgR6HmJ%V{k& zfpaM-<9)DFM(1Ld?j+c*x$40*G!NS) z7r*owY&I^2vpsRz*swlIg;KI>WT`_E?myaL>k&-{^>(;a?$r<$wRJ-+ygh@k4WRY1 z46wNUf}26uOU17PHkC1U^FhzAziXIzJ zR*J3XI4j`u+X08p$Gc^vXt5lb%Uio;NBN;2?=(cFG8L3;8B@ovaws?efXHo1zZDjh zGuR^o0NKf);zu_4@HIGq*1WtCwuoc6Ics$Wb2EC6^ZxuTfmKrrkK8LmlUr(C>jlXn zGh)krN+cfC%yYo9Q_@kQb*q~pl2GCm-XRf%)5G4Z*>M&>?g-QblKEnK778m^Q}MS= zfOzz8n}B#3IZb_#Y+E3i_&;odcKXYN)H$jRzS9{_Pophbj@hZi;x&VzD1(MmkJtfq zd7*QZr}GQd(8~1&qiX2Xf7=A&BNK(tz3EL7vVv}eq2*|^er)ts|Dv^? z{?R)-xpRtG^^n6&UqqqWTa^VO1$<~57n1yoFsOn6BfyfZo}L#<_9O)01$6JRCc1)E zI?;>4(ukYl(P9@q;ej|3KWtpkD2_>PjYqj#Yzz4V4@*3Wb~@5af}mnK*aDoBwf8hj;pfqy-!U+hzR?L_G!3kIb+K4VW-}TeBiv9p@(#n$aiN+gX z1+q$l1&=tO@M6y>I_8^nRho5thyhAET}Sy__2pGeHzT~&{Zf>cEpp7yOA>_5y0{i5 zsz#bj3J;Q|OuYg^=ri*bnAnzLPZUM?C0Ihf5gV30vBTgP;~TOO9g02vC+wh0WyoCK zlmWe{;FG&`eCe8Q(al@E$*g?}7n7kP&@l8j0UskkFBMM+Se>9otuDvQfQ-k|Oa#v1 z3GTprlH0=1sf`)(i1Mf?XEm%?uK68hp;@^6Bvl8hOy%)zNt94kCj*MbD@^1e7jD~Y z|H!F%ff@Fc0gODTtDJZse$$du786ME7d^?_9jWRe&5{<^iG|PWxUAzNOMy=SzrE=cmc?Hrmjej zc-M?wmL44ltvM7~n3ECidSrLCC*xg6jW<6Re7gX#EQ(pxYTC?{{vUQF;~C zxqMgpf{e^uNArng$}U3=qJb)Po_l3oC` zM}jbn>|kJffX}E)NzN7w`VxsRF5_%R@Dh3OBJ0SM+%321s(52+Um8C}y_%>0++7~F z=*gP!8^nx3eFxnMb1Txbbhge997qFDUnu?qKSm&^hVjcHw_n0v`gEMTU)@{01?fi+ zS(a&eZ6FALaI%Oqv|6H{xTRw1msv2f6H=o{jp4FDcQE*;=|#VvSqg zD~Y0(z27gtjtC(0vs{g_3uoNUZ=7JU{TnC9PM`B19eOZ#)Mo+;VH%y-z%4G?hB$R0 zSGJy^ZHIV}W_p-U+Y`=tzvINN928GcTjOIOB@a7 z>>N7KH?tC&dJYhck{xsP>Myz@C+5MY6Cu!ggGbWH8qi&G`eBg;bv$S5UtCnkm50+t?5Bt-twO7Lny01#O2e9s z{=&_-xkQ_ng6EvrQ{=&d8_6#A{cAC+LidnPCLi#8#YS@=Ie=cpu^VCQ+Y%r{ zp{oad{=x~UE&jm?oEj~H|H28bfoT}PtgKfI%ApFpw!0-TF_rC;@WEpdSuB20OR(`B<-zAD@%r zHcT$~put6%P31k)yOe%~h=1h42A)@j(MzkL6+Lgg>;W;+H}Q zeEcsdd&LZAe>*Ax0W}W059-J^h`?6?x^&0Y32WrQ%t1fh%*+JBiIyi#h3fmeZooMj zNTC9!R6g3o_nBvt#14HPE{7&wr)_ghg@6f@{NYl$diU*{+9#`FGSD_mqXhzX?Ot!b)PR(y1 zww{>WW#p9b*ZurG(`EnD822lOG`^R(jiED3imMsSMVT1b7B0mzP_1T29o98PtmJ-B zJrp$z(rSY&8Iqw&3#tFrOK+(v@T)QLN0m=VTLKT3px~nNs&~fuVIY|Z7hC!Rn2;%3|x zH8)78Vcs!yq?%d^B?sL9?wun~cH@og0I@A1a4G?bU|bEf#@~t)N^c`4cemT< z5GsCx7bnjUj(x2}e?Uh&8g_osfNGEEjUYTeiOGAQz%gSLwsmfZBapHbm?r2XxqE#E zAOO`sqDXrrG6`cBtPPr|GVTf+PaUq|nkVE^R&Q_nQo+vEfP@X6qf^5GVUC-_UL~Q& z#@G}niVHbL3cUw6jc;sA6mwq>XbJSuHa9po+i{DQVL(4GCftB6ujSKAQiT|OLEwC* z)F2-6_F4bQC?M?4TzTl6AVHZ_RxYE84PVjnQ2$Ae86qy);G*;qmTXOtZv|PxhJGpi zV5M4{>pJdjqnBD!HCmb)D%J;8^Ms*uJ4=~5xJrd4mke4t4w}smaUgsbh|RQwFRMa_ zB$W+5Nx}1#A)p6Esk2l|FD_Ul@CGF=omq7?wm*C)KgHPCzN3ILu^BBIVNm4|UkqAhQBds0;PRzUew>|W$=IWp zWJ4mP>@j`Z0I_=q6Y2@G)(z#RcGw!*@vm#;W(d^{ul*iWNmEo3&fL=)D|LL5^uKH1 zorn|<+F^BoqWLg`2114abF?>fiz2e7WsMrnMbTH)@f~rbDdG?zF$GEo_?5^w0~!p{ zxU0FoM9bO4v?7l%GA5+8@!N{m1s}DrL>gHIs`$ha1?3PQQtJ)-W-+3%)7BZuAQ`4IL zL~KnjP#&RqR{?Wd`~dEn>H?hlvctzUb5nH^@~-UGkSL~x%vgS;O<99c{E!-ITnAlz zY)*H_*iXDH?Q;u|3;(Jo*uajyy@1IzH=WjX3N~Bd*wpS#@mS;}lyH_l7vczGnsrE2j zg3#BkplUVZ#HASn&6D55eb-tD}s1KVOHPe+nf4P5mh3hI&28}7g+m`S? zp7A)ybxtGv0g%`k^%vvXxae=4!=f2}1fU}1H~NmXDI=^Z7yKdM#8fV|oL+<*b!yxw zNc%c%h#v^Jtc1f#@1y8>GHRBx2Yf$-CRv6%J@rs?9MoX!8Xz{T2164N<8(6Z=ombeWOa>Uw@ zP3cF!M5uS}1HR<105RZECJwA$eA&@o<=TEOuy{EG6i#b;``|*cU+GsPzqoJcJHGB`azA9z z2~Cc+dT!~=bJ z>&eDyBdRxZpGWdJu7EI&BMB5$BPh7XXt_KYSjs6Y7Ta z6C(@_wUSLQ;EgsC^&8O-rEIeyF5-)bE*&t=2oxH9CM){{f3`y*kNS@XjqJUg`xXUy zfr?T~!nMx3CZ-9kQl^bXW|FL$%DDsNS_FVsBT$8cz8lWk4r8vo{}z(yyOZ&9kb1-2 zFdU^}+~C#Tq55J+t>TthhhFJP)@~APk2QtF=kei^i%`A((RE7u`oKSsftdlWwkx~j zl=v$;%1NQ}4@5yo&~+G>!Oc&EkSh>E^tjF3Pi&%ckLp8UXD+K&D&H5?<}K7j_4DXg zX3zU*P1CQ-BHIqZGRx1(A-08?mo2dckYt~dNc|7RH2FGGU!zD_6PFfMfLMrTzy;C0prfu@jRjZ2-^%yL_F^dyfnFuXIw}5whgdSB}d_Kz2wd^z_0u zz+TMYWCp-lN_I$sK^OnaJ}a6@`R)l(04kJzJVfFFFcdM~hlTe=G5XcDod7L|)4tP2 zvO2l8fL!*47{yk3llB7_5&|n`B20$zT7QBu65|gKQ5U`a+s2+y%r(a~LJf8Hs`03e z@E*5k&#YgHmV`WvGFa@T`U`h1q3SE#pnssUWMSiwNa-nTXcy_e$& zzbMQcW#yE)_Fbm^GzI#ox(W(%IHCWnz`A^5n}DLcUb@=F)2cnLB7EnZ*B{ZUA)%aAOly5jj;jMK5-A`( zm^2E3Uk-)5nkNs~HUieatBk#JVc&BF+&YJII2V+$TUaj;Rir7wr@@1eJCaALJb^{I zh=(=M5_lT|8~lAYOD3F4`?IF|Cj5++;+81c`#~3za)LjtZ}i^t#d)Ol7k>Y)Rpm~A zjd2gdNI04V90ps-0W+UAs47ATc@NW#_y`M-h12fs?5%*YY8ELS))F^cNp0!IWjA5} zdIAB^EXF==8H_+Mu-c2B_eF@2({CJRCA0Y~Zbq$ZMgP+HL;mZcqsyD|2FO@|=`!K% zgZYgECtlf4@e$PrQR!*d=NLFFCi*B<*)1jMH0v%b5X&H-y;dY5Fiq-_09}`gKKvij zWFR^&Vc327+t0X)lPzz-vXL#amU{P>*#!l@#kLHXta_YW8CMDm%;gL6PcTe!Fy2+R zxDwgB($a+>IXqdcx_DyMWw+^{L}73Eb8{yLsW8`Jfjr?K9?9P*fy$pE6sb?e?CIWC zh@DP1J%!Tq%+oWC!FKX7Nh+mxoA2HS^42maR{Qe?m0sbqN)zp+aL|a&c7r1YN$-+k zJC9=Nb-^THt-&KTosAyaukxTeWJ93!980J%Ltx!P@y#~PL6zo4uL-JC?HfrP&=fdu zZeduxUh=b5D@~~<%+JrJlJ$|!VfNYBFPF>IY)vV0>*paKbUZ{xjzokm$IYwEIPlZy zgZ4;jHg4-n;?>f7NZ(xnUbG~>o3QU%he3{clfeMA20xq#R-FSg-k)^I={Phj2Lmv+ zS>P2q+k3$U8D7E~1T_ZE1AQ*A0=fmBu(=Ybu-lxAs~M+t=BHr2fXA$q3y9A>a%hNC z3rr;Z>%YoGWMpE1jLs;`nA~wMmQSVjZp1~JvqKT}t5Z@}y=@B&OCbju3tI6Vh?-Qj zGB0FsPZ@s|imX-NE;D4P*Dfu$fWBqukYp=zEZ)%JuT+rzSf3fgrsgSj08X5+#i||B z?cXTu3&v{(tmm+@(u?INBRtFUVK!@Ufy|)f3F;^sl+&byv54O2wN4MLievjLtlXsO)b$K4@zx-+jp^Edw@OaRPE4kr`$&|3>JlkPti@yZaaTVV~qG-9(Pm zvs)ll=h8oTNSs$6f-4a8x)rw!Psb`vB zB0XW!4bl3sgNvQYyshO}DNv$OxIg!67z$er`qC~mXRWouhX8B*BQ)|5*g=ah;o=8w z9Q$7+!E4UaDVs@fm@{ae5s%2@46ih*(EpPpuxH;2F^X|iKfG?=oKMdaSu#lR4NA3W zpIx0x6@YXM^oY-rUt~w} z-SEA3^+zw2z@&gP+UKScya{1uaMgPt11MNj+5r+Et~E8Hl09y~3}L^Wydn_&ZW?US z@+zOC(;?-{A#dcxJ&)Ns1NO`KSxYb3>Bw6g_~h$HZvG1GceQLqt&HohcGVgVtFZU> z4n3}2bF*gzZwHNIYnj=Q7!OeYpD0+rH%}PtI^BaT_yESHyq*s&G!`t#x9oqtS_l3P zH?Lk<>a2A_)erIQ%#IJs{sjYQw~QH>wDJ^ z7GS5?C3J}~qkQh-(lCWQI+Yz{(vKj&DmHt-|FIT1Yxw~MISHgJAnMWlE}gJW*BOQb z*TCJOj<6AfLHUcPzyJ{_qi0CLvhv0c9ESE0EAE7TEP5#nWV*|BP3`5Fmj{2> z3fbUN=dHMudK3Gn1k1yB9T=3@hrMZ+?;VlpTH}wo7aG+Z{Ns`-a#;l@;7#=W2Z)2Y0K*l$zUIb7nd)YybGp4GS1V)5 z6;*mp;bjS+Cdkj)uubzN^Qh|mEog_^*>xNC&l7;v{Uup87pGrFUUQTv!l4J#)_hKu z)SlqPZ?5eJqSIIeSY-L1OC7_a8SHKp_q5-ZIQUa*-*A zAxQ%g)3#ZR7eiGjA2F@I$&srJFI8;V4lOB83XIy6tT2 z)q6Ptfw!NzKIP57ViBttdW9Mf9__9`yn@Ko$lcF>Um=3sZ)q++Jw@n?q2NYNkGU=T z<^9wMCfpiAJ8j8eF$M|dRDdQ)emQn{iD@)Ejs$Y%F2Us8wwJ52F;30CYm?LtyO&A>%r{czbH~xgb6WES$Vxc zDTiTe5RsH*7@hzLY0lxic5qe2Zx%10G+sfp!;cCS+Oe@!_SXsLqmj#)|EBSJ7k;k- zPyEKxEXMI@Do;c&5;FgcFg|(g0m(t{?}6h%vfDTRGcZL=EThn&P?NKb1}q4t;FDxI z_SXk)sGl2!NEv)ohhO$YcqB41-;fCzplM*zo4xIn6jG zpaNHg36406Khw8Y*JU}{)t#UcX>3`<_wf+dJ!x9~eL&6s<^ynv@TFcTew84wsLa2u zv#|}LyibnMBOM2-LWXehR=Q3)rEK6BH&l2XwyeOE7r09Y z-w;m?wwF%4F{i>bIym)=eY0P&XSlZ-ANZ#htAdnQFg-Cp?jq+Sq%nUOjMpZoYnHka zndB=|qfVu8XaZ165~iFL&lg`0R*)Afqhv_mE7JGYr*RtIxwV0uVU_(d??R;8w5_HZ+?N^_>3H3v-faUDo8htz=8~53^Q*O~PsdwV;pJf8 zamjHr$Sf=GLCQrx zBDu~7$&{f^Hx2YPK+KT#`PsK&{WhXrjkFiIi#*W)dTDS6WmaU% z=Cj7)6CC2~0qg=p{^saVkahVxLE&2$0Hw!~Q) zxi8<*TZXb5uNAq*PqU95HjX_32?246@pqR1rVF)jaEZJT<4*D=j#w<#*)gBJC0XAZ zoZ|i@7r60w8jHkEhJSOBSR>6vW@^WSDX8i6P#n7(<WTGJxCtsQ}y;IH}%HqCezU_8PN<>&;SY%7aa~-Ajp{VDM`2#Enwo5lIwC=0lFK9 z56HNVLCT^+?F2~50YhR9%T)XVX5-xe>6ZU$cT}2IT`i4G zL*W(EkkfGI%bc#>z5|V<;=8Ka!JqfR~^Tjb|0Oo@QSlev~MBZdz zRSprMIH?#aRHhF4@jc)}>hC~#b}~3SGwy?>0Cq zB21lbBKJIGDS7OY8^N#gDMCF@duC1`u)Z( z3ImERv6)g#a%Df0+Siz*I&^kL$=zj#LJ1gq=MEMQHyO(sY>Bys!%79U~JLVLLP%BSN1tXseUkmdMfNJ97ycL0-n}jJ*`4bHWy` z@4y8)RKQJezXBbsW(%=QxfY0A8h<>RbH=Xc>ku*_UOCgZv(PpDq3Zc~FgSiV!C2}O zoY7d|_p1UL$ohv6GIb|maX}p&afN~lFZ-jP)YO$~ZF|3t(j@%r=?U!A>0v3uN&HY_ z9dzuosqT4&0hs;UxiYOi3`Zc-tXrw{qEiC?c=nDdo!N#u1tYFj1o*PaCtt)u98_t| zgEg~Dz-|-zjr0Bi36b`uFznwh!No^7+utq$MnNyH`bEq(J`*jQKRz1KOKlm?ZsTmi zAZUT~;D|svff9G*Hcj`^%q<5~e<|mHpXy9kZmtVTpsu8VfLANr>;U;6yKLYr1Nn+G zg^mza2QpBal|Q&P0bycvAxMY6wvfkH=d0554DWmsIVrdPtr4VXZK(d#UHCXcc{T2oZ9N7a_dvkOp)sAkFVj_ z$4k}pXsVYPS=kvne&Kv)MB8ih@t80xx^;exyu!Fhx>Q=|0a#t5rRTAl={UH=ZP=xP z7x*#?N-E`mr--XjzD?Bkbf7>?s+n`K847j~ES&-hPG|Qp;+{jbPGEytwliJ54#c)<_0iNfTDbkw6sER!6#Iy0#JsGzxSGvJ4V2T zoTPTabLsFFXxIfuLJ}g=$cfV)H_rEA?sRt#xl8&uIj=PJ=ODm20%)Rc8`YX{I7g-%H|;&HOzoh#OOg<^HJ zg`g}?bQQxjNsaZJIG$glrS2~^=f4a1Wvw8F818_?NDD3vHg`|%X6d9fArE^#H$I$9 zg8XWHpUWRi9^`Iz{Sc|ORqwKf*~nnxuKjUQukdXWtS@Z2!OeAjn*@=UjoK~)gFsH7 zUs`8X(el8A^qtoXQ8qeaxfbQT%cM@W*WLHS(BCM*C?i=S-uZs6-AO>#UnqgI)t4n? z7~X>rsHR0-l8Jium`2}3OyvnQ?)It7lp4_jAqvGGMD<>9O+jP(Azj_noMZwt^k|x` zlPCrD-z>#F7ALS27QJxU9Z>oqQ36J_UB`mw)37>X*|*Am=j=5 z_kXJdR~`?9biwRZ`K$dB(DYtA#DQU5^V4sl5FteNt9pz)`NNGBIXOEmhzoG&mABE+ z_=r1-MLzAL(W2?0`aopl5KD>DR~KsaB;w_fHi>v2)kswUp^uCOG^)}HP8#Gkb@j`N z-qm`x`BtYO&WR;aIkwS-hYd>n(KGrrxg{yz#XJ2|wgpkM01TmX#QH1Lv zQ5|L|-rSaqaiU7U>=XAR0hyCm8gj4zlKw&o-Y2728Q&&5R^K;Rd0wre5}LXOZ2uQ) z?-*TKx3z1p*tTsu726ftwr$(C?WAJcsw5SoVmql=C(pC@^PYF_ALqx{zMreL+1i@@ z?tS(-#+YMV_q{qF?%_R8^wgfwEsNTBQEsd!6r$ZmTBL@YLH#n9T5DdN3ftu$Itx-R zM|@&j2_)qt2`CC);Y5ER-iXw=pL?t9EBetlhDh-|s#aA zl)%9NC3cpjc3?*D8GgHsV`~|$VomY~$u~k7y>}ry#(Qw9@|J?@NHV;9@a)SKyfLfK z!T>YPoe6|gf!mot8N-#QSf-Cz?`)QyuspI!-sxY7R;aINgg&}&;3VT|A+>Jk%qZNg z)T16+wf0e?WNA|a7{ma>kfrmtRj>9Zq_Pss>z?ooP4CwJvS6p=FI$2OU=ShAc(7k7 zzQl#h)Cp*a?IEz|HB^Nm7Q83ZJnZz4llR!2~?T^UT={ zZD~~q?ONghh-4UEi67D4D8yyqIT!q5_KS8|$Qs=Y^2H53TNi^x{?PG=3y5yf^LaQ| z1#Q2j<&qqVcHe__e<3R9ir7j4OQ2*<%fO-qXY0q(?X2<+Y>h4Up71Q$!M)~;* za=(E=wk@hhgL#i0Zx^7LAbUFhWMBhx)({Ww@L0`V3e7ea5f)MEatlXs0Vp9Y#7ci> zZ(`da`*9UyIN9>2h97DFleOse6SaPPb?jf~&4DAn9Pa;Lr_KNC@cIAupFsTa#yzvB zxl9=CzzNBWut4ieghPqEyB^KK#Z^SbOP#!e4EsAjfGUA2Rw<-fVjtSLPjnH8R$)YW zn0YW+h_<8kJ}7YW%o;3+T$G2WTH=x>e!XKRU&%ZwUCavKQ_xGF`w0%?dN!^aNVN;~T zHo2>6u~!Q&NU-sh`>hb0L3vj^%#P^J!u}l;o$vzEQRF_>?-b*k-L`|zWU5YQbMew| z6J1OcJ;!5jTv?9&9d!y$s97&8HwBz;tcOhRs5#HefY18nsMgyyxp#nZuD`$#+Vrr+tHv$^=-3s};!nrE()pi`X!zIdbIRl@GVuw`%+Dw7bRe`Lwx^3$FPm^yi7t zBFpQGQVVx7!$D1kNyjv})aD8riI^6r~yayYD2%$)nSGx z?v;5Xsk&?XQBcj3f{%UK&^e;Dvx%}*VYS)s{?JsN>&|?LTv}QPzgA2C4&~AWaIAI0 zP(>)RoU^cplc4ek7A4+cu_t$k-pdBFyHh#KG-0(X@;fQwOicWOr$j(y+y31Ye~Hs{ zVFa>fX`7h+3YHTwx%|^(PUsZ{jYWWGtkJNDkS9f8NOajs;K5q0>2`6anNC7H>*_CqAQU! zAszOPT}I0bIgKvfBJ>{0q8dluE?vvxA{m8VINOEo#!c#>GT9_?%5AHLD7S?5VnPHQ zcL`ohDj#0LQ}gqHpx*PZ8;6b)6f)oH&NJE_=obEi_U<4Bd^xRy+(RTmw{}ArSIyjP zE!bf_-|6^!Wm7*vc8$9b7Ifr-we4F|hSJjgY|msumk;kx@rnd|hnK|O5x9o9LVHY9 zeK840Y4d$w)wOC%TN$jL>(yf8j4e!7g>qhs?&9fA)R1qE>9z4(GbCQilTvO`@Ogx# zw^<~-dCIU;K*8c)yb%OCEq0ERK8%ytlT$oL&}Hmg zaRAUl?Bh2W_s?hLwJLQ2l>nMo{u(SW=Gf`&_HJp%iY&Iay$bi1B}U-wS64!bF!0 zst$Q|f$30+er_0^P$xW2Jz#NdMz(b+dF_F!qcE~Kw) zP|J!I{%Ulo*vEXT>QY4lqs9({3V__qE00c*Eeu8`UaG~0%d{ok4!vMRu2%C+qpw-3 zc@_WwuWz-;T1x?*F`@*9WeU>hnV`#?1lL&%{6R*U$>8Uqq9HS2x22poQu17*)`f|3 z)r%4m?9)oFVxsMS6O4Z3QbT&5>dB_rOvXHzL`s>zz+{s%JGy?O>1-J4kF9~d4!JgD zgkMc_>Fw3`arhhyT7R?^F<}Fj_M3n8)kmHuVJsfrYMT|t^xLP_tVY9kmlq+mLLcClIZbnLXOu3yW^zDLwGw=@V&3R~m;IpUQKJ(QPttZ3;M32hh zJ;A}wa_Vak-3l2Bi{jXkt4&NzbeZ58x)0KzDmB%*bvCL#g1l|n>R$_e1Ixb^yf<^I?1{+UXNYd&|V4*WAAJzu9LG$Wmv}p0EE3y zWkSQ`hE=Q=I5ymyQM_)6z<9Sr910r9}w z#>(<{%1r|QkPeFV6rY@0{LffK_adH?h$3VFjAu#&xAsRe&8)!@^lj>+0Y%@%ly9{# zal6yit0gzgcfe#8Piy%mAHEv&PKN}5I7Q%#Bc!X-z{5^4d1xv^w+eAHBp zw^k($DcGbgOx-JEp6gyBMRbtgvCw=iNp+(CC&*{%_nVmNu$HlF=;J|5aY#aboS}#v z4}7|{lwR9)<@|oJ9g^C(`vOf9KB*C&DgD8Qb0c@ya{7!*Fs*Og1U}c){P_o$N8Kgf z7W6ZZulf?7lt}ycDD2kPNF9u|4=}2pUvY}9IpYR3Aj7|f_8#U*Hhw!|og~-vHRulU z$6O%GArd<+0dI^EV1UluRFA$Yq&?V={?<9CQj4P-ava4jNx>NP1GuFKnq&)iJr`{U z(An71U3F|1!+1#+m@U4svYM`fJA&if8fSdhdhGc+Jo#5YX6X&8`hm{i=@tfI%Wwx0 z$rEh}-w~a?5x`^=(0Hg3@f5JYN|NZ6aNJb3w>w?)$J~A#kdHOCR>+S0=$zK4Lhzy1 z$F@7AF;z{=4k=&S$^x0RkAg##N+kzXo(81UD1JhxW%h$DF-6`7;NkM4N+o4y&nmJT zR?qU|%Wf#*KL7oo9#JyOrslI6 z)Nk;7 z=Z!4LUHP}Kvo`k?v^qXc-a0?n4}JQ0L1a&vpSei2o{8m4;QH_1Nc5&vP_=JB@QRV2 zzARqCF_d81_Kn+OY}W)IK!b`$q_gGY;JPTViIyA)vmHipeS!kk`t*jUO;oCuC#!`b zik8M_+kQW;9Jo>V2&EF4)~-35lwvf~O~e#8sIifp0kew#JtzSIG>6fDKA|;}HhM(F zP8csyQ$OF{B^dAeRegx;Y0blMMcLb9>*jY!%e~k1QA%|jK!ONC;0{1Q=DE2KxIhl^ z!;)oZ@OCy{cU0Y#&c3MQ%nPB^-^$}_V??Eq^I2w7OkH@JEgp65|8k>Q;U^TX|M%bi zuekv5?|(>OKw;?3f8UTCwl!+*iGF87`ztmCH-C&Ca!<(WGe~XOkx&RzOr*Tgq7u($09?f zg=-Qpu8S~4ScO%l#Z7AilqDm@ezQ zSp|`0N&);gal-9`E+^OM!UU8(yaz)#hqLaQ_k$Sfa4s!|JI?9Vg$D;d)9AW^ja&Ax z@G(JGzUe95;OTN6v-PKMguX)x{n=PPsNrY4_#A&qp9*0w#F1PuZ&GxaE1_!=iT&%K zH(mjNFm*EfKXsy*oYQY$nf5wEx07!(?<|--?oW4CRc6@u4K2?~vH!1x{7+iG{e!^Y6qkI!5cd6@Na%8h^Ca~mV)fSaCmq7Q zkw%*g#7WYB361HwP!HG$I>Q7uir}5;8mCd04n_&u@Y^LhN=wzV#e^uJc|+~}^oi!* z@c~135cwl68O1a`*5%zxT+i?RXW$@!HgTF?ANagYunX0izssGmK|= z`>fH?zGa0BN;0y=_&A^-lJXz9w)yhXl{QueUUB9q8J3eCsdra1MzR^Ao3ns6pG_~C z`9MvuMi|M!%wsGKNL_buP%ouF4~VMf_g8puI+>|u*qL6_MhqCs*mUEEb$1UESw;0a z<{Od+f%!*5!2UlRd}KWnb4grQx`1YWUZ4hQZH0x5U_wkO`rylfvB0j8l#bXb?kydA z`Sk~9^!M&U$llI?mZeOuI#1d!nk;T8LJ|RsVM1qp^1oyP5mtl7`MYuc-v^5G+b=2Y zZ^6F!2%5>`*eXCYYo1@Y;+0#>`?_sQMCST#J~`Z4MNvh&^mjy{>{0T}^XMag>U2Wu zW|JcU3L|#nh{KdD<;dVHwR4dRWlQ?g{?RQDHX21Eq}lWqF{iCjh=h!K8t9tZoHWmL z%&k-_+&=a$Pys;8zY(MV#0&)VZ@4`W-umQSyaBc-l|S%|fr~!vHdn87@xg#~o>t0X zLBzS{K;h^mh%uDt>5O2wM?cRO}X0m*idD3;0k0LAZe7h0r%Wk?IhGA>M zDxU4Br7Jd1rc+OypC=M{acgDIchf^!yiZYFlXy2&BEghCmTte7kjQ}bC!(0W1 z{@-YNAoyqOgFompQ~=5!sxPf}!mci-PRs9J?_qu5}awPmZ6LxXS4;_FtY z!sr4D&e#C#L#QZ=RAiB`K>e{hy7e;K=nf&kLI@*dvQOCGiOaKU0f^%}(~a9t%w0n? zlZa(9kj%|_nP*kA8~DcPL3_&7Jp^%fIWgg@>t+e(h7B%)XWwP37}BF4x$nalZ`OuqC{Pci z!GCtan~Eid3`-{+-gO(@nz!k(ds#5yNHReHnz9ozI;AU%fT^|AO#05kq+mKDW0$ zpc6O&=uWAy8AuZpG3!TMIlQR%XPlUBd9nzoL&Q@m>O+kuIMkk(bv@0A>pBnErZ-B&%ZBEzQ>xrH_Vtx$6=IZmZ4^g`p<{u07F!Krj4?8;Rw zFuP@PjH84Ym_wp2nD1?H>wb{c_8QK2Ja`KLXx>qImYcBeBx;=F zCm+^S!G&kX(|r6{!dCWM3hIP!DDPt0%b3xB=8yq=3p5dEfrK63Z26s(-p6=X=4re; z{NG6^dq@1wbbx^T5lMZdq(j_sV9sKgEDIF4QSgCs=i3_-VQZGJJDna|^WF($SmP7CnBpIMMjcxmop8nm9# zMmNh~sa683B#U&`M!~hKGU5Ry%=Ld9n2N!+p@qgiE-pSME=5nGNVrF;RXrJ=L%_?P z{&@=hq!uXZjHNJEXh@rFt=$43H~6RD1J3>->!YX^h?XfUEd&`Yg#%HyI`e%)%q>#5 zI~Q~%Io>KnX~#mdW@QC{v&$BFHO@m}cGIw4IWwlw?ccx@ow&g)>+%$*){wiL;=OF* z67FApu~Gl7(nKfs{}#w0FBXZ9nStr&Rl{-jP(t>}ypab=BbRaO8P`IFjJmO5{S_1V zpi&Gm7WebwqUW9nih{{5NM}l;$?gZ{do*&-wQ5)Q*_({^J&xR>!Ej1C?^QrKGtWna zP^i{O#c64d><0ODUyNDu+9MHK$*}p}Y7Cz+CjbED{-2cr-25L%eiU25>&#_)KuhR~ z>#-ssdE_`Zc#K{?$O>?ose{X!_h2T2x=xwD(apF>9Xya0lR%*Nwe+y#f$_+=O3itK z(7h=$P0cw8vT4PyX7V}U zkle~4Le+hgJ8tg_4m){!dFMLO5~9FwMe_j^7U$QgAY|q>>KWHBBCrz4MAS{RA;%gh zNv=ug$(DY5HCXV=UNUKO3E?eE@{C$lyuskFeH;8f%*Ii%?aiNRV1>0{7h|Pk(PtX1 z2)tk^PdGk2Ct;A0(dHYc0y|dxkcnO(dwlCOC9aj&dK<2iKM54(j0Zve031V|6i~ zncfdnEgV4Cp&CpoWE~jG$YcnElHqTBu-+p|G%mZ*taziJVfwV&v< zu|2nR*=pd10{6@8R^598%ly@c@ACJz^ckLwc|TQ=LxiC9PJevX2MsUn@%s6n@J}%# zmKqH?=-Z)Aa@P+y7FF94GXZPK1LnsIW`~QL#5JGx#WN+0ss}<6L{R;9e+JjGaRt%A zV=}tV_uYyVNTI*{8C{H+%S4~^CzSWldSIK%m+uohgHTL=+=LsHF?gI z&@~*T+1A*N%~zW9E4}DtrAz7vkoExpy8nBSjq?x9ALRlMln)6k-2sG@K#ixo<@9q? z#2i-!tA6qty9#&=9@GNHbYiG7-P75t@5f=bDN*eN2C7@rjO5pP>)-vAq|E5wN%|)# zxTB8{vGaVb6r78>LP&kjF+r%x*>2cu8s-ae4nO6AE^It}>rjmkY9>hdCXqIVkLs#S zKsT_ZidH}HmzENUM)bj;VJ%|)3w2eH)E7vH5%Z0swLT|5M*!|KRyiy`1MMiQ%kY zlRy{zL5(}MLgKwp*1V}#+3rF{wF^7+sqFWc0@_9yl<#u^V0$GiSlt zcRiOM%dNe1uNS39yz7UdhD2nHxrQ>}VfGAv9SbG?OFY9nA<%wG#Di$qg&8r)wbt% zL&42!T&(p44~2btXYo?=y}pJP^gI9g&I0WI2ge_X%rx~)n^;-1IieYW6ZN=8M|&Rf zj(;v0mDiIH&VBD7w!;NUjZ6JYHpS+Gd`fIWi;0|Q5*V~#ULIW#E!nCcWlq9X>0`S& z?d+!M-?KE}{a4YK>fbrLNNq{xxBo3mVtB`K6UR0JslTthE$3&cnka1-7Si%%{tFx%Gg|&CkTa~xD4Wa6o@8==klai9G z8&bAwD<@>mKAvl;x_iWNctxB4Gh!gHf3y05_*y6Py45lnsstcEl5WtdpdwJyUXm@#S8Eg`O+nL$Nsc7^#{q8YoV+u4-(jqaj2)8RTVBE(Dyot{VUaQSOa$d8Ps=e z?PN~()y(2SvY+KfFq_hZXu)j+S`h0D{?lydU63s zAHQ9+7ZHBR3?#+AV%7!fZY2SLJkc9Gw`l}`v4#*T0C0qT`Q(AH>0xdFh)!$0vw%+_ ztXcs2zEV~`kiDqJ000A2EqfaP@M8Iz03%qseVHX)79xKf+3OG!%q!0tHj>AW7x_jf zT#(i8TEM`023c^DvHQ7vgdVv0-Ew-sO*E4{@I)rbzfdygTfTdrz&o_eq)cYGF8o@o zo<$xOhg|aC|OJ;>W**FN&Ttm-n^QFt2cH#%=7F zJe4=lL%sHp)JP*KSj0!s$Mdp%E%h{qgDdr4J$CsEb%2dF?(9_g3B=yjfJ`yg6~ukH zNQht=#!;}c#4I;5FWrIm9Tx8!aFMVRydSXe2Lq6V;8n1v-`;7>QTRSI!=lN#7L=O~ zJw?_?{rd!2;Yl^NV+v*iMh?){$?V1KG7!V2gc|BUilhaH61WJ)3nQ_W;)0-qwm2L5 zV20z!^8rA~Y=59C9dKwD;CpTW3`0U`j0{VP08X~EQivDEH`q&N_mfQ_`AEB$#}*vh z9^~8VE856hSgA@=C=PE{3F$>-f_RNIg*APLSWG&|Ygm|^h%*djI_oypai%Z=p9wLq zn8y>vhMJ%1^*YD=f7<%u4N{;pnhO4*gV=*d{_;{Vf;ay4q~Fd zrIu<*l16odOZdUVcDJuZkkCF?VMUq&Z9n5!DI0bk0TcE&UB8ER;M}ZZ;yR|)kfd`Z z*A>QaMLp&nvSNw=c4GDUwhgsYR>v zon_~+aR8`~yVU#Ss;2Y`BHm_2Ne^eQvMw_NoPY84{cN9zaDDI)vcCBG<7K05le1Hq zM-?UCpnu!<`FY9K#^mDk(qQ%Kz9n1$${6r?d?i5=fAJ#6zx1~7fZ&erQ(XR?+T#Hx2j-iM4RwPkYtZY$`e)LSG# z5{q0aEZ2WL8_aub$xZO5hg6^+qO|4}xZKw#$SJ`C+S$2`KlV7L&Bpq7Ed)2lt=@Cg z8LgIdV-b>`VCHaWWFVU>;75{QdM|`M1>53>w~RTt-;y_&h$LzEj)WeD)3mY(M}zTG z1|-gnf7JW>iywieV@kb53Fz_aJgQdGP;r{nf`1q&d}Q^Z`qVbT_h%&k#}jDokGN>+ zu|IdU1i@PQ#%U5gHK~apiQ4gP&@iJQ7^TFHrZ-hQ%!Dzg$gJka#S~XLVf8n{e>k?o z{pBg6KpB);h8>69$?P~>=z34$l=7|;@|fn+xA?34>rf*pYwo@f5IU4?$pL7z8OD2| zA$kr5wOT$B*_^=~sdELV-JjH;+h7Gm*xl+xVK4%e5AuJ z8^VfaO%!c)kkkc~j(6R*&@v~Sv<_LIRaq$Kp z%!^tq5|P$|$a&q>74gtHD-wopNKGHA0JmXdhy10@HRMTEWe-g3re1Ih#>viaau<{A zj65Cy1MVbTm8ZpG2g0mHKU{-be@eBA84nTxqiRn)ls-D*g8K}E3&j2-?LbjCdDfcA zEZzY%fy1>Lt4E-7rM>h=sghx-u;FaE27Owtw8A_DIO$H3k~)KyQJ`a~O0MMW)G2J9 zVaIB2$7~L#|z*DMW>uf z6;Gl`p+ZV?6vwkghJP5!7va;Dpc`uxZiAtOdkwP2HltPnj%}%cVvcu*&qp4G57O}I z_u*m?fu``#ZQVlfykRygL2MM3>l@jKTtZ3W67*1lhanz<`lx}P?knrpd4Z(ebJZ{! zNau))3$bXrO29YR#On*ly%0H2nQ9=vZ~b#YP?|NHbG%fy_e3f6?X_}1kG#PZ@08#JsRN!J_xe#qzVuu^ z01aL#NYH=YYidWb3aC%zo40D9VF29hQLrR?BZ1Tk~2y%FW7FZXVD{1jv%T z!L7nm7ebQnnQ`3_Oc4Sy~Q*l z70$v{v8SaY08WiyZ=qbtDU3j?6(z~g}s_lt1(hpdXM$y3h)T-@S?R4JwD1sI19$~YI$@0_DU zP^s$i>Q`EXcN@sk{>AnWYZPzrq5bV#T_$Di30WYWYoznn7aC((-cmn@W^qX7fTiQO zp`(eQD}T`-U|aBdv6Y^7R0G!+lop_zM5+UqNaisFM~-&xi{0cgL6$6FQ&Q3y+1jKf zl;e$P`K`)nf0UO8>ApNEETUzr8?RlQf45ld6;n&Mb`rMIxD-YZReW}L>(02Nz>VqE zfSS)nn+zc{irm89E|H_YfI`Nr5DQ^n^YN#s+xQ2IZwT}^79V=JHW?p&jcvT<5k=qP zeggFCTra^#PMLa3wq62W92A+u1Y{XJrhxSUiI|hWfG6iZ?x0xB#Vdb*dr)fA*?Jao zf+Xt=L2+o4eFc|6P0k3Zw$BI2-kVLN-ALryeM)OS+Vmn?MV$i~?- zh}W{0^6jor`uE)ZvBQF$YIjUP@4$9z6q8-7$ zJ~Bij6zx^~3r_kQ=R+$gDPmIeKwA<3c>XZl>Jf}xTTmZmx&=5C=5a=(;C<60Ovr(d zJt%*%x=b!;5kALIc1PhZ^$BY?^HH!%989@C$stdWDT>C1$Aq(d+Sl?l9TN;W(+Y4x zeoLq)aMSlwL@!Q6 zB4k{LxWWvQTEn5Z^4pg&4ppR;TI^FyC9o^MD54)6pRZsTa&3~`r$KTw&ls*<{y5+2 zusPFIxNwye@OLr0ekoq&HNG?_Fn&iFkBo>uTi#n^dSecufeUmBYnfk?AvNjTXt@;G$NCMy zi}~^VNA>}%w(tVkfC@a+3L-C@AD~DHGno z)W*~6Kw(CyG2Z+UP;@xWs_+_UaZXt~+=}Zf-qHm&4qo9d#nLb$ONUqlOx|Lo|;!)r-7Y4$18j$ONCWi>-J5Z{y>a#da z7X%Wq!aNdnd)c8$K&q+>T-u(Ueq~BPN90lW%WfUSNqM;T)>i zLRCkNUM@nN%(4EdYiL`>g;kAh_85Lf+o*J04PIhMeC^=T9ZhY1BtLX++=t zl(4kwhKLk#4;=W$+|ZBbSyTSl4&pA+>J_qyikoZJLh+aoIR3-GZGeTA8ceEo1Hspal{;L`vC0P-BM&puiO_hjXOR5{<&@98= zhEL+k3QlI@_KH#18}s;+CL-ajMfRGd9fC(4LGr z<-5AOPp(+HRv6Fh&I-JgJmD4LS138tL3`KJ!4KG{hJ1_HFQSbg+(k^0jisC*7xszV?NA6k`j=k zqjxVqO;qEVpOxr!KVJJeWZ{w)1_fYCw#~A2lgKr&=p|HmT~>ig-6Xy7?o)tt!4|qF3a2|% zG0r8;5d4O*{fn0ukxmCy&94K6I<)s4s&~U){KV9p!HfKpljHn5Q{jhWemywz$!Sl_ zU|%Cv#Lei%L92V)yTq*qzvNCu737la^HlCkSAzX&`_v7p zL};Px4hl8)erfm{{ElOF01~ghjcQ`plE;?0VFERR)F)tQ#tmunh|w1tibu9sgM68v z6?Wkn-bk0Z96=$f8F&o#%;&pdK#Gtne5qjvxgO=a)3lt$ZBudY)BaBCF965GVJq+k|)Wl@KIGhlm9b1>d*OoXetjP1+~G z-nLPo>k{PVlc3Jjs$&0DjmxpYoRvdluD^scgz1d~m+@7(zzWL*^FokeAi=7*^IQ9}TW{40WKGPwMEZ}e< z3?#h%IkM2Z^>Y6;GdnuuGZ0l>sS?Rs1SCIqkY6(LbUG*G^il2eh(uja!rBDq4$jUi zTA+RSL3;Wr8}}Q!*_rM)HH-YU;|@JEg&Ib#cjz2f%0?_YNYBqPfjF*xM0k3wtfqv$ zONJv!JzYLAF%1RJJ33D#cJyxW%P?)o^64bn&mmm>?b-v994?araQT@k z@WPb&xpi*!r^j~rTyBDhTx>^TX;54v_&t?m2z`RZg!vtxkl&8jVcsZ%n;iPG`#o(A z0<<+nnJ}c1hw|B7EY@u;E72;iE%T!6f_F8KsLcoKUSm;q#_K(wl z6NTSoY6j_q#Su~f^%stOF#4-JMx-Z9z-%<7e_sbc7qRye2FP@ z0Upmg;~X-ueE^|Nipa-bWAkrU%L@QJmOq0}P4N$4=T7xGJsuWBqWMe_j78=uknK#8 zs(#-?54-#6BVK+6Hs2660%OP*{F}eL&|GbqAk0LwdGxu9Spf>xxQ*;djzo(FF;!za za7)Y(wH|FC`LqLDk{Q&1*0EfX(e{dY3$xPY<^heJQyedY_>t8{hj-z}Dack+ zu@85)a{9GbldJ))EME#r==)YC0N9NOed0Eu{qYV;;@zy1K3;~8Vw7IhjxWJa3^gJF z_lt@)Hy^Px5@87yRFirwwu5Nfd8C!Z$$%xY@z&qG^d20Z_P;IoQ8SAmZNfUbg> zrA$FvF*-m4fQ|Pe;Qq8CX?Z#x)GWq5S7CkwNNJ+!&PmAQyJ|M_@Ln4RskmZPW5l{i1UcJm@C4AfFn+&eSoYj+#O8Po;~U@eu}l5Tp`}>YyDoM5S}t z)kA^atm8gDx_i3-%S)VJvj2;BFSo_;7w>*FmKF|!z%>{mselq3Htdu{{dP@6Y0k>x-f@f44V`A(&qa;0g47ebh#~)pEj*=e{TX@z zt_7Wo`;QfkMQ~^|Y@DQ|O<`8beD#vo(>wrm_Z7+}uMTc_iRc(GJCORdzNa!1N>xR3=~HxD!8U_|}vNx0`kq&Puk^XToJ;|Dvo zMYQf<&%>t%CZ2ROr$+I^S#ir{01sf(8_J8kePY|4Y&ojv;~iw#j&~=v-jd0ScbLCE z3aLCs8Jm;F)E2ysJsM~mNe<^Qe)ck4mt^F*>@q&!KqkQ1oOYVh4z?KFHuF2wl7EHd zgXWMYbu5p@#dooveKcYolJQrBXbN;>-FV}9^a?nK&hI&VHK0JrAncyz&$|QLi7D{K zTP~or12tiGEQcjoWte-J)&g}G239rQgqbi9=1qnkIu^X6rvJATAg@xlujSC z1V047A1ZOU?{)nK`c@Smv%xk#$1IWoNgRUflj=arZCYT%W+gkUD}6T_ZROz z5C#VAk#;zaP*aYlW7fZJ`uuYGmP8Jx@eK44=SV|{0C`#xxlyYiGU#d*oLqnQxxwV- zce&`x&sk(#WwAg-HWYbi>Xl}Pd%D{zy7r(bPih(19}*Jb_Lm=eRcg_}&PEEZYEu{O z6frd~WzaP1vb7gbSJ)rk4L9HcFF!tFmRFnO@9flupf zw;U&WViFZ3_>a8H7934yH!6UnG5b1~%9!la zLB$bJj)EBL=Z}AW%&**{p4AGf@8ZX;|G9I5%CQ9VZN(Yb3#hGZo2YLnV@yAcRQS~W zgj?g?xOD3YlUEV8s}cs1e?li)oj@Ofu>DRq1tK<)ab~bjklrXa##**Oz~r6fP^+Y7 z4z)gW%DZyOM%SJ5lJ2ai6A_!BG!jV2!pg5+VHS-Kgg2X|DUEyq4MKQVj0}Gb=VDA2 z5%&gzIcV0ZlfD+Rxp3+~TDd^PddH@d%*NW0!iBE1%}NW*A0GYlPN9XGS`cH-a1?qr z5M|gSYp)nGa|`{?{I~{dL(i^&E~A_(u?#h;>{a|AhjYxtJ=Y-?1)7Y4OHD9VDd|$ zeAv8X1S}diC(^n^+e{MU>CG$iG5$bmc-HS3ZP{J1OC6GabYHy4;=8gEj^z2G916-I zf;~TB41D=z*0nD6kvfg6e$kmoVfZL$%ls0|w|=Eb%dD|nxHLnZ+po>!$<+C|T8l>N zXiWVWiKlVnnvE(BO83*b8%8+;XUwl&7qwn8RgT51;~w7*5-reaz<3!nf|+D=@PW(I z5lz^XZmBGYKSS2-=j0v_3|ofOnd)OaZVvPU-r^=klp@;aMfbUSGJEYa9K&u`W;j#y zTec%av=6t~{`n}qkzzN{udHa&w_~Rp$W7_H{ZA+m=b& zTKG^`HkY=j{DjvjM(<;F4!UVWGC15J>JwM)0e)6WIQefL@skvSo6 z^!^mm&AJ!!Q{D|J!CgjiRVZ*!wQqB+ghT?dgBRRU^QRk`(p4Qg$P(0{b8mk$>o5AV zRZFxX(u_n-_Xq2ck2D=0;A}ygY{?E{C6O@a9B08Zi*yH)I=jbEQ}$Mc*c&!>-#df(54Q4@u)oSX{w?^bAV8~&uUMQ!L z$Q%hEneQezqBEUs>5*F!VXv4h>#gk8j&-y)K_T9SivwxIXu4;O|J+292mDS?~G40tK>YgO2LVl{*ohfF58Zl}1|Mz*bRi7e0SnOFfw0{w6nE zz#?_cEz7i_?VkAhL_1(5w->RY6zZMG%^ZpKDlMqE>TLX>k)V@>w#3aOeHyN8k z0hXwWDv&{)t%;c~LqTNeo7>Rb;d4Pj1V%)xwy7mYFBZuZD@jQgN**iWB|(pnQYa4eS%jgK z3|xn=JGlAyjMq|lW9tBIoiQ?KDk6_1Y2b_Nm``~>Y;yzpL1Jue#;T&I9VH9q&g~Ag zHgDILQve|-b8mYcO#`++Y<1wk+qQNct72B{6{}*~wr$%^#TBPw+fFLB zZQHhev-dgQx!>9M|7~ryw&uUNMjySuLw!75tkwC5Vua5qu6ov|N&&Ls5ntwme|k&? zC~RXMzuAwQLC2?fJMlIXUmgQvnO}{w0)MB1F#l;_%URNa1(EO9zL{yU>J zWAPx{a4gnHt-^SXxPyX1@~1I?p!o0pvcNcm`;J{tf8viAWm$?*LknJq`BW-ibM7=} z@;B0TsX;cen`!h`Uyca}+8Vtdmk8{2<<<(3l@OT2WSg(~O_^s7Io(p~s|Q7^hl7PS zqY%*-v_rV3%*46$8#1dX88vg$egqL+0}=5Vu;8+c(%Y?`?m zI`?(ZDCTU&%u1ImxAWTiNYa;=4Ir&(vQqO1X=y{Ge~kz2FN7C^3C5~V7J_2MtHp_B z?wBi;1w>BXYC=H#fIz~IBTG+#q{ksheBY)hf`O_WOG>5NeBHkx%id?F(Cg3jz~pQ_ zz)Xx3f}fe7yRu=rPwR*RC7WTi20QtH^PC2(E!uEu?l$QW$M4?AW*$hfTWVO0JRBrA zrVArCDh+qZd=}=b4gkQ6r|8Y(9HCeJ{XJxJl8#-?*`#TJ0*4& zxYE}wR>+780mhb7_9+ETyE8WX=6E?)bH3>mXjzJR`Z9(<*3eZ;3}&OE?TH;ok#Kmi zdm`QbVXW1QZcd~Qk6wN~Z>D^l+o;e5L`~lxx~ZUlz$Ooy53ZBC>!crqJ;3AfSAn^D z65e7^^}WDvDylI76J-nyrHDw=Q+|}tT}Z;Bl+<$x+ow?*YMIQH=Oi(L!BEz4BnjDY z%9?_w$ozw}>E5qsX-*^!ox zDfaw~c7k|&qi$mz+xM@4duUUU=)?&MY#aT(H?Bo9s2zAVfgr%qs`p9wD_;V1*3nM) zq!A=poTGWF+i?8ses0a&4ixIAXtQQH(xlO2f=6*IoV9%GC*j!=FHiaF zZxt}84|vd~PJC<)l7QO8wN~zMCh`wL;|3TT+tQXpMIrN3wCUS&QMXPh9f(h#Qhe2b zf<$ACY7R?DrgZY97b1@6t+*te*8J9uy$ch8C$G*lhOcEU0espfN{f#mJKBe@=lR)orL_w)+(t_TV zL*)PtVJJu~;`VxUaPM2kfxuFfy$|L}E)KUST@M%{aE8deo{#Kb;QQ|^EveB~{#Cy+ z7lAyz)Zauhf-^grGU%U}Ol#yL|43X6v^FY1Cg+PFNo=?~RoFb-J|=>rdgqGpe5=;+ zruVyxc?DRZDRo&Ql{z}aTi(`D-5VR45NhObru?)t+F8&hj52c41(-kcyyvUgQcSiG6_`Q+Pc$fg+EPKob89(B!XWw7w}C%Ut-b)CiJp zeH3gqS$en30fmiF9S zUIRACanvDY(fv;#h&OGtLTHu~H#FGlNVZL+!p`lljaE`RELCBy5N-&q%ni`P^r)4y zKh>>bck)*a4G^7;OCDRa|+E!iPGIWPm?BQ*=C#Sp(Z!2 z7pF%==kE%)_u`~q48OSm!w%0?T{d4Y$la`RlKi|NmE~HMKpyI0&pXG>c+%yk;wg)E zN)VaE-lJYOJ)n%Fh96(Az-b>A1NHvXUq5Wn=9*E7Tt1%$$E2}Dqp0u;k1SkSBC{$# z&R5;Pe~oq1*tNja&Jd1sd!1n7K);=%*;&8bNW}n!p&U>#(v_w0Dhh8kIj#>JOsy_l zx%wSV+;{$wT?4xiMIja%GBUA*qzYn9AO0m=4nvlv+VICvD41*BIi(kxiI)z3PH49sJ`Zc9(S z6a`)^%C^FfO60~8)VmjVml9ty4O_1XX)k*6gvq!YyV3rb*x>n7vmFCB9S4~5uI-hT zsd^*(&q0r`WC?E=15yYTL?6e(_d7lgd2A(`(PaODHUoS`=zl@(#Y5b?!7DsLvE``k zeyM{LoC}_!8dCP$j#@wp?4b+H3*Ay`>7s=-hYXf0h+JPSsLe(NiGX=fF98l!JJjXk zkv(zW9l@71a|Bma~-|D35DOxW{T+8zX;}0I@+k6`j;%y;t`SBpVT1~N}q}_8^ zaB)bcC;gS#i%AAdr|uV(&k1;fFVB=k3tQs{cNNvuR01RPvBghu0susT8SZ0KRkbBO zOrOkPY;|{ZDGB|y?Hg76qu0fzeOHY=hF!{^^m?`Q5b&dcZ0}Mfs$>W()#7`7!tQ1J zgRb5bmwG}$nv!dLv@TZ;514^AUT>T0U zI_RB-%(r=W|<$% z8W-H?7q_5YDkyVbaOZoUlFNvvzUcXcFb%B_~AeU&d@_ zF00ON%ZQBED$+5f*e&p}53Q*n_NYZuyRCG)or?MH_ey~`BrwPRgla|q&9cXTYnnga z`q@#j-AgIWD%J{&0#kiL?7JmpDEs5FA$mb!mq4Tr*p2V%G+vbhx>6zBlAHkHiW6IU zq7K@y%Pc{$ZMBOUpi|}JWo_jI&6uTtVR%9m(UL8kwPlvM%%SrDQussoC2WU~2B%4w z`#~^W)%LNb41Ph3G!!&>d=0BeuGddY{N|gJIH2sMj!*LToH_G3O%G(S3%%7Hk<=L@ zyIY1!+}`t+1)=z=Qk24r|6%K9&3$aCi>r9LH#4@g>1nV8cr;d{(coFGtx&UI{m>S9 z&wg`?b0sHLw?N(6Q?KmT^f@Mv0eyXm?SfYt2n|fEvy*y!LZlOGX@|pak=cQ~t}9 zpp?8@%s645EEs>L4{Eih7)D)tq$D6sIQ)U3QEFHQ6Z_SXoBDvq$;#o?j>Cvr%jH={l(1Mwoib$11$U;Cu9WmDoU(|tb zsmc@H*FbaLkhl_-~7H3XNup)GUu$4_sqs^acjsP!Cl8N#(Xlpj=boGA^V zj1k_CFWDg0_JhngsaoRA7LcIenhI8nA3R1cYm^l79K8(3U?OQ*1!tA`$jmk<)=}>L zV4MG(5MxSp(BNFMmuG^@f{XA~K=DC<=M9L(sR zZU3rKNx(2i8Z4R^EjO2U2kdA*CyzS{0#2oE4f@D`Ex#=vbTfJw2Z3(X2pPt}^~@E&G(zLyI^gmk{z zi~s0T=BYBC^E6KpvNWHlMM}iVNGVfy%GU;nW#K)4duZCkMf)$j{dg=SCdP^<4+2Kl z*D4Me=IbpNR%_U8;uuh;}pm|k}yDF?61qa?7Ph2Y&EHF`NYWK-XR)7VCo5;khfug z-<6xgN$bEDF|F)i0n9+mU4XlveUj- z$*7;Q7`qr8Y&%>iR*>T06{$70AY8k4E@Lb5gMB` z?-Cw`&dhL~Njy0owa%kAMA5#ipEJy1aQ^OF7~nE%mhhufYUL=x)Tg4w`U6ZXC(StkU% zW3Oh-(sDU=Prvk3npTlRMjx(g@CgN9g5})rVIrP|bF#*7P*z^GW55!OV+zVR+;P_; z-Dm9UqmG}c<13LHOVS`(#t%{>dDh>xnMcpsU=nyaD(8JHm7*|aYDvyw_QvJ6vxX{z zo+=0z6|0XJHxhN861$PZ5wNp1jY7daz)~O*9A3dmt;SOcnfD5(oR(_)nGT%ijtfWQ z>}GLyPY%Q=cLUXET<&K}btUc?XOOaR4X363N1xTt8nQmAl!lNW{J&#!8n%nNGT01D z)kbg(%cW}DJT$YLw==_Prbuz{{wUV<0-bX;&$%5PLs7)?>U`5LNpcAP@ww-cKvLW- zvYpyi#);+2;Q0!2He+D7@q}sT^iD6=j0nheF9PH3cNCmnZm3nPy3h>3{OmGghPcFh z%MA9%K1jm0yqH4e`f`*QoXbyy{tOatxd5sS1Di0-NY6A^ygl-aLA9}ER#=yb1y=$zYkG~d~hHtm&^^WnZd zM)qtn7J+-aeqq@qjb^zr8onP4!#l6`aw(8{@L5^5J@KXOL`c(fY-bg~(j~1(cdid# zw}QWAYRs>}ko;sjG0&B6Z8G{Bc+f^}<<*Bb95~QYSbEvdXTB>;pp0fFQdcm(jvkz2 z9-1=mxr?^_MJdUW(vGh9I(GYF6q*Vmqw?-zY7l;8t{d`Z$zrBjqM}#~bVs`9F0nU% z<5th2vI#d)s27S&pD&Uhnq$Lm>sAxYg?!~8t~--RUqtc@@uKaOd>^`CieknP|`L8vb=GZL(qBW z#N`S_W9=>q6{<@jp}Ac)AgjjIO~Cm3*!#5A3*cQl7oT7;wo2H*zScN$l09BNZ75Wu zL8l@z#Ivbg5_%Yhigg{-x%Js5dsVHkwt9JufhGhY*b%a&F|4Ri7)0=kO_Vlhqxze% z;`3p@Akpr*-ePv;PKm|{IcHM7ip0PNwuOkZ$$hKSOe1f_ZB%pjLW3CTYJTiiIGC&T zusgTUP3eT3O(F@!IZ!p6;AbvawUAgQ;80suw@ffRT6Mn1gU-Ll*Y zf``!?$wUHzsI~mkIu}+2P`_!7>^?h6!U0e8uD>Rg;IdWG3^U0A$L4~508qA=!Ke6_ zk*#9GSJ@BFq-^?7P~(JItk!m;p(@E>`*xjbPV4H(nx)Sa(i ztnx28H~zr>zWBiBwWA=!N3yQ(ptx^ar`l+C3Y*!w``A8z)5x5C3WlFZ@lpc*+$x#T zvKCc41~Y(*}|EL zXAEpJzL-nex5>V8!XH^OAZ^)k=H7?ZGqwwgi*D+!<~IoLxhmUuCf>kDOyHCKt3^X0 z3oxa|M(}80`HN*wzMj|jI&lx4^*vK-CUjls!SQR(q3s4E5c0&T_Qd6-cE3zIMyA>4 z7%D9Ue4|+ukd?>G+lzD&^Nz^FPK)pW%}Xuc?c7I=q?#P9!0oz)lVe$nny_nj0|iW{ zC^Suz%K2~TBI6hiwiiLrviBua$g_ONytOT^=qFoHQ=KpD)9LN%Y}4g#;=)TTRI^z- z1JY756wkKp)*UVwrbHPtu~e?Z3Od3G1G|7sX^^gv-N*VR7O1B%1zU@vV<3(0Iw0a=1ZCp%2~%^pI0)h z-RlQVaI5{I<~O7C?Le1|i46tg(BQR*eDSe1rk*=G_Sye-t|LtT+-PHMEe_?i!An0_ z`#M*}t6-|_6`_qDMXxKE*%gbO5(N5rPoL%nlWPnMQw=H)xFA;OcLPl&Ef)s`Vg|`Bzt2gyqZ~XMqDm@2k zX()2&F>=IeTmP_QRf~*JV4{OVeF1FGTI^dCiAFx-LtFu{lP#f^*;CX;=??HggFb-F z-tp`^r14f>cSW#?3SP)D*)lik0*@ebL{aHwW1&!rJJCQN!>jnP;Q7O832H*sK%{gO zZH3!!%il1c_19j(miZ4qNWy6L^Z8|m6$c&eycZ^p6Llewaa=g1gR5;aX@=*nfP8_`=Co&^0v$3w-ut@ zW|*=sf`Y{=UuP`yLiqzwpHH<;hzE$)KJl=`NM`S z%(3|^Gb~h$K1Z-1sa6Sg*b6^!TXOl_#J|Di2{gSh!T@nwWI`g>NVpIb*{*(6>?hlI zC>}8`G3B-((nl$%bEZ24N<4>JtM^RyexEOBeSU-Z>s>#N@ud+qtvR?fJA$YC@*`u? zFT*Nh0L%@xy>2s<)z)+UZxqi2*_L{qbHhIcGfTLmSG8T;rW%=q%C=593%9w~R$e*` z)ey$>QG6EzAV=e4&ZM*0V59&A=0LrIaRxn!)R?vpO+hq6mq^bxrv$bbN`x7rQ(x4e*N)xMO% zcMEUFUZz<{d>veUgIaZ~d;ipmS|pEbI;s;xbpM zPSt-Me4)`i|Mm0#;lUW&WOKcLVifOSVg)5AGwvBz0};{YQI8nmgH$cqOrFZP!6JW0 z)fgR2w`MKBBN7K+6@-WGpNF2UU~s$fptRhnHX&blI|bWVBHOm9q4s`T@$_+vA%|>J zdGLIrsfqxW-E~!>kJLy#c#wUqB+jW;TA+eF{kS&hy3~poJqUYj%V@x2nV>LF(<g$OXd5f26V$t?4bc^u6Etu{U>qG1iy>)fa9J3>FDA#1~9xV|KsX{^;Q z{&ZufI*bQptIzu=H3=33KE*-NCUBET;teLkZr}auxT?4T#DONp{T0Ah5aE{hHp!VP zPAH{lCCX`SR@UXe!V(4I0%@w!N8{%F;FWcCZ3TI7-vT1=h&TVc!oe&DE z4ue8x^QiZ2UR8p3GQU~bR+$q{x!2)nD@OcMJ4e7#c;?VH^vV_YmEt<`*is(BPC4qw}~d zZf0NiI)e@>c7B#}lJd^c)@FCryLdNW9XL;)O=w`k=<2|y(jV<&>AA1?LGM-HhH|1i z4{6H*qrPz(-*wri{U_NL4TE;<$Ln4rnGwM6MCZ&VU8>5SnBUg-d36>eN0O8~019YN zHwNVmbYrtqvxEhh`df^o31+}=RsOIAX^1^3_olaoH&x+tTGDGe8JgN?m%LO?LNpVK zGSzLP4BM&l4JmVR|DcgCbP;LBroy@g_}doLq-vOy$&ghf#5t9sDEUqP?Y3lrDo-1x z)r7_}RjdHNU3G)UC(mk~pe@4F>A9ZiT}}F-;NoUDqp2Q+npy@Yahm95!inB)X)BB! z&V2MoYBU z@i0@K=$<&7DJxG&TIsj^y`v^0Wt=-QF}VPT+v6)ilH?3$o$=g zZUUpVW)(PlxaJz4a5&X%fnpFSas_)W#-S8e^+z~Y;~(3wu;Xo*27q9ot`CuQ6*^=GRNYoy{M#jxZg3yPSofQ;9&O4Z z4=9vN4kaHD5Jq9>Bm(H>&h&#y-}b^ULDSJ$I?8vvyGH8Zly{He)Z-ipL?6B)BsHXH zlp16qG$#t-8hZuBDnXX>Kt?u6y!+8DKV4k#u1^91+tXLuOI@>BYBF6IzBNj5%9Q6- zaBkaXAH9|mlF?J~;wF6swOJPd*1moG9JBr_Oaty&%uXMHg&F-c7HlIIS>aE8kXt)f zH9!tuW!4EDt4;yC=scbu5D?_-%3(B5BHw+Tfer+!vfu<^#Qer>`^BI`>hZTtxi&nv<>ViGT^wCIQ zp^<+NMnq)J?`Ra6l0VI}?aa+go==A8C^*=?K4nxMi7c~L{|MDS_)9+oVd`*royyS>4;4=hn_E{lp z5+lhqo-@CZe}huJZ7oK7x@*)g;1#)NxmOM0A&>dysUHI&F=Pw%s^-P?EBncIxWJ6h zbm|aHO}Slw5N$Ppk4irMxBx8l7nI1oqk;2AonXZ;mL)s-$#94eSejc!di?ce;OmcV zYpGAgnm{58iCXxAVMK=UP0iS0Qod9Ai~19eNOYvL{M?9uK#D(=Ke%KyXzC&X>f)Uj zba1zX`fs!Y?Je1rk>RoK3xwa$#*JP;PpjF6!;&hLo#ikhD?-)@7%}BDdJBRa?JL`! zH9~2LT6_%Bl|IaT`f(0`fUt&dU%~}snSb^jGk;u!>$lK!@Ud4M)|OC_dN`+*?kl1W zm-&#Bw~m{1Fd%V6ycx2R}^EPiJxp)VlnIhCT_leZ z+~_eR;fW>-ZP_`Fs z2;>Ew8ScOG{(%=_Lt9j*>p00GcB?d;Xq5c*?E&=axcfR0s5js-0YvH89%oV zb_t8LLPR7Dadc9cs7y0#@y&Q9pg|PFE5gB!;2O$-pd=U%OT?;!axY|{U*Lkr6kSpW zp7_VJjba4M7=T`p<*gg?_I<-7>Tk^av)xO+{PYA*5;DBd%u!FxB-Jhf}O zvx;JCjk!*)d~4hLNd@UnZr-7~1U2?t_$F~`nzT)1lZe0F;#eA9lS%o5v}Jl~-K?+F z75wG{Uh~>`b19P_UTFT6TJ?l)dmyQ<&dgoWzv_{Svp*Yj0vKIstAKr?L1 zoJAtXr5K|9@guPaLWZH>Way=I{qH`%$y0n4-5g1co21qaxPV))X3bBvdaq(eGEs0q z-GM9!>|cW+cq8q}Wr+|0E|2P)_X07x;d=}Sy1Q}_lxnQ_=sc*4hXRDbSmaM(-s(RY zyvKvTn|%>R5D$3N_o)QW%)P9CyQx_QLlC7;UF+BDSNFDIBB*=>01Use-O1JIanOv2 zJPMJ_NaIR32A2vS3_L|g^ordK7093bVy^{DW!BfoK6?w}*M)Wdq8x8iHn5}%=|iEh ztOrHxYVn=&XpkscYV6G$KBT666G-x2U52x7G?APiqUJAJo3ZH&Gb$FnF)1O1grXYW zfe|%J2CvqXE`g zC|kxfPTxZw=F&~wzZ;#gFc*F}h1T&wKmjrNTz(#2{qY0md+n_j+r$7_YnmC-0MnK^ z1o6v^8}hVO?HZzF33((doX(VB{9yzd-bOWh?@hxM_9<_%156~09TI8&X)_5%bVgz0 zCu1l~k*r{+5*YSvLE;BB<)fkCBRzW~;77&bOlTxZkkY$v01MwO`BNSkYn#-Bz*(Hq z-z-wFn70>8A!Td$y@vwBF=c|X{7m;t zc0qwqtQ@JcW8oY=CPA)IEt6_bQ!sih*d+aM_2qjWbHPLFsl!c2GNe@w2e{}{?i>r* zK}tcj-;C<*yFgdg5~aAQ8#qXRO8lZa4+v1q9t3M`W#RrbL`l$%#<>td!X3xHKymLd z@b~?>!BRNeOg7-|HTq{0A=DZ>w_Q^rYXFIj2m|dWsR{WiPWgWs4hTp$wKr6*Jc$vm zyy;;~|C3#yZDrg`Io1+O=E|i+Ox!gpSNHRs!Uu$RB^H?1(PGqQ|0_`%HDV>zv~NQ* zHGF8`+9}yHc&i#VU9GM`A~OMNBuT#%DSfh!u>coGqMZdc2V$jl$s0zjQsKpxQU~4`sKsSyQiTh`s$6KVgMxH^=MT3K zx$zO63?mvHe}6fg9R&o*|A5qh4)`elFjjiOIXsN(>|6#kl0T#QsqAC%2u zN?ml!u;FmFu(9@&=oNOwV~_Ut8uanZrF54RcsKIw>vBPGpp|pqbH{NS2zjyGLwt%N z^e2)&7rC2W==-bRKn-}MDALouM}1)p|F6G&y#vRH1ItoJwvH(+Azf$$0<=Uwo?eaF zG*#rSyR8!<>(p=nzk<4J=!0Z8ao#Vgn9C@fD_3Q(6`Wb`(_{|lZJ{A1 zLbO?iQb@bSCx^MjNb4#gpmXfg@qznm4EdVTA{6=kcWZI5TX1p$T^9{*`f0M+bdz53 z9)dM~bNnL7i@!k5W=_q>W*5+6KSEJ&^F@Knm}%XVuoY z;^cr+RYnyf;tv?3B7F~d$GFW5hS5~NVo&_rf4=PLG=7iK%Fb>-`jlA%_wp9z^xw7x z1x5{*BXbE%C;sFNL&f|NOu}<`5SoS5S(!m!Y5k&j2Pw}m?kjXX1STz|!8B}RS>v&l znWgr&`jLoMWMbAC%HYg z2^laHd#dP``QE)KkQo7sBXuulu-tkt;8hlVK7o9@_w)e$nu^y;tH!T%LcBg^1+sL>9LhNs;3U%VLxr&c`^D;uR@t5eI@iI zpUw$~QY@OX23*{DF){&3@M1TwQpaKv!@O^ zINB%NfN)eovJgq2G{diU2^_8-aUdK-Mrjo+2)z)(ap4So`e%|Pj;eU!Fb5Qj)~@M&m#{&k>nXQl2U3FhBYsDD z=6)LUeQgU_yBq+8;8XT?lGJ&O}5?)o(;X zsR%Cij#b(w*TLbq>t)|}&UbW1W@kQ;Wif>AI)pa2T=8k-Jyu%rj?a`1+5&u$< z|0pGvXszfU1Rw}zL{e~d)Ps#WoPFhFJAI_@fIVR{<%EX8`eRIqX~%|ge=;j{t`I$+ zp5Xfu#?HlM%-J~x$#nF(7#(t~pSm4Am!~j3qy+0{R|gp)NL+zEWg=3UW_IaW^&5O? z>mrmFjDyYf75ud><5sPqy8c7h(#W2MUxe|Z(@`aXF68U~88CoxfHMpKmQHjMeOYo; zt|vR%e8%|Ji4mY68Uyhjuy!-h0kdo_1BXViC6*sD)XnD_zTNAk5it?KZ-PfjT>oxq zkh3NEe}yp^#&MyaSAe}@llYkyw|C!e&Oly$03NsY%{RgIK#eO9ImO47+2NlN$dCR9 zDCQs-e2qwRsgyY15G!;EGipOASdufmB9jAwTJe|bJ9J>Ou2CH6TcrWLb%JB=)A(o( zvNoJ#d5%Z$`(hr+!wcC2lA(rIjr2R0-JB!VBhyWHF6=)upnOSWXIJ)=F3kZ2IR0BJ!el{_I6$y!|LrXx zEc~fuUo!gb^CO}f2uRg+z|H=t3kZGuvKM^R;iALQ1GaH!GNlo>4wud~B7it6%T+Jx zR4Yzx*Z1_S)|py-fK_gV4q22~>7^vD0dTi1gEvOon-1S>d0ER?F5&z04rJGPSEt__ zp(&Ajx%w$|$LObrIrJY#rk-vcv_BM|iikH;S>lQYoLE&L<9b2Xb6~@zqE28cx$9V} zLugyYdQU^tg7+nKpPWoK%Is{Prrqrh|Lzp%$IJ#ie{zcZrp-T}%o9U;|J7Mv3#Fk~ z$3NzDYr#2&t8sF>PHAl^6KPY1D&vG6R20Uud6hwj7ZB(%%flT!vI5~ zfnF_F1B0cPiC`+ zSW4N_0xcbe?~}`yk9vL(2aGm(^VxO=3l~elz_H&Pd13w;;gEA|uS|kW#T%t%CRYxv zs~>8X0`>K&_d8l=h3!?W0o$34$gpwZ_scjUv&RrD5R3>TUdkTq_c7fCE<}?b^67N2 z&N5uq;mxRJRI+`&n@Tkf}(SghFV=xhSqoAPx?>d7#+A&M+~2zn_r|%VhuA zk^zTN7fkJuuyt0^+)nz~#Ts(k=MO@c75;hDU|lw zWga`l)gWSpgzJRxUTViawUF|PbKE+Ht;zatPB5;*hdUwYkI>olc$WCyJ2XLBzDPpprgb3@jv+7!p} z&20&v+uk9BxV^Ox$~h|9+IKh;XhZ-EN6?r-T;I{WcruJ@ss@VrW7NGlLU)GmZ=1mk zs{X@FTuby8xh%fBM(UqXhIXO@eeUdVT%^)iOeHWcT82x!%l)lal!Ur~FF@{VF55aW zHZD#~wu&=q0t}m1`1c1#$NsHFE&e7G;z4WT-$Iy~f(hqTFWV$Zqz=ICSgxk#nWt8l z=*cabE;YTN$Ka~UCoJ@wpkc@nzberm>bgYDnFQHflLsl^!k;{LXtr#fA>Zdky1#|3 z?!f~Af#LqgrC5P8PyUu~42cYiH}NU|k95c1#NVgNlOAn-(wRKOz|C0w9x5?$->!V2un>Usul}y=e1PS; zAxjWzLg`}{$fIN8vi+y&j_ovPt8rsgs<_ za}t0hZULCcRBuui787a*VcP;E%C+%0oaVOOjFxDux^+4s#a6i87LG?9bSm+ePBIKt zBb;CjLA>}znxiM>KDZ~HXnLx7-Y=_@to_^8#!vrnM)HaM9YdH{z^!_`i$dBIi zcc!B#ASQ*-Dml4rfr~I#66;ym5BNIpKMMt*GyW{te=8KSw%Z7?5||3m*Jf-b(eZ0o zx$YVG+i-Wa8Pe0k`cE_mf(B=T7I^U-vbJv916HsDV4NWZ;N4vk}TKGn3cz`m0DCU zz6<*z_!&_#rTQHA)hn6yl}+*PN*{iQg#GrnZ(#XfUk2{{hqiD7XVLsCln(qbW6XfS zf;I}LYKGgn;PpQXnGE!l1pwbvE~2SQZQKGnv1-U`aFq{1>^;64Y??p`N^x_qzL8WK z7?rSb7GuBhiFFglr797igmb**_CHY@w(^0^#-p9WKSkkS~QlpQ*AwUtT62omY9-01IK z(i&S!re0($xyQ-e!%g02AsVcR5S!{l0PA-_SkJA~F|{fQspo8O!IZ|9%C$}-=s-hpS-0mz{1e;f1veg8j3tPX4d diff --git a/services/capture/src/fixtures/mock-stream2.mp4 b/services/capture/src/fixtures/mock-stream2.mp4 deleted file mode 100644 index 0e9dc7ff043f977edbdb652931ce21410dff921f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 165027 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

(FHYs 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�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 do}ZCrB1s*O|FIuDXWmnZgQEHO { - if (err) { - server.log.error(err) - process.exit(1) - } - }) -} - -async function doRunWorker(workerUtils: WorkerUtils) { - - let workerIds: string[] = [] - - const runnerOptions: RunnerOptions = { - preset, - concurrency: configs.workerConcurrency, - taskDirectory: join(__dirname, 'tasks'), - } - - - - const runner = await graphileRun(runnerOptions) - if (!runner) throw new Error('failed to initialize graphile worker'); - - /** - * This is likely only relevant during development. - * if nodemon restarts us, we need to unlock the graphile-worker job so it gets retried immediately by another worker. - * - */ - runner.events.on('worker:create', ({ worker }) => { - // There is no way to get workerIds on demand when the SIGUSR2 comes in, so we log the IDs ahead of time. - workerIds.push(worker.workerId) - }) - process.on('SIGUSR2', async () => { - console.warn(`SIGUSR2 detected! ulocking ${workerIds.length} workers, workerIds=${workerIds}`) - await workerUtils.forceUnlockWorkers(workerIds) - process.kill(process.pid, 'SIGTERM'); - }) - runner.events.on("pool:gracefulShutdown", async ({ workerPool, message }) => { - const workerIds = workerPool._workers.map((w) => w.workerId) - console.warn(`gracefulShutdown detected. releasing job locks on ${workerIds.length} workers, workerIds=${workerIds}, message=${message}`); - await workerUtils.forceUnlockWorkers(workerIds) - }); - - - await runner.promise -} - - -async function main() { - - const workerUtils = await makeWorkerUtils({ connectionString }) - await workerUtils.migrate() - - console.log(`@futureporn/capture version ${version} (FUNCTION=${process.env.FUNCTION})`) - if (process.env.FUNCTION === 'api') { - doRunApi() - } else if (process.env.FUNCTION === 'worker') { - doRunWorker(workerUtils) - } else { - throw new Error('process.env.FUNCTION must be either api or worker. got '+process.env.FUNCTION) - } -} - -main().catch((err) => { - console.error('there was an error!') - console.error(err); - process.exit(874); -}); diff --git a/services/capture/src/integration/Capture.spec.js b/services/capture/src/integration/Capture.spec.js deleted file mode 100644 index a946d53..0000000 --- a/services/capture/src/integration/Capture.spec.js +++ /dev/null @@ -1,123 +0,0 @@ -import 'dotenv/config' -import chai, { expect } from 'chai' -import sinon from 'sinon' -import sinonChai from 'sinon-chai' -import { CID } from 'multiformats/cid' -import EventEmitter from 'node:events' -import { fileURLToPath } from 'url'; -import path from 'node:path' -import postgres from 'postgres' -import Capture from '../src/Capture.js' -import Voddo from '../src/Voddo.js' -import Video from '../src/Video.js' - -chai.use(sinonChai) - -if (typeof process.env.POSTGRES_PASSWORD === 'undefined') throw new Error('missing POSTGRES_PASSWORD'); -if (typeof process.env.POSTGRES_USERNAME === 'undefined') throw new Error('missing POSTGRES_USERNAME'); - - -const cidFixture = 'bafybeid3mg5lzrvnmpfi5ftwhiupp7i5bgkmdo7dnlwrvklbv33telrrry' -const inputFixture = 'projektmelody 3021-10-16 06-16.mp4' -const outputFixture = 'projektmelody-chaturbate-30211016T000000Z.mp4' -const timestampFixture = 33191316900000 - -describe('Capture integration', function () { - - let clock - - beforeEach(() => { - clock = sinon.useFakeTimers({ - toFake: ["setTimeout", "setInterval"], - shouldAdvanceTime: false - }); - }) - - afterEach(() => { - sinon.restore() - clock.restore() - }) - - it('end of stream behavior', async function() { - const ipfsClusterUpload = sinon.mock() - .withExactArgs(outputFixture) - .resolves(cidFixture) - - const sql = postgres({ - username: process.env.POSTGRES_USERNAME, - password: process.env.POSTGRES_PASSWORD, - host: process.env.POSTGRES_HOST, - database: 'futureporn', - idle_timeout: 1 - }) - - - const voddo = sinon.createStubInstance(Voddo) - voddo.on.callThrough() - voddo.off.callThrough() - voddo.emit.callThrough() - voddo.listeners.callThrough() - voddo.listenerCount.callThrough() - voddo.getFilenames.returns([{ - timestamp: timestampFixture, - filename: inputFixture - }]) - - const video = sinon.createStubInstance(Video) - video.concat.resolves(outputFixture) - - const capture = new Capture({ - voddo, - sql, - ipfsClusterUpload, - video - }) - - capture.download() - voddo.emit('stop', { - reason: 'close', - stats: { - filenames: [ - inputFixture - ] - } - }) - - clock.next() // actionTimer elapse - - - expect(clock.countTimers()).to.equal(0) - clock.restore() - - // gotta wait to verify otherwise verification - // occurs before ipfsClusterUpload has a chance - // to be invoked. - // - // (not sure why) - // - // maybe we're waiting for the - // concat promise to resolve? - - await Promise.resolve(() => { - expect(ipfsClusterUpload).calledOnce - }) - - // Capture.save is called as a side effect - // of Capture.process - // which is called as a side effect of Capture.download - // so we have to wait for it to complete - // this is not ideal because there is potential - // to not wait long enough - await new Promise((resolve) => { - setTimeout(resolve, 1000) - }) - - const rows = await sql`SELECT "videoSrcHash" FROM vod WHERE "videoSrcHash" = ${cidFixture}` - - expect(rows[0]).to.exist - expect(rows[0]).to.have.property('videoSrcHash', cidFixture) - - - - }) -}) \ No newline at end of file diff --git a/services/capture/src/integration/Ipfs.spec.js b/services/capture/src/integration/Ipfs.spec.js deleted file mode 100644 index 0b23477..0000000 --- a/services/capture/src/integration/Ipfs.spec.js +++ /dev/null @@ -1,18 +0,0 @@ - -import Ipfs from '../src/Ipfs.js' -import { expect } from 'chai' -import path, { dirname } from 'path'; -import { fileURLToPath } from 'url'; -const __dirname = dirname(fileURLToPath(import.meta.url)); - -const ipfsExecutable = '/home/chris/.local/bin/ipfs' - -describe('Ipfs', function() { - describe('hash', function () { - it('should hash a file and return the v1 CID', async function () { - const ipfs = new Ipfs({ ipfsExecutable }) - const cid = await ipfs.hash(path.join(__dirname, '../test/fixtures/mock-stream0.mp4')) - expect(cid).to.equal('bafkreihfbftehabfrakhr6tmbx72inewwpayw6cypwgm6lbhbf7mxm7wni') - }) - }) -}) \ No newline at end of file diff --git a/services/capture/src/integration/Voddo.spec.js b/services/capture/src/integration/Voddo.spec.js deleted file mode 100644 index 7cc3742..0000000 --- a/services/capture/src/integration/Voddo.spec.js +++ /dev/null @@ -1,62 +0,0 @@ -import 'dotenv/config' -import Voddo from '../src/Voddo.js' -import { - expect -} from 'chai' -import sinon from 'sinon' -import YoutubeDlWrap from 'youtube-dl-wrap' -import { - EventEmitter -} from 'events' -import { getRandomRoom } from '../src/cb.js' -import path, { dirname } from 'path'; -import { fileURLToPath } from 'url'; -const __dirname = dirname(fileURLToPath(import.meta.url)); - - -describe('voddo', function() { - - - describe('getVideoLength', function () { - it('should return the video length in ms', async function () { - const fixtureFile = path.join(__dirname, '..', 'test', 'fixtures', 'mock-stream0.mp4') - const length = await Voddo.getVideoLength(fixtureFile) - expect(length).to.equal(3819) - }) - }) - - it('aborted stream', function(done) { - this.timeout(10000) - - getRandomRoom().then((room) => { - console.log(room) - const abortController = new AbortController() - - const url = `https://chaturbate.com/${room}` - const format = 'best' - const cwd = '/tmp' - const voddo = new Voddo({ - url, - format, - cwd - }) - - - voddo.once('stop', function(data) { - console.log('f in chat') - expect(voddo.stats.files[0]).to.have.property('size') - done() - }) - - voddo.start() - - setTimeout(() => { - voddo.stop() - }, 5000) - }) - - - }) - - -}) \ No newline at end of file diff --git a/services/capture/src/integration/record.spec.js b/services/capture/src/integration/record.spec.js deleted file mode 100644 index 4a625f4..0000000 --- a/services/capture/src/integration/record.spec.js +++ /dev/null @@ -1,35 +0,0 @@ -import { record, assertDependencyDirectory } from '../record.js' -import { getRandomRoom } from '../cb.js' -import path from 'node:path' -import os from 'node:os' -import { execa } from 'execa' - -describe('record', function() { - it('should record a file to disk', async function () { - this.timeout(1000*60) - const roomName = await getRandomRoom() - console.log(`roomName:${roomName}`) - const appContext = { - env: { - FUTUREPORN_WORKDIR: os.tmpdir(), - DOWNLOADER_UA: "Mozilla/5.0 (X11; Linux x86_64; rv:105.0) Gecko/20100101 Firefox/105.0" - }, - logger: { - log: (msg) => { console.log(JSON.stringify(msg)) } - } - } - console.log(appContext) - const { stdout } = await execa('yt-dlp', ['-g', `https://chaturbate.com/${roomName}`]) - const playlistUrl = stdout.trim() - console.log(`playlistUrl:${playlistUrl}`) - assertDependencyDirectory(appContext) - const ffmpegProc = record(appContext, playlistUrl, roomName) - // console.log(ffmpegProc) - return new Promise((resolve) => { - setTimeout(() => { - ffmpegProc.kill('SIGINT') - resolve() - }, 1000*10) - }) - }) -}) \ No newline at end of file diff --git a/services/capture/src/integration/video.spec.js b/services/capture/src/integration/video.spec.js deleted file mode 100644 index b3b7f3f..0000000 --- a/services/capture/src/integration/video.spec.js +++ /dev/null @@ -1,33 +0,0 @@ - -import 'dotenv/config' -import Video from '../src/Video.js' -import { expect } from 'chai' -import { dirname } from 'path'; -import { fileURLToPath } from 'url'; -import path from 'node:path' - -const __dirname = dirname(fileURLToPath(import.meta.url)); - -const dataFixture = [ - { - timestamp: 1, - file: 'mock-stream0.mp4' - }, { - timestamp: 2, - file: 'mock-stream1.mp4' - }, { - timestamp: 3, - file: 'mock-stream2.mp4' - } -] - -xdescribe('video', function () { - describe('concat', function () { - it('should combine several videos into one', async function() { - const cwd = path.join(__dirname, './fixtures') - const outputFile = await concat(dataFixture, { - cwd - }) - }) - }) -}) \ No newline at end of file diff --git a/services/capture/src/logger.js b/services/capture/src/logger.js deleted file mode 100644 index 84c387f..0000000 --- a/services/capture/src/logger.js +++ /dev/null @@ -1,25 +0,0 @@ -import winston from 'winston' - -export const loggerFactory = (options) => { - const mergedOptions = Object.assign({}, { - level: 'info', - defaultMeta: { service: 'futureporn' }, - format: winston.format.timestamp() - }, options) - const logger = winston.createLogger(mergedOptions); - - if (process.env.NODE_ENV !== 'production') { - logger.add(new winston.transports.Console({ - level: 'debug', - format: winston.format.simple() - })) - } else { - logger.add(new winston.transports.Console({ - level: 'info', - format: winston.format.json() - })) - } - - return logger -} - diff --git a/services/capture/src/poc-lite.js b/services/capture/src/poc-lite.js deleted file mode 100644 index 6bcef30..0000000 --- a/services/capture/src/poc-lite.js +++ /dev/null @@ -1,25 +0,0 @@ -import { createWriteStream } from 'node:fs' -import ffmpeg from 'fluent-ffmpeg' - -// test stream from https://ottverse.com/free-hls-m3u8-test-urls/ -const playlistUrl = 'https://demo.unified-streaming.com/k8s/features/stable/video/tears-of-steel/tears-of-steel.ism/.m3u8' -const fileOutputStream = createWriteStream('/tmp/test-stream.ts') - -ffmpeg() - .input(playlistUrl) - .audioCodec('copy') - .videoCodec('copy') - .addOutputOptions('-movflags faststart') - .output(fileOutputStream) - .format('mpegts') - .on('end', () => { - console.log('Finished'); - }) - .on('error', (err, stdout, stderr) => { - console.error(`there was an error`); - console.error(err); - console.error(stdout); - console.error(stderr); - throw new Error(err.message); - }) - .run(); diff --git a/services/capture/src/poc-s3-alt.ts b/services/capture/src/poc-s3-alt.ts deleted file mode 100644 index e4b1a21..0000000 --- a/services/capture/src/poc-s3-alt.ts +++ /dev/null @@ -1,135 +0,0 @@ -import { PassThrough, pipeline, Readable } from "stream"; -import { type Progress, Upload } from "@aws-sdk/lib-storage"; -import { S3Client } from "@aws-sdk/client-s3"; -import { createReadStream, createWriteStream } from 'fs'; -import { createId } from '@paralleldrive/cuid2'; -import prettyBytes from 'pretty-bytes'; -import dotenv from 'dotenv' -dotenv.config({ - path: '../../.env.development' -}) - -if (!process.env.S3_BUCKET_NAME) throw new Error('S3_BUCKET_NAME missing in env'); -if (!process.env.S3_BUCKET_KEY_ID) throw new Error('S3_BUCKET_KEY_ID missing in env'); -if (!process.env.S3_BUCKET_APPLICATION_KEY) throw new Error('S3_BUCKET_APPLICATION_KEY missing in env'); - -async function uploadStream(client: S3Client, stream: NodeJS.ReadableStream, keyName: string) { - // const pass = new PassThrough() - // Create a stream to the S3 bucket. We use this stream to upload the livestream to Backblaze S3 service - console.log(`keyName=${keyName}`) - const target = { - Bucket: process.env.S3_BUCKET_NAME!, - Key: keyName, - Body: new Readable().wrap(stream) - } - console.log(target) - // greets https://stackoverflow.com/a/70159394/1004931 - try { - const parallelUploads3 = new Upload({ - client: client, - partSize: 1024 * 1024 * 5, - // tags: [...], // optional tags - queueSize: 1, // optional concurrency configuration - leavePartsOnError: false, // optional manually handle dropped parts - params: target, - }); - - - parallelUploads3.on("httpUploadProgress", (progress) => { - if (progress?.loaded) { - console.log(`loaded ${progress.loaded} bytes (${prettyBytes(progress.loaded)})`); - } else { - console.log(`httpUploadProgress ${JSON.stringify(progress, null, 2)}`) - } - }); - - const res = await parallelUploads3.done(); - return res - - } catch (e) { - if (e instanceof Error) { - console.error(`while uploading a file to s3, we encountered an error`) - throw new Error(e.message); - } else { - throw new Error(`error of some sort ${JSON.stringify(e, null, 2)}`) - } - } -} - -async function main() { - try { - const client = new S3Client({ - endpoint: 'https://s3.us-west-000.backblazeb2.com', - region: 'us-west-000', - credentials: { - accessKeyId: process.env.S3_BUCKET_KEY_ID!, - secretAccessKey: process.env.S3_BUCKET_APPLICATION_KEY! - } - }); - - - - // let debugCounter = 0 - // let uploadStream = new PassThrough() - // uploadStream.on('data', (data) => { - // debugCounter += data.length - // console.log(`[data] uploadStream. ${debugCounter} aggregated bytes (${prettyBytes(debugCounter)}).`) - // }) - // uploadStream.on('drain', () => { - // console.log('[drain] uploadStream') - // }) - // uploadStream.on('close', () => { - // console.log(`[close] uploadStream closed`) - // }) - // uploadStream.on('error', (err) => { - // console.error('[error] uploadStream had an error as follows') - // console.error(err) - // }) - // uploadStream.on('exit', (code) => { - // console.log(`[exit] uploadStream exited with code ${code}`) - // }) - // uploadStream.on('disconnect', () => { - // console.log('[disconnect] uploadStream disconnected') - // }) - // uploadStream.on('message', (msg) => { - // console.log('[message] uploadStream sent a message as follows') - // console.log(msg) - // }) - - - - const fileStream = createReadStream('/home/cj/Downloads/stream-23894234.ts') - const datestamp = new Date().toISOString() - const keyName = `${datestamp}-stream3-chaturbate-${createId()}.ts` - await uploadStream(client, fileStream, keyName) - - // // we set up a pipeline which has an readable stream (ffmpeg), a transform stream (debug), and a writable stream (s3 Upload) - // pipeline( - // fileStream, - // uploadStream, - // (err) => { - // if (err) { - // console.error(`pipeline errored.`) - // console.error(err) - // } else { - // console.log('pipeline succeeded.') - // } - // } - // ) - - - - } catch (e) { - if (e instanceof Error) { - console.error(`Eyy lookat me, I'm a big nastry try/catch block and I did my jorb!`) - console.error(e) - } else { - console.error('err or some sort') - console.error(e) - } - } - -} -main() - - diff --git a/services/capture/src/poc-s3.ts b/services/capture/src/poc-s3.ts deleted file mode 100644 index ea592b2..0000000 --- a/services/capture/src/poc-s3.ts +++ /dev/null @@ -1,122 +0,0 @@ -import { PassThrough, pipeline, Readable } from "stream"; -import { Upload } from "@aws-sdk/lib-storage"; -import { S3Client } from "@aws-sdk/client-s3"; -import { createReadStream } from 'fs'; -import { createId } from '@paralleldrive/cuid2'; -import prettyBytes from 'pretty-bytes'; -import dotenv from 'dotenv' -dotenv.config({ - path: '../../.env.development' -}) - -if (!process.env.S3_BUCKET_NAME) throw new Error('S3_BUCKET_NAME missing in env'); -if (!process.env.S3_BUCKET_KEY_ID) throw new Error('S3_BUCKET_KEY_ID missing in env'); -if (!process.env.S3_BUCKET_APPLICATION_KEY) throw new Error('S3_BUCKET_APPLICATION_KEY missing in env'); - -function makeProgressTicker(counter: number) { - const ticker = setInterval(() => { - console.log(`[progress] ${counter} bytes (aggregate) (${prettyBytes(counter)}) have passed through the pipeline.`) - }, 1000 * 30) - return ticker -} - -function makeS3Client() { - const client = new S3Client({ - endpoint: 'https://s3.us-west-000.backblazeb2.com', - region: 'us-west-000', - credentials: { - accessKeyId: process.env.S3_BUCKET_KEY_ID!, - secretAccessKey: process.env.S3_BUCKET_APPLICATION_KEY! - } - }) - return client -} - - -async function uploadToS3({ client, uploadStream, keyName }: { client: S3Client, uploadStream: NodeJS.ReadableStream, keyName: string }) { - - const target = { - Bucket: process.env.S3_BUCKET_NAME!, - Key: keyName, - Body: new Readable().wrap(uploadStream) - } - - // greets https://stackoverflow.com/a/70159394/1004931 - try { - const parallelUploads3 = new Upload({ - client: client, - partSize: 1024 * 1024 * 5, - queueSize: 1, - leavePartsOnError: false, - params: target, - }); - - - parallelUploads3.on("httpUploadProgress", (progress) => { - console.log(progress) - if (progress?.loaded) { - console.log(`loaded ${progress.loaded} bytes (${prettyBytes(progress.loaded)})`); - } else { - console.log(`httpUploadProgress ${JSON.stringify(progress, null, 2)}`) - } - }); - - await parallelUploads3.done(); - - } catch (e) { - if (e instanceof Error) { - console.error(`while uploading a file to s3, we encountered an error`) - throw new Error(e.message); - } else { - throw new Error(`error of some sort ${JSON.stringify(e, null, 2)}`) - } - } -} - -async function main() { - - let counter = 0 - const client = makeS3Client() - const ticker = makeProgressTicker(counter) - const datestamp = new Date().toISOString() - const keyName = `${datestamp}-stream3-chaturbate-${createId()}.ts` - console.log(`Uploading ${keyName} to S3`) - - /** - * setup the streams which process the data - */ - const ffmpegStream = createReadStream('/home/cj/Downloads/stream-23894234.ts') - const uploadStream = new PassThrough() - - // update the progress ticker data - uploadStream.on('data', (data) => { - counter += data.length - }) - - /** - * we set up a pipeline which has an readable stream (ffmpeg), a transform stream (debug), and a writable stream (s3 Upload) - */ - pipeline( - ffmpegStream, - uploadStream, - (err) => { - if (err) { - console.error(`pipeline errored.`) - console.error(err) - } else { - console.log('pipeline succeeded.') - } - } - ) - - - await uploadToS3({client, uploadStream, keyName }) - clearInterval(ticker) - - -} - - -main() - - diff --git a/services/capture/src/poc.ts b/services/capture/src/poc.ts deleted file mode 100644 index e59242f..0000000 --- a/services/capture/src/poc.ts +++ /dev/null @@ -1,177 +0,0 @@ -/** - * - * @todo if we have the ffmpeg stream send an end event, does the ffmpegStream close? - * so far, we have observed the end of a CB stream, and the uploadStream is what shows as closed. - * It would be better to have the ffmpeg stream do the closing, amirite? or does it even matter? - * Here's what the console.log shows when the CB stream ended while we were not using { end: true } on the ffmpeg stream - * - * - * [data] uploadStream. 118018880 aggregated bytes (118 MB). - [data] uploadStream. 118067384 aggregated bytes (118 MB). - [data] uploadStream. 118101224 aggregated bytes (118 MB). - [data] uploadStream. 118119648 aggregated bytes (118 MB). - [close] uploadStream closed - pipeline succeeded. - */ - - - -import { getRandomRoom } from '@futureporn/scout/cb.js' -import { ua0 } from "@futureporn/scout/ua.js"; -import { spawn } from "child_process"; -import { PassThrough, pipeline } from "stream"; -import { Upload } from "@aws-sdk/lib-storage"; -import { S3Client } from "@aws-sdk/client-s3"; -import { createId } from '@paralleldrive/cuid2'; -import prettyBytes from 'pretty-bytes'; -import dotenv from 'dotenv' -dotenv.config({ - path: '../../.env.development' -}) - -if (!process.env.S3_BUCKET_NAME) throw new Error('S3_BUCKET_NAME missing in env'); -if (!process.env.S3_ACCESS_KEY_ID) throw new Error('S3_ACCESS_KEY_ID missing in env'); -if (!process.env.S3_SECRET_ACCESS_KEY) throw new Error('S3_SECRET_ACCESS_KEY missing in env'); - -async function main() { - const client = new S3Client({ - endpoint: 'https://s3.us-west-000.backblazeb2.com', - region: 'us-west-000', - credentials: { - accessKeyId: process.env.S3_ACCESS_KEY_ID!, - secretAccessKey: process.env.S3_SECRET_ACCESS_KEY! - } - }); - - - const randomRoom = await getRandomRoom() - console.log(`randomRoom=${randomRoom.name}`) - - const playlistUrl: string = await new Promise((resolve, reject) => { - - // get the m3u8 playlist for the livestream - const ytdlp = spawn('yt-dlp', [ '-g', randomRoom.url ]) - let output = '' - ytdlp.on('error', (err) => { - console.error(err) - }) - ytdlp.once('exit', (code) => { - console.log(`code=${code}`) - if (code !== 0) reject(`yt-dlp exited with code ${code}. stderr as follows ${JSON.stringify(ytdlp.stderr, null, 2)}`); - resolve(output) - }) - ytdlp.stderr.on('data', (data) => { - console.error('stderr data as follows') - console.error(data.toString()) - }) - ytdlp.stdout.on('data', (data) => { - output = data - }) - - }) - console.log(`playlistUrl=${playlistUrl}`) - if (!playlistUrl) throw new Error(`failed to get playlistUrl from yt-dlp -g ${randomRoom.url}`); - - let debugCounter = 0 - // let ffmpegLogStream = createWriteStream('/tmp/ffmpeg-log.txt') - let uploadStream = new PassThrough() - uploadStream.on('data', (data) => { - debugCounter += data.length - if (debugCounter % (1 * 1024 * 1024) < 1024) { - console.log(`Received ${debugCounter} bytes (${prettyBytes(debugCounter)}) [${debugCounter % (1*1024*1024)}]`); - } - }) - // uploadStream.on('drain', () => { - // console.log('[drain] uploadStream') - // }) - uploadStream.on('close', () => { - console.log(`[close] uploadStream closed`) - }) - uploadStream.on('error', (err) => { - console.error('[error] uploadStream had an error as follows') - console.error(err) - }) - uploadStream.on('exit', (code) => { - console.log(`[exit] uploadStream exited with code ${code}`) - }) - uploadStream.on('disconnect', () => { - console.log('[disconnect] uploadStream disconnected') - }) - uploadStream.on('message', (msg) => { - console.log('[message] uploadStream sent a message as follows') - console.log(msg) - }) - - - const datestamp = new Date().toISOString() - - - - const ffmpegProc = spawn('ffmpeg', [ - '-headers', `"User-Agent: ${ua0}"`, - '-i', playlistUrl, - '-c:v', 'copy', - '-c:a', 'copy', - '-movflags', 'faststart', - '-y', - '-f', 'mpegts', - '-loglevel', 'quiet', - 'pipe:1' - ], { - // ignoring stderr is important because if not, ffmpeg will fill that buffer and node will hang - stdio: ['pipe', 'pipe', 'ignore'] - }) - - console.log('the following is the ffmpegProc.stdout') - console.log(ffmpegProc.stdout.constructor.name) - - // we set up a pipeline which has an readable stream (ffmpeg), a transform stream (debug), and a writable stream (s3 Upload) - pipeline( - ffmpegProc.stdout, - uploadStream, - (err) => { - if (err) { - console.error(`pipeline errored.`) - console.error(err) - } else { - console.log('pipeline succeeded.') - } - } - ) - - // Create a stream to the S3 bucket. We use this stream to upload the livestream to Backblaze S3 service - const keyName = `${datestamp}-${randomRoom.name}-chaturbate-${createId()}.ts` - console.log(`keyName=${keyName}`) - const target = { - Bucket: process.env.S3_BUCKET_NAME, - Key: keyName, - Body: uploadStream - } - // greets https://stackoverflow.com/a/70159394/1004931 - try { - const parallelUploads3 = new Upload({ - client: client, - //tags: [...], // optional tags - queueSize: 4, // optional concurrency configuration - leavePartsOnError: false, // optional manually handle dropped parts - params: target, - }); - - await parallelUploads3.done(); - - } catch (e) { - if (e instanceof Error) { - console.error(`while uploading a file to s3, we encountered an error`) - throw new Error(e.message); - } else { - throw new Error(`error of some sort ${JSON.stringify(e, null, 2)}`) - } - } - - - - -} -main() - - diff --git a/services/capture/src/record.js.old b/services/capture/src/record.js.old deleted file mode 100644 index 5c799eb..0000000 --- a/services/capture/src/record.js.old +++ /dev/null @@ -1,117 +0,0 @@ -import { join } from 'path'; -import { spawn } from 'child_process'; -import fs from 'node:fs'; - -export const getFilename = (appContext, roomName) => { - const name = `${roomName}_${new Date().toISOString()}.ts` - return join(appContext.env.FUTUREPORN_WORKDIR, 'recordings', name); -} - - -export const assertDirectory = (directoryPath) => { - if (fs.statSync(directoryPath, { throwIfNoEntry: false }) === undefined) fs.mkdirSync(directoryPath); -} - -export const checkFFmpeg = async (appContext) => { - return new Promise((resolve, reject) => { - const childProcess = spawn('ffmpeg', ['-version']); - - childProcess.on('error', (err) => { - appContext.logger.log({ - level: 'error', - message: `ffmpeg -version failed, which likely means ffmpeg is not installed or not on $PATH`, - }); - throw new Error('ffmpeg is missing') - }); - - childProcess.on('exit', (code) => { - if (code !== 0) reject(`'ffmpeg -version' exited with code ${code}`) - if (code === 0) { - appContext.logger.log({ level: 'info', message: `ffmpeg PRESENT.` }); - resolve() - } - }); - }) -}; - -export const assertDependencyDirectory = (appContext) => { - // Extract the directory path from the filename - const directoryPath = join(appContext.env.FUTUREPORN_WORKDIR, 'recordings'); - console.log(`asserting ${directoryPath} exists`) - - // Check if the directory exists, and create it if it doesn't - if (!fs.existsSync(directoryPath)) { - fs.mkdirSync(directoryPath, { recursive: true }); - console.log(`Created directory: ${directoryPath}`); - } -} - -export const record = async (appContext, playlistUrl, roomName) => { - if (appContext === undefined) throw new Error('appContext undef'); - if (playlistUrl === undefined) throw new Error('playlistUrl undef'); - if (roomName === undefined) throw new Error('roomName undef'); - - const filename = getFilename(appContext, roomName); - console.log(`downloading to ${filename}`) - - // example: `ffmpeg -headers "User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:105.0) Gecko/20100101 Firefox/105.0" - // -i ${chunkPlaylist} - // -c:v copy - // -c:a copy - // -movflags faststart - // -y - // -f mpegts - // ./my-recording.ts` - const ffmpegProcess = spawn('ffmpeg', [ - '-headers', `"User-Agent: ${appContext.env.DOWNLOADER_UA}"`, - '-i', playlistUrl, - '-c:v', 'copy', - '-c:a', 'copy', - '-movflags', 'faststart', - '-y', - '-f', 'mpegts', - filename - ], { - stdio: 'inherit' - }); - - - - return new Promise((resolve, reject) => { - ffmpegProcess.once('exit', (code) => { - resolve(code) - }) - }) - - // ffmpegProcess.on('data', (data) => { - // console.log(data.toString()); - // }); - - - // Optional: Handle other events such as 'error', 'close', etc. - // @todo this needs to be handled outside this function - // otherwise this function is not testable - // ffmpegProcess.on('exit', (code, signal) => { - // // Retry the download using exponential backoff if the process exits for any reason - // console.log(`ffmpeg exited with code ${code} and signal ${signal}`) - // retryDownload(appContext, playlistUrl, roomName); - // }); - - // return ffmpegProcess; -} - - -const calculateExponentialBackoffDelay = (attemptNumber) => { - return Math.pow(2, attemptNumber) * 1000; -}; - -const retryDownload = (appContext, playlistUrl, roomName, attemptNumber = 1, maxAttempts = 3) => { - const delay = calculateExponentialBackoffDelay(attemptNumber); - - appContext.logger.log({ level: 'debug', message: `Retrying download in ${delay / 1000} seconds...` }); - - setTimeout(() => { - console.log('Retrying download...'); - record(appContext, playlistUrl, roomName, attemptNumber + 1); - }, delay); -}; diff --git a/services/capture/src/s3.ts b/services/capture/src/s3.ts deleted file mode 100644 index e649b76..0000000 --- a/services/capture/src/s3.ts +++ /dev/null @@ -1,90 +0,0 @@ -import { S3Client, HeadObjectCommand } from "@aws-sdk/client-s3" -import { Upload } from "@aws-sdk/lib-storage" -import { basename } from 'node:path' -import fs from 'node:fs' - -if (!process.env.S3_BUCKET_NAME) throw new Error('S3_BUCKET_NAME was undefined in env'); -if (!process.env.S3_BUCKET_KEY_ID) throw new Error('S3_BUCKET_KEY_ID was undefined in env'); -if (!process.env.S3_BUCKET_APPLICATION_KEY) throw new Error('S3_BUCKET_APPLICATION_KEY was undefined in env'); - - - -export async function uploadFile(filePath: string) { - if (!filePath) throw new Error("first argument, 'filePath' is undefined"); - - const fileName = basename(filePath) - let isFileAlreadyUploaded = false - - - const client = new S3Client({ - endpoint: 'https://s3.us-west-000.backblazeb2.com', - region: 'us-west-000', - credentials: { - accessKeyId: process.env.S3_BUCKET_KEY_ID!, - secretAccessKey: process.env.S3_BUCKET_APPLICATION_KEY! - } - }); - - // try { - // console.log(`> Checking to see if ${fileName} exists in the S3 bucket`) - // const input = { - // "Bucket": process.env.S3_BUCKET_NAME, - // "Key": fileName - // }; - - // const command = new HeadObjectCommand(input); - // const response = await client.send(command); - - // if (response.ContentLength && response.ContentLength > 0) isFileAlreadyUploaded = true; - - // // console.log('does this following file exist?') - - // return { Key: fileName } - // } catch (e) { - // console.log(`> failed to get ${fileName} which means it probably doesnt exist in the bucket`) - // console.log(e) - // } - - - console.log('> Uploading file') - - const target = { - Bucket: process.env.S3_BUCKET_NAME, - Key: fileName, - Body: fs.createReadStream(filePath) - } - - // greets https://stackoverflow.com/a/70159394/1004931 - try { - const parallelUploads3 = new Upload({ - client: client, - //tags: [...], // optional tags - queueSize: 4, // optional concurrency configuration - leavePartsOnError: false, // optional manually handle dropped parts - params: target, - }); - - // parallelUploads3.on("httpUploadProgress", (progress) => { - // console.log(progress); - // }); - - const res = await parallelUploads3.done(); - return res - - } catch (e) { - if (e instanceof Error) { - console.error(`while uploading a file to s3, we encountered an error`) - throw new Error(e.message); - } else { - throw new Error(`error of some sort ${JSON.stringify(e, null, 2)}`) - } - } - - - - -} - - - - diff --git a/services/capture/src/sample.ts b/services/capture/src/sample.ts deleted file mode 100644 index 3028cee..0000000 --- a/services/capture/src/sample.ts +++ /dev/null @@ -1,28 +0,0 @@ -import { setInterval, setTimeout } from 'node:timers/promises'; - -async function task() { - console.log('Task is running:', new Date().toISOString()); - await setTimeout(1000); // Simulating async work with 1 second delay - console.log('Task completed:', new Date().toISOString()); -} - -async function main() { - console.log('Main function started'); - - // Run the async function on an interval - (async () => { - for await (const _ of setInterval(5000)) { // 5 seconds interval - task(); // Running async task without waiting for it to complete - } - })(); - - // Non-blocking logic in the main function with a for loop - for (let i = 1; i <= 5; i++) { - console.log(`Main function loop iteration ${i}`); - await setTimeout(2000); // Simulate async work (non-blocking) with 2 seconds delay - } - - console.log('Main function loop completed'); -} - -main(); diff --git a/services/capture/src/server.ts.old b/services/capture/src/server.ts.old deleted file mode 100755 index a2c9f4c..0000000 --- a/services/capture/src/server.ts.old +++ /dev/null @@ -1,154 +0,0 @@ -#!/usr/bin/env node - -// import Capture from './src/Capture.js' -// import Video from './src/Video.js' - -import dotenv from 'dotenv' -dotenv.config() -import { createId } from '@paralleldrive/cuid2' -import os from 'os' -import fs from 'node:fs' -import { loggerFactory } from "./logger.js" -import { verifyStorage } from './disk.js' -import { record, assertDependencyDirectory, checkFFmpeg } from './record.js' -import fastq from 'fastq' -import pRetry from 'p-retry'; -import Fastify from 'fastify'; - - - -// Create a map to store the work queues -const workQueues = new Map(); - - - -async function captureTask (args, cb) { - const { appContext, playlistUrl, roomName } = args; - - try { - const downloadStream = async () => { - const rc = await record(appContext, playlistUrl, roomName) - if (rc !== 0) throw new Error('ffmpeg exited irregularly (return code was other than zero)') - } - await pRetry(downloadStream, { - retries: 3, - onFailedAttempt: error => { - appContext.logger.log({ level: 'error', message: `downloadStream attempt ${error.attemptNumber} failed. There are ${error.retriesLeft} retries left.` }); - }, - }) - } catch (e) { - // we can get here if all retries are exhausted. - // this could be that the stream is over, the playlistUrl might be different, etc. - // we might have queued tasks so we don't want to crash. - appContext.logger.log({ level: 'error', message: `downloadStream exhausted all retries.` }) - appContext.logger.log({ level: 'error', message: e }) - } - - verifyStorage(appContext) - - appContext.logger.log({ level: 'info', message: 'Capture task complete'}) - cb(null, null) -} - -/** - * - * Fastify is used to facilitate Docker health checks - * - */ -async function initFastify(appContext) { - appContext.fastify = Fastify({ - logger: true - }) - - // Declare a route - appContext.fastify.get('/health', function (_, reply) { - reply.send({ message: 'futureporn-capture sneed' }); - }) - - // Run the server! - appContext.fastify.listen({ port: appContext.env.PORT }, function (err, address) { - if (err) { - appContext.fastify.log.error(err) - process.exit(1) - } - }) -} - -async function init () { - - const appEnv = new Array( - 'FUTUREPORN_WORKDIR', - 'DOWNLOADER_UA', - 'PORT' - ) - - const logger = loggerFactory({ - service: 'futureporn/capture' - }) - - - const appContext = { - env: appEnv.reduce((acc, ev) => { - if (typeof process.env[ev] === 'undefined') throw new Error(`${ev} is undefined in env`); - acc[ev] = process.env[ev]; - return acc; - }, {}), - logger, - pkg: JSON.parse(fs.readFileSync('./package.json', { encoding: 'utf-8'})), - workerId: `${os.hostname}-${createId()}`, - }; - - await initFastify(appContext); - - assertDependencyDirectory(appContext) - await checkFFmpeg(appContext) - verifyStorage(appContext) - - return appContext -} - - -async function main () { - - const appContext = await init() - - - - appContext.logger.log({ level: 'info', message: `capture version: ${appContext.pkg.version}` }) - appContext.logger.log({ level: 'info', message: `my capture directory is ${appContext.env.FUTUREPORN_WORKDIR}` }) - - - // connect to realtime server - appContext.pubsub.subscribe('/signals', (message) => { - appContext.logger.log({ level: 'debug', message: JSON.stringify(message) }) - - if ( - (message?.signal === 'start') && - (message?.room) && - (message?.url.startsWith('https://')) - ) { - - const roomName = message.room; - const playlistUrl = message.url; - - // Check if a work queue for the room already exists, otherwise create a new one - if (!workQueues.has(roomName)) { - workQueues.set(roomName, fastq(captureTask, 1)); - } - - - // Push the task to the corresponding work queue - workQueues.get(roomName).push({ appContext, playlistUrl, roomName }); - } - - }) - - - -} - -main() - - - - diff --git a/services/capture/src/tasks/record.ts b/services/capture/src/tasks/record.ts deleted file mode 100644 index 3c5c47b..0000000 --- a/services/capture/src/tasks/record.ts +++ /dev/null @@ -1,73 +0,0 @@ - - -import { Helpers, type Task } from 'graphile-worker' -import RecordNextGeneration from '../RecordNextGeneration.ts' -import createVod from '@futureporn/fetchers/createVod.ts' -import findOrCreateStream from '@futureporn/fetchers/findOrCreateStream.ts' -import findOrCreateVtuber from '@futureporn/fetchers/findOrCreateVtuber.ts' -import getRecording from '@futureporn/fetchers/getRecording.ts' -import patchRecording from '@futureporn/fetchers/patchRecording.ts' - - -interface Payload { - recording_id: string; - // url: string; - // discord_message_id?: string; - // vod_id?: string; - // date?: string; -} - - - -function assertPayload(payload: any): asserts payload is Payload { - if (typeof payload !== "object" || !payload) throw new Error("invalid payload"); - if (typeof payload.recording_id !== "string") throw new Error("invalid recording_id"); -} - -async function assertVod(payload: Payload) { - - let { recording_id } = payload - const recording = await getRecording(recording_id) - if (!recording) throw new Error(`failed to getRecording id=${recording_id}`); - let { url, vod_id, date } = recording - if (vod_id) return { vodId: vod_id }; - if (!date) date = new Date().toISOString(); - - const vtuberId = await findOrCreateVtuber({ url }) - const streamId = await findOrCreateStream({ vtuberId, date: new Date(date) }) - if (!streamId) throw new Error(`failed to find or create a Stream in database`); - const vod = await createVod({ stream_id: streamId, vtuber: vtuberId, url, date, recording_id }) - if (!vod) throw new Error('failed to createVod. please try again.') - - return { vodId: vod.id } -} - - - - - -export const record: Task = async function (payload: unknown, helpers: Helpers) { - - - assertPayload(payload) - const { recording_id: recordingId } = payload - const recording = await getRecording(recordingId) - if (!recording) throw new Error(`failed to getRecording() ${recordingId}`); - const { url } = recording - const { vodId } = await assertVod(payload) - // await patchRecording(recording.id, { vod_id: vodId }) - - /** - * RecordNextGeneration handles errors for us and re-throws ones that should fail the Task. - * We intentionally do not use a try/catch block here. - */ - const recordNG = new RecordNextGeneration({ url, vodId }) - await recordNG.done() - - - - return; -} - - -export default record \ No newline at end of file diff --git a/services/capture/tsconfig.json b/services/capture/tsconfig.json deleted file mode 100644 index e93bad7..0000000 --- a/services/capture/tsconfig.json +++ /dev/null @@ -1,32 +0,0 @@ -{ - "compilerOptions": { - // Base Options recommended for all projects - "allowImportingTsExtensions": true, - "noEmit": true, // tsup does the emissions - "esModuleInterop": true, - "skipLibCheck": true, - "target": "ESNext", - "allowJs": true, - "moduleResolution": "Bundler", - "resolveJsonModule": true, - "moduleDetection": "force", - "isolatedModules": true, - // Enable strict type checking so you can catch bugs early - "strict": true, - "noUncheckedIndexedAccess": true, - "noImplicitOverride": true, - // Transpile our TypeScript code to JavaScript - "module": "ESNext", - "outDir": "dist", - "lib": [ - "ESNext" - ] - }, - // Include the necessary files for your project - "include": [ - "src/**/*.ts" -, "../bot/src/tasks/restart_failed_recordings.ts" ], - "exclude": [ - "node_modules" - ] -} \ No newline at end of file diff --git a/services/capture/tsup.config.ts b/services/capture/tsup.config.ts deleted file mode 100644 index ed5ff52..0000000 --- a/services/capture/tsup.config.ts +++ /dev/null @@ -1,30 +0,0 @@ -import { defineConfig } from "tsup"; -import { exec } from 'child_process'; - -export default defineConfig({ - entry: [ - "src/index.ts", - "src/tasks/**.ts" - ], - format: ["esm"], - target: "node20", - clean: true, - sourcemap: true, - /** - * These packages are using the internal packages approach, so it needs to - * be transpiled / bundled together with the deployed code. - */ - noExternal: [ - "@futureporn/utils", - "@futureporn/scout", - "@futureporn/types", - ], - /** - * We do not use tsup for generating d.ts files because it can not generate type - * the definition maps required for go-to-definition to work in our IDE. We - * use tsc for that. - */ - onSuccess: async () => { - exec('tsc --emitDeclarationOnly'); - }, -}); \ No newline at end of file diff --git a/services/factory/.mocharc.json b/services/factory/.mocharc.json deleted file mode 100644 index 8c76e02..0000000 --- a/services/factory/.mocharc.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "$schema": "https://json.schemastore.org/mocharc.json", - "extensions": ["ts"], - "spec": "./src/**/*.spec.ts", - "require": "tsx" -} \ No newline at end of file diff --git a/services/factory/README.md b/services/factory/README.md deleted file mode 100644 index a0f4bc7..0000000 --- a/services/factory/README.md +++ /dev/null @@ -1,17 +0,0 @@ -# @futureporn/factory - -Factory takes raw materials (video segments) and produces an end product (encoded video, thumbnail) - -factory has a big disk and lots of RAM in order to do transcoding tasks - - - -## 240p encodes - - ffmpeg -i ./pmel-cb-2023-03-04.mp4 -vf scale=-2:240 -b:v 368k -b:a 45k ./projektmelody-chaturbatep2023-03-04_240p.mp4 - - -## potential helpful tools - -* https://github.com/superstreamerapp/superstreamer -* https://github.com/StreamPot/StreamPot diff --git a/services/factory/crontab b/services/factory/crontab deleted file mode 100644 index 79439f5..0000000 --- a/services/factory/crontab +++ /dev/null @@ -1,17 +0,0 @@ -# @see https://worker.graphile.org/docs/cron -# -# ┌───────────── UTC minute (0 - 59) -# │ ┌───────────── UTC hour (0 - 23) -# │ │ ┌───────────── UTC day of the month (1 - 31) -# │ │ │ ┌───────────── UTC month (1 - 12) -# │ │ │ │ ┌───────────── UTC day of the week (0 - 6) (Sunday to Saturday) -# │ │ │ │ │ ┌───────────── task (identifier) to schedule -# │ │ │ │ │ │ ┌────────── optional scheduling options -# │ │ │ │ │ │ │ ┌────── optional payload to merge -# │ │ │ │ │ │ │ │ -# │ │ │ │ │ │ │ │ -# * * * * * task ?opts {payload} - - -## every n minutes, we update the patrons table to contain patron names who have opted-in to the Website Shoutout perk -*/1 * * * * synchronize_patrons_list ?max=1 \ No newline at end of file diff --git a/services/factory/package.json b/services/factory/package.json deleted file mode 100644 index 8a3c195..0000000 --- a/services/factory/package.json +++ /dev/null @@ -1,47 +0,0 @@ -{ - "name": "@futureporn/factory", - "type": "module", - "version": "2.0.0", - "description": "", - "main": "src/index.ts", - "scripts": { - "test": "mocha", - "dev": "pnpm run dev.nodemon # yes this is crazy to have nodemon execute tsx, but it's the only way I have found to get live reloading in TS/ESM/docker with Graphile Worker's way of loading tasks", - "dev.tsx": "tsx ./src/index.ts", - "dev.nodemon": "nodemon --verbose --exitcrash --ext ts --exec \"pnpm run dev.tsx\"", - "dev.node": "node --no-warnings=ExperimentalWarning --loader ts-node/esm src/index.ts" - }, - "keywords": [ - "transcode", - "transcoding", - "process", - "processing", - "build", - "factory" - ], - "author": "@cj_clippy", - "license": "Unlicense", - "dependencies": { - "@aws-sdk/client-s3": "^3.637.0", - "@aws-sdk/lib-storage": "^3.637.0", - "@futureporn/fetchers": "workspace:^", - "@futureporn/storage": "workspace:^", - "@futureporn/utils": "workspace:^", - "@paralleldrive/cuid2": "^2.2.2", - "@types/chai": "^4.3.19", - "@types/mocha": "^10.0.7", - "@types/node": "^22.5.2", - "chai": "^5.1.1", - "dotenv": "^16.4.5", - "graphile-config": "0.0.1-beta.9", - "graphile-worker": "^0.16.6", - "p-retry": "^6.2.0" - }, - "devDependencies": { - "@futureporn/types": "workspace:^", - "mocha": "^10.7.3", - "nodemon": "^3.1.4", - "ts-node": "^10.9.2", - "tsx": "^4.19.0" - } -} diff --git a/services/factory/pnpm-lock.yaml b/services/factory/pnpm-lock.yaml deleted file mode 100644 index 4117d98..0000000 --- a/services/factory/pnpm-lock.yaml +++ /dev/null @@ -1,2974 +0,0 @@ -lockfileVersion: '9.0' - -settings: - autoInstallPeers: true - excludeLinksFromLockfile: false - -importers: - - .: - dependencies: - '@aws-sdk/client-s3': - specifier: ^3.637.0 - version: 3.637.0 - '@aws-sdk/lib-storage': - specifier: ^3.637.0 - version: 3.637.0(@aws-sdk/client-s3@3.637.0) - '@futureporn/fetchers': - specifier: workspace:^ - version: link:../../packages/fetchers - '@futureporn/storage': - specifier: workspace:^ - version: link:../../packages/storage - '@futureporn/utils': - specifier: workspace:^ - version: link:../../packages/utils - '@paralleldrive/cuid2': - specifier: ^2.2.2 - version: 2.2.2 - '@types/chai': - specifier: ^4.3.19 - version: 4.3.20 - '@types/mocha': - specifier: ^10.0.7 - version: 10.0.9 - '@types/node': - specifier: ^22.5.2 - version: 22.5.2 - chai: - specifier: ^5.1.1 - version: 5.1.2 - dotenv: - specifier: ^16.4.5 - version: 16.4.5 - graphile-config: - specifier: 0.0.1-beta.9 - version: 0.0.1-beta.9 - graphile-worker: - specifier: ^0.16.6 - version: 0.16.6(typescript@5.5.4) - p-retry: - specifier: ^6.2.0 - version: 6.2.0 - devDependencies: - '@futureporn/types': - specifier: workspace:^ - version: link:../../packages/types - mocha: - specifier: ^10.7.3 - version: 10.7.3 - nodemon: - specifier: ^3.1.4 - version: 3.1.4 - ts-node: - specifier: ^10.9.2 - version: 10.9.2(@types/node@22.5.2)(typescript@5.5.4) - tsx: - specifier: ^4.19.0 - version: 4.19.0 - -packages: - - '@aws-crypto/crc32@5.2.0': - resolution: {integrity: sha512-nLbCWqQNgUiwwtFsen1AdzAtvuLRsQS8rYgMuxCrdKf9kOssamGLuPwyTY9wyYblNr9+1XM8v6zoDTPPSIeANg==} - engines: {node: '>=16.0.0'} - - '@aws-crypto/crc32c@5.2.0': - resolution: {integrity: sha512-+iWb8qaHLYKrNvGRbiYRHSdKRWhto5XlZUEBwDjYNf+ly5SVYG6zEoYIdxvf5R3zyeP16w4PLBn3rH1xc74Rag==} - - '@aws-crypto/sha1-browser@5.2.0': - resolution: {integrity: sha512-OH6lveCFfcDjX4dbAvCFSYUjJZjDr/3XJ3xHtjn3Oj5b9RjojQo8npoLeA/bNwkOkrSQ0wgrHzXk4tDRxGKJeg==} - - '@aws-crypto/sha256-browser@5.2.0': - resolution: {integrity: sha512-AXfN/lGotSQwu6HNcEsIASo7kWXZ5HYWvfOmSNKDsEqC4OashTp8alTmaz+F7TC2L083SFv5RdB+qU3Vs1kZqw==} - - '@aws-crypto/sha256-js@5.2.0': - resolution: {integrity: sha512-FFQQyu7edu4ufvIZ+OadFpHHOt+eSTBaYaki44c+akjg7qZg9oOQeLlk77F6tSYqjDAFClrHJk9tMf0HdVyOvA==} - engines: {node: '>=16.0.0'} - - '@aws-crypto/supports-web-crypto@5.2.0': - resolution: {integrity: sha512-iAvUotm021kM33eCdNfwIN//F77/IADDSs58i+MDaOqFrVjZo9bAal0NK7HurRuWLLpF1iLX7gbWrjHjeo+YFg==} - - '@aws-crypto/util@5.2.0': - resolution: {integrity: sha512-4RkU9EsI6ZpBve5fseQlGNUWKMa1RLPQ1dnjnQoe07ldfIzcsGb5hC5W0Dm7u423KWzawlrpbjXBrXCEv9zazQ==} - - '@aws-sdk/client-s3@3.637.0': - resolution: {integrity: sha512-y6UC94fsMvhKbf0dzfnjVP1HePeGjplfcYfilZU1COIJLyTkMcUv4XcT4I407CGIrvgEafONHkiC09ygqUauNA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/client-sso-oidc@3.637.0': - resolution: {integrity: sha512-27bHALN6Qb6m6KZmPvRieJ/QRlj1lyac/GT2Rn5kJpre8Mpp+yxrtvp3h9PjNBty4lCeFEENfY4dGNSozBuBcw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.637.0 - - '@aws-sdk/client-sso@3.637.0': - resolution: {integrity: sha512-+KjLvgX5yJYROWo3TQuwBJlHCY0zz9PsLuEolmXQn0BVK1L/m9GteZHtd+rEdAoDGBpE0Xqjy1oz5+SmtsaRUw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/client-sts@3.637.0': - resolution: {integrity: sha512-xUi7x4qDubtA8QREtlblPuAcn91GS/09YVEY/RwU7xCY0aqGuFwgszAANlha4OUIqva8oVj2WO4gJuG+iaSnhw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/core@3.635.0': - resolution: {integrity: sha512-i1x/E/sgA+liUE1XJ7rj1dhyXpAKO1UKFUcTTHXok2ARjWTvszHnSXMOsB77aPbmn0fUp1JTx2kHUAZ1LVt5Bg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-env@3.620.1': - resolution: {integrity: sha512-ExuILJ2qLW5ZO+rgkNRj0xiAipKT16Rk77buvPP8csR7kkCflT/gXTyzRe/uzIiETTxM7tr8xuO9MP/DQXqkfg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-http@3.635.0': - resolution: {integrity: sha512-iJyRgEjOCQlBMXqtwPLIKYc7Bsc6nqjrZybdMDenPDa+kmLg7xh8LxHsu9088e+2/wtLicE34FsJJIfzu3L82g==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-ini@3.637.0': - resolution: {integrity: sha512-h+PFCWfZ0Q3Dx84SppET/TFpcQHmxFW8/oV9ArEvMilw4EBN+IlxgbL0CnHwjHW64szcmrM0mbebjEfHf4FXmw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.637.0 - - '@aws-sdk/credential-provider-node@3.637.0': - resolution: {integrity: sha512-yoEhoxJJfs7sPVQ6Is939BDQJZpZCoUgKr/ySse4YKOZ24t4VqgHA6+wV7rYh+7IW24Rd91UTvEzSuHYTlxlNA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-process@3.620.1': - resolution: {integrity: sha512-hWqFMidqLAkaV9G460+1at6qa9vySbjQKKc04p59OT7lZ5cO5VH5S4aI05e+m4j364MBROjjk2ugNvfNf/8ILg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-sso@3.637.0': - resolution: {integrity: sha512-Mvz+h+e62/tl+dVikLafhv+qkZJ9RUb8l2YN/LeKMWkxQylPT83CPk9aimVhCV89zth1zpREArl97+3xsfgQvA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-web-identity@3.621.0': - resolution: {integrity: sha512-w7ASSyfNvcx7+bYGep3VBgC3K6vEdLmlpjT7nSIHxxQf+WSdvy+HynwJosrpZax0sK5q0D1Jpn/5q+r5lwwW6w==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.621.0 - - '@aws-sdk/lib-storage@3.637.0': - resolution: {integrity: sha512-HiNGOP4a1QrCWwO1joKw4mCp19nLXoF9K52PislBaYDI35IlHC3DP6MeOg5zmElwtL1GtEHFBy5olfPWPsLyLg==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-s3': ^3.637.0 - - '@aws-sdk/middleware-bucket-endpoint@3.620.0': - resolution: {integrity: sha512-eGLL0W6L3HDb3OACyetZYOWpHJ+gLo0TehQKeQyy2G8vTYXqNTeqYhuI6up9HVjBzU9eQiULVQETmgQs7TFaRg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-expect-continue@3.620.0': - resolution: {integrity: sha512-QXeRFMLfyQ31nAHLbiTLtk0oHzG9QLMaof5jIfqcUwnOkO8YnQdeqzakrg1Alpy/VQ7aqzIi8qypkBe2KXZz0A==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-flexible-checksums@3.620.0': - resolution: {integrity: sha512-ftz+NW7qka2sVuwnnO1IzBku5ccP+s5qZGeRTPgrKB7OzRW85gthvIo1vQR2w+OwHFk7WJbbhhWwbCbktnP4UA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-host-header@3.620.0': - resolution: {integrity: sha512-VMtPEZwqYrII/oUkffYsNWY9PZ9xpNJpMgmyU0rlDQ25O1c0Hk3fJmZRe6pEkAJ0omD7kLrqGl1DUjQVxpd/Rg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-location-constraint@3.609.0': - resolution: {integrity: sha512-xzsdoTkszGVqGVPjUmgoP7TORiByLueMHieI1fhQL888WPdqctwAx3ES6d/bA9Q/i8jnc6hs+Fjhy8UvBTkE9A==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-logger@3.609.0': - resolution: {integrity: sha512-S62U2dy4jMDhDFDK5gZ4VxFdWzCtLzwbYyFZx2uvPYTECkepLUfzLic2BHg2Qvtu4QjX+oGE3P/7fwaGIsGNuQ==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-recursion-detection@3.620.0': - resolution: {integrity: sha512-nh91S7aGK3e/o1ck64sA/CyoFw+gAYj2BDOnoNa6ouyCrVJED96ZXWbhye/fz9SgmNUZR2g7GdVpiLpMKZoI5w==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-sdk-s3@3.635.0': - resolution: {integrity: sha512-RLdYJPEV4JL/7NBoFUs7VlP90X++5FlJdxHz0DzCjmiD3qCviKy+Cym3qg1gBgHwucs5XisuClxDrGokhAdTQw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-ssec@3.609.0': - resolution: {integrity: sha512-GZSD1s7+JswWOTamVap79QiDaIV7byJFssBW68GYjyRS5EBjNfwA/8s+6uE6g39R3ojyTbYOmvcANoZEhSULXg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-user-agent@3.637.0': - resolution: {integrity: sha512-EYo0NE9/da/OY8STDsK2LvM4kNa79DBsf4YVtaG4P5pZ615IeFsD8xOHZeuJmUrSMlVQ8ywPRX7WMucUybsKug==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/region-config-resolver@3.614.0': - resolution: {integrity: sha512-vDCeMXvic/LU0KFIUjpC3RiSTIkkvESsEfbVHiHH0YINfl8HnEqR5rj+L8+phsCeVg2+LmYwYxd5NRz4PHxt5g==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/signature-v4-multi-region@3.635.0': - resolution: {integrity: sha512-J6QY4/invOkpogCHjSaDON1hF03viPpOnsrzVuCvJMmclS/iG62R4EY0wq1alYll0YmSdmKlpJwHMWwGtqK63Q==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/token-providers@3.614.0': - resolution: {integrity: sha512-okItqyY6L9IHdxqs+Z116y5/nda7rHxLvROxtAJdLavWTYDydxrZstImNgGWTeVdmc0xX2gJCI77UYUTQWnhRw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sso-oidc': ^3.614.0 - - '@aws-sdk/types@3.609.0': - resolution: {integrity: sha512-+Tqnh9w0h2LcrUsdXyT1F8mNhXz+tVYBtP19LpeEGntmvHwa2XzvLUCWpoIAIVsHp5+HdB2X9Sn0KAtmbFXc2Q==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-arn-parser@3.568.0': - resolution: {integrity: sha512-XUKJWWo+KOB7fbnPP0+g/o5Ulku/X53t7i/h+sPHr5xxYTJJ9CYnbToo95mzxe7xWvkLrsNtJ8L+MnNn9INs2w==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-endpoints@3.637.0': - resolution: {integrity: sha512-pAqOKUHeVWHEXXDIp/qoMk/6jyxIb6GGjnK1/f8dKHtKIEs4tKsnnL563gceEvdad53OPXIt86uoevCcCzmBnw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-locate-window@3.568.0': - resolution: {integrity: sha512-3nh4TINkXYr+H41QaPelCceEB2FXP3fxp93YZXB/kqJvX0U9j0N0Uk45gvsjmEPzG8XxkPEeLIfT2I1M7A6Lig==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-user-agent-browser@3.609.0': - resolution: {integrity: sha512-fojPU+mNahzQ0YHYBsx0ZIhmMA96H+ZIZ665ObU9tl+SGdbLneVZVikGve+NmHTQwHzwkFsZYYnVKAkreJLAtA==} - - '@aws-sdk/util-user-agent-node@3.614.0': - resolution: {integrity: sha512-15ElZT88peoHnq5TEoEtZwoXTXRxNrk60TZNdpl/TUBJ5oNJ9Dqb5Z4ryb8ofN6nm9aFf59GVAerFDz8iUoHBA==} - engines: {node: '>=16.0.0'} - peerDependencies: - aws-crt: '>=1.0.0' - peerDependenciesMeta: - aws-crt: - optional: true - - '@aws-sdk/xml-builder@3.609.0': - resolution: {integrity: sha512-l9XxNcA4HX98rwCC2/KoiWcmEiRfZe4G+mYwDbCFT87JIMj6GBhLDkAzr/W8KAaA2IDr8Vc6J8fZPgVulxxfMA==} - engines: {node: '>=16.0.0'} - - '@babel/code-frame@7.24.7': - resolution: {integrity: sha512-BcYH1CVJBO9tvyIZ2jVeXgSIMvGZ2FDRvDdOIVQyuklNKSsx+eppDEBq/g47Ayw+RqNFE+URvOShmf+f/qwAlA==} - engines: {node: '>=6.9.0'} - - '@babel/helper-validator-identifier@7.24.7': - resolution: {integrity: sha512-rR+PBcQ1SMQDDyF6X0wxtG8QyLCgUB0eRAGguqRLfkCA87l7yAP7ehq8SNj96OOGTO8OBV70KhuFYcIkHXOg0w==} - engines: {node: '>=6.9.0'} - - '@babel/highlight@7.24.7': - resolution: {integrity: sha512-EStJpq4OuY8xYfhGVXngigBJRWxftKX9ksiGDnmlY3o7B/V7KIAc9X4oiK87uPJSc/vs5L869bem5fhZa8caZw==} - engines: {node: '>=6.9.0'} - - '@cspotcode/source-map-support@0.8.1': - resolution: {integrity: sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==} - engines: {node: '>=12'} - - '@esbuild/aix-ppc64@0.23.1': - resolution: {integrity: sha512-6VhYk1diRqrhBAqpJEdjASR/+WVRtfjpqKuNw11cLiaWpAT/Uu+nokB+UJnevzy/P9C/ty6AOe0dwueMrGh/iQ==} - engines: {node: '>=18'} - cpu: [ppc64] - os: [aix] - - '@esbuild/android-arm64@0.23.1': - resolution: {integrity: sha512-xw50ipykXcLstLeWH7WRdQuysJqejuAGPd30vd1i5zSyKK3WE+ijzHmLKxdiCMtH1pHz78rOg0BKSYOSB/2Khw==} - engines: {node: '>=18'} - cpu: [arm64] - os: [android] - - '@esbuild/android-arm@0.23.1': - resolution: {integrity: sha512-uz6/tEy2IFm9RYOyvKl88zdzZfwEfKZmnX9Cj1BHjeSGNuGLuMD1kR8y5bteYmwqKm1tj8m4cb/aKEorr6fHWQ==} - engines: {node: '>=18'} - cpu: [arm] - os: [android] - - '@esbuild/android-x64@0.23.1': - resolution: {integrity: sha512-nlN9B69St9BwUoB+jkyU090bru8L0NA3yFvAd7k8dNsVH8bi9a8cUAUSEcEEgTp2z3dbEDGJGfP6VUnkQnlReg==} - engines: {node: '>=18'} - cpu: [x64] - os: [android] - - '@esbuild/darwin-arm64@0.23.1': - resolution: {integrity: sha512-YsS2e3Wtgnw7Wq53XXBLcV6JhRsEq8hkfg91ESVadIrzr9wO6jJDMZnCQbHm1Guc5t/CdDiFSSfWP58FNuvT3Q==} - engines: {node: '>=18'} - cpu: [arm64] - os: [darwin] - - '@esbuild/darwin-x64@0.23.1': - resolution: {integrity: sha512-aClqdgTDVPSEGgoCS8QDG37Gu8yc9lTHNAQlsztQ6ENetKEO//b8y31MMu2ZaPbn4kVsIABzVLXYLhCGekGDqw==} - engines: {node: '>=18'} - cpu: [x64] - os: [darwin] - - '@esbuild/freebsd-arm64@0.23.1': - resolution: {integrity: sha512-h1k6yS8/pN/NHlMl5+v4XPfikhJulk4G+tKGFIOwURBSFzE8bixw1ebjluLOjfwtLqY0kewfjLSrO6tN2MgIhA==} - engines: {node: '>=18'} - cpu: [arm64] - os: [freebsd] - - '@esbuild/freebsd-x64@0.23.1': - resolution: {integrity: sha512-lK1eJeyk1ZX8UklqFd/3A60UuZ/6UVfGT2LuGo3Wp4/z7eRTRYY+0xOu2kpClP+vMTi9wKOfXi2vjUpO1Ro76g==} - engines: {node: '>=18'} - cpu: [x64] - os: [freebsd] - - '@esbuild/linux-arm64@0.23.1': - resolution: {integrity: sha512-/93bf2yxencYDnItMYV/v116zff6UyTjo4EtEQjUBeGiVpMmffDNUyD9UN2zV+V3LRV3/on4xdZ26NKzn6754g==} - engines: {node: '>=18'} - cpu: [arm64] - os: [linux] - - '@esbuild/linux-arm@0.23.1': - resolution: {integrity: sha512-CXXkzgn+dXAPs3WBwE+Kvnrf4WECwBdfjfeYHpMeVxWE0EceB6vhWGShs6wi0IYEqMSIzdOF1XjQ/Mkm5d7ZdQ==} - engines: {node: '>=18'} - cpu: [arm] - os: [linux] - - '@esbuild/linux-ia32@0.23.1': - resolution: {integrity: sha512-VTN4EuOHwXEkXzX5nTvVY4s7E/Krz7COC8xkftbbKRYAl96vPiUssGkeMELQMOnLOJ8k3BY1+ZY52tttZnHcXQ==} - engines: {node: '>=18'} - cpu: [ia32] - os: [linux] - - '@esbuild/linux-loong64@0.23.1': - resolution: {integrity: sha512-Vx09LzEoBa5zDnieH8LSMRToj7ir/Jeq0Gu6qJ/1GcBq9GkfoEAoXvLiW1U9J1qE/Y/Oyaq33w5p2ZWrNNHNEw==} - engines: {node: '>=18'} - cpu: [loong64] - os: [linux] - - '@esbuild/linux-mips64el@0.23.1': - resolution: {integrity: sha512-nrFzzMQ7W4WRLNUOU5dlWAqa6yVeI0P78WKGUo7lg2HShq/yx+UYkeNSE0SSfSure0SqgnsxPvmAUu/vu0E+3Q==} - engines: {node: '>=18'} - cpu: [mips64el] - os: [linux] - - '@esbuild/linux-ppc64@0.23.1': - resolution: {integrity: sha512-dKN8fgVqd0vUIjxuJI6P/9SSSe/mB9rvA98CSH2sJnlZ/OCZWO1DJvxj8jvKTfYUdGfcq2dDxoKaC6bHuTlgcw==} - engines: {node: '>=18'} - cpu: [ppc64] - os: [linux] - - '@esbuild/linux-riscv64@0.23.1': - resolution: {integrity: sha512-5AV4Pzp80fhHL83JM6LoA6pTQVWgB1HovMBsLQ9OZWLDqVY8MVobBXNSmAJi//Csh6tcY7e7Lny2Hg1tElMjIA==} - engines: {node: '>=18'} - cpu: [riscv64] - os: [linux] - - '@esbuild/linux-s390x@0.23.1': - resolution: {integrity: sha512-9ygs73tuFCe6f6m/Tb+9LtYxWR4c9yg7zjt2cYkjDbDpV/xVn+68cQxMXCjUpYwEkze2RcU/rMnfIXNRFmSoDw==} - engines: {node: '>=18'} - cpu: [s390x] - os: [linux] - - '@esbuild/linux-x64@0.23.1': - resolution: {integrity: sha512-EV6+ovTsEXCPAp58g2dD68LxoP/wK5pRvgy0J/HxPGB009omFPv3Yet0HiaqvrIrgPTBuC6wCH1LTOY91EO5hQ==} - engines: {node: '>=18'} - cpu: [x64] - os: [linux] - - '@esbuild/netbsd-x64@0.23.1': - resolution: {integrity: sha512-aevEkCNu7KlPRpYLjwmdcuNz6bDFiE7Z8XC4CPqExjTvrHugh28QzUXVOZtiYghciKUacNktqxdpymplil1beA==} - engines: {node: '>=18'} - cpu: [x64] - os: [netbsd] - - '@esbuild/openbsd-arm64@0.23.1': - resolution: {integrity: sha512-3x37szhLexNA4bXhLrCC/LImN/YtWis6WXr1VESlfVtVeoFJBRINPJ3f0a/6LV8zpikqoUg4hyXw0sFBt5Cr+Q==} - engines: {node: '>=18'} - cpu: [arm64] - os: [openbsd] - - '@esbuild/openbsd-x64@0.23.1': - resolution: {integrity: sha512-aY2gMmKmPhxfU+0EdnN+XNtGbjfQgwZj43k8G3fyrDM/UdZww6xrWxmDkuz2eCZchqVeABjV5BpildOrUbBTqA==} - engines: {node: '>=18'} - cpu: [x64] - os: [openbsd] - - '@esbuild/sunos-x64@0.23.1': - resolution: {integrity: sha512-RBRT2gqEl0IKQABT4XTj78tpk9v7ehp+mazn2HbUeZl1YMdaGAQqhapjGTCe7uw7y0frDi4gS0uHzhvpFuI1sA==} - engines: {node: '>=18'} - cpu: [x64] - os: [sunos] - - '@esbuild/win32-arm64@0.23.1': - resolution: {integrity: sha512-4O+gPR5rEBe2FpKOVyiJ7wNDPA8nGzDuJ6gN4okSA1gEOYZ67N8JPk58tkWtdtPeLz7lBnY6I5L3jdsr3S+A6A==} - engines: {node: '>=18'} - cpu: [arm64] - os: [win32] - - '@esbuild/win32-ia32@0.23.1': - resolution: {integrity: sha512-BcaL0Vn6QwCwre3Y717nVHZbAa4UBEigzFm6VdsVdT/MbZ38xoj1X9HPkZhbmaBGUD1W8vxAfffbDe8bA6AKnQ==} - engines: {node: '>=18'} - cpu: [ia32] - os: [win32] - - '@esbuild/win32-x64@0.23.1': - resolution: {integrity: sha512-BHpFFeslkWrXWyUPnbKm+xYYVYruCinGcftSBaa8zoF9hZO4BcSCFUvHVTtzpIY6YzUnYtuEhZ+C9iEXjxnasg==} - engines: {node: '>=18'} - cpu: [x64] - os: [win32] - - '@graphile/logger@0.2.0': - resolution: {integrity: sha512-jjcWBokl9eb1gVJ85QmoaQ73CQ52xAaOCF29ukRbYNl6lY+ts0ErTaDYOBlejcbUs2OpaiqYLO5uDhyLFzWw4w==} - - '@jridgewell/resolve-uri@3.1.2': - resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==} - engines: {node: '>=6.0.0'} - - '@jridgewell/sourcemap-codec@1.5.0': - resolution: {integrity: sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==} - - '@jridgewell/trace-mapping@0.3.9': - resolution: {integrity: sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==} - - '@noble/hashes@1.5.0': - resolution: {integrity: sha512-1j6kQFb7QRru7eKN3ZDvRcP13rugwdxZqCjbiAVZfIJwgj2A65UmT4TgARXGlXgnRkORLTDTrO19ZErt7+QXgA==} - engines: {node: ^14.21.3 || >=16} - - '@paralleldrive/cuid2@2.2.2': - resolution: {integrity: sha512-ZOBkgDwEdoYVlSeRbYYXs0S9MejQofiVYoTbKzy/6GQa39/q5tQU2IX46+shYnUkpEl3wc+J6wRlar7r2EK2xA==} - - '@smithy/abort-controller@3.1.1': - resolution: {integrity: sha512-MBJBiidoe+0cTFhyxT8g+9g7CeVccLM0IOKKUMCNQ1CNMJ/eIfoo0RTfVrXOONEI1UCN1W+zkiHSbzUNE9dZtQ==} - engines: {node: '>=16.0.0'} - - '@smithy/chunked-blob-reader-native@3.0.0': - resolution: {integrity: sha512-VDkpCYW+peSuM4zJip5WDfqvg2Mo/e8yxOv3VF1m11y7B8KKMKVFtmZWDe36Fvk8rGuWrPZHHXZ7rR7uM5yWyg==} - - '@smithy/chunked-blob-reader@3.0.0': - resolution: {integrity: sha512-sbnURCwjF0gSToGlsBiAmd1lRCmSn72nu9axfJu5lIx6RUEgHu6GwTMbqCdhQSi0Pumcm5vFxsi9XWXb2mTaoA==} - - '@smithy/config-resolver@3.0.5': - resolution: {integrity: sha512-SkW5LxfkSI1bUC74OtfBbdz+grQXYiPYolyu8VfpLIjEoN/sHVBlLeGXMQ1vX4ejkgfv6sxVbQJ32yF2cl1veA==} - engines: {node: '>=16.0.0'} - - '@smithy/core@2.4.0': - resolution: {integrity: sha512-cHXq+FneIF/KJbt4q4pjN186+Jf4ZB0ZOqEaZMBhT79srEyGDDBV31NqBRBjazz8ppQ1bJbDJMY9ba5wKFV36w==} - engines: {node: '>=16.0.0'} - - '@smithy/credential-provider-imds@3.2.0': - resolution: {integrity: sha512-0SCIzgd8LYZ9EJxUjLXBmEKSZR/P/w6l7Rz/pab9culE/RWuqelAKGJvn5qUOl8BgX8Yj5HWM50A5hiB/RzsgA==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-codec@3.1.2': - resolution: {integrity: sha512-0mBcu49JWt4MXhrhRAlxASNy0IjDRFU+aWNDRal9OtUJvJNiwDuyKMUONSOjLjSCeGwZaE0wOErdqULer8r7yw==} - - '@smithy/eventstream-serde-browser@3.0.6': - resolution: {integrity: sha512-2hM54UWQUOrki4BtsUI1WzmD13/SeaqT/AB3EUJKbcver/WgKNaiJ5y5F5XXuVe6UekffVzuUDrBZVAA3AWRpQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-config-resolver@3.0.3': - resolution: {integrity: sha512-NVTYjOuYpGfrN/VbRQgn31x73KDLfCXCsFdad8DiIc3IcdxL+dYA9zEQPyOP7Fy2QL8CPy2WE4WCUD+ZsLNfaQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-node@3.0.5': - resolution: {integrity: sha512-+upXvnHNyZP095s11jF5dhGw/Ihzqwl5G+/KtMnoQOpdfC3B5HYCcDVG9EmgkhJMXJlM64PyN5gjJl0uXFQehQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-universal@3.0.5': - resolution: {integrity: sha512-5u/nXbyoh1s4QxrvNre9V6vfyoLWuiVvvd5TlZjGThIikc3G+uNiG9uOTCWweSRjv1asdDIWK7nOmN7le4RYHQ==} - engines: {node: '>=16.0.0'} - - '@smithy/fetch-http-handler@3.2.4': - resolution: {integrity: sha512-kBprh5Gs5h7ug4nBWZi1FZthdqSM+T7zMmsZxx0IBvWUn7dK3diz2SHn7Bs4dQGFDk8plDv375gzenDoNwrXjg==} - - '@smithy/hash-blob-browser@3.1.2': - resolution: {integrity: sha512-hAbfqN2UbISltakCC2TP0kx4LqXBttEv2MqSPE98gVuDFMf05lU+TpC41QtqGP3Ff5A3GwZMPfKnEy0VmEUpmg==} - - '@smithy/hash-node@3.0.3': - resolution: {integrity: sha512-2ctBXpPMG+B3BtWSGNnKELJ7SH9e4TNefJS0cd2eSkOOROeBnnVBnAy9LtJ8tY4vUEoe55N4CNPxzbWvR39iBw==} - engines: {node: '>=16.0.0'} - - '@smithy/hash-stream-node@3.1.2': - resolution: {integrity: sha512-PBgDMeEdDzi6JxKwbfBtwQG9eT9cVwsf0dZzLXoJF4sHKHs5HEo/3lJWpn6jibfJwT34I1EBXpBnZE8AxAft6g==} - engines: {node: '>=16.0.0'} - - '@smithy/invalid-dependency@3.0.3': - resolution: {integrity: sha512-ID1eL/zpDULmHJbflb864k72/SNOZCADRc9i7Exq3RUNJw6raWUSlFEQ+3PX3EYs++bTxZB2dE9mEHTQLv61tw==} - - '@smithy/is-array-buffer@2.2.0': - resolution: {integrity: sha512-GGP3O9QFD24uGeAXYUjwSTXARoqpZykHadOmA8G5vfJPK0/DC67qa//0qvqrJzL1xc8WQWX7/yc7fwudjPHPhA==} - engines: {node: '>=14.0.0'} - - '@smithy/is-array-buffer@3.0.0': - resolution: {integrity: sha512-+Fsu6Q6C4RSJiy81Y8eApjEB5gVtM+oFKTffg+jSuwtvomJJrhUJBu2zS8wjXSgH/g1MKEWrzyChTBe6clb5FQ==} - engines: {node: '>=16.0.0'} - - '@smithy/md5-js@3.0.3': - resolution: {integrity: sha512-O/SAkGVwpWmelpj/8yDtsaVe6sINHLB1q8YE/+ZQbDxIw3SRLbTZuRaI10K12sVoENdnHqzPp5i3/H+BcZ3m3Q==} - - '@smithy/middleware-content-length@3.0.5': - resolution: {integrity: sha512-ILEzC2eyxx6ncej3zZSwMpB5RJ0zuqH7eMptxC4KN3f+v9bqT8ohssKbhNR78k/2tWW+KS5Spw+tbPF4Ejyqvw==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-endpoint@3.1.0': - resolution: {integrity: sha512-5y5aiKCEwg9TDPB4yFE7H6tYvGFf1OJHNczeY10/EFF8Ir8jZbNntQJxMWNfeQjC1mxPsaQ6mR9cvQbf+0YeMw==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-retry@3.0.15': - resolution: {integrity: sha512-iTMedvNt1ApdvkaoE8aSDuwaoc+BhvHqttbA/FO4Ty+y/S5hW6Ci/CTScG7vam4RYJWZxdTElc3MEfHRVH6cgQ==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-serde@3.0.3': - resolution: {integrity: sha512-puUbyJQBcg9eSErFXjKNiGILJGtiqmuuNKEYNYfUD57fUl4i9+mfmThtQhvFXU0hCVG0iEJhvQUipUf+/SsFdA==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-stack@3.0.3': - resolution: {integrity: sha512-r4klY9nFudB0r9UdSMaGSyjyQK5adUyPnQN/ZM6M75phTxOdnc/AhpvGD1fQUvgmqjQEBGCwpnPbDm8pH5PapA==} - engines: {node: '>=16.0.0'} - - '@smithy/node-config-provider@3.1.4': - resolution: {integrity: sha512-YvnElQy8HR4vDcAjoy7Xkx9YT8xZP4cBXcbJSgm/kxmiQu08DwUwj8rkGnyoJTpfl/3xYHH+d8zE+eHqoDCSdQ==} - engines: {node: '>=16.0.0'} - - '@smithy/node-http-handler@3.1.4': - resolution: {integrity: sha512-+UmxgixgOr/yLsUxcEKGH0fMNVteJFGkmRltYFHnBMlogyFdpzn2CwqWmxOrfJELhV34v0WSlaqG1UtE1uXlJg==} - engines: {node: '>=16.0.0'} - - '@smithy/property-provider@3.1.3': - resolution: {integrity: sha512-zahyOVR9Q4PEoguJ/NrFP4O7SMAfYO1HLhB18M+q+Z4KFd4V2obiMnlVoUFzFLSPeVt1POyNWneHHrZaTMoc/g==} - engines: {node: '>=16.0.0'} - - '@smithy/protocol-http@4.1.0': - resolution: {integrity: sha512-dPVoHYQ2wcHooGXg3LQisa1hH0e4y0pAddPMeeUPipI1tEOqL6A4N0/G7abeq+K8wrwSgjk4C0wnD1XZpJm5aA==} - engines: {node: '>=16.0.0'} - - '@smithy/querystring-builder@3.0.3': - resolution: {integrity: sha512-vyWckeUeesFKzCDaRwWLUA1Xym9McaA6XpFfAK5qI9DKJ4M33ooQGqvM4J+LalH4u/Dq9nFiC8U6Qn1qi0+9zw==} - engines: {node: '>=16.0.0'} - - '@smithy/querystring-parser@3.0.3': - resolution: {integrity: sha512-zahM1lQv2YjmznnfQsWbYojFe55l0SLG/988brlLv1i8z3dubloLF+75ATRsqPBboUXsW6I9CPGE5rQgLfY0vQ==} - engines: {node: '>=16.0.0'} - - '@smithy/service-error-classification@3.0.3': - resolution: {integrity: sha512-Jn39sSl8cim/VlkLsUhRFq/dKDnRUFlfRkvhOJaUbLBXUsLRLNf9WaxDv/z9BjuQ3A6k/qE8af1lsqcwm7+DaQ==} - engines: {node: '>=16.0.0'} - - '@smithy/shared-ini-file-loader@3.1.4': - resolution: {integrity: sha512-qMxS4hBGB8FY2GQqshcRUy1K6k8aBWP5vwm8qKkCT3A9K2dawUwOIJfqh9Yste/Bl0J2lzosVyrXDj68kLcHXQ==} - engines: {node: '>=16.0.0'} - - '@smithy/signature-v4@4.1.0': - resolution: {integrity: sha512-aRryp2XNZeRcOtuJoxjydO6QTaVhxx/vjaR+gx7ZjaFgrgPRyZ3HCTbfwqYj6ZWEBHkCSUfcaymKPURaByukag==} - engines: {node: '>=16.0.0'} - - '@smithy/smithy-client@3.2.0': - resolution: {integrity: sha512-pDbtxs8WOhJLJSeaF/eAbPgXg4VVYFlRcL/zoNYA5WbG3wBL06CHtBSg53ppkttDpAJ/hdiede+xApip1CwSLw==} - engines: {node: '>=16.0.0'} - - '@smithy/types@3.3.0': - resolution: {integrity: sha512-IxvBBCTFDHbVoK7zIxqA1ZOdc4QfM5HM7rGleCuHi7L1wnKv5Pn69xXJQ9hgxH60ZVygH9/JG0jRgtUncE3QUA==} - engines: {node: '>=16.0.0'} - - '@smithy/url-parser@3.0.3': - resolution: {integrity: sha512-pw3VtZtX2rg+s6HMs6/+u9+hu6oY6U7IohGhVNnjbgKy86wcIsSZwgHrFR+t67Uyxvp4Xz3p3kGXXIpTNisq8A==} - - '@smithy/util-base64@3.0.0': - resolution: {integrity: sha512-Kxvoh5Qtt0CDsfajiZOCpJxgtPHXOKwmM+Zy4waD43UoEMA+qPxxa98aE/7ZhdnBFZFXMOiBR5xbcaMhLtznQQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-body-length-browser@3.0.0': - resolution: {integrity: sha512-cbjJs2A1mLYmqmyVl80uoLTJhAcfzMOyPgjwAYusWKMdLeNtzmMz9YxNl3/jRLoxSS3wkqkf0jwNdtXWtyEBaQ==} - - '@smithy/util-body-length-node@3.0.0': - resolution: {integrity: sha512-Tj7pZ4bUloNUP6PzwhN7K386tmSmEET9QtQg0TgdNOnxhZvCssHji+oZTUIuzxECRfG8rdm2PMw2WCFs6eIYkA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-buffer-from@2.2.0': - resolution: {integrity: sha512-IJdWBbTcMQ6DA0gdNhh/BwrLkDR+ADW5Kr1aZmd4k3DIF6ezMV4R2NIAmT08wQJ3yUK82thHWmC/TnK/wpMMIA==} - engines: {node: '>=14.0.0'} - - '@smithy/util-buffer-from@3.0.0': - resolution: {integrity: sha512-aEOHCgq5RWFbP+UDPvPot26EJHjOC+bRgse5A8V3FSShqd5E5UN4qc7zkwsvJPPAVsf73QwYcHN1/gt/rtLwQA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-config-provider@3.0.0': - resolution: {integrity: sha512-pbjk4s0fwq3Di/ANL+rCvJMKM5bzAQdE5S/6RL5NXgMExFAi6UgQMPOm5yPaIWPpr+EOXKXRonJ3FoxKf4mCJQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-defaults-mode-browser@3.0.15': - resolution: {integrity: sha512-FZ4Psa3vjp8kOXcd3HJOiDPBCWtiilLl57r0cnNtq/Ga9RSDrM5ERL6xt+tO43+2af6Pn5Yp92x2n5vPuduNfg==} - engines: {node: '>= 10.0.0'} - - '@smithy/util-defaults-mode-node@3.0.15': - resolution: {integrity: sha512-KSyAAx2q6d0t6f/S4XB2+3+6aQacm3aLMhs9aLMqn18uYGUepbdssfogW5JQZpc6lXNBnp0tEnR5e9CEKmEd7A==} - engines: {node: '>= 10.0.0'} - - '@smithy/util-endpoints@2.0.5': - resolution: {integrity: sha512-ReQP0BWihIE68OAblC/WQmDD40Gx+QY1Ez8mTdFMXpmjfxSyz2fVQu3A4zXRfQU9sZXtewk3GmhfOHswvX+eNg==} - engines: {node: '>=16.0.0'} - - '@smithy/util-hex-encoding@3.0.0': - resolution: {integrity: sha512-eFndh1WEK5YMUYvy3lPlVmYY/fZcQE1D8oSf41Id2vCeIkKJXPcYDCZD+4+xViI6b1XSd7tE+s5AmXzz5ilabQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-middleware@3.0.3': - resolution: {integrity: sha512-l+StyYYK/eO3DlVPbU+4Bi06Jjal+PFLSMmlWM1BEwyLxZ3aKkf1ROnoIakfaA7mC6uw3ny7JBkau4Yc+5zfWw==} - engines: {node: '>=16.0.0'} - - '@smithy/util-retry@3.0.3': - resolution: {integrity: sha512-AFw+hjpbtVApzpNDhbjNG5NA3kyoMs7vx0gsgmlJF4s+yz1Zlepde7J58zpIRIsdjc+emhpAITxA88qLkPF26w==} - engines: {node: '>=16.0.0'} - - '@smithy/util-stream@3.1.3': - resolution: {integrity: sha512-FIv/bRhIlAxC0U7xM1BCnF2aDRPq0UaelqBHkM2lsCp26mcBbgI0tCVTv+jGdsQLUmAMybua/bjDsSu8RQHbmw==} - engines: {node: '>=16.0.0'} - - '@smithy/util-uri-escape@3.0.0': - resolution: {integrity: sha512-LqR7qYLgZTD7nWLBecUi4aqolw8Mhza9ArpNEQ881MJJIU2sE5iHCK6TdyqqzcDLy0OPe10IY4T8ctVdtynubg==} - engines: {node: '>=16.0.0'} - - '@smithy/util-utf8@2.3.0': - resolution: {integrity: sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A==} - engines: {node: '>=14.0.0'} - - '@smithy/util-utf8@3.0.0': - resolution: {integrity: sha512-rUeT12bxFnplYDe815GXbq/oixEGHfRFFtcTF3YdDi/JaENIM6aSYYLJydG83UNzLXeRI5K8abYd/8Sp/QM0kA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-waiter@3.1.2': - resolution: {integrity: sha512-4pP0EV3iTsexDx+8PPGAKCQpd/6hsQBaQhqWzU4hqKPHN5epPsxKbvUTIiYIHTxaKt6/kEaqPBpu/ufvfbrRzw==} - engines: {node: '>=16.0.0'} - - '@tsconfig/node10@1.0.11': - resolution: {integrity: sha512-DcRjDCujK/kCk/cUe8Xz8ZSpm8mS3mNNpta+jGCA6USEDfktlNvm1+IuZ9eTcDbNk41BHwpHHeW+N1lKCz4zOw==} - - '@tsconfig/node12@1.0.11': - resolution: {integrity: sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==} - - '@tsconfig/node14@1.0.3': - resolution: {integrity: sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==} - - '@tsconfig/node16@1.0.4': - resolution: {integrity: sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==} - - '@types/chai@4.3.20': - resolution: {integrity: sha512-/pC9HAB5I/xMlc5FP77qjCnI16ChlJfW0tGa0IUcFn38VJrTV6DeZ60NU5KZBtaOZqjdpwTWohz5HU1RrhiYxQ==} - - '@types/debug@4.1.12': - resolution: {integrity: sha512-vIChWdVG3LG1SMxEvI/AK+FWJthlrqlTu7fbrlywTkkaONwk/UAGaULXRlf8vkzFBLVm0zkMdCquhL5aOjhXPQ==} - - '@types/interpret@1.1.3': - resolution: {integrity: sha512-uBaBhj/BhilG58r64mtDb/BEdH51HIQLgP5bmWzc5qCtFMja8dCk/IOJmk36j0lbi9QHwI6sbtUNGuqXdKCAtQ==} - - '@types/mocha@10.0.9': - resolution: {integrity: sha512-sicdRoWtYevwxjOHNMPTl3vSfJM6oyW8o1wXeI7uww6b6xHg8eBznQDNSGBCDJmsE8UMxP05JgZRtsKbTqt//Q==} - - '@types/ms@0.7.34': - resolution: {integrity: sha512-nG96G3Wp6acyAgJqGasjODb+acrI7KltPiRxzHPXnP3NgI28bpQDRv53olbqGXbfcgF5aiiHmO3xpwEpS5Ld9g==} - - '@types/node@20.16.3': - resolution: {integrity: sha512-/wdGiWRkMOm53gAsSyFMXFZHbVg7C6CbkrzHNpaHoYfsUWPg7m6ZRKtvQjgvQ9i8WT540a3ydRlRQbxjY30XxQ==} - - '@types/node@22.5.2': - resolution: {integrity: sha512-acJsPTEqYqulZS/Yp/S3GgeE6GZ0qYODUR8aVr/DkhHQ8l9nd4j5x1/ZJy9/gHrRlFMqkO6i0I3E27Alu4jjPg==} - - '@types/pg@8.11.8': - resolution: {integrity: sha512-IqpCf8/569txXN/HoP5i1LjXfKZWL76Yr2R77xgeIICUbAYHeoaEZFhYHo2uDftecLWrTJUq63JvQu8q3lnDyA==} - - '@types/retry@0.12.2': - resolution: {integrity: sha512-XISRgDJ2Tc5q4TRqvgJtzsRkFYNJzZrhTdtMoGVBttwzzQJkPnS3WWTFc7kuDRoPtPakl+T+OfdEUjYJj7Jbow==} - - '@types/semver@7.5.8': - resolution: {integrity: sha512-I8EUhyrgfLrcTkzV3TSsGyl1tSuPrEDzr0yd5m90UgNxQkyDXULk3b6MlQqTCpZpNtWe1K0hzclnZkTcLBe2UQ==} - - acorn-walk@8.3.3: - resolution: {integrity: sha512-MxXdReSRhGO7VlFe1bRG/oI7/mdLV9B9JJT0N8vZOhF7gFRR5l3M8W9G8JxmKV+JC5mGqJ0QvqfSOLsCPa4nUw==} - engines: {node: '>=0.4.0'} - - acorn@8.12.1: - resolution: {integrity: sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==} - engines: {node: '>=0.4.0'} - hasBin: true - - ansi-colors@4.1.3: - resolution: {integrity: sha512-/6w/C21Pm1A7aZitlI5Ni/2J6FFQN8i1Cvz3kHABAAbw93v/NlvKdVOqz7CCWz/3iv/JplRSEEZ83XION15ovw==} - engines: {node: '>=6'} - - ansi-regex@5.0.1: - resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==} - engines: {node: '>=8'} - - ansi-styles@3.2.1: - resolution: {integrity: sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==} - engines: {node: '>=4'} - - ansi-styles@4.3.0: - resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} - engines: {node: '>=8'} - - anymatch@3.1.3: - resolution: {integrity: sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==} - engines: {node: '>= 8'} - - arg@4.1.3: - resolution: {integrity: sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==} - - argparse@2.0.1: - resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==} - - assertion-error@2.0.1: - resolution: {integrity: sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==} - engines: {node: '>=12'} - - balanced-match@1.0.2: - resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} - - base64-js@1.5.1: - resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==} - - binary-extensions@2.3.0: - resolution: {integrity: sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==} - engines: {node: '>=8'} - - bowser@2.11.0: - resolution: {integrity: sha512-AlcaJBi/pqqJBIQ8U9Mcpc9i8Aqxn88Skv5d+xBX006BY5u8N3mGLHa5Lgppa7L/HfwgwLgZ6NYs+Ag6uUmJRA==} - - brace-expansion@1.1.11: - resolution: {integrity: sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==} - - brace-expansion@2.0.1: - resolution: {integrity: sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==} - - braces@3.0.3: - resolution: {integrity: sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==} - engines: {node: '>=8'} - - browser-stdout@1.3.1: - resolution: {integrity: sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==} - - buffer@5.6.0: - resolution: {integrity: sha512-/gDYp/UtU0eA1ys8bOs9J6a+E/KWIY+DZ+Q2WESNUA0jFRsJOc0SNUO6xJ5SGA1xueg3NL65W6s+NY5l9cunuw==} - - callsites@3.1.0: - resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==} - engines: {node: '>=6'} - - camelcase@6.3.0: - resolution: {integrity: sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==} - engines: {node: '>=10'} - - chai@5.1.2: - resolution: {integrity: sha512-aGtmf24DW6MLHHG5gCx4zaI3uBq3KRtxeVs0DjFH6Z0rDNbsvTxFASFvdj79pxjxZ8/5u3PIiN3IwEIQkiiuPw==} - engines: {node: '>=12'} - - chalk@2.4.2: - resolution: {integrity: sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==} - engines: {node: '>=4'} - - chalk@4.1.2: - resolution: {integrity: sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==} - engines: {node: '>=10'} - - check-error@2.1.1: - resolution: {integrity: sha512-OAlb+T7V4Op9OwdkjmguYRqncdlx5JiofwOAUkmTF+jNdHwzTaTs4sRAGpzLF3oOz5xAyDGrPgeIDFQmDOTiJw==} - engines: {node: '>= 16'} - - chokidar@3.6.0: - resolution: {integrity: sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==} - engines: {node: '>= 8.10.0'} - - cliui@7.0.4: - resolution: {integrity: sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==} - - cliui@8.0.1: - resolution: {integrity: sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==} - engines: {node: '>=12'} - - color-convert@1.9.3: - resolution: {integrity: sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==} - - color-convert@2.0.1: - resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} - engines: {node: '>=7.0.0'} - - color-name@1.1.3: - resolution: {integrity: sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==} - - color-name@1.1.4: - resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} - - concat-map@0.0.1: - resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} - - cosmiconfig@8.3.6: - resolution: {integrity: sha512-kcZ6+W5QzcJ3P1Mt+83OUv/oHFqZHIx8DuxG6eZ5RGMERoLqp4BuGjhHLYGK+Kf5XVkQvqBSmAy/nGWN3qDgEA==} - engines: {node: '>=14'} - peerDependencies: - typescript: '>=4.9.5' - peerDependenciesMeta: - typescript: - optional: true - - create-require@1.1.1: - resolution: {integrity: sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==} - - debug@4.3.6: - resolution: {integrity: sha512-O/09Bd4Z1fBrU4VzkhFqVgpPzaGbw6Sm9FEkBT1A/YBXQFGuuSxa1dN2nxgxS34JmKXqYx8CZAwEVoJFImUXIg==} - engines: {node: '>=6.0'} - peerDependencies: - supports-color: '*' - peerDependenciesMeta: - supports-color: - optional: true - - decamelize@4.0.0: - resolution: {integrity: sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==} - engines: {node: '>=10'} - - deep-eql@5.0.2: - resolution: {integrity: sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==} - engines: {node: '>=6'} - - diff@4.0.2: - resolution: {integrity: sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==} - engines: {node: '>=0.3.1'} - - diff@5.2.0: - resolution: {integrity: sha512-uIFDxqpRZGZ6ThOk84hEfqWoHx2devRFvpTZcTHur85vImfaxUbTW9Ryh4CpCuDnToOP1CEtXKIgytHBPVff5A==} - engines: {node: '>=0.3.1'} - - dotenv@16.4.5: - resolution: {integrity: sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==} - engines: {node: '>=12'} - - emoji-regex@8.0.0: - resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==} - - error-ex@1.3.2: - resolution: {integrity: sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==} - - esbuild@0.23.1: - resolution: {integrity: sha512-VVNz/9Sa0bs5SELtn3f7qhJCDPCF5oMEl5cO9/SSinpE9hbPVvxbd572HH5AKiP7WD8INO53GgfDDhRjkylHEg==} - engines: {node: '>=18'} - hasBin: true - - escalade@3.2.0: - resolution: {integrity: sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==} - engines: {node: '>=6'} - - escape-string-regexp@1.0.5: - resolution: {integrity: sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==} - engines: {node: '>=0.8.0'} - - escape-string-regexp@4.0.0: - resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} - engines: {node: '>=10'} - - events@3.3.0: - resolution: {integrity: sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==} - engines: {node: '>=0.8.x'} - - fast-xml-parser@4.4.1: - resolution: {integrity: sha512-xkjOecfnKGkSsOwtZ5Pz7Us/T6mrbPQrq0nh+aCO5V9nk5NLWmasAHumTKjiPJPWANe+kAZ84Jc8ooJkzZ88Sw==} - hasBin: true - - fill-range@7.1.1: - resolution: {integrity: sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==} - engines: {node: '>=8'} - - find-up@5.0.0: - resolution: {integrity: sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==} - engines: {node: '>=10'} - - flat@5.0.2: - resolution: {integrity: sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==} - hasBin: true - - fs.realpath@1.0.0: - resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==} - - fsevents@2.3.3: - resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} - engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} - os: [darwin] - - get-caller-file@2.0.5: - resolution: {integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==} - engines: {node: 6.* || 8.* || >= 10.*} - - get-tsconfig@4.8.0: - resolution: {integrity: sha512-Pgba6TExTZ0FJAn1qkJAjIeKoDJ3CsI2ChuLohJnZl/tTU8MVrq3b+2t5UOPfRa4RMsorClBjJALkJUMjG1PAw==} - - glob-parent@5.1.2: - resolution: {integrity: sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==} - engines: {node: '>= 6'} - - glob@8.1.0: - resolution: {integrity: sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==} - engines: {node: '>=12'} - deprecated: Glob versions prior to v9 are no longer supported - - graphile-config@0.0.1-beta.9: - resolution: {integrity: sha512-7vNxXZ24OAgXxDKXYi9JtgWPMuNbBL3057Yf32Ux+/rVP4+EePgySCc+NNnn0tORi8qwqVreN8bdWqGIcSwNXg==} - engines: {node: '>=16'} - - graphile-worker@0.16.6: - resolution: {integrity: sha512-e7gGYDmGqzju2l83MpzX8vNG/lOtVJiSzI3eZpAFubSxh/cxs7sRrRGBGjzBP1kNG0H+c95etPpNRNlH65PYhw==} - engines: {node: '>=14.0.0'} - hasBin: true - - has-flag@3.0.0: - resolution: {integrity: sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==} - engines: {node: '>=4'} - - has-flag@4.0.0: - resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} - engines: {node: '>=8'} - - he@1.2.0: - resolution: {integrity: sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==} - hasBin: true - - ieee754@1.2.1: - resolution: {integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==} - - ignore-by-default@1.0.1: - resolution: {integrity: sha512-Ius2VYcGNk7T90CppJqcIkS5ooHUZyIQK+ClZfMfMNFEF9VSE73Fq+906u/CWu92x4gzZMWOwfFYckPObzdEbA==} - - import-fresh@3.3.0: - resolution: {integrity: sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==} - engines: {node: '>=6'} - - inflight@1.0.6: - resolution: {integrity: sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==} - deprecated: This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful. - - inherits@2.0.4: - resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} - - interpret@3.1.1: - resolution: {integrity: sha512-6xwYfHbajpoF0xLW+iwLkhwgvLoZDfjYfoFNu8ftMoXINzwuymNLd9u/KmwtdT2GbR+/Cz66otEGEVVUHX9QLQ==} - engines: {node: '>=10.13.0'} - - is-arrayish@0.2.1: - resolution: {integrity: sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==} - - is-binary-path@2.1.0: - resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==} - engines: {node: '>=8'} - - is-extglob@2.1.1: - resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} - engines: {node: '>=0.10.0'} - - is-fullwidth-code-point@3.0.0: - resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==} - engines: {node: '>=8'} - - is-glob@4.0.3: - resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==} - engines: {node: '>=0.10.0'} - - is-network-error@1.1.0: - resolution: {integrity: sha512-tUdRRAnhT+OtCZR/LxZelH/C7QtjtFrTu5tXCA8pl55eTUElUHT+GPYV8MBMBvea/j+NxQqVt3LbWMRir7Gx9g==} - engines: {node: '>=16'} - - is-number@7.0.0: - resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==} - engines: {node: '>=0.12.0'} - - is-plain-obj@2.1.0: - resolution: {integrity: sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==} - engines: {node: '>=8'} - - is-unicode-supported@0.1.0: - resolution: {integrity: sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==} - engines: {node: '>=10'} - - js-tokens@4.0.0: - resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} - - js-yaml@4.1.0: - resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==} - hasBin: true - - json-parse-even-better-errors@2.3.1: - resolution: {integrity: sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==} - - json5@2.2.3: - resolution: {integrity: sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==} - engines: {node: '>=6'} - hasBin: true - - lines-and-columns@1.2.4: - resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} - - locate-path@6.0.0: - resolution: {integrity: sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==} - engines: {node: '>=10'} - - log-symbols@4.1.0: - resolution: {integrity: sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==} - engines: {node: '>=10'} - - loupe@3.1.2: - resolution: {integrity: sha512-23I4pFZHmAemUnz8WZXbYRSKYj801VDaNv9ETuMh7IrMc7VuVVSo+Z9iLE3ni30+U48iDWfi30d3twAXBYmnCg==} - - make-error@1.3.6: - resolution: {integrity: sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==} - - minimatch@3.1.2: - resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==} - - minimatch@5.1.6: - resolution: {integrity: sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==} - engines: {node: '>=10'} - - mocha@10.7.3: - resolution: {integrity: sha512-uQWxAu44wwiACGqjbPYmjo7Lg8sFrS3dQe7PP2FQI+woptP4vZXSMcfMyFL/e1yFEeEpV4RtyTpZROOKmxis+A==} - engines: {node: '>= 14.0.0'} - hasBin: true - - ms@2.1.2: - resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==} - - ms@2.1.3: - resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} - - nodemon@3.1.4: - resolution: {integrity: sha512-wjPBbFhtpJwmIeY2yP7QF+UKzPfltVGtfce1g/bB15/8vCGZj8uxD62b/b9M9/WVgme0NZudpownKN+c0plXlQ==} - engines: {node: '>=10'} - hasBin: true - - normalize-path@3.0.0: - resolution: {integrity: sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==} - engines: {node: '>=0.10.0'} - - obuf@1.1.2: - resolution: {integrity: sha512-PX1wu0AmAdPqOL1mWhqmlOd8kOIZQwGZw6rh7uby9fTc5lhaOWFLX3I6R1hrF9k3zUY40e6igsLGkDXK92LJNg==} - - once@1.4.0: - resolution: {integrity: sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==} - - p-limit@3.1.0: - resolution: {integrity: sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==} - engines: {node: '>=10'} - - p-locate@5.0.0: - resolution: {integrity: sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==} - engines: {node: '>=10'} - - p-retry@6.2.0: - resolution: {integrity: sha512-JA6nkq6hKyWLLasXQXUrO4z8BUZGUt/LjlJxx8Gb2+2ntodU/SS63YZ8b0LUTbQ8ZB9iwOfhEPhg4ykKnn2KsA==} - engines: {node: '>=16.17'} - - parent-module@1.0.1: - resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} - engines: {node: '>=6'} - - parse-json@5.2.0: - resolution: {integrity: sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==} - engines: {node: '>=8'} - - path-exists@4.0.0: - resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==} - engines: {node: '>=8'} - - path-type@4.0.0: - resolution: {integrity: sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==} - engines: {node: '>=8'} - - pathval@2.0.0: - resolution: {integrity: sha512-vE7JKRyES09KiunauX7nd2Q9/L7lhok4smP9RZTDeD4MVs72Dp2qNFVz39Nz5a0FVEW0BJR6C0DYrq6unoziZA==} - engines: {node: '>= 14.16'} - - pg-cloudflare@1.1.1: - resolution: {integrity: sha512-xWPagP/4B6BgFO+EKz3JONXv3YDgvkbVrGw2mTo3D6tVDQRh1e7cqVGvyR3BE+eQgAvx1XhW/iEASj4/jCWl3Q==} - - pg-connection-string@2.6.4: - resolution: {integrity: sha512-v+Z7W/0EO707aNMaAEfiGnGL9sxxumwLl2fJvCQtMn9Fxsg+lPpPkdcyBSv/KFgpGdYkMfn+EI1Or2EHjpgLCA==} - - pg-int8@1.0.1: - resolution: {integrity: sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw==} - engines: {node: '>=4.0.0'} - - pg-numeric@1.0.2: - resolution: {integrity: sha512-BM/Thnrw5jm2kKLE5uJkXqqExRUY/toLHda65XgFTBTFYZyopbKjBe29Ii3RbkvlsMoFwD+tHeGaCjjv0gHlyw==} - engines: {node: '>=4'} - - pg-pool@3.6.2: - resolution: {integrity: sha512-Htjbg8BlwXqSBQ9V8Vjtc+vzf/6fVUuak/3/XXKA9oxZprwW3IMDQTGHP+KDmVL7rtd+R1QjbnCFPuTHm3G4hg==} - peerDependencies: - pg: '>=8.0' - - pg-protocol@1.6.1: - resolution: {integrity: sha512-jPIlvgoD63hrEuihvIg+tJhoGjUsLPn6poJY9N5CnlPd91c2T18T/9zBtLxZSb1EhYxBRoZJtzScCaWlYLtktg==} - - pg-types@2.2.0: - resolution: {integrity: sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==} - engines: {node: '>=4'} - - pg-types@4.0.2: - resolution: {integrity: sha512-cRL3JpS3lKMGsKaWndugWQoLOCoP+Cic8oseVcbr0qhPzYD5DWXK+RZ9LY9wxRf7RQia4SCwQlXk0q6FCPrVng==} - engines: {node: '>=10'} - - pg@8.12.0: - resolution: {integrity: sha512-A+LHUSnwnxrnL/tZ+OLfqR1SxLN3c/pgDztZ47Rpbsd4jUytsTtwQo/TLPRzPJMp/1pbhYVhH9cuSZLAajNfjQ==} - engines: {node: '>= 8.0.0'} - peerDependencies: - pg-native: '>=3.0.1' - peerDependenciesMeta: - pg-native: - optional: true - - pgpass@1.0.5: - resolution: {integrity: sha512-FdW9r/jQZhSeohs1Z3sI1yxFQNFvMcnmfuj4WBMUTxOrAyLMaTcE1aAMBiTlbMNaXvBCQuVi0R7hd8udDSP7ug==} - - picocolors@1.1.0: - resolution: {integrity: sha512-TQ92mBOW0l3LeMeyLV6mzy/kWr8lkd/hp3mTg7wYK7zJhuBStmGMBG0BdeDZS/dZx1IukaX6Bk11zcln25o1Aw==} - - picomatch@2.3.1: - resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==} - engines: {node: '>=8.6'} - - postgres-array@2.0.0: - resolution: {integrity: sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA==} - engines: {node: '>=4'} - - postgres-array@3.0.2: - resolution: {integrity: sha512-6faShkdFugNQCLwucjPcY5ARoW1SlbnrZjmGl0IrrqewpvxvhSLHimCVzqeuULCbG0fQv7Dtk1yDbG3xv7Veog==} - engines: {node: '>=12'} - - postgres-bytea@1.0.0: - resolution: {integrity: sha512-xy3pmLuQqRBZBXDULy7KbaitYqLcmxigw14Q5sj8QBVLqEwXfeybIKVWiqAXTlcvdvb0+xkOtDbfQMOf4lST1w==} - engines: {node: '>=0.10.0'} - - postgres-bytea@3.0.0: - resolution: {integrity: sha512-CNd4jim9RFPkObHSjVHlVrxoVQXz7quwNFpz7RY1okNNme49+sVyiTvTRobiLV548Hx/hb1BG+iE7h9493WzFw==} - engines: {node: '>= 6'} - - postgres-date@1.0.7: - resolution: {integrity: sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q==} - engines: {node: '>=0.10.0'} - - postgres-date@2.1.0: - resolution: {integrity: sha512-K7Juri8gtgXVcDfZttFKVmhglp7epKb1K4pgrkLxehjqkrgPhfG6OO8LHLkfaqkbpjNRnra018XwAr1yQFWGcA==} - engines: {node: '>=12'} - - postgres-interval@1.2.0: - resolution: {integrity: sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==} - engines: {node: '>=0.10.0'} - - postgres-interval@3.0.0: - resolution: {integrity: sha512-BSNDnbyZCXSxgA+1f5UU2GmwhoI0aU5yMxRGO8CdFEcY2BQF9xm/7MqKnYoM1nJDk8nONNWDk9WeSmePFhQdlw==} - engines: {node: '>=12'} - - postgres-range@1.1.4: - resolution: {integrity: sha512-i/hbxIE9803Alj/6ytL7UHQxRvZkI9O4Sy+J3HGc4F4oo/2eQAjTSNJ0bfxyse3bH0nuVesCk+3IRLaMtG3H6w==} - - pstree.remy@1.1.8: - resolution: {integrity: sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==} - - randombytes@2.1.0: - resolution: {integrity: sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==} - - readable-stream@3.6.2: - resolution: {integrity: sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==} - engines: {node: '>= 6'} - - readdirp@3.6.0: - resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==} - engines: {node: '>=8.10.0'} - - require-directory@2.1.1: - resolution: {integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==} - engines: {node: '>=0.10.0'} - - resolve-from@4.0.0: - resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==} - engines: {node: '>=4'} - - resolve-pkg-maps@1.0.0: - resolution: {integrity: sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==} - - retry@0.13.1: - resolution: {integrity: sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg==} - engines: {node: '>= 4'} - - safe-buffer@5.2.1: - resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} - - semver@7.6.3: - resolution: {integrity: sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==} - engines: {node: '>=10'} - hasBin: true - - serialize-javascript@6.0.2: - resolution: {integrity: sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g==} - - simple-update-notifier@2.0.0: - resolution: {integrity: sha512-a2B9Y0KlNXl9u/vsW6sTIu9vGEpfKu2wRV6l1H3XEas/0gUIzGzBoP/IouTcUQbm9JWZLH3COxyn03TYlFax6w==} - engines: {node: '>=10'} - - split2@4.2.0: - resolution: {integrity: sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==} - engines: {node: '>= 10.x'} - - stream-browserify@3.0.0: - resolution: {integrity: sha512-H73RAHsVBapbim0tU2JwwOiXUj+fikfiaoYAKHF3VJfA0pe2BCzkhAHBlLG6REzE+2WNZcxOXjK7lkso+9euLA==} - - string-width@4.2.3: - resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} - engines: {node: '>=8'} - - string_decoder@1.3.0: - resolution: {integrity: sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==} - - strip-ansi@6.0.1: - resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} - engines: {node: '>=8'} - - strip-json-comments@3.1.1: - resolution: {integrity: sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==} - engines: {node: '>=8'} - - strnum@1.0.5: - resolution: {integrity: sha512-J8bbNyKKXl5qYcR36TIO8W3mVGVHrmmxsd5PAItGkmyzwJvybiw2IVq5nqd0i4LSNSkB/sx9VHllbfFdr9k1JA==} - - supports-color@5.5.0: - resolution: {integrity: sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==} - engines: {node: '>=4'} - - supports-color@7.2.0: - resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==} - engines: {node: '>=8'} - - supports-color@8.1.1: - resolution: {integrity: sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==} - engines: {node: '>=10'} - - to-regex-range@5.0.1: - resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} - engines: {node: '>=8.0'} - - touch@3.1.1: - resolution: {integrity: sha512-r0eojU4bI8MnHr8c5bNo7lJDdI2qXlWWJk6a9EAFG7vbhTjElYhBVS3/miuE0uOuoLdb8Mc/rVfsmm6eo5o9GA==} - hasBin: true - - ts-node@10.9.2: - resolution: {integrity: sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==} - hasBin: true - peerDependencies: - '@swc/core': '>=1.2.50' - '@swc/wasm': '>=1.2.50' - '@types/node': '*' - typescript: '>=2.7' - peerDependenciesMeta: - '@swc/core': - optional: true - '@swc/wasm': - optional: true - - tslib@2.7.0: - resolution: {integrity: sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA==} - - tsx@4.19.0: - resolution: {integrity: sha512-bV30kM7bsLZKZIOCHeMNVMJ32/LuJzLVajkQI/qf92J2Qr08ueLQvW00PUZGiuLPP760UINwupgUj8qrSCPUKg==} - engines: {node: '>=18.0.0'} - hasBin: true - - typescript@5.5.4: - resolution: {integrity: sha512-Mtq29sKDAEYP7aljRgtPOpTvOfbwRWlS6dPRzwjdE+C0R4brX/GUyhHSecbHMFLNBLcJIPt9nl9yG5TZ1weH+Q==} - engines: {node: '>=14.17'} - hasBin: true - - undefsafe@2.0.5: - resolution: {integrity: sha512-WxONCrssBM8TSPRqN5EmsjVrsv4A8X12J4ArBiiayv3DyyG3ZlIg6yysuuSYdZsVz3TKcTg2fd//Ujd4CHV1iA==} - - undici-types@6.19.8: - resolution: {integrity: sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==} - - util-deprecate@1.0.2: - resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==} - - uuid@9.0.1: - resolution: {integrity: sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==} - hasBin: true - - v8-compile-cache-lib@3.0.1: - resolution: {integrity: sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==} - - workerpool@6.5.1: - resolution: {integrity: sha512-Fs4dNYcsdpYSAfVxhnl1L5zTksjvOJxtC5hzMNl+1t9B8hTJTdKDyZ5ju7ztgPy+ft9tBFXoOlDNiOT9WUXZlA==} - - wrap-ansi@7.0.0: - resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} - engines: {node: '>=10'} - - wrappy@1.0.2: - resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==} - - xtend@4.0.2: - resolution: {integrity: sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==} - engines: {node: '>=0.4'} - - y18n@5.0.8: - resolution: {integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==} - engines: {node: '>=10'} - - yargs-parser@20.2.9: - resolution: {integrity: sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==} - engines: {node: '>=10'} - - yargs-parser@21.1.1: - resolution: {integrity: sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==} - engines: {node: '>=12'} - - yargs-unparser@2.0.0: - resolution: {integrity: sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==} - engines: {node: '>=10'} - - yargs@16.2.0: - resolution: {integrity: sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==} - engines: {node: '>=10'} - - yargs@17.7.2: - resolution: {integrity: sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==} - engines: {node: '>=12'} - - yn@3.1.1: - resolution: {integrity: sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==} - engines: {node: '>=6'} - - yocto-queue@0.1.0: - resolution: {integrity: sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==} - engines: {node: '>=10'} - -snapshots: - - '@aws-crypto/crc32@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/crc32c@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/sha1-browser@5.2.0': - dependencies: - '@aws-crypto/supports-web-crypto': 5.2.0 - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-locate-window': 3.568.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-crypto/sha256-browser@5.2.0': - dependencies: - '@aws-crypto/sha256-js': 5.2.0 - '@aws-crypto/supports-web-crypto': 5.2.0 - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-locate-window': 3.568.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-crypto/sha256-js@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/supports-web-crypto@5.2.0': - dependencies: - tslib: 2.7.0 - - '@aws-crypto/util@5.2.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-sdk/client-s3@3.637.0': - dependencies: - '@aws-crypto/sha1-browser': 5.2.0 - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-bucket-endpoint': 3.620.0 - '@aws-sdk/middleware-expect-continue': 3.620.0 - '@aws-sdk/middleware-flexible-checksums': 3.620.0 - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-location-constraint': 3.609.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-sdk-s3': 3.635.0 - '@aws-sdk/middleware-ssec': 3.609.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/signature-v4-multi-region': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@aws-sdk/xml-builder': 3.609.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/eventstream-serde-browser': 3.0.6 - '@smithy/eventstream-serde-config-resolver': 3.0.3 - '@smithy/eventstream-serde-node': 3.0.5 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-blob-browser': 3.1.2 - '@smithy/hash-node': 3.0.3 - '@smithy/hash-stream-node': 3.1.2 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/md5-js': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-stream': 3.1.3 - '@smithy/util-utf8': 3.0.0 - '@smithy/util-waiter': 3.1.2 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sso@3.637.0': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sts@3.637.0': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/core@3.635.0': - dependencies: - '@smithy/core': 2.4.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-middleware': 3.0.3 - fast-xml-parser: 4.4.1 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-env@3.620.1': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-http@3.635.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-stream': 3.1.3 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-ini@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/credential-provider-env': 3.620.1 - '@aws-sdk/credential-provider-http': 3.635.0 - '@aws-sdk/credential-provider-process': 3.620.1 - '@aws-sdk/credential-provider-sso': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/credential-provider-web-identity': 3.621.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - aws-crt - - '@aws-sdk/credential-provider-node@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/credential-provider-env': 3.620.1 - '@aws-sdk/credential-provider-http': 3.635.0 - '@aws-sdk/credential-provider-ini': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/credential-provider-process': 3.620.1 - '@aws-sdk/credential-provider-sso': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/credential-provider-web-identity': 3.621.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - '@aws-sdk/client-sts' - - aws-crt - - '@aws-sdk/credential-provider-process@3.620.1': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-sso@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))': - dependencies: - '@aws-sdk/client-sso': 3.637.0 - '@aws-sdk/token-providers': 3.614.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - aws-crt - - '@aws-sdk/credential-provider-web-identity@3.621.0(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/lib-storage@3.637.0(@aws-sdk/client-s3@3.637.0)': - dependencies: - '@aws-sdk/client-s3': 3.637.0 - '@smithy/abort-controller': 3.1.1 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/smithy-client': 3.2.0 - buffer: 5.6.0 - events: 3.3.0 - stream-browserify: 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-bucket-endpoint@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-arn-parser': 3.568.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-expect-continue@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-flexible-checksums@3.620.0': - dependencies: - '@aws-crypto/crc32': 5.2.0 - '@aws-crypto/crc32c': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@smithy/is-array-buffer': 3.0.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-host-header@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-location-constraint@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-logger@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-recursion-detection@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-sdk-s3@3.635.0': - dependencies: - '@aws-sdk/core': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-arn-parser': 3.568.0 - '@smithy/core': 2.4.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-stream': 3.1.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-ssec@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-user-agent@3.637.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/region-config-resolver@3.614.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@aws-sdk/signature-v4-multi-region@3.635.0': - dependencies: - '@aws-sdk/middleware-sdk-s3': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/token-providers@3.614.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))': - dependencies: - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/types@3.609.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/util-arn-parser@3.568.0': - dependencies: - tslib: 2.7.0 - - '@aws-sdk/util-endpoints@3.637.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - '@smithy/util-endpoints': 2.0.5 - tslib: 2.7.0 - - '@aws-sdk/util-locate-window@3.568.0': - dependencies: - tslib: 2.7.0 - - '@aws-sdk/util-user-agent-browser@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - bowser: 2.11.0 - tslib: 2.7.0 - - '@aws-sdk/util-user-agent-node@3.614.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/xml-builder@3.609.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@babel/code-frame@7.24.7': - dependencies: - '@babel/highlight': 7.24.7 - picocolors: 1.1.0 - - '@babel/helper-validator-identifier@7.24.7': {} - - '@babel/highlight@7.24.7': - dependencies: - '@babel/helper-validator-identifier': 7.24.7 - chalk: 2.4.2 - js-tokens: 4.0.0 - picocolors: 1.1.0 - - '@cspotcode/source-map-support@0.8.1': - dependencies: - '@jridgewell/trace-mapping': 0.3.9 - - '@esbuild/aix-ppc64@0.23.1': - optional: true - - '@esbuild/android-arm64@0.23.1': - optional: true - - '@esbuild/android-arm@0.23.1': - optional: true - - '@esbuild/android-x64@0.23.1': - optional: true - - '@esbuild/darwin-arm64@0.23.1': - optional: true - - '@esbuild/darwin-x64@0.23.1': - optional: true - - '@esbuild/freebsd-arm64@0.23.1': - optional: true - - '@esbuild/freebsd-x64@0.23.1': - optional: true - - '@esbuild/linux-arm64@0.23.1': - optional: true - - '@esbuild/linux-arm@0.23.1': - optional: true - - '@esbuild/linux-ia32@0.23.1': - optional: true - - '@esbuild/linux-loong64@0.23.1': - optional: true - - '@esbuild/linux-mips64el@0.23.1': - optional: true - - '@esbuild/linux-ppc64@0.23.1': - optional: true - - '@esbuild/linux-riscv64@0.23.1': - optional: true - - '@esbuild/linux-s390x@0.23.1': - optional: true - - '@esbuild/linux-x64@0.23.1': - optional: true - - '@esbuild/netbsd-x64@0.23.1': - optional: true - - '@esbuild/openbsd-arm64@0.23.1': - optional: true - - '@esbuild/openbsd-x64@0.23.1': - optional: true - - '@esbuild/sunos-x64@0.23.1': - optional: true - - '@esbuild/win32-arm64@0.23.1': - optional: true - - '@esbuild/win32-ia32@0.23.1': - optional: true - - '@esbuild/win32-x64@0.23.1': - optional: true - - '@graphile/logger@0.2.0': {} - - '@jridgewell/resolve-uri@3.1.2': {} - - '@jridgewell/sourcemap-codec@1.5.0': {} - - '@jridgewell/trace-mapping@0.3.9': - dependencies: - '@jridgewell/resolve-uri': 3.1.2 - '@jridgewell/sourcemap-codec': 1.5.0 - - '@noble/hashes@1.5.0': {} - - '@paralleldrive/cuid2@2.2.2': - dependencies: - '@noble/hashes': 1.5.0 - - '@smithy/abort-controller@3.1.1': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/chunked-blob-reader-native@3.0.0': - dependencies: - '@smithy/util-base64': 3.0.0 - tslib: 2.7.0 - - '@smithy/chunked-blob-reader@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/config-resolver@3.0.5': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@smithy/core@2.4.0': - dependencies: - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/credential-provider-imds@3.2.0': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - tslib: 2.7.0 - - '@smithy/eventstream-codec@3.1.2': - dependencies: - '@aws-crypto/crc32': 5.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-hex-encoding': 3.0.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-browser@3.0.6': - dependencies: - '@smithy/eventstream-serde-universal': 3.0.5 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-config-resolver@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-node@3.0.5': - dependencies: - '@smithy/eventstream-serde-universal': 3.0.5 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-universal@3.0.5': - dependencies: - '@smithy/eventstream-codec': 3.1.2 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/fetch-http-handler@3.2.4': - dependencies: - '@smithy/protocol-http': 4.1.0 - '@smithy/querystring-builder': 3.0.3 - '@smithy/types': 3.3.0 - '@smithy/util-base64': 3.0.0 - tslib: 2.7.0 - - '@smithy/hash-blob-browser@3.1.2': - dependencies: - '@smithy/chunked-blob-reader': 3.0.0 - '@smithy/chunked-blob-reader-native': 3.0.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/hash-node@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/hash-stream-node@3.1.2': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/invalid-dependency@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/is-array-buffer@2.2.0': - dependencies: - tslib: 2.7.0 - - '@smithy/is-array-buffer@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/md5-js@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/middleware-content-length@3.0.5': - dependencies: - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/middleware-endpoint@3.1.0': - dependencies: - '@smithy/middleware-serde': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@smithy/middleware-retry@3.0.15': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/service-error-classification': 3.0.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - tslib: 2.7.0 - uuid: 9.0.1 - - '@smithy/middleware-serde@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/middleware-stack@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/node-config-provider@3.1.4': - dependencies: - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/node-http-handler@3.1.4': - dependencies: - '@smithy/abort-controller': 3.1.1 - '@smithy/protocol-http': 4.1.0 - '@smithy/querystring-builder': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/property-provider@3.1.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/protocol-http@4.1.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/querystring-builder@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-uri-escape': 3.0.0 - tslib: 2.7.0 - - '@smithy/querystring-parser@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/service-error-classification@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - - '@smithy/shared-ini-file-loader@3.1.4': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/signature-v4@4.1.0': - dependencies: - '@smithy/is-array-buffer': 3.0.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-hex-encoding': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-uri-escape': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/smithy-client@3.2.0': - dependencies: - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-stack': 3.0.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-stream': 3.1.3 - tslib: 2.7.0 - - '@smithy/types@3.3.0': - dependencies: - tslib: 2.7.0 - - '@smithy/url-parser@3.0.3': - dependencies: - '@smithy/querystring-parser': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-base64@3.0.0': - dependencies: - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-body-length-browser@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-body-length-node@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-buffer-from@2.2.0': - dependencies: - '@smithy/is-array-buffer': 2.2.0 - tslib: 2.7.0 - - '@smithy/util-buffer-from@3.0.0': - dependencies: - '@smithy/is-array-buffer': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-config-provider@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-defaults-mode-browser@3.0.15': - dependencies: - '@smithy/property-provider': 3.1.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - bowser: 2.11.0 - tslib: 2.7.0 - - '@smithy/util-defaults-mode-node@3.0.15': - dependencies: - '@smithy/config-resolver': 3.0.5 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-endpoints@2.0.5': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-hex-encoding@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-middleware@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-retry@3.0.3': - dependencies: - '@smithy/service-error-classification': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-stream@3.1.3': - dependencies: - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-base64': 3.0.0 - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-hex-encoding': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-uri-escape@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-utf8@2.3.0': - dependencies: - '@smithy/util-buffer-from': 2.2.0 - tslib: 2.7.0 - - '@smithy/util-utf8@3.0.0': - dependencies: - '@smithy/util-buffer-from': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-waiter@3.1.2': - dependencies: - '@smithy/abort-controller': 3.1.1 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@tsconfig/node10@1.0.11': {} - - '@tsconfig/node12@1.0.11': {} - - '@tsconfig/node14@1.0.3': {} - - '@tsconfig/node16@1.0.4': {} - - '@types/chai@4.3.20': {} - - '@types/debug@4.1.12': - dependencies: - '@types/ms': 0.7.34 - - '@types/interpret@1.1.3': - dependencies: - '@types/node': 22.5.2 - - '@types/mocha@10.0.9': {} - - '@types/ms@0.7.34': {} - - '@types/node@20.16.3': - dependencies: - undici-types: 6.19.8 - - '@types/node@22.5.2': - dependencies: - undici-types: 6.19.8 - - '@types/pg@8.11.8': - dependencies: - '@types/node': 22.5.2 - pg-protocol: 1.6.1 - pg-types: 4.0.2 - - '@types/retry@0.12.2': {} - - '@types/semver@7.5.8': {} - - acorn-walk@8.3.3: - dependencies: - acorn: 8.12.1 - - acorn@8.12.1: {} - - ansi-colors@4.1.3: {} - - ansi-regex@5.0.1: {} - - ansi-styles@3.2.1: - dependencies: - color-convert: 1.9.3 - - ansi-styles@4.3.0: - dependencies: - color-convert: 2.0.1 - - anymatch@3.1.3: - dependencies: - normalize-path: 3.0.0 - picomatch: 2.3.1 - - arg@4.1.3: {} - - argparse@2.0.1: {} - - assertion-error@2.0.1: {} - - balanced-match@1.0.2: {} - - base64-js@1.5.1: {} - - binary-extensions@2.3.0: {} - - bowser@2.11.0: {} - - brace-expansion@1.1.11: - dependencies: - balanced-match: 1.0.2 - concat-map: 0.0.1 - - brace-expansion@2.0.1: - dependencies: - balanced-match: 1.0.2 - - braces@3.0.3: - dependencies: - fill-range: 7.1.1 - - browser-stdout@1.3.1: {} - - buffer@5.6.0: - dependencies: - base64-js: 1.5.1 - ieee754: 1.2.1 - - callsites@3.1.0: {} - - camelcase@6.3.0: {} - - chai@5.1.2: - dependencies: - assertion-error: 2.0.1 - check-error: 2.1.1 - deep-eql: 5.0.2 - loupe: 3.1.2 - pathval: 2.0.0 - - chalk@2.4.2: - dependencies: - ansi-styles: 3.2.1 - escape-string-regexp: 1.0.5 - supports-color: 5.5.0 - - chalk@4.1.2: - dependencies: - ansi-styles: 4.3.0 - supports-color: 7.2.0 - - check-error@2.1.1: {} - - chokidar@3.6.0: - dependencies: - anymatch: 3.1.3 - braces: 3.0.3 - glob-parent: 5.1.2 - is-binary-path: 2.1.0 - is-glob: 4.0.3 - normalize-path: 3.0.0 - readdirp: 3.6.0 - optionalDependencies: - fsevents: 2.3.3 - - cliui@7.0.4: - dependencies: - string-width: 4.2.3 - strip-ansi: 6.0.1 - wrap-ansi: 7.0.0 - - cliui@8.0.1: - dependencies: - string-width: 4.2.3 - strip-ansi: 6.0.1 - wrap-ansi: 7.0.0 - - color-convert@1.9.3: - dependencies: - color-name: 1.1.3 - - color-convert@2.0.1: - dependencies: - color-name: 1.1.4 - - color-name@1.1.3: {} - - color-name@1.1.4: {} - - concat-map@0.0.1: {} - - cosmiconfig@8.3.6(typescript@5.5.4): - dependencies: - import-fresh: 3.3.0 - js-yaml: 4.1.0 - parse-json: 5.2.0 - path-type: 4.0.0 - optionalDependencies: - typescript: 5.5.4 - - create-require@1.1.1: {} - - debug@4.3.6: - dependencies: - ms: 2.1.2 - - debug@4.3.6(supports-color@5.5.0): - dependencies: - ms: 2.1.2 - optionalDependencies: - supports-color: 5.5.0 - - debug@4.3.6(supports-color@8.1.1): - dependencies: - ms: 2.1.2 - optionalDependencies: - supports-color: 8.1.1 - - decamelize@4.0.0: {} - - deep-eql@5.0.2: {} - - diff@4.0.2: {} - - diff@5.2.0: {} - - dotenv@16.4.5: {} - - emoji-regex@8.0.0: {} - - error-ex@1.3.2: - dependencies: - is-arrayish: 0.2.1 - - esbuild@0.23.1: - optionalDependencies: - '@esbuild/aix-ppc64': 0.23.1 - '@esbuild/android-arm': 0.23.1 - '@esbuild/android-arm64': 0.23.1 - '@esbuild/android-x64': 0.23.1 - '@esbuild/darwin-arm64': 0.23.1 - '@esbuild/darwin-x64': 0.23.1 - '@esbuild/freebsd-arm64': 0.23.1 - '@esbuild/freebsd-x64': 0.23.1 - '@esbuild/linux-arm': 0.23.1 - '@esbuild/linux-arm64': 0.23.1 - '@esbuild/linux-ia32': 0.23.1 - '@esbuild/linux-loong64': 0.23.1 - '@esbuild/linux-mips64el': 0.23.1 - '@esbuild/linux-ppc64': 0.23.1 - '@esbuild/linux-riscv64': 0.23.1 - '@esbuild/linux-s390x': 0.23.1 - '@esbuild/linux-x64': 0.23.1 - '@esbuild/netbsd-x64': 0.23.1 - '@esbuild/openbsd-arm64': 0.23.1 - '@esbuild/openbsd-x64': 0.23.1 - '@esbuild/sunos-x64': 0.23.1 - '@esbuild/win32-arm64': 0.23.1 - '@esbuild/win32-ia32': 0.23.1 - '@esbuild/win32-x64': 0.23.1 - - escalade@3.2.0: {} - - escape-string-regexp@1.0.5: {} - - escape-string-regexp@4.0.0: {} - - events@3.3.0: {} - - fast-xml-parser@4.4.1: - dependencies: - strnum: 1.0.5 - - fill-range@7.1.1: - dependencies: - to-regex-range: 5.0.1 - - find-up@5.0.0: - dependencies: - locate-path: 6.0.0 - path-exists: 4.0.0 - - flat@5.0.2: {} - - fs.realpath@1.0.0: {} - - fsevents@2.3.3: - optional: true - - get-caller-file@2.0.5: {} - - get-tsconfig@4.8.0: - dependencies: - resolve-pkg-maps: 1.0.0 - - glob-parent@5.1.2: - dependencies: - is-glob: 4.0.3 - - glob@8.1.0: - dependencies: - fs.realpath: 1.0.0 - inflight: 1.0.6 - inherits: 2.0.4 - minimatch: 5.1.6 - once: 1.4.0 - - graphile-config@0.0.1-beta.9: - dependencies: - '@types/interpret': 1.1.3 - '@types/node': 20.16.3 - '@types/semver': 7.5.8 - chalk: 4.1.2 - debug: 4.3.6 - interpret: 3.1.1 - semver: 7.6.3 - tslib: 2.7.0 - yargs: 17.7.2 - transitivePeerDependencies: - - supports-color - - graphile-worker@0.16.6(typescript@5.5.4): - dependencies: - '@graphile/logger': 0.2.0 - '@types/debug': 4.1.12 - '@types/pg': 8.11.8 - cosmiconfig: 8.3.6(typescript@5.5.4) - graphile-config: 0.0.1-beta.9 - json5: 2.2.3 - pg: 8.12.0 - tslib: 2.7.0 - yargs: 17.7.2 - transitivePeerDependencies: - - pg-native - - supports-color - - typescript - - has-flag@3.0.0: {} - - has-flag@4.0.0: {} - - he@1.2.0: {} - - ieee754@1.2.1: {} - - ignore-by-default@1.0.1: {} - - import-fresh@3.3.0: - dependencies: - parent-module: 1.0.1 - resolve-from: 4.0.0 - - inflight@1.0.6: - dependencies: - once: 1.4.0 - wrappy: 1.0.2 - - inherits@2.0.4: {} - - interpret@3.1.1: {} - - is-arrayish@0.2.1: {} - - is-binary-path@2.1.0: - dependencies: - binary-extensions: 2.3.0 - - is-extglob@2.1.1: {} - - is-fullwidth-code-point@3.0.0: {} - - is-glob@4.0.3: - dependencies: - is-extglob: 2.1.1 - - is-network-error@1.1.0: {} - - is-number@7.0.0: {} - - is-plain-obj@2.1.0: {} - - is-unicode-supported@0.1.0: {} - - js-tokens@4.0.0: {} - - js-yaml@4.1.0: - dependencies: - argparse: 2.0.1 - - json-parse-even-better-errors@2.3.1: {} - - json5@2.2.3: {} - - lines-and-columns@1.2.4: {} - - locate-path@6.0.0: - dependencies: - p-locate: 5.0.0 - - log-symbols@4.1.0: - dependencies: - chalk: 4.1.2 - is-unicode-supported: 0.1.0 - - loupe@3.1.2: {} - - make-error@1.3.6: {} - - minimatch@3.1.2: - dependencies: - brace-expansion: 1.1.11 - - minimatch@5.1.6: - dependencies: - brace-expansion: 2.0.1 - - mocha@10.7.3: - dependencies: - ansi-colors: 4.1.3 - browser-stdout: 1.3.1 - chokidar: 3.6.0 - debug: 4.3.6(supports-color@8.1.1) - diff: 5.2.0 - escape-string-regexp: 4.0.0 - find-up: 5.0.0 - glob: 8.1.0 - he: 1.2.0 - js-yaml: 4.1.0 - log-symbols: 4.1.0 - minimatch: 5.1.6 - ms: 2.1.3 - serialize-javascript: 6.0.2 - strip-json-comments: 3.1.1 - supports-color: 8.1.1 - workerpool: 6.5.1 - yargs: 16.2.0 - yargs-parser: 20.2.9 - yargs-unparser: 2.0.0 - - ms@2.1.2: {} - - ms@2.1.3: {} - - nodemon@3.1.4: - dependencies: - chokidar: 3.6.0 - debug: 4.3.6(supports-color@5.5.0) - ignore-by-default: 1.0.1 - minimatch: 3.1.2 - pstree.remy: 1.1.8 - semver: 7.6.3 - simple-update-notifier: 2.0.0 - supports-color: 5.5.0 - touch: 3.1.1 - undefsafe: 2.0.5 - - normalize-path@3.0.0: {} - - obuf@1.1.2: {} - - once@1.4.0: - dependencies: - wrappy: 1.0.2 - - p-limit@3.1.0: - dependencies: - yocto-queue: 0.1.0 - - p-locate@5.0.0: - dependencies: - p-limit: 3.1.0 - - p-retry@6.2.0: - dependencies: - '@types/retry': 0.12.2 - is-network-error: 1.1.0 - retry: 0.13.1 - - parent-module@1.0.1: - dependencies: - callsites: 3.1.0 - - parse-json@5.2.0: - dependencies: - '@babel/code-frame': 7.24.7 - error-ex: 1.3.2 - json-parse-even-better-errors: 2.3.1 - lines-and-columns: 1.2.4 - - path-exists@4.0.0: {} - - path-type@4.0.0: {} - - pathval@2.0.0: {} - - pg-cloudflare@1.1.1: - optional: true - - pg-connection-string@2.6.4: {} - - pg-int8@1.0.1: {} - - pg-numeric@1.0.2: {} - - pg-pool@3.6.2(pg@8.12.0): - dependencies: - pg: 8.12.0 - - pg-protocol@1.6.1: {} - - pg-types@2.2.0: - dependencies: - pg-int8: 1.0.1 - postgres-array: 2.0.0 - postgres-bytea: 1.0.0 - postgres-date: 1.0.7 - postgres-interval: 1.2.0 - - pg-types@4.0.2: - dependencies: - pg-int8: 1.0.1 - pg-numeric: 1.0.2 - postgres-array: 3.0.2 - postgres-bytea: 3.0.0 - postgres-date: 2.1.0 - postgres-interval: 3.0.0 - postgres-range: 1.1.4 - - pg@8.12.0: - dependencies: - pg-connection-string: 2.6.4 - pg-pool: 3.6.2(pg@8.12.0) - pg-protocol: 1.6.1 - pg-types: 2.2.0 - pgpass: 1.0.5 - optionalDependencies: - pg-cloudflare: 1.1.1 - - pgpass@1.0.5: - dependencies: - split2: 4.2.0 - - picocolors@1.1.0: {} - - picomatch@2.3.1: {} - - postgres-array@2.0.0: {} - - postgres-array@3.0.2: {} - - postgres-bytea@1.0.0: {} - - postgres-bytea@3.0.0: - dependencies: - obuf: 1.1.2 - - postgres-date@1.0.7: {} - - postgres-date@2.1.0: {} - - postgres-interval@1.2.0: - dependencies: - xtend: 4.0.2 - - postgres-interval@3.0.0: {} - - postgres-range@1.1.4: {} - - pstree.remy@1.1.8: {} - - randombytes@2.1.0: - dependencies: - safe-buffer: 5.2.1 - - readable-stream@3.6.2: - dependencies: - inherits: 2.0.4 - string_decoder: 1.3.0 - util-deprecate: 1.0.2 - - readdirp@3.6.0: - dependencies: - picomatch: 2.3.1 - - require-directory@2.1.1: {} - - resolve-from@4.0.0: {} - - resolve-pkg-maps@1.0.0: {} - - retry@0.13.1: {} - - safe-buffer@5.2.1: {} - - semver@7.6.3: {} - - serialize-javascript@6.0.2: - dependencies: - randombytes: 2.1.0 - - simple-update-notifier@2.0.0: - dependencies: - semver: 7.6.3 - - split2@4.2.0: {} - - stream-browserify@3.0.0: - dependencies: - inherits: 2.0.4 - readable-stream: 3.6.2 - - string-width@4.2.3: - dependencies: - emoji-regex: 8.0.0 - is-fullwidth-code-point: 3.0.0 - strip-ansi: 6.0.1 - - string_decoder@1.3.0: - dependencies: - safe-buffer: 5.2.1 - - strip-ansi@6.0.1: - dependencies: - ansi-regex: 5.0.1 - - strip-json-comments@3.1.1: {} - - strnum@1.0.5: {} - - supports-color@5.5.0: - dependencies: - has-flag: 3.0.0 - - supports-color@7.2.0: - dependencies: - has-flag: 4.0.0 - - supports-color@8.1.1: - dependencies: - has-flag: 4.0.0 - - to-regex-range@5.0.1: - dependencies: - is-number: 7.0.0 - - touch@3.1.1: {} - - ts-node@10.9.2(@types/node@22.5.2)(typescript@5.5.4): - dependencies: - '@cspotcode/source-map-support': 0.8.1 - '@tsconfig/node10': 1.0.11 - '@tsconfig/node12': 1.0.11 - '@tsconfig/node14': 1.0.3 - '@tsconfig/node16': 1.0.4 - '@types/node': 22.5.2 - acorn: 8.12.1 - acorn-walk: 8.3.3 - arg: 4.1.3 - create-require: 1.1.1 - diff: 4.0.2 - make-error: 1.3.6 - typescript: 5.5.4 - v8-compile-cache-lib: 3.0.1 - yn: 3.1.1 - - tslib@2.7.0: {} - - tsx@4.19.0: - dependencies: - esbuild: 0.23.1 - get-tsconfig: 4.8.0 - optionalDependencies: - fsevents: 2.3.3 - - typescript@5.5.4: {} - - undefsafe@2.0.5: {} - - undici-types@6.19.8: {} - - util-deprecate@1.0.2: {} - - uuid@9.0.1: {} - - v8-compile-cache-lib@3.0.1: {} - - workerpool@6.5.1: {} - - wrap-ansi@7.0.0: - dependencies: - ansi-styles: 4.3.0 - string-width: 4.2.3 - strip-ansi: 6.0.1 - - wrappy@1.0.2: {} - - xtend@4.0.2: {} - - y18n@5.0.8: {} - - yargs-parser@20.2.9: {} - - yargs-parser@21.1.1: {} - - yargs-unparser@2.0.0: - dependencies: - camelcase: 6.3.0 - decamelize: 4.0.0 - flat: 5.0.2 - is-plain-obj: 2.1.0 - - yargs@16.2.0: - dependencies: - cliui: 7.0.4 - escalade: 3.2.0 - get-caller-file: 2.0.5 - require-directory: 2.1.1 - string-width: 4.2.3 - y18n: 5.0.8 - yargs-parser: 20.2.9 - - yargs@17.7.2: - dependencies: - cliui: 8.0.1 - escalade: 3.2.0 - get-caller-file: 2.0.5 - require-directory: 2.1.1 - string-width: 4.2.3 - y18n: 5.0.8 - yargs-parser: 21.1.1 - - yn@3.1.1: {} - - yocto-queue@0.1.0: {} diff --git a/services/factory/src/config.ts b/services/factory/src/config.ts deleted file mode 100644 index a454d6c..0000000 --- a/services/factory/src/config.ts +++ /dev/null @@ -1,70 +0,0 @@ -import { config } from 'dotenv' -import { join, dirname } from 'node:path' -import { fileURLToPath } from 'node:url'; -const __dirname = dirname(fileURLToPath(import.meta.url)); -config({ path: join(__dirname, '../../../.env.development') }) - - -if (!process.env.WORKER_CONNECTION_STRING) throw new Error("WORKER_CONNECTION_STRING was missing from env"); -if (!process.env.POSTGREST_URL) throw new Error('Missing POSTGREST_URL env var'); -if (!process.env.AUTOMATION_USER_JWT) throw new Error('Missing AUTOMATION_USER_JWT env var'); -if (!process.env.S3_ACCESS_KEY_ID) throw new Error('Missing S3_ACCESS_KEY_ID env var'); -if (!process.env.S3_SECRET_ACCESS_KEY) throw new Error('Missing S3_BUCKET_APPLICATION_KEY env var'); -if (!process.env.S3_REGION) throw new Error('Missing S3_REGION env var'); -if (!process.env.S3_ENDPOINT) throw new Error('Missing S3_REGION env var'); -if (!process.env.S3_MAIN_BUCKET) throw new Error('Missing S3_BUCKET env var'); -if (!process.env.S3_USC_BUCKET) throw new Error('Missing S3_USC_BUCKET env var'); -if (!process.env.CACHE_DIR) throw new Error('Missing CACHE_DIR env var'); -if (!process.env.PATREON_CREATOR_ACCESS_TOKEN) throw new Error('Missing PATREON_CREATOR_ACCESS_TOKEN env var'); -if (!process.env.PATREON_CREATOR_REFRESH_TOKEN) throw new Error('Missing PATREON_CREATOR_REFRESH_TOKEN env var'); -if (!process.env.SUPERTOKENS_URL) throw new Error('Missing SUPERTOKENS_URL env var'); -if (!process.env.SUPERTOKENS_API_KEY) throw new Error('Missing SUPERTOKENS_API_KEY env var'); -const postgrestUrl = process.env.POSTGREST_URL! -const automationUserJwt = process.env.AUTOMATION_USER_JWT! -const connectionString = process.env.WORKER_CONNECTION_STRING! -const s3AccessKeyId = process.env.S3_ACCESS_KEY_ID! -const s3Region = process.env.S3_REGION! -const s3Endpoint = process.env.S3_ENDPOINT! -const s3SecretAccessKey = process.env.S3_SECRET_ACCESS_KEY! -const s3MainBucket = process.env.S3_MAIN_BUCKET! -const s3UscBucket = process.env.S3_USC_BUCKET! -const cacheDir = process.env.CACHE_DIR! -const patreonCreatorAccessToken = process.env.PATREON_CREATOR_ACCESS_TOKEN! -const patreonCreatorRefreshToken = process.env.PATREON_CREATOR_REFRESH_TOKEN! -const supertokensUrl = process.env.SUPERTOKENS_URL! -const supertokensApiKey = process.env.SUPERTOKENS_API_KEY! - -export interface Config { - postgrestUrl: string; - automationUserJwt: string; - connectionString: string; - s3AccessKeyId: string; - s3SecretAccessKey: string; - s3Region: string; - s3Endpoint: string; - s3UscBucket: string; - s3MainBucket: string; - cacheDir: string; - patreonCreatorAccessToken: string; - patreonCreatorRefreshToken: string; - supertokensUrl: string; - supertokensApiKey: string; -} - - -export const configs: Config = { - postgrestUrl, - automationUserJwt, - connectionString, - s3AccessKeyId, - s3SecretAccessKey, - s3Endpoint, - s3Region, - s3MainBucket, - s3UscBucket, - cacheDir, - patreonCreatorAccessToken, - patreonCreatorRefreshToken, - supertokensUrl, - supertokensApiKey, -} diff --git a/services/factory/src/index.ts b/services/factory/src/index.ts deleted file mode 100644 index e961f07..0000000 --- a/services/factory/src/index.ts +++ /dev/null @@ -1,45 +0,0 @@ - -import type { RunnerOptions } from 'graphile-worker' -import type { GraphileConfig } from "graphile-config"; -import { run } from 'graphile-worker' -import { join, dirname } from 'node:path' -import { fileURLToPath } from 'url' -import { configs } from './config.ts' - -const __dirname = dirname(fileURLToPath(import.meta.url)); - - -async function setupGraphileWorker() { - try { - const preset: GraphileConfig.Preset = { - worker: { - connectionString: configs.connectionString, - concurrentJobs: 3, - fileExtensions: [".js", ".ts"], - taskDirectory: join(__dirname, 'tasks') - }, - }; - // console.log('worker preset as follows') - // console.log(preset) - const runnerOptions: RunnerOptions = { - preset - } - - const runner = await run(runnerOptions) - if (!runner) throw new Error('failed to initialize graphile worker'); - await runner.promise - } catch (e) { - console.error('error caught during setupGraphileWorker') - console.error(e) - } -} - -async function main() { - await setupGraphileWorker() -} - -main().catch((e) => { - console.error("error during main() function") - console.error(e) - process.exit(3) -}) \ No newline at end of file diff --git a/services/factory/src/tasks/combine_video_segments.ts b/services/factory/src/tasks/combine_video_segments.ts deleted file mode 100644 index 8781c2d..0000000 --- a/services/factory/src/tasks/combine_video_segments.ts +++ /dev/null @@ -1,337 +0,0 @@ - -import { Helpers, type Task } from 'graphile-worker' -import { basename, join } from 'node:path'; -import { S3Client } from "@aws-sdk/client-s3" -import { Upload } from "@aws-sdk/lib-storage" -import { createId } from '@paralleldrive/cuid2'; -import { execFile } from 'node:child_process'; -import { configs } from '../config'; -import { PassThrough, Readable } from 'node:stream'; -import { createReadStream } from 'node:fs'; -import { stat, writeFile } from 'node:fs/promises'; -import { pipeline } from 'node:stream/promises'; -import { tmpdir } from 'node:os'; -import patchVodInDatabase from '@futureporn/fetchers/patchVodInDatabase.ts' -import { downloadFile } from '@futureporn/storage/s3.ts'; -import { S3FileRecord, SegmentResponse } from '@futureporn/types'; -import getVod from '@futureporn/fetchers/getVod.ts'; -import { type S3ClientConfig } from "@aws-sdk/client-s3"; -import pRetry, {AbortError} from 'p-retry' -import { getFileChecksum, getTmpFile } from '@futureporn/utils/file.ts'; - -interface s3ManifestEntry { - key: string; - bytes: number; -} - -interface Payload { - vod_id?: string; - s3_manifest?: S3FileRecord[]; -} - -interface S3Target { - Bucket: string; - Key: string; - Body: Readable; -} - -interface S3UploadParameters { - bucket: string; - keyName: string; - uploadStream: PassThrough; - client: S3Client; - onProgress?: Function; -} - - -function assertPayload(payload: any): asserts payload is Payload { - if (typeof payload !== "object" || !payload) throw new Error("invalid payload"); -} - -async function withRetry(fn: () => Promise, retries = 3): Promise { - return pRetry(fn, { - onFailedAttempt: (e: Error) => { - console.error(`Error during attempt:`, e); - }, - retries - }); -} - -/** - * - * generate a txt file on disk which ffmpeg's concat filter will use to concatenate files. - * example: ffmpeg -f concat -safe 0 -i ./files.txt -c copy ./projektmelody-chaturbate-2023-07-18.mp4 - * the text file is written to os.tmpdir() - * the text file contents looks like the following. - * - * file './cb-recording-part-1.mp4' - * file './cb-recording-part-2.mp4' - * file './cb-recording-part-3.mp4' - */ -const getFFmpegConcatSpecFile = async function (inputVideoFilePaths: string[]): Promise { - if (!inputVideoFilePaths) throw new Error('getFFmpegConcatSpec file requires an array of filepaths as argument, but it was undefined'); - if (inputVideoFilePaths.length < 1) throw new Error('getFFmpegConcatSpec arg0, inputVideoFilePaths was length 0 which is unsupported.'); - let lines = [] - for (const filePath of inputVideoFilePaths) { - lines.push(`file '${filePath}'`) - } - const specFilePath = join(tmpdir(), `concat-spec-${createId()}`) - await writeFile(specFilePath, lines.join('\n'), { encoding: 'utf-8' }) - return specFilePath -} - -const getFFmpegConcatenation = async function (specFilePath: string, outputPath: string) { - if (!specFilePath) throw new Error('getFFmpegStream requires specFilePath as arg'); - const execFileP = promisify(execFile) - const { stdout, stderr } = await execFileP('ffmpeg', [ - '-f', 'concat', - '-safe', '0', - '-i', specFilePath, - '-c', 'copy', - outputPath - ]) - console.log(stdout) - console.log(stderr) - return outputPath -} - -const concatVideos = async function (videoFilePaths: string[]): Promise { - console.log(`concatVideos with ${JSON.stringify(videoFilePaths)}`) - if (!videoFilePaths || videoFilePaths.length < 1 || typeof videoFilePaths[0] !== 'string') throw new Error('concatVideos requires videoFilePaths as arg, but it was undefined or not an array of strings'); - if (videoFilePaths.length === 1) { - // if there is only one video, we don't need to do anything. - return videoFilePaths[0] - } - const concatSpec = await getFFmpegConcatSpecFile(videoFilePaths) - const outputVideoPath = join(tmpdir(), `${basename(videoFilePaths[0], '.mp4')}-${createId()}.mp4`) - try { - await getFFmpegConcatenation(concatSpec, outputVideoPath) - } catch (err) { - console.error(`error encountered while concatenating video files together`) - console.error(err) - throw err - } - return outputVideoPath -} - - - - -const getS3ParallelUpload = async function ({ - filePath, - client, - s3KeyName -}: { - s3KeyName: string, - filePath: string, - client: S3Client, -}): Promise<{upload: Upload, uploadStream: PassThrough}> { - if (!filePath) throw new Error("first argument passed to uploadToS3, 'filePath' is undefined"); - - console.log(`uploading ${s3KeyName} to S3`) - - const uploadStream = new PassThrough() - const target: S3Target = { - Bucket: configs.s3UscBucket, - Key: s3KeyName, - Body: uploadStream - } - - const upload = new Upload({ - client, - partSize: 1024 * 1024 * 5, - queueSize: 1, - leavePartsOnError: false, - params: target, - }); - - - return { upload, uploadStream } -} - - -/** - * Checks if a file exists at the specified path. - */ -async function fileExists(path: string): Promise { - try { - await stat(path); - return true; - } catch { - return false; - } -} - -/** - * Validates checksum if an expected checksum is provided. - */ -async function validateChecksumIfNeeded(filePath: string, segment: SegmentResponse) { - const expectedChecksum = segment?.checksum; - if (expectedChecksum) { - const actualChecksum = await getFileChecksum(filePath, 'md5'); - if (expectedChecksum !== actualChecksum) { - throw new Error(`Downloaded segment ${segment.id} but the expected checksum ${expectedChecksum} did not match actual ${actualChecksum}.`); - } - } -} - -/** - * downloadSegment - * - * - [x] If the file is already in local cache, that file is used. - * - [x] Validates checksum to ensure file is whole - */ -async function downloadSegment(client: S3Client, segment: SegmentResponse) { - if (!segment) throw new Error('segment passed to downloadSegment was missing'); - if (!segment.s3_key) throw new Error('segment passed to downloadSegment was missing s3_key'); - - const cachePath = join(configs.cacheDir, segment.s3_key); - const isFileInCache = await fileExists(cachePath); - - // Check and return cache if available - if (isFileInCache) { - await validateChecksumIfNeeded(cachePath, segment); - return cachePath; - } - - // Download segment and validate checksum - const tmpFilePath = await withRetry(() => downloadFile(client, configs.s3UscBucket, segment.s3_key), 3); - await validateChecksumIfNeeded(tmpFilePath, segment); - - return tmpFilePath; -} - -/** - * downloadSegments - * - * Download a list of segments from S3. - - */ -async function downloadSegments(client: S3Client, segments: SegmentResponse[]) { - for (const segment of segments) { - const segmentFilePath = await downloadSegment(client, segment) - } -} - -/** - * doIntegratedCombine - * - * Integrated combine_video_segments is when the requester is giving us only a vod_id - * It's our job to inspect the vod and get it's segments and combine them - * Then upload the result to S3 and update the vod's s3_file - * - */ -async function doIntegratedCombine(helpers: Helpers, vod_id: string) { - /** - * Here we take a manifest of S3 files and we download each of them. - * Then we combine them all, preserving original order using `ffmpeg -f concat` - * Then we upload the resulting video to S3 - * Then we create records in Postgrest - * * s3_file - * * vod - * - * After the records are created, - * if we were told about a stream record that this recording belongs to, - * we edit the stream record, adding a relation to the vod we just created. - */ - - try { - - const vod = await getVod(vod_id) - if (!vod) throw new Error('doIntegratedCombine failed to get vod'); - - // * [x] Get a list of segments associated with this vod. - // * [ ] Download the segments (or pull from disk cache) - // * [ ] VALIDATE segment checksums(?) - // * [ ] Combine segments using ffmpeg - // * [ ] Upload resulting video - // * [ ] Create s3_file with resulting video information - // * [ ] Update s3_file vod record to reference s3_file - - const options: S3ClientConfig = { - endpoint: configs.s3Endpoint, - region: configs.s3Region, - credentials: { - accessKeyId: configs.s3AccessKeyId, - secretAccessKey: configs.s3SecretAccessKey - } - } - const client = new S3Client(options); - - if (!vod.segments) throw new Error('vod.segments was missing'); - const segments = vod.segments - - const localSegments = await downloadSegments(client, segments) - - const s3Manifest = s3_manifest - - - const inputVideoFilePaths = await Promise.all(s3Manifest.filter((m: S3FileRecord) => (m.bytes !== 0)).map((m) => downloadFile(client, configs.s3UscBucket, m.key))) - const concatenatedVideoFile = await concatVideos(inputVideoFilePaths) - const s3KeyName = basename(concatenatedVideoFile) - const inputStream = createReadStream(concatenatedVideoFile) - const filePath = concatenatedVideoFile - const { uploadStream, upload } = await getS3ParallelUpload({ client, s3KeyName, filePath }) - pipeline(inputStream, uploadStream) - await upload.done() - - if (!vod_id) throw new Error('vod_id was missing from payload'); - // update the vod with the s3_file of the combined video - const payload = { - s3_file: s3KeyName, - vod_id - } - await patchVodInDatabase(vod_id, payload) - - - } catch (e: any) { - helpers.logger.error('combine_video_segments failed') - if (e instanceof Error) { - helpers.logger.error(e.message) - } else { - helpers.logger.error(e) - } - throw e - } - -} - - -/** - * doSoloRequest - * - * Solo combine_video_segments is when the requester is giving us a s3_manifest - * Which is just a S3FileRecord[] - * - * It's our job to download those files and combine them, then upload the result to S3 - */ -async function doSoloCombine(helpers: Helpers, s3_manifest: S3FileRecord[]) { - const s3Manifest = s3_manifest - const inputVideoFilePaths = await Promise.all(s3Manifest.filter((m) => (m.bytes !== 0)).map((m) => downloadFile(client, configs.s3UscBucket, m.key))) -} - - -export const combine_video_segments: Task = async function (payload: unknown, helpers: Helpers) { - // helpers.logger.info('the following is the raw Task payload') - // helpers.logger.info(payload) - // helpers.logger.info(JSON.stringify(payload?.s3_manifest)) - assertPayload(payload) - const { s3_manifest, vod_id } = payload - if (!vod_id) throw new Error('combine_video_segments was called without a vod_id.'); - helpers.logger.info(`🏗️ combine_video_segments started with s3_manifest=${JSON.stringify(s3_manifest)}, vod_id=${vod_id}`) - - const isSoloRequest = (!!vod_id && !s3_manifest) - const isIntegratedRequest = (!!s3_manifest?.length && !vod_id) - - if (isSoloRequest && !isIntegratedRequest) { - await doSoloCombine(helpers, s3_manifest!) - } else if (isIntegratedRequest && !isSoloRequest) { - await doIntegratedCombine(helpers, vod_id) - } else { - throw new Error(`Ambiguous request. Use either s3_manifest or vod_id argument, not both.`); - } - -} - - -export default combine_video_segments \ No newline at end of file diff --git a/services/factory/src/tasks/generate_thumbnail.ts b/services/factory/src/tasks/generate_thumbnail.ts deleted file mode 100644 index 469dba1..0000000 --- a/services/factory/src/tasks/generate_thumbnail.ts +++ /dev/null @@ -1,92 +0,0 @@ -import type { Task, Helpers } from "graphile-worker"; -import getVod from "@futureporn/fetchers/getVod.ts"; -import { getStoryboard } from '@futureporn/utils/image.ts' -import { getCdnUrl, uploadFile, type S3FileArgs } from '@futureporn/storage/s3.ts' -import patchVodInDatabase from "@futureporn/fetchers/patchVodInDatabase.ts"; -import { configs } from "../config"; - -interface Payload { - vod_id?: string; - video_url?: string; -} - -function assertPayload(payload: any): asserts payload is Payload { - if (typeof payload !== "object" || !payload) throw new Error("invalid payload (it must be an object)"); - if (typeof payload.vod_id !== "string" && typeof payload.video_url !== "string") throw new Error("payload requires either vod_id or video_url, however both were not a string."); -} - - -async function doIntegratedRequest(vodId: string): Promise { - - const vod = await getVod(vodId) - const s3_file = vod?.s3_file - if (!s3_file) throw new Error(`vod ${vodId} was missing a s3_file.`); - - // we need to get a CDN url to the vod so we can download chunks of the file in order for Prevvy to create the storyboard image. - const cdnUrl = getCdnUrl(configs.s3MainBucket, s3_file.s3_key) - - const thumbnailCdnUrl = await getThumbnailCdnUrl(cdnUrl) - - await patchVodInDatabase(vodId, { thumbnail: thumbnailCdnUrl }) - - - -} - - -async function getThumbnailCdnUrl(videoUrl: string): Promise { - - const tmpImagePath = await getStoryboard(videoUrl) - - // we need to upload the image to S3 - const uploadArgs: S3FileArgs = { - filePath: tmpImagePath, - s3AccessKeyId: configs.s3AccessKeyId, - s3SecretAccessKey: configs.s3SecretAccessKey, - s3BucketName: configs.s3MainBucket, - s3Endpoint: configs.s3Region, - s3Region: configs.s3Region - } - - const upload = await uploadFile(uploadArgs) - if (!upload) throw new Error(`failed to upload ${tmpImagePath} to S3`); - if (!upload.Key) throw new Error(`failed to upload ${tmpImagePath} to S3 (upload.Key was missing)`); - - // we need to create a S3 file in the db - const thumbnail = getCdnUrl(configs.s3MainBucket, upload.Key) - - return thumbnail - -} - -async function doSoloRequest(videoUrl: string): Promise { - await getThumbnailCdnUrl(videoUrl) -} - - -export const generate_thumbnail: Task = async function (payload: unknown, helpers: Helpers) { - assertPayload(payload) - const { vod_id, video_url } = payload - - helpers.logger.info(`🏗️ generate_thumbnail started with vod_id=${vod_id}, video_url=${video_url}`); - - - // Determine what kind of request is being made. - // It could be one of two scenarios - // * Here is a VOD record in the database, please update it's thumbnail. (integratedRequest) - // * Here is a video URL, please give us a thumbnail URL. (soloRequest) - // - const integratedRequest = (!!vod_id && !video_url) - const soloRequest = (!!video_url && !vod_id) - - if (integratedRequest) { - await doIntegratedRequest(vod_id) - } else if (soloRequest) { - await getThumbnailCdnUrl(video_url) - } else { - throw new Error(`unsupported ambiguous request!`) - } - -} - -export default generate_thumbnail \ No newline at end of file diff --git a/services/factory/src/tasks/process_video.ts b/services/factory/src/tasks/process_video.ts deleted file mode 100644 index 1530b03..0000000 --- a/services/factory/src/tasks/process_video.ts +++ /dev/null @@ -1,72 +0,0 @@ -import type { Helpers, Task } from "graphile-worker" -import getVod from "@futureporn/fetchers/getVod.ts" - -interface Payload { - vod_id: string; -} - -function assertPayload(payload: any): asserts payload is Payload { - if (typeof payload !== "object" || !payload) throw new Error("invalid payload (it must be an object)"); - if (typeof payload.vod_id !== "string") throw new Error("payload.vod_id was not a string"); -} - - - - -/** - * - * # process_video - * - * We just recorded a livestream. Now what? - * process_video takes a /vods record and runs a bunch of processes to get it ready for publishing. - * - * The following are graphile-worker tasks which process_video is responsible for adding to the job queue. - * Some of these tasks are run conditionally based on the structure of the /vods record. - * For example, combine_video_segments is only useful on a vod recording which ended up with multiple segments. - * - * - combine_video_segments - * - remux_video - * - generate_thumbnail - * - queue_moderator_review - * - create_mux_asset - * - create_torrent - * - * Some of the above Tasks are dependent on others. generate_thumbnail and everything following it depends on combine_video_segments. - * graphile-worker doesn't have support for dependent tasks, - * thus our solution is to invoke ALL the above jobs immediately, and build those jobs to fail if pre-conditions are unmet. - * all listed jobs are also idempotent, which means if they don't need to run (previously already performed their tasks), they will exit without doing anything. - * - */ -const process_video: Task = async function (payload: unknown, helpers: Helpers) { - assertPayload(payload) - const { vod_id } = payload - helpers.logger.info(`🏗️ process_video task has begun for vod_id=${vod_id}`) - - - const vod = await getVod(vod_id) - if (!vod) throw new Error(`failed to get vod from database.`); - if (!vod.segments) throw new Error(`vod ${vod_id} fetched from database lacked any segments.`); - - const maxAttempts = 6 - const isCombinationNeeded = (vod.segments.length > 1) - if (isCombinationNeeded) { - const s3_manifest = vod.segments.map((segment) => ({ key: segment.s3_key, bytes: segment.bytes })) - helpers.logger.info(`There are ${vod.segments.length} segments; Concatenation is needed.`) - helpers.addJob('combine_video_segments', { s3_manifest, vod_id }, { maxAttempts }) - } else { - helpers.addJob('remux_video', { vod_id }, { maxAttempts }) - } - - helpers.addJob('generate_thumbnail', { vod_id }, { maxAttempts }) - // helpers.addJob('queue_moderator_review', { }) - // helpers.addJob('create_mux_asset', { }) - // helpers.addJob('create_torrent', { }) - // helpers.addJob('create_ipfs', { }) - - - - - -} - -export default process_video; diff --git a/services/factory/src/tasks/remux_video.ts b/services/factory/src/tasks/remux_video.ts deleted file mode 100644 index ac057f4..0000000 --- a/services/factory/src/tasks/remux_video.ts +++ /dev/null @@ -1,83 +0,0 @@ -import type { Helpers, Task } from "graphile-worker" -import { configs } from "../config" -import getVod from "@futureporn/fetchers/getVod.ts" -import { downloadFile, uploadFile, type S3FileArgs } from "@futureporn/storage/s3.ts" -import { remux } from '@futureporn/utils/video.ts' -import { getTmpFile } from "@futureporn/utils/file.ts" -import { basename } from "node:path" -import patchVodInDatabase from "@futureporn/fetchers/patchVodInDatabase.ts" -import { S3Client, S3ClientConfig } from "@aws-sdk/client-s3" - -interface Payload { - vod_id: string; -} - -function assertPayload(payload: any): asserts payload is Payload { - if (typeof payload !== "object" || !payload) throw new Error("invalid payload (it must be an object)"); - if (typeof payload.vod_id !== "string") throw new Error("payload.vod_id was not a string"); -} - - - - -/** - * - * # remux_video - * - * convert a .ts video file to a .mp4 video file (safely, without loss of quality) - */ -const remux_video: Task = async function (payload: unknown, helpers: Helpers) { - assertPayload(payload) - const { vod_id } = payload - helpers.logger.info(`🏗️ remux_video task has begun for vod_id=${vod_id}`) - - - const vod = await getVod(vod_id) - if (!vod) throw new Error(`failed to get vod from database.`); - if (!vod.segments) throw new Error(`vod ${vod_id} fetched from database lacked any segments.`); - if (!vod.segments.at(0)) throw new Error(`vod ${vod_id} lacked a recorded segment.`); - if (!vod.segments.at(0)?.s3_key) throw new Error(`vod ${vod_id} lacked a recorded segment s3_key name.`); - - const segmentFileName = vod.segments.at(0)!.s3_key - if (!segmentFileName.endsWith('.ts')) throw new Error(`vod ${vod_id} recording segment ${segmentFileName} did not end with .ts, which is expected.`); - - const moreThanOneSegment = (vod.segments.length > 1) - if (moreThanOneSegment) { - throw new Error(`remux_video was given vod=${vod_id} which has more than one segment. This is unsupported. (use combine_video_segments Task to combine multiple videos)`); - } - - const tmpFilePath = getTmpFile(segmentFileName) - const s3ClientConfig: S3ClientConfig = { - credentials: { - accessKeyId: configs.s3AccessKeyId, - secretAccessKey: configs.s3SecretAccessKey, - }, - endpoint: configs.s3Endpoint, - region: configs.s3Region, - } - const client = new S3Client(s3ClientConfig) - await downloadFile(client, configs.s3UscBucket, segmentFileName) - - helpers.logger.info('Remuxing the video') - const outputVideoPath = getTmpFile(`${basename(segmentFileName, '.ts')}.mp4`) - await remux(tmpFilePath, outputVideoPath) - - helpers.logger.info('Uploading the remuxed video') - const uploadArgs: S3FileArgs = { - filePath: outputVideoPath, - s3AccessKeyId: configs.s3AccessKeyId, - s3SecretAccessKey: configs.s3SecretAccessKey, - s3BucketName: configs.s3MainBucket, - s3Endpoint: configs.s3Region, - s3Region: configs.s3Region - } - const upload = await uploadFile(uploadArgs) - - helpers.logger.info('Patching the vod in the database') - await patchVodInDatabase(vod_id, { s3_file: upload?.Key }) - -} - - - -export default remux_video; diff --git a/services/factory/src/tasks/synchronize_patrons_list.ts b/services/factory/src/tasks/synchronize_patrons_list.ts deleted file mode 100644 index a6f6ee2..0000000 --- a/services/factory/src/tasks/synchronize_patrons_list.ts +++ /dev/null @@ -1,140 +0,0 @@ -import type { Task, Helpers } from "graphile-worker"; -import { configs } from '../config.ts' -import { formatPatronUsername } from "@futureporn/utils/patron.ts"; -import { PublicPatron } from "@futureporn/utils/patron.ts"; - - - -// async function fetchAllPages(url: string, headers: HeadersInit = {}): Promise { -// const responses: Patreon.APIResponse[] = []; -// let cursor: string | null = null; - -// do { -// const fetchUrl = cursor ? `${url}&page[cursor]=${cursor}` : url; -// const response = await fetch(fetchUrl, { headers }); -// const jsonResponse: Patreon.APIResponse = await response.json(); - -// responses.push(jsonResponse); -// cursor = jsonResponse.meta?.pagination?.cursors?.next || null; -// } while (cursor); - -// return responses; -// } - - -/** - * getOptInPatrons - * - * Get a list of users who have the 'patron' role and have opted-in to having their username displayed on the site - * - * @see https://app.swaggerhub.com/apis/supertokens/CDI/5.1.1#/User%20Roles%20Recipe/getRoleUsers - * - * @todo this rapid-fires 600+ fetch() requests to supertokens. Do we need to throttle this? - */ -async function getPublicPatrons(): Promise { - const allPatrons = await getPatronIds() - let publicPatrons = [] - for (const patronId of allPatrons) { - const res = await fetch(`${configs.supertokensUrl}/recipe/user/metadata?userId=${patronId}`, { - method: 'GET', - headers: { - 'rid': 'usermetadata', - 'Authorization': configs.supertokensApiKey, - 'cdi-version': '5.1', - } - }) - if (!res.ok) { - throw new Error(`failed to fetch usermetadata from supertokens. res.status=${res.status}, res.statusText=${res.statusText}`) - } - const data = await res.json() - const metadata = data.metadata - const isUsernamePublic = metadata?.isUsernamePublic - if (isUsernamePublic) { - const patron: PublicPatron = { - full_name: formatPatronUsername(metadata.first_name, metadata.last_name), - id: patronId, - link: metadata?.vanityLink - } - publicPatrons.push(patron) - } - } - return publicPatrons -} - - -/** - * getPatronIds - * - * get the UUIDs of all patrons - */ -async function getPatronIds(): Promise { - const res = await fetch(`${configs.supertokensUrl}/recipe/role/users?role=patron`, { - method: 'GET', - headers: { - 'rid': 'userroles', - 'Authorization': configs.supertokensApiKey, - 'cdi-version': '5.1' - } - }) - if (!res.ok) { - throw new Error(`failed to getPublicPatrons() with res.status=${res.status} res.statusText=${res.statusText}`) - } - const data = await res.json() - const patrons = data?.users - return patrons -} - - - - -const updatePatronsTable = async function (patronsList: PublicPatron[]): Promise { - console.log(`Syncronizing ${patronsList.length} patron name to the public db`) - // console.log(patronsList) - const res = await fetch(`${configs.postgrestUrl}/patrons`, { - method: 'POST', - headers: { - 'Authorization': `Bearer ${configs.automationUserJwt}`, - 'Prefer': 'resolution=merge-duplicates,return=representation', - 'Content-Type': 'application/json', - }, - body: JSON.stringify(patronsList) - }) - if (!res.ok) { - const body = await res.text() - throw new Error(`failed to update patrons. res.status=${res.status} res.statusText=${res.statusText} body=${body}`); - } -} - -const deleteExpiredPatrons = async function (activePatronIds: string[]): Promise { - const res = await fetch(`${configs.postgrestUrl}/patrons?id=not.in.(${activePatronIds.join(',')})`, { - method: 'DELETE', - headers: { - 'Authorization': `Bearer ${configs.automationUserJwt}`, - 'Prefer': 'return=representation', - } - }); - - if (!res.ok) { - const body = await res.text(); - throw new Error(`Failed to delete expired patrons. res.status=${res.status} res.statusText=${res.statusText} body=${body}`); - } -}; - - -const synchronize_patrons_list: Task = async function (payload: unknown, helpers: Helpers) { - try { - - const patrons = await getPublicPatrons() - await updatePatronsTable(patrons) - await deleteExpiredPatrons(patrons.map((p) => p.id)) - - } catch (e) { - helpers.logger.error('failed to synchronize_patrons_list') - helpers.logger.error(JSON.stringify(e)) - - throw e - } -} - - -export default synchronize_patrons_list; \ No newline at end of file diff --git a/services/factory/src/utils/importDirectory.ts b/services/factory/src/utils/importDirectory.ts deleted file mode 100644 index a6dc622..0000000 --- a/services/factory/src/utils/importDirectory.ts +++ /dev/null @@ -1,22 +0,0 @@ -// @see https://github.com/discordeno/discordeno/blob/main/examples/advanced/src/utils/loader.ts - -import { readdir } from 'node:fs/promises' -import { join } from 'node:path' - -export async function importDirectory(folder: string): Promise { - const files = await readdir(folder, { recursive: true }) - - // bot.logger.info(files) - for (const filename of files) { - if (!filename.endsWith('.js') && !filename.endsWith('.ts')) continue - console.log(`loading ${filename}`) - - // Using `file://` and `process.cwd()` to avoid weird issues with relative paths and/or Windows - // await import(`file://${process.cwd()}/${folder}/${filename}`).catch((x) => - await import(join(folder, filename)).catch((x) => - // console.error(x) - console.error(`cannot import ${filename} for reason: ${x}`) - // logger.fatal(`Cannot import file (${folder}/${filename}) for reason: ${x}`), - ) - } -} \ No newline at end of file diff --git a/services/factory/tsconfig.json b/services/factory/tsconfig.json deleted file mode 100644 index 3bd8fd4..0000000 --- a/services/factory/tsconfig.json +++ /dev/null @@ -1,33 +0,0 @@ -{ - "compilerOptions": { - // Base Options recommended for all projects - "allowImportingTsExtensions": true, - "noEmit": true, // tsup does the emissions - "esModuleInterop": true, - "skipLibCheck": true, - "target": "ESNext", - "allowJs": true, - "moduleResolution": "Bundler", - "resolveJsonModule": true, - "moduleDetection": "force", - "isolatedModules": true, - // Enable strict type checking so you can catch bugs early - "strict": true, - "noUncheckedIndexedAccess": true, - "noImplicitOverride": true, - // Transpile our TypeScript code to JavaScript - "module": "ESNext", - "outDir": "dist", - "lib": [ - "ESNext", - "dom" - ] - }, - // Include the necessary files for your project - "include": [ - "src/**/*.ts" - ], - "exclude": [ - "node_modules" - ] -} \ No newline at end of file diff --git a/services/mailbox/README.md b/services/mailbox/README.md deleted file mode 100644 index 6d42cb0..0000000 --- a/services/mailbox/README.md +++ /dev/null @@ -1,17 +0,0 @@ -# @futureporn/mailbox - -A daemon which connects to an IMAP e-mail account and listens for new e-mails. - -When an e-mail is received, the daemon checks to see if the e-mail is from chaturbate or fansly. - -If the e-mail is from one of the aforementioned platforms, the e-mail is parsed. - -If the e-mail is a going live notification for a streamer on said platforms, the daemon creates or updates records in Strapi. - -Creates or updates the following record types in Strapi. - -* Platform Notification -* VTuber -* Stream - -With this data, we have a log of all the lewdtuber streams that have occured. \ No newline at end of file diff --git a/services/mailbox/package.json b/services/mailbox/package.json deleted file mode 100644 index c1f0d70..0000000 --- a/services/mailbox/package.json +++ /dev/null @@ -1,43 +0,0 @@ -{ - "name": "@futureporn/mailbox", - "type": "module", - "version": "1.0.0", - "description": "", - "scripts": { - "superclean": "rm -rf node_modules && rm -rf pnpm-lock.yaml && rm -rf dist", - "start": "node dist/index.cjs", - "test": "echo \"Error: no test specified\" && exit 1", - "dev": "tsx ", - "build": "tsup" - }, - "keywords": [], - "author": "@CJ_Clippy", - "license": "Unlicense", - "devDependencies": { - "@esbuild-plugins/esm-externals": "^0.1.2", - "@futureporn/scout": "workspace:^", - "@futureporn/storage": "workspace:^", - "@futureporn/types": "workspace:^", - "@futureporn/utils": "workspace:^", - "@types/chai": "^4.3.19", - "@types/imapflow": "^1.0.19", - "@types/mailparser": "^3.4.4", - "@types/mocha": "^10.0.7", - "chai": "^5.1.1", - "mocha": "^10.7.3", - "tsup": "^8.2.4", - "typescript": "^5.5.4" - }, - "dependencies": { - "@types/node": "^22.5.2", - "@types/qs": "^6.9.15", - "cheerio": "1.0.0", - "date-fns": "^3.6.0", - "dotenv": "^16.4.5", - "fastify": "^4.28.1", - "graphile-worker": "^0.16.6", - "imapflow": "^1.0.164", - "mailparser": "^3.7.1", - "qs": "^6.13.0" - } -} diff --git a/services/mailbox/pnpm-lock.yaml b/services/mailbox/pnpm-lock.yaml deleted file mode 100644 index 4475893..0000000 --- a/services/mailbox/pnpm-lock.yaml +++ /dev/null @@ -1,3207 +0,0 @@ -lockfileVersion: '9.0' - -settings: - autoInstallPeers: true - excludeLinksFromLockfile: false - -importers: - - .: - dependencies: - '@types/node': - specifier: ^22.5.2 - version: 22.5.2 - '@types/qs': - specifier: ^6.9.15 - version: 6.9.15 - cheerio: - specifier: 1.0.0 - version: 1.0.0 - date-fns: - specifier: ^3.6.0 - version: 3.6.0 - dotenv: - specifier: ^16.4.5 - version: 16.4.5 - fastify: - specifier: ^4.28.1 - version: 4.28.1 - graphile-worker: - specifier: ^0.16.6 - version: 0.16.6(typescript@5.5.4) - imapflow: - specifier: ^1.0.164 - version: 1.0.164 - mailparser: - specifier: ^3.7.1 - version: 3.7.1 - qs: - specifier: ^6.13.0 - version: 6.13.0 - devDependencies: - '@esbuild-plugins/esm-externals': - specifier: ^0.1.2 - version: 0.1.2(esbuild@0.23.1) - '@futureporn/scout': - specifier: workspace:^ - version: link:../scout - '@futureporn/storage': - specifier: workspace:^ - version: link:../../packages/storage - '@futureporn/types': - specifier: workspace:^ - version: link:../../packages/types - '@futureporn/utils': - specifier: workspace:^ - version: link:../../packages/utils - '@types/chai': - specifier: ^4.3.19 - version: 4.3.19 - '@types/imapflow': - specifier: ^1.0.19 - version: 1.0.19 - '@types/mailparser': - specifier: ^3.4.4 - version: 3.4.4 - '@types/mocha': - specifier: ^10.0.7 - version: 10.0.7 - chai: - specifier: ^5.1.1 - version: 5.1.1 - mocha: - specifier: ^10.7.3 - version: 10.7.3 - tsup: - specifier: ^8.2.4 - version: 8.2.4(typescript@5.5.4) - typescript: - specifier: ^5.5.4 - version: 5.5.4 - -packages: - - '@babel/code-frame@7.24.7': - resolution: {integrity: sha512-BcYH1CVJBO9tvyIZ2jVeXgSIMvGZ2FDRvDdOIVQyuklNKSsx+eppDEBq/g47Ayw+RqNFE+URvOShmf+f/qwAlA==} - engines: {node: '>=6.9.0'} - - '@babel/helper-validator-identifier@7.24.7': - resolution: {integrity: sha512-rR+PBcQ1SMQDDyF6X0wxtG8QyLCgUB0eRAGguqRLfkCA87l7yAP7ehq8SNj96OOGTO8OBV70KhuFYcIkHXOg0w==} - engines: {node: '>=6.9.0'} - - '@babel/highlight@7.24.7': - resolution: {integrity: sha512-EStJpq4OuY8xYfhGVXngigBJRWxftKX9ksiGDnmlY3o7B/V7KIAc9X4oiK87uPJSc/vs5L869bem5fhZa8caZw==} - engines: {node: '>=6.9.0'} - - '@esbuild-plugins/esm-externals@0.1.2': - resolution: {integrity: sha512-Ij/xQYmzHE5yu6N2s/OFAVOtQUrS4Edpjg4hwBnN+aiQ4SryOEv7Gyr1t++PxniOFHGcvGAOORNooQ+nswzmuQ==} - peerDependencies: - esbuild: '*' - - '@esbuild/aix-ppc64@0.23.1': - resolution: {integrity: sha512-6VhYk1diRqrhBAqpJEdjASR/+WVRtfjpqKuNw11cLiaWpAT/Uu+nokB+UJnevzy/P9C/ty6AOe0dwueMrGh/iQ==} - engines: {node: '>=18'} - cpu: [ppc64] - os: [aix] - - '@esbuild/android-arm64@0.23.1': - resolution: {integrity: sha512-xw50ipykXcLstLeWH7WRdQuysJqejuAGPd30vd1i5zSyKK3WE+ijzHmLKxdiCMtH1pHz78rOg0BKSYOSB/2Khw==} - engines: {node: '>=18'} - cpu: [arm64] - os: [android] - - '@esbuild/android-arm@0.23.1': - resolution: {integrity: sha512-uz6/tEy2IFm9RYOyvKl88zdzZfwEfKZmnX9Cj1BHjeSGNuGLuMD1kR8y5bteYmwqKm1tj8m4cb/aKEorr6fHWQ==} - engines: {node: '>=18'} - cpu: [arm] - os: [android] - - '@esbuild/android-x64@0.23.1': - resolution: {integrity: sha512-nlN9B69St9BwUoB+jkyU090bru8L0NA3yFvAd7k8dNsVH8bi9a8cUAUSEcEEgTp2z3dbEDGJGfP6VUnkQnlReg==} - engines: {node: '>=18'} - cpu: [x64] - os: [android] - - '@esbuild/darwin-arm64@0.23.1': - resolution: {integrity: sha512-YsS2e3Wtgnw7Wq53XXBLcV6JhRsEq8hkfg91ESVadIrzr9wO6jJDMZnCQbHm1Guc5t/CdDiFSSfWP58FNuvT3Q==} - engines: {node: '>=18'} - cpu: [arm64] - os: [darwin] - - '@esbuild/darwin-x64@0.23.1': - resolution: {integrity: sha512-aClqdgTDVPSEGgoCS8QDG37Gu8yc9lTHNAQlsztQ6ENetKEO//b8y31MMu2ZaPbn4kVsIABzVLXYLhCGekGDqw==} - engines: {node: '>=18'} - cpu: [x64] - os: [darwin] - - '@esbuild/freebsd-arm64@0.23.1': - resolution: {integrity: sha512-h1k6yS8/pN/NHlMl5+v4XPfikhJulk4G+tKGFIOwURBSFzE8bixw1ebjluLOjfwtLqY0kewfjLSrO6tN2MgIhA==} - engines: {node: '>=18'} - cpu: [arm64] - os: [freebsd] - - '@esbuild/freebsd-x64@0.23.1': - resolution: {integrity: sha512-lK1eJeyk1ZX8UklqFd/3A60UuZ/6UVfGT2LuGo3Wp4/z7eRTRYY+0xOu2kpClP+vMTi9wKOfXi2vjUpO1Ro76g==} - engines: {node: '>=18'} - cpu: [x64] - os: [freebsd] - - '@esbuild/linux-arm64@0.23.1': - resolution: {integrity: sha512-/93bf2yxencYDnItMYV/v116zff6UyTjo4EtEQjUBeGiVpMmffDNUyD9UN2zV+V3LRV3/on4xdZ26NKzn6754g==} - engines: {node: '>=18'} - cpu: [arm64] - os: [linux] - - '@esbuild/linux-arm@0.23.1': - resolution: {integrity: sha512-CXXkzgn+dXAPs3WBwE+Kvnrf4WECwBdfjfeYHpMeVxWE0EceB6vhWGShs6wi0IYEqMSIzdOF1XjQ/Mkm5d7ZdQ==} - engines: {node: '>=18'} - cpu: [arm] - os: [linux] - - '@esbuild/linux-ia32@0.23.1': - resolution: {integrity: sha512-VTN4EuOHwXEkXzX5nTvVY4s7E/Krz7COC8xkftbbKRYAl96vPiUssGkeMELQMOnLOJ8k3BY1+ZY52tttZnHcXQ==} - engines: {node: '>=18'} - cpu: [ia32] - os: [linux] - - '@esbuild/linux-loong64@0.23.1': - resolution: {integrity: sha512-Vx09LzEoBa5zDnieH8LSMRToj7ir/Jeq0Gu6qJ/1GcBq9GkfoEAoXvLiW1U9J1qE/Y/Oyaq33w5p2ZWrNNHNEw==} - engines: {node: '>=18'} - cpu: [loong64] - os: [linux] - - '@esbuild/linux-mips64el@0.23.1': - resolution: {integrity: sha512-nrFzzMQ7W4WRLNUOU5dlWAqa6yVeI0P78WKGUo7lg2HShq/yx+UYkeNSE0SSfSure0SqgnsxPvmAUu/vu0E+3Q==} - engines: {node: '>=18'} - cpu: [mips64el] - os: [linux] - - '@esbuild/linux-ppc64@0.23.1': - resolution: {integrity: sha512-dKN8fgVqd0vUIjxuJI6P/9SSSe/mB9rvA98CSH2sJnlZ/OCZWO1DJvxj8jvKTfYUdGfcq2dDxoKaC6bHuTlgcw==} - engines: {node: '>=18'} - cpu: [ppc64] - os: [linux] - - '@esbuild/linux-riscv64@0.23.1': - resolution: {integrity: sha512-5AV4Pzp80fhHL83JM6LoA6pTQVWgB1HovMBsLQ9OZWLDqVY8MVobBXNSmAJi//Csh6tcY7e7Lny2Hg1tElMjIA==} - engines: {node: '>=18'} - cpu: [riscv64] - os: [linux] - - '@esbuild/linux-s390x@0.23.1': - resolution: {integrity: sha512-9ygs73tuFCe6f6m/Tb+9LtYxWR4c9yg7zjt2cYkjDbDpV/xVn+68cQxMXCjUpYwEkze2RcU/rMnfIXNRFmSoDw==} - engines: {node: '>=18'} - cpu: [s390x] - os: [linux] - - '@esbuild/linux-x64@0.23.1': - resolution: {integrity: sha512-EV6+ovTsEXCPAp58g2dD68LxoP/wK5pRvgy0J/HxPGB009omFPv3Yet0HiaqvrIrgPTBuC6wCH1LTOY91EO5hQ==} - engines: {node: '>=18'} - cpu: [x64] - os: [linux] - - '@esbuild/netbsd-x64@0.23.1': - resolution: {integrity: sha512-aevEkCNu7KlPRpYLjwmdcuNz6bDFiE7Z8XC4CPqExjTvrHugh28QzUXVOZtiYghciKUacNktqxdpymplil1beA==} - engines: {node: '>=18'} - cpu: [x64] - os: [netbsd] - - '@esbuild/openbsd-arm64@0.23.1': - resolution: {integrity: sha512-3x37szhLexNA4bXhLrCC/LImN/YtWis6WXr1VESlfVtVeoFJBRINPJ3f0a/6LV8zpikqoUg4hyXw0sFBt5Cr+Q==} - engines: {node: '>=18'} - cpu: [arm64] - os: [openbsd] - - '@esbuild/openbsd-x64@0.23.1': - resolution: {integrity: sha512-aY2gMmKmPhxfU+0EdnN+XNtGbjfQgwZj43k8G3fyrDM/UdZww6xrWxmDkuz2eCZchqVeABjV5BpildOrUbBTqA==} - engines: {node: '>=18'} - cpu: [x64] - os: [openbsd] - - '@esbuild/sunos-x64@0.23.1': - resolution: {integrity: sha512-RBRT2gqEl0IKQABT4XTj78tpk9v7ehp+mazn2HbUeZl1YMdaGAQqhapjGTCe7uw7y0frDi4gS0uHzhvpFuI1sA==} - engines: {node: '>=18'} - cpu: [x64] - os: [sunos] - - '@esbuild/win32-arm64@0.23.1': - resolution: {integrity: sha512-4O+gPR5rEBe2FpKOVyiJ7wNDPA8nGzDuJ6gN4okSA1gEOYZ67N8JPk58tkWtdtPeLz7lBnY6I5L3jdsr3S+A6A==} - engines: {node: '>=18'} - cpu: [arm64] - os: [win32] - - '@esbuild/win32-ia32@0.23.1': - resolution: {integrity: sha512-BcaL0Vn6QwCwre3Y717nVHZbAa4UBEigzFm6VdsVdT/MbZ38xoj1X9HPkZhbmaBGUD1W8vxAfffbDe8bA6AKnQ==} - engines: {node: '>=18'} - cpu: [ia32] - os: [win32] - - '@esbuild/win32-x64@0.23.1': - resolution: {integrity: sha512-BHpFFeslkWrXWyUPnbKm+xYYVYruCinGcftSBaa8zoF9hZO4BcSCFUvHVTtzpIY6YzUnYtuEhZ+C9iEXjxnasg==} - engines: {node: '>=18'} - cpu: [x64] - os: [win32] - - '@fastify/ajv-compiler@3.6.0': - resolution: {integrity: sha512-LwdXQJjmMD+GwLOkP7TVC68qa+pSSogeWWmznRJ/coyTcfe9qA05AHFSe1eZFwK6q+xVRpChnvFUkf1iYaSZsQ==} - - '@fastify/error@3.4.1': - resolution: {integrity: sha512-wWSvph+29GR783IhmvdwWnN4bUxTD01Vm5Xad4i7i1VuAOItLvbPAb69sb0IQ2N57yprvhNIwAP5B6xfKTmjmQ==} - - '@fastify/fast-json-stringify-compiler@4.3.0': - resolution: {integrity: sha512-aZAXGYo6m22Fk1zZzEUKBvut/CIIQe/BapEORnxiD5Qr0kPHqqI69NtEMCme74h+at72sPhbkb4ZrLd1W3KRLA==} - - '@fastify/merge-json-schemas@0.1.1': - resolution: {integrity: sha512-fERDVz7topgNjtXsJTTW1JKLy0rhuLRcquYqNR9rF7OcVpCa2OVW49ZPDIhaRRCaUuvVxI+N416xUoF76HNSXA==} - - '@graphile/logger@0.2.0': - resolution: {integrity: sha512-jjcWBokl9eb1gVJ85QmoaQ73CQ52xAaOCF29ukRbYNl6lY+ts0ErTaDYOBlejcbUs2OpaiqYLO5uDhyLFzWw4w==} - - '@isaacs/cliui@8.0.2': - resolution: {integrity: sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==} - engines: {node: '>=12'} - - '@jridgewell/gen-mapping@0.3.5': - resolution: {integrity: sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==} - engines: {node: '>=6.0.0'} - - '@jridgewell/resolve-uri@3.1.2': - resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==} - engines: {node: '>=6.0.0'} - - '@jridgewell/set-array@1.2.1': - resolution: {integrity: sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==} - engines: {node: '>=6.0.0'} - - '@jridgewell/sourcemap-codec@1.5.0': - resolution: {integrity: sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==} - - '@jridgewell/trace-mapping@0.3.25': - resolution: {integrity: sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==} - - '@nodelib/fs.scandir@2.1.5': - resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==} - engines: {node: '>= 8'} - - '@nodelib/fs.stat@2.0.5': - resolution: {integrity: sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==} - engines: {node: '>= 8'} - - '@nodelib/fs.walk@1.2.8': - resolution: {integrity: sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==} - engines: {node: '>= 8'} - - '@pkgjs/parseargs@0.11.0': - resolution: {integrity: sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==} - engines: {node: '>=14'} - - '@rollup/rollup-android-arm-eabi@4.21.2': - resolution: {integrity: sha512-fSuPrt0ZO8uXeS+xP3b+yYTCBUd05MoSp2N/MFOgjhhUhMmchXlpTQrTpI8T+YAwAQuK7MafsCOxW7VrPMrJcg==} - cpu: [arm] - os: [android] - - '@rollup/rollup-android-arm64@4.21.2': - resolution: {integrity: sha512-xGU5ZQmPlsjQS6tzTTGwMsnKUtu0WVbl0hYpTPauvbRAnmIvpInhJtgjj3mcuJpEiuUw4v1s4BimkdfDWlh7gA==} - cpu: [arm64] - os: [android] - - '@rollup/rollup-darwin-arm64@4.21.2': - resolution: {integrity: sha512-99AhQ3/ZMxU7jw34Sq8brzXqWH/bMnf7ZVhvLk9QU2cOepbQSVTns6qoErJmSiAvU3InRqC2RRZ5ovh1KN0d0Q==} - cpu: [arm64] - os: [darwin] - - '@rollup/rollup-darwin-x64@4.21.2': - resolution: {integrity: sha512-ZbRaUvw2iN/y37x6dY50D8m2BnDbBjlnMPotDi/qITMJ4sIxNY33HArjikDyakhSv0+ybdUxhWxE6kTI4oX26w==} - cpu: [x64] - os: [darwin] - - '@rollup/rollup-linux-arm-gnueabihf@4.21.2': - resolution: {integrity: sha512-ztRJJMiE8nnU1YFcdbd9BcH6bGWG1z+jP+IPW2oDUAPxPjo9dverIOyXz76m6IPA6udEL12reYeLojzW2cYL7w==} - cpu: [arm] - os: [linux] - - '@rollup/rollup-linux-arm-musleabihf@4.21.2': - resolution: {integrity: sha512-flOcGHDZajGKYpLV0JNc0VFH361M7rnV1ee+NTeC/BQQ1/0pllYcFmxpagltANYt8FYf9+kL6RSk80Ziwyhr7w==} - cpu: [arm] - os: [linux] - - '@rollup/rollup-linux-arm64-gnu@4.21.2': - resolution: {integrity: sha512-69CF19Kp3TdMopyteO/LJbWufOzqqXzkrv4L2sP8kfMaAQ6iwky7NoXTp7bD6/irKgknDKM0P9E/1l5XxVQAhw==} - cpu: [arm64] - os: [linux] - - '@rollup/rollup-linux-arm64-musl@4.21.2': - resolution: {integrity: sha512-48pD/fJkTiHAZTnZwR0VzHrao70/4MlzJrq0ZsILjLW/Ab/1XlVUStYyGt7tdyIiVSlGZbnliqmult/QGA2O2w==} - cpu: [arm64] - os: [linux] - - '@rollup/rollup-linux-powerpc64le-gnu@4.21.2': - resolution: {integrity: sha512-cZdyuInj0ofc7mAQpKcPR2a2iu4YM4FQfuUzCVA2u4HI95lCwzjoPtdWjdpDKyHxI0UO82bLDoOaLfpZ/wviyQ==} - cpu: [ppc64] - os: [linux] - - '@rollup/rollup-linux-riscv64-gnu@4.21.2': - resolution: {integrity: sha512-RL56JMT6NwQ0lXIQmMIWr1SW28z4E4pOhRRNqwWZeXpRlykRIlEpSWdsgNWJbYBEWD84eocjSGDu/XxbYeCmwg==} - cpu: [riscv64] - os: [linux] - - '@rollup/rollup-linux-s390x-gnu@4.21.2': - resolution: {integrity: sha512-PMxkrWS9z38bCr3rWvDFVGD6sFeZJw4iQlhrup7ReGmfn7Oukrr/zweLhYX6v2/8J6Cep9IEA/SmjXjCmSbrMQ==} - cpu: [s390x] - os: [linux] - - '@rollup/rollup-linux-x64-gnu@4.21.2': - resolution: {integrity: sha512-B90tYAUoLhU22olrafY3JQCFLnT3NglazdwkHyxNDYF/zAxJt5fJUB/yBoWFoIQ7SQj+KLe3iL4BhOMa9fzgpw==} - cpu: [x64] - os: [linux] - - '@rollup/rollup-linux-x64-musl@4.21.2': - resolution: {integrity: sha512-7twFizNXudESmC9oneLGIUmoHiiLppz/Xs5uJQ4ShvE6234K0VB1/aJYU3f/4g7PhssLGKBVCC37uRkkOi8wjg==} - cpu: [x64] - os: [linux] - - '@rollup/rollup-win32-arm64-msvc@4.21.2': - resolution: {integrity: sha512-9rRero0E7qTeYf6+rFh3AErTNU1VCQg2mn7CQcI44vNUWM9Ze7MSRS/9RFuSsox+vstRt97+x3sOhEey024FRQ==} - cpu: [arm64] - os: [win32] - - '@rollup/rollup-win32-ia32-msvc@4.21.2': - resolution: {integrity: sha512-5rA4vjlqgrpbFVVHX3qkrCo/fZTj1q0Xxpg+Z7yIo3J2AilW7t2+n6Q8Jrx+4MrYpAnjttTYF8rr7bP46BPzRw==} - cpu: [ia32] - os: [win32] - - '@rollup/rollup-win32-x64-msvc@4.21.2': - resolution: {integrity: sha512-6UUxd0+SKomjdzuAcp+HAmxw1FlGBnl1v2yEPSabtx4lBfdXHDVsW7+lQkgz9cNFJGY3AWR7+V8P5BqkD9L9nA==} - cpu: [x64] - os: [win32] - - '@selderee/plugin-htmlparser2@0.11.0': - resolution: {integrity: sha512-P33hHGdldxGabLFjPPpaTxVolMrzrcegejx+0GxjrIb9Zv48D8yAIA/QTDR2dFl7Uz7urX8aX6+5bCZslr+gWQ==} - - '@types/chai@4.3.19': - resolution: {integrity: sha512-2hHHvQBVE2FiSK4eN0Br6snX9MtolHaTo/batnLjlGRhoQzlCL61iVpxoqO7SfFyOw+P/pwv+0zNHzKoGWz9Cw==} - - '@types/debug@4.1.12': - resolution: {integrity: sha512-vIChWdVG3LG1SMxEvI/AK+FWJthlrqlTu7fbrlywTkkaONwk/UAGaULXRlf8vkzFBLVm0zkMdCquhL5aOjhXPQ==} - - '@types/estree@1.0.5': - resolution: {integrity: sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==} - - '@types/imapflow@1.0.19': - resolution: {integrity: sha512-U48VZXe4XYhS3AFNI+4ougW8OXI4VaKjyF0nGXgVzIN8SPs9lh2LPNRM0HmIM+hUTw60l7MHgbQO8hsf+Q4U5w==} - - '@types/interpret@1.1.3': - resolution: {integrity: sha512-uBaBhj/BhilG58r64mtDb/BEdH51HIQLgP5bmWzc5qCtFMja8dCk/IOJmk36j0lbi9QHwI6sbtUNGuqXdKCAtQ==} - - '@types/mailparser@3.4.4': - resolution: {integrity: sha512-C6Znp2QVS25JqtuPyxj38Qh+QoFcLycdxsvcc6IZCGekhaMBzbdTXzwGzhGoYb3TfKu8IRCNV0sV1o3Od97cEQ==} - - '@types/mocha@10.0.7': - resolution: {integrity: sha512-GN8yJ1mNTcFcah/wKEFIJckJx9iJLoMSzWcfRRuxz/Jk+U6KQNnml+etbtxFK8lPjzOw3zp4Ha/kjSst9fsHYw==} - - '@types/ms@0.7.34': - resolution: {integrity: sha512-nG96G3Wp6acyAgJqGasjODb+acrI7KltPiRxzHPXnP3NgI28bpQDRv53olbqGXbfcgF5aiiHmO3xpwEpS5Ld9g==} - - '@types/node@20.16.3': - resolution: {integrity: sha512-/wdGiWRkMOm53gAsSyFMXFZHbVg7C6CbkrzHNpaHoYfsUWPg7m6ZRKtvQjgvQ9i8WT540a3ydRlRQbxjY30XxQ==} - - '@types/node@22.5.2': - resolution: {integrity: sha512-acJsPTEqYqulZS/Yp/S3GgeE6GZ0qYODUR8aVr/DkhHQ8l9nd4j5x1/ZJy9/gHrRlFMqkO6i0I3E27Alu4jjPg==} - - '@types/pg@8.11.8': - resolution: {integrity: sha512-IqpCf8/569txXN/HoP5i1LjXfKZWL76Yr2R77xgeIICUbAYHeoaEZFhYHo2uDftecLWrTJUq63JvQu8q3lnDyA==} - - '@types/qs@6.9.15': - resolution: {integrity: sha512-uXHQKES6DQKKCLh441Xv/dwxOq1TVS3JPUMlEqoEglvlhR6Mxnlew/Xq/LRVHpLyk7iK3zODe1qYHIMltO7XGg==} - - '@types/semver@7.5.8': - resolution: {integrity: sha512-I8EUhyrgfLrcTkzV3TSsGyl1tSuPrEDzr0yd5m90UgNxQkyDXULk3b6MlQqTCpZpNtWe1K0hzclnZkTcLBe2UQ==} - - abort-controller@3.0.0: - resolution: {integrity: sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==} - engines: {node: '>=6.5'} - - abstract-logging@2.0.1: - resolution: {integrity: sha512-2BjRTZxTPvheOvGbBslFSYOUkr+SjPtOnrLP33f+VIWLzezQpZcqVg7ja3L4dBXmzzgwT+a029jRx5PCi3JuiA==} - - ajv-formats@2.1.1: - resolution: {integrity: sha512-Wx0Kx52hxE7C18hkMEggYlEifqWZtYaRgouJor+WMdPnQyEK13vgEWyVNup7SoeeoLMsr4kf5h6dOW11I15MUA==} - peerDependencies: - ajv: ^8.0.0 - peerDependenciesMeta: - ajv: - optional: true - - ajv-formats@3.0.1: - resolution: {integrity: sha512-8iUql50EUR+uUcdRQ3HDqa6EVyo3docL8g5WJ3FNcWmu62IbkGUue/pEyLBW8VGKKucTPgqeks4fIU1DA4yowQ==} - peerDependencies: - ajv: ^8.0.0 - peerDependenciesMeta: - ajv: - optional: true - - ajv@8.17.1: - resolution: {integrity: sha512-B/gBuNg5SiMTrPkC+A2+cW0RszwxYmn6VYxB/inlBStS5nx6xHIt/ehKRhIMhqusl7a8LjQoZnjCs5vhwxOQ1g==} - - ansi-colors@4.1.3: - resolution: {integrity: sha512-/6w/C21Pm1A7aZitlI5Ni/2J6FFQN8i1Cvz3kHABAAbw93v/NlvKdVOqz7CCWz/3iv/JplRSEEZ83XION15ovw==} - engines: {node: '>=6'} - - ansi-regex@5.0.1: - resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==} - engines: {node: '>=8'} - - ansi-regex@6.0.1: - resolution: {integrity: sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==} - engines: {node: '>=12'} - - ansi-styles@3.2.1: - resolution: {integrity: sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==} - engines: {node: '>=4'} - - ansi-styles@4.3.0: - resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} - engines: {node: '>=8'} - - ansi-styles@6.2.1: - resolution: {integrity: sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==} - engines: {node: '>=12'} - - any-promise@1.3.0: - resolution: {integrity: sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==} - - anymatch@3.1.3: - resolution: {integrity: sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==} - engines: {node: '>= 8'} - - argparse@2.0.1: - resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==} - - array-union@2.1.0: - resolution: {integrity: sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==} - engines: {node: '>=8'} - - assertion-error@2.0.1: - resolution: {integrity: sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==} - engines: {node: '>=12'} - - atomic-sleep@1.0.0: - resolution: {integrity: sha512-kNOjDqAh7px0XWNI+4QbzoiR/nTkHAWNud2uvnJquD1/x5a7EQZMJT0AczqK0Qn67oY/TTQ1LbUKajZpp3I9tQ==} - engines: {node: '>=8.0.0'} - - avvio@8.4.0: - resolution: {integrity: sha512-CDSwaxINFy59iNwhYnkvALBwZiTydGkOecZyPkqBpABYR1KqGEsET0VOOYDwtleZSUIdeY36DC2bSZ24CO1igA==} - - balanced-match@1.0.2: - resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} - - base64-js@1.5.1: - resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==} - - binary-extensions@2.3.0: - resolution: {integrity: sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==} - engines: {node: '>=8'} - - boolbase@1.0.0: - resolution: {integrity: sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==} - - brace-expansion@2.0.1: - resolution: {integrity: sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==} - - braces@3.0.3: - resolution: {integrity: sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==} - engines: {node: '>=8'} - - browser-stdout@1.3.1: - resolution: {integrity: sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==} - - buffer@6.0.3: - resolution: {integrity: sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==} - - bundle-require@5.0.0: - resolution: {integrity: sha512-GuziW3fSSmopcx4KRymQEJVbZUfqlCqcq7dvs6TYwKRZiegK/2buMxQTPs6MGlNv50wms1699qYO54R8XfRX4w==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - peerDependencies: - esbuild: '>=0.18' - - cac@6.7.14: - resolution: {integrity: sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==} - engines: {node: '>=8'} - - call-bind@1.0.7: - resolution: {integrity: sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==} - engines: {node: '>= 0.4'} - - callsites@3.1.0: - resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==} - engines: {node: '>=6'} - - camelcase@6.3.0: - resolution: {integrity: sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==} - engines: {node: '>=10'} - - chai@5.1.1: - resolution: {integrity: sha512-pT1ZgP8rPNqUgieVaEY+ryQr6Q4HXNg8Ei9UnLUrjN4IA7dvQC5JB+/kxVcPNDHyBcc/26CXPkbNzq3qwrOEKA==} - engines: {node: '>=12'} - - chalk@2.4.2: - resolution: {integrity: sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==} - engines: {node: '>=4'} - - chalk@4.1.2: - resolution: {integrity: sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==} - engines: {node: '>=10'} - - check-error@2.1.1: - resolution: {integrity: sha512-OAlb+T7V4Op9OwdkjmguYRqncdlx5JiofwOAUkmTF+jNdHwzTaTs4sRAGpzLF3oOz5xAyDGrPgeIDFQmDOTiJw==} - engines: {node: '>= 16'} - - cheerio-select@2.1.0: - resolution: {integrity: sha512-9v9kG0LvzrlcungtnJtpGNxY+fzECQKhK4EGJX2vByejiMX84MFNQw4UxPJl3bFbTMw+Dfs37XaIkCwTZfLh4g==} - - cheerio@1.0.0: - resolution: {integrity: sha512-quS9HgjQpdaXOvsZz82Oz7uxtXiy6UIsIQcpBj7HRw2M63Skasm9qlDocAM7jNuaxdhpPU7c4kJN+gA5MCu4ww==} - engines: {node: '>=18.17'} - - chokidar@3.6.0: - resolution: {integrity: sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==} - engines: {node: '>= 8.10.0'} - - cliui@7.0.4: - resolution: {integrity: sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==} - - cliui@8.0.1: - resolution: {integrity: sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==} - engines: {node: '>=12'} - - color-convert@1.9.3: - resolution: {integrity: sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==} - - color-convert@2.0.1: - resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} - engines: {node: '>=7.0.0'} - - color-name@1.1.3: - resolution: {integrity: sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==} - - color-name@1.1.4: - resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} - - commander@4.1.1: - resolution: {integrity: sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==} - engines: {node: '>= 6'} - - consola@3.2.3: - resolution: {integrity: sha512-I5qxpzLv+sJhTVEoLYNcTW+bThDCPsit0vLNKShZx6rLtpilNpmmeTPaeqJb9ZE9dV3DGaeby6Vuhrw38WjeyQ==} - engines: {node: ^14.18.0 || >=16.10.0} - - cookie@0.6.0: - resolution: {integrity: sha512-U71cyTamuh1CRNCfpGY6to28lxvNwPG4Guz/EVjgf3Jmzv0vlDp1atT9eS5dDjMYHucpHbWns6Lwf3BKz6svdw==} - engines: {node: '>= 0.6'} - - cosmiconfig@8.3.6: - resolution: {integrity: sha512-kcZ6+W5QzcJ3P1Mt+83OUv/oHFqZHIx8DuxG6eZ5RGMERoLqp4BuGjhHLYGK+Kf5XVkQvqBSmAy/nGWN3qDgEA==} - engines: {node: '>=14'} - peerDependencies: - typescript: '>=4.9.5' - peerDependenciesMeta: - typescript: - optional: true - - cross-spawn@7.0.3: - resolution: {integrity: sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==} - engines: {node: '>= 8'} - - css-select@5.1.0: - resolution: {integrity: sha512-nwoRF1rvRRnnCqqY7updORDsuqKzqYJ28+oSMaJMMgOauh3fvwHqMS7EZpIPqK8GL+g9mKxF1vP/ZjSeNjEVHg==} - - css-what@6.1.0: - resolution: {integrity: sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw==} - engines: {node: '>= 6'} - - date-fns@3.6.0: - resolution: {integrity: sha512-fRHTG8g/Gif+kSh50gaGEdToemgfj74aRX3swtiouboip5JDLAyDE9F11nHMIcvOaXeOC6D7SpNhi7uFyB7Uww==} - - debug@4.3.6: - resolution: {integrity: sha512-O/09Bd4Z1fBrU4VzkhFqVgpPzaGbw6Sm9FEkBT1A/YBXQFGuuSxa1dN2nxgxS34JmKXqYx8CZAwEVoJFImUXIg==} - engines: {node: '>=6.0'} - peerDependencies: - supports-color: '*' - peerDependenciesMeta: - supports-color: - optional: true - - decamelize@4.0.0: - resolution: {integrity: sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==} - engines: {node: '>=10'} - - deep-eql@5.0.2: - resolution: {integrity: sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==} - engines: {node: '>=6'} - - deepmerge@4.3.1: - resolution: {integrity: sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==} - engines: {node: '>=0.10.0'} - - define-data-property@1.1.4: - resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==} - engines: {node: '>= 0.4'} - - diff@5.2.0: - resolution: {integrity: sha512-uIFDxqpRZGZ6ThOk84hEfqWoHx2devRFvpTZcTHur85vImfaxUbTW9Ryh4CpCuDnToOP1CEtXKIgytHBPVff5A==} - engines: {node: '>=0.3.1'} - - dir-glob@3.0.1: - resolution: {integrity: sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==} - engines: {node: '>=8'} - - dom-serializer@2.0.0: - resolution: {integrity: sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg==} - - domelementtype@2.3.0: - resolution: {integrity: sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==} - - domhandler@5.0.3: - resolution: {integrity: sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==} - engines: {node: '>= 4'} - - domutils@3.1.0: - resolution: {integrity: sha512-H78uMmQtI2AhgDJjWeQmHwJJ2bLPD3GMmO7Zja/ZZh84wkm+4ut+IUnUdRa8uCGX88DiVx1j6FRe1XfxEgjEZA==} - - dotenv@16.4.5: - resolution: {integrity: sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==} - engines: {node: '>=12'} - - eastasianwidth@0.2.0: - resolution: {integrity: sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==} - - emoji-regex@8.0.0: - resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==} - - emoji-regex@9.2.2: - resolution: {integrity: sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==} - - encoding-japanese@2.0.0: - resolution: {integrity: sha512-++P0RhebUC8MJAwJOsT93dT+5oc5oPImp1HubZpAuCZ5kTLnhuuBhKHj2jJeO/Gj93idPBWmIuQ9QWMe5rX3pQ==} - engines: {node: '>=8.10.0'} - - encoding-japanese@2.1.0: - resolution: {integrity: sha512-58XySVxUgVlBikBTbQ8WdDxBDHIdXucB16LO5PBHR8t75D54wQrNo4cg+58+R1CtJfKnsVsvt9XlteRaR8xw1w==} - engines: {node: '>=8.10.0'} - - encoding-japanese@2.2.0: - resolution: {integrity: sha512-EuJWwlHPZ1LbADuKTClvHtwbaFn4rOD+dRAbWysqEOXRc2Uui0hJInNJrsdH0c+OhJA4nrCBdSkW4DD5YxAo6A==} - engines: {node: '>=8.10.0'} - - encoding-sniffer@0.2.0: - resolution: {integrity: sha512-ju7Wq1kg04I3HtiYIOrUrdfdDvkyO9s5XM8QAj/bN61Yo/Vb4vgJxy5vi4Yxk01gWHbrofpPtpxM8bKger9jhg==} - - entities@4.5.0: - resolution: {integrity: sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==} - engines: {node: '>=0.12'} - - error-ex@1.3.2: - resolution: {integrity: sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==} - - es-define-property@1.0.0: - resolution: {integrity: sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==} - engines: {node: '>= 0.4'} - - es-errors@1.3.0: - resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==} - engines: {node: '>= 0.4'} - - esbuild@0.23.1: - resolution: {integrity: sha512-VVNz/9Sa0bs5SELtn3f7qhJCDPCF5oMEl5cO9/SSinpE9hbPVvxbd572HH5AKiP7WD8INO53GgfDDhRjkylHEg==} - engines: {node: '>=18'} - hasBin: true - - escalade@3.2.0: - resolution: {integrity: sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==} - engines: {node: '>=6'} - - escape-string-regexp@1.0.5: - resolution: {integrity: sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==} - engines: {node: '>=0.8.0'} - - escape-string-regexp@4.0.0: - resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} - engines: {node: '>=10'} - - event-target-shim@5.0.1: - resolution: {integrity: sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==} - engines: {node: '>=6'} - - events@3.3.0: - resolution: {integrity: sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==} - engines: {node: '>=0.8.x'} - - execa@5.1.1: - resolution: {integrity: sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==} - engines: {node: '>=10'} - - fast-content-type-parse@1.1.0: - resolution: {integrity: sha512-fBHHqSTFLVnR61C+gltJuE5GkVQMV0S2nqUO8TJ+5Z3qAKG8vAx4FKai1s5jq/inV1+sREynIWSuQ6HgoSXpDQ==} - - fast-decode-uri-component@1.0.1: - resolution: {integrity: sha512-WKgKWg5eUxvRZGwW8FvfbaH7AXSh2cL+3j5fMGzUMCxWBJ3dV3a7Wz8y2f/uQ0e3B6WmodD3oS54jTQ9HVTIIg==} - - fast-deep-equal@3.1.3: - resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==} - - fast-glob@3.3.2: - resolution: {integrity: sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==} - engines: {node: '>=8.6.0'} - - fast-json-stringify@5.16.1: - resolution: {integrity: sha512-KAdnLvy1yu/XrRtP+LJnxbBGrhN+xXu+gt3EUvZhYGKCr3lFHq/7UFJHHFgmJKoqlh6B40bZLEv7w46B0mqn1g==} - - fast-querystring@1.1.2: - resolution: {integrity: sha512-g6KuKWmFXc0fID8WWH0jit4g0AGBoJhCkJMb1RmbsSEUNvQ+ZC8D6CUZ+GtF8nMzSPXnhiePyyqqipzNNEnHjg==} - - fast-redact@3.5.0: - resolution: {integrity: sha512-dwsoQlS7h9hMeYUq1W++23NDcBLV4KqONnITDV9DjfS3q1SgDGVrBdvvTLUotWtPSD7asWDV9/CmsZPy8Hf70A==} - engines: {node: '>=6'} - - fast-uri@2.4.0: - resolution: {integrity: sha512-ypuAmmMKInk5q7XcepxlnUWDLWv4GFtaJqAzWKqn62IpQ3pejtr5dTVbt3vwqVaMKmkNR55sTT+CqUKIaT21BA==} - - fast-uri@3.0.1: - resolution: {integrity: sha512-MWipKbbYiYI0UC7cl8m/i/IWTqfC8YXsqjzybjddLsFjStroQzsHXkc73JutMvBiXmOvapk+axIl79ig5t55Bw==} - - fastify@4.28.1: - resolution: {integrity: sha512-kFWUtpNr4i7t5vY2EJPCN2KgMVpuqfU4NjnJNCgiNB900oiDeYqaNDRcAfeBbOF5hGixixxcKnOU4KN9z6QncQ==} - - fastq@1.17.1: - resolution: {integrity: sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==} - - fill-range@7.1.1: - resolution: {integrity: sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==} - engines: {node: '>=8'} - - find-my-way@8.2.0: - resolution: {integrity: sha512-HdWXgFYc6b1BJcOBDBwjqWuHJj1WYiqrxSh25qtU4DabpMFdj/gSunNBQb83t+8Zt67D7CXEzJWTkxaShMTMOA==} - engines: {node: '>=14'} - - find-up@5.0.0: - resolution: {integrity: sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==} - engines: {node: '>=10'} - - flat@5.0.2: - resolution: {integrity: sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==} - hasBin: true - - foreground-child@3.3.0: - resolution: {integrity: sha512-Ld2g8rrAyMYFXBhEqMz8ZAHBi4J4uS1i/CxGMDnjyFWddMXLVcDp051DZfu+t7+ab7Wv6SMqpWmyFIj5UbfFvg==} - engines: {node: '>=14'} - - forwarded@0.2.0: - resolution: {integrity: sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==} - engines: {node: '>= 0.6'} - - fs.realpath@1.0.0: - resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==} - - fsevents@2.3.3: - resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} - engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} - os: [darwin] - - function-bind@1.1.2: - resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} - - get-caller-file@2.0.5: - resolution: {integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==} - engines: {node: 6.* || 8.* || >= 10.*} - - get-func-name@2.0.2: - resolution: {integrity: sha512-8vXOvuE167CtIc3OyItco7N/dpRtBbYOsPsXCz7X/PMnlGjYjSGuZJgM1Y7mmew7BKf9BqvLX2tnOVy1BBUsxQ==} - - get-intrinsic@1.2.4: - resolution: {integrity: sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==} - engines: {node: '>= 0.4'} - - get-stream@6.0.1: - resolution: {integrity: sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==} - engines: {node: '>=10'} - - glob-parent@5.1.2: - resolution: {integrity: sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==} - engines: {node: '>= 6'} - - glob@10.4.5: - resolution: {integrity: sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg==} - hasBin: true - - glob@8.1.0: - resolution: {integrity: sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==} - engines: {node: '>=12'} - deprecated: Glob versions prior to v9 are no longer supported - - globby@11.1.0: - resolution: {integrity: sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==} - engines: {node: '>=10'} - - gopd@1.0.1: - resolution: {integrity: sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==} - - graphile-config@0.0.1-beta.9: - resolution: {integrity: sha512-7vNxXZ24OAgXxDKXYi9JtgWPMuNbBL3057Yf32Ux+/rVP4+EePgySCc+NNnn0tORi8qwqVreN8bdWqGIcSwNXg==} - engines: {node: '>=16'} - - graphile-worker@0.16.6: - resolution: {integrity: sha512-e7gGYDmGqzju2l83MpzX8vNG/lOtVJiSzI3eZpAFubSxh/cxs7sRrRGBGjzBP1kNG0H+c95etPpNRNlH65PYhw==} - engines: {node: '>=14.0.0'} - hasBin: true - - has-flag@3.0.0: - resolution: {integrity: sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==} - engines: {node: '>=4'} - - has-flag@4.0.0: - resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} - engines: {node: '>=8'} - - has-property-descriptors@1.0.2: - resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==} - - has-proto@1.0.3: - resolution: {integrity: sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==} - engines: {node: '>= 0.4'} - - has-symbols@1.0.3: - resolution: {integrity: sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==} - engines: {node: '>= 0.4'} - - hasown@2.0.2: - resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} - engines: {node: '>= 0.4'} - - he@1.2.0: - resolution: {integrity: sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==} - hasBin: true - - html-to-text@9.0.5: - resolution: {integrity: sha512-qY60FjREgVZL03vJU6IfMV4GDjGBIoOyvuFdpBDIX9yTlDw0TjxVBQp+P8NvpdIXNJvfWBTNul7fsAQJq2FNpg==} - engines: {node: '>=14'} - - htmlparser2@8.0.2: - resolution: {integrity: sha512-GYdjWKDkbRLkZ5geuHs5NY1puJ+PXwP7+fHPRz06Eirsb9ugf6d8kkXav6ADhcODhFFPMIXyxkxSuMf3D6NCFA==} - - htmlparser2@9.1.0: - resolution: {integrity: sha512-5zfg6mHUoaer/97TxnGpxmbR7zJtPwIYFMZ/H5ucTlPZhKvtum05yiPK3Mgai3a0DyVxv7qYqoweaEd2nrYQzQ==} - - human-signals@2.1.0: - resolution: {integrity: sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==} - engines: {node: '>=10.17.0'} - - iconv-lite@0.6.3: - resolution: {integrity: sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==} - engines: {node: '>=0.10.0'} - - ieee754@1.2.1: - resolution: {integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==} - - ignore@5.3.2: - resolution: {integrity: sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==} - engines: {node: '>= 4'} - - imapflow@1.0.164: - resolution: {integrity: sha512-+KAmLrpqq2Q0Ts1imMP4svydfhYznlvlLHhgtTb8NiIcccZvdRNfdHVP8/RrGaw0hy0TOaluawsm/6q+TqdLPw==} - - import-fresh@3.3.0: - resolution: {integrity: sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==} - engines: {node: '>=6'} - - inflight@1.0.6: - resolution: {integrity: sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==} - deprecated: This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful. - - inherits@2.0.4: - resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} - - interpret@3.1.1: - resolution: {integrity: sha512-6xwYfHbajpoF0xLW+iwLkhwgvLoZDfjYfoFNu8ftMoXINzwuymNLd9u/KmwtdT2GbR+/Cz66otEGEVVUHX9QLQ==} - engines: {node: '>=10.13.0'} - - ip-address@9.0.5: - resolution: {integrity: sha512-zHtQzGojZXTwZTHQqra+ETKd4Sn3vgi7uBmlPoXVWZqYvuKmtI0l/VZTjqGmJY9x88GGOaZ9+G9ES8hC4T4X8g==} - engines: {node: '>= 12'} - - ipaddr.js@1.9.1: - resolution: {integrity: sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==} - engines: {node: '>= 0.10'} - - is-arrayish@0.2.1: - resolution: {integrity: sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==} - - is-binary-path@2.1.0: - resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==} - engines: {node: '>=8'} - - is-extglob@2.1.1: - resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} - engines: {node: '>=0.10.0'} - - is-fullwidth-code-point@3.0.0: - resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==} - engines: {node: '>=8'} - - is-glob@4.0.3: - resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==} - engines: {node: '>=0.10.0'} - - is-number@7.0.0: - resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==} - engines: {node: '>=0.12.0'} - - is-plain-obj@2.1.0: - resolution: {integrity: sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==} - engines: {node: '>=8'} - - is-stream@2.0.1: - resolution: {integrity: sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==} - engines: {node: '>=8'} - - is-unicode-supported@0.1.0: - resolution: {integrity: sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==} - engines: {node: '>=10'} - - isexe@2.0.0: - resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} - - jackspeak@3.4.3: - resolution: {integrity: sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==} - - joycon@3.1.1: - resolution: {integrity: sha512-34wB/Y7MW7bzjKRjUKTa46I2Z7eV62Rkhva+KkopW7Qvv/OSWBqvkSY7vusOPrNuZcUG3tApvdVgNB8POj3SPw==} - engines: {node: '>=10'} - - js-tokens@4.0.0: - resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} - - js-yaml@4.1.0: - resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==} - hasBin: true - - jsbn@1.1.0: - resolution: {integrity: sha512-4bYVV3aAMtDTTu4+xsDYa6sy9GyJ69/amsu9sYF2zqjiEoZA5xJi3BrfX3uY+/IekIu7MwdObdbDWpoZdBv3/A==} - - json-parse-even-better-errors@2.3.1: - resolution: {integrity: sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==} - - json-schema-ref-resolver@1.0.1: - resolution: {integrity: sha512-EJAj1pgHc1hxF6vo2Z3s69fMjO1INq6eGHXZ8Z6wCQeldCuwxGK9Sxf4/cScGn3FZubCVUehfWtcDM/PLteCQw==} - - json-schema-traverse@1.0.0: - resolution: {integrity: sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==} - - json5@2.2.3: - resolution: {integrity: sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==} - engines: {node: '>=6'} - hasBin: true - - leac@0.6.0: - resolution: {integrity: sha512-y+SqErxb8h7nE/fiEX07jsbuhrpO9lL8eca7/Y1nuWV2moNlXhyd59iDGcRf6moVyDMbmTNzL40SUyrFU/yDpg==} - - libbase64@1.2.1: - resolution: {integrity: sha512-l+nePcPbIG1fNlqMzrh68MLkX/gTxk/+vdvAb388Ssi7UuUN31MI44w4Yf33mM3Cm4xDfw48mdf3rkdHszLNew==} - - libbase64@1.3.0: - resolution: {integrity: sha512-GgOXd0Eo6phYgh0DJtjQ2tO8dc0IVINtZJeARPeiIJqge+HdsWSuaDTe8ztQ7j/cONByDZ3zeB325AHiv5O0dg==} - - libmime@5.2.0: - resolution: {integrity: sha512-X2U5Wx0YmK0rXFbk67ASMeqYIkZ6E5vY7pNWRKtnNzqjvdYYG8xtPDpCnuUEnPU9vlgNev+JoSrcaKSUaNvfsw==} - - libmime@5.3.5: - resolution: {integrity: sha512-nSlR1yRZ43L3cZCiWEw7ali3jY29Hz9CQQ96Oy+sSspYnIP5N54ucOPHqooBsXzwrX1pwn13VUE05q4WmzfaLg==} - - libqp@2.0.1: - resolution: {integrity: sha512-Ka0eC5LkF3IPNQHJmYBWljJsw0UvM6j+QdKRbWyCdTmYwvIDE6a7bCm0UkTAL/K+3KXK5qXT/ClcInU01OpdLg==} - - libqp@2.1.0: - resolution: {integrity: sha512-O6O6/fsG5jiUVbvdgT7YX3xY3uIadR6wEZ7+vy9u7PKHAlSEB6blvC1o5pHBjgsi95Uo0aiBBdkyFecj6jtb7A==} - - light-my-request@5.13.0: - resolution: {integrity: sha512-9IjUN9ZyCS9pTG+KqTDEQo68Sui2lHsYBrfMyVUTTZ3XhH8PMZq7xO94Kr+eP9dhi/kcKsx4N41p2IXEBil1pQ==} - - lilconfig@3.1.2: - resolution: {integrity: sha512-eop+wDAvpItUys0FWkHIKeC9ybYrTGbU41U5K7+bttZZeohvnY7M9dZ5kB21GNWiFT2q1OoPTvncPCgSOVO5ow==} - engines: {node: '>=14'} - - lines-and-columns@1.2.4: - resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} - - linkify-it@5.0.0: - resolution: {integrity: sha512-5aHCbzQRADcdP+ATqnDuhhJ/MRIqDkZX5pyjFHRRysS8vZ5AbqGEoFIb6pYHPZ+L/OC2Lc+xT8uHVVR5CAK/wQ==} - - load-tsconfig@0.2.5: - resolution: {integrity: sha512-IXO6OCs9yg8tMKzfPZ1YmheJbZCiEsnBdcB03l0OcfK9prKnJb96siuHCr5Fl37/yo9DnKU+TLpxzTUspw9shg==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - locate-path@6.0.0: - resolution: {integrity: sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==} - engines: {node: '>=10'} - - lodash.sortby@4.7.0: - resolution: {integrity: sha512-HDWXG8isMntAyRF5vZ7xKuEvOhT4AhlRt/3czTSjvGUxjYCBVRQY48ViDHyfYz9VIoBkW4TMGQNapx+l3RUwdA==} - - log-symbols@4.1.0: - resolution: {integrity: sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==} - engines: {node: '>=10'} - - loupe@3.1.1: - resolution: {integrity: sha512-edNu/8D5MKVfGVFRhFf8aAxiTM6Wumfz5XsaatSxlD3w4R1d/WEKUTydCdPGbl9K7QG/Ca3GnDV2sIKIpXRQcw==} - - lru-cache@10.4.3: - resolution: {integrity: sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==} - - mailparser@3.7.1: - resolution: {integrity: sha512-RCnBhy5q8XtB3mXzxcAfT1huNqN93HTYYyL6XawlIKycfxM/rXPg9tXoZ7D46+SgCS1zxKzw+BayDQSvncSTTw==} - - mailsplit@5.4.0: - resolution: {integrity: sha512-wnYxX5D5qymGIPYLwnp6h8n1+6P6vz/MJn5AzGjZ8pwICWssL+CCQjWBIToOVHASmATot4ktvlLo6CyLfOXWYA==} - - merge-stream@2.0.0: - resolution: {integrity: sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==} - - merge2@1.4.1: - resolution: {integrity: sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==} - engines: {node: '>= 8'} - - micromatch@4.0.8: - resolution: {integrity: sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==} - engines: {node: '>=8.6'} - - mimic-fn@2.1.0: - resolution: {integrity: sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==} - engines: {node: '>=6'} - - minimatch@5.1.6: - resolution: {integrity: sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==} - engines: {node: '>=10'} - - minimatch@9.0.5: - resolution: {integrity: sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==} - engines: {node: '>=16 || 14 >=14.17'} - - minipass@7.1.2: - resolution: {integrity: sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==} - engines: {node: '>=16 || 14 >=14.17'} - - mocha@10.7.3: - resolution: {integrity: sha512-uQWxAu44wwiACGqjbPYmjo7Lg8sFrS3dQe7PP2FQI+woptP4vZXSMcfMyFL/e1yFEeEpV4RtyTpZROOKmxis+A==} - engines: {node: '>= 14.0.0'} - hasBin: true - - ms@2.1.2: - resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==} - - ms@2.1.3: - resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} - - mz@2.7.0: - resolution: {integrity: sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==} - - nodemailer@6.9.13: - resolution: {integrity: sha512-7o38Yogx6krdoBf3jCAqnIN4oSQFx+fMa0I7dK1D+me9kBxx12D+/33wSb+fhOCtIxvYJ+4x4IMEhmhCKfAiOA==} - engines: {node: '>=6.0.0'} - - nodemailer@6.9.14: - resolution: {integrity: sha512-Dobp/ebDKBvz91sbtRKhcznLThrKxKt97GI2FAlAyy+fk19j73Uz3sBXolVtmcXjaorivqsbbbjDY+Jkt4/bQA==} - engines: {node: '>=6.0.0'} - - normalize-path@3.0.0: - resolution: {integrity: sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==} - engines: {node: '>=0.10.0'} - - npm-run-path@4.0.1: - resolution: {integrity: sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==} - engines: {node: '>=8'} - - nth-check@2.1.1: - resolution: {integrity: sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==} - - object-assign@4.1.1: - resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} - engines: {node: '>=0.10.0'} - - object-inspect@1.13.2: - resolution: {integrity: sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==} - engines: {node: '>= 0.4'} - - obuf@1.1.2: - resolution: {integrity: sha512-PX1wu0AmAdPqOL1mWhqmlOd8kOIZQwGZw6rh7uby9fTc5lhaOWFLX3I6R1hrF9k3zUY40e6igsLGkDXK92LJNg==} - - on-exit-leak-free@2.1.2: - resolution: {integrity: sha512-0eJJY6hXLGf1udHwfNftBqH+g73EU4B504nZeKpz1sYRKafAghwxEJunB2O7rDZkL4PGfsMVnTXZ2EjibbqcsA==} - engines: {node: '>=14.0.0'} - - once@1.4.0: - resolution: {integrity: sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==} - - onetime@5.1.2: - resolution: {integrity: sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==} - engines: {node: '>=6'} - - p-limit@3.1.0: - resolution: {integrity: sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==} - engines: {node: '>=10'} - - p-locate@5.0.0: - resolution: {integrity: sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==} - engines: {node: '>=10'} - - package-json-from-dist@1.0.0: - resolution: {integrity: sha512-dATvCeZN/8wQsGywez1mzHtTlP22H8OEfPrVMLNr4/eGa+ijtLn/6M5f0dY8UKNrC2O9UCU6SSoG3qRKnt7STw==} - - parent-module@1.0.1: - resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} - engines: {node: '>=6'} - - parse-json@5.2.0: - resolution: {integrity: sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==} - engines: {node: '>=8'} - - parse5-htmlparser2-tree-adapter@7.0.0: - resolution: {integrity: sha512-B77tOZrqqfUfnVcOrUvfdLbz4pu4RopLD/4vmu3HUPswwTA8OH0EMW9BlWR2B0RCoiZRAHEUu7IxeP1Pd1UU+g==} - - parse5-parser-stream@7.1.2: - resolution: {integrity: sha512-JyeQc9iwFLn5TbvvqACIF/VXG6abODeB3Fwmv/TGdLk2LfbWkaySGY72at4+Ty7EkPZj854u4CrICqNk2qIbow==} - - parse5@7.1.2: - resolution: {integrity: sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==} - - parseley@0.12.1: - resolution: {integrity: sha512-e6qHKe3a9HWr0oMRVDTRhKce+bRO8VGQR3NyVwcjwrbhMmFCX9KszEV35+rn4AdilFAq9VPxP/Fe1wC9Qjd2lw==} - - path-exists@4.0.0: - resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==} - engines: {node: '>=8'} - - path-key@3.1.1: - resolution: {integrity: sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==} - engines: {node: '>=8'} - - path-scurry@1.11.1: - resolution: {integrity: sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==} - engines: {node: '>=16 || 14 >=14.18'} - - path-type@4.0.0: - resolution: {integrity: sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==} - engines: {node: '>=8'} - - pathval@2.0.0: - resolution: {integrity: sha512-vE7JKRyES09KiunauX7nd2Q9/L7lhok4smP9RZTDeD4MVs72Dp2qNFVz39Nz5a0FVEW0BJR6C0DYrq6unoziZA==} - engines: {node: '>= 14.16'} - - peberminta@0.9.0: - resolution: {integrity: sha512-XIxfHpEuSJbITd1H3EeQwpcZbTLHc+VVr8ANI9t5sit565tsI4/xK3KWTUFE2e6QiangUkh3B0jihzmGnNrRsQ==} - - pg-cloudflare@1.1.1: - resolution: {integrity: sha512-xWPagP/4B6BgFO+EKz3JONXv3YDgvkbVrGw2mTo3D6tVDQRh1e7cqVGvyR3BE+eQgAvx1XhW/iEASj4/jCWl3Q==} - - pg-connection-string@2.6.4: - resolution: {integrity: sha512-v+Z7W/0EO707aNMaAEfiGnGL9sxxumwLl2fJvCQtMn9Fxsg+lPpPkdcyBSv/KFgpGdYkMfn+EI1Or2EHjpgLCA==} - - pg-int8@1.0.1: - resolution: {integrity: sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw==} - engines: {node: '>=4.0.0'} - - pg-numeric@1.0.2: - resolution: {integrity: sha512-BM/Thnrw5jm2kKLE5uJkXqqExRUY/toLHda65XgFTBTFYZyopbKjBe29Ii3RbkvlsMoFwD+tHeGaCjjv0gHlyw==} - engines: {node: '>=4'} - - pg-pool@3.6.2: - resolution: {integrity: sha512-Htjbg8BlwXqSBQ9V8Vjtc+vzf/6fVUuak/3/XXKA9oxZprwW3IMDQTGHP+KDmVL7rtd+R1QjbnCFPuTHm3G4hg==} - peerDependencies: - pg: '>=8.0' - - pg-protocol@1.6.1: - resolution: {integrity: sha512-jPIlvgoD63hrEuihvIg+tJhoGjUsLPn6poJY9N5CnlPd91c2T18T/9zBtLxZSb1EhYxBRoZJtzScCaWlYLtktg==} - - pg-types@2.2.0: - resolution: {integrity: sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==} - engines: {node: '>=4'} - - pg-types@4.0.2: - resolution: {integrity: sha512-cRL3JpS3lKMGsKaWndugWQoLOCoP+Cic8oseVcbr0qhPzYD5DWXK+RZ9LY9wxRf7RQia4SCwQlXk0q6FCPrVng==} - engines: {node: '>=10'} - - pg@8.12.0: - resolution: {integrity: sha512-A+LHUSnwnxrnL/tZ+OLfqR1SxLN3c/pgDztZ47Rpbsd4jUytsTtwQo/TLPRzPJMp/1pbhYVhH9cuSZLAajNfjQ==} - engines: {node: '>= 8.0.0'} - peerDependencies: - pg-native: '>=3.0.1' - peerDependenciesMeta: - pg-native: - optional: true - - pgpass@1.0.5: - resolution: {integrity: sha512-FdW9r/jQZhSeohs1Z3sI1yxFQNFvMcnmfuj4WBMUTxOrAyLMaTcE1aAMBiTlbMNaXvBCQuVi0R7hd8udDSP7ug==} - - picocolors@1.1.0: - resolution: {integrity: sha512-TQ92mBOW0l3LeMeyLV6mzy/kWr8lkd/hp3mTg7wYK7zJhuBStmGMBG0BdeDZS/dZx1IukaX6Bk11zcln25o1Aw==} - - picomatch@2.3.1: - resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==} - engines: {node: '>=8.6'} - - pino-abstract-transport@1.2.0: - resolution: {integrity: sha512-Guhh8EZfPCfH+PMXAb6rKOjGQEoy0xlAIn+irODG5kgfYV+BQ0rGYYWTIel3P5mmyXqkYkPmdIkywsn6QKUR1Q==} - - pino-std-serializers@7.0.0: - resolution: {integrity: sha512-e906FRY0+tV27iq4juKzSYPbUj2do2X2JX4EzSca1631EB2QJQUqGbDuERal7LCtOpxl6x3+nvo9NPZcmjkiFA==} - - pino@9.2.0: - resolution: {integrity: sha512-g3/hpwfujK5a4oVbaefoJxezLzsDgLcNJeITvC6yrfwYeT9la+edCK42j5QpEQSQCZgTKapXvnQIdgZwvRaZug==} - hasBin: true - - pino@9.4.0: - resolution: {integrity: sha512-nbkQb5+9YPhQRz/BeQmrWpEknAaqjpAqRK8NwJpmrX/JHu7JuZC5G1CeAwJDJfGes4h+YihC6in3Q2nGb+Y09w==} - hasBin: true - - pirates@4.0.6: - resolution: {integrity: sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==} - engines: {node: '>= 6'} - - postcss-load-config@6.0.1: - resolution: {integrity: sha512-oPtTM4oerL+UXmx+93ytZVN82RrlY/wPUV8IeDxFrzIjXOLF1pN+EmKPLbubvKHT2HC20xXsCAH2Z+CKV6Oz/g==} - engines: {node: '>= 18'} - peerDependencies: - jiti: '>=1.21.0' - postcss: '>=8.0.9' - tsx: ^4.8.1 - yaml: ^2.4.2 - peerDependenciesMeta: - jiti: - optional: true - postcss: - optional: true - tsx: - optional: true - yaml: - optional: true - - postgres-array@2.0.0: - resolution: {integrity: sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA==} - engines: {node: '>=4'} - - postgres-array@3.0.2: - resolution: {integrity: sha512-6faShkdFugNQCLwucjPcY5ARoW1SlbnrZjmGl0IrrqewpvxvhSLHimCVzqeuULCbG0fQv7Dtk1yDbG3xv7Veog==} - engines: {node: '>=12'} - - postgres-bytea@1.0.0: - resolution: {integrity: sha512-xy3pmLuQqRBZBXDULy7KbaitYqLcmxigw14Q5sj8QBVLqEwXfeybIKVWiqAXTlcvdvb0+xkOtDbfQMOf4lST1w==} - engines: {node: '>=0.10.0'} - - postgres-bytea@3.0.0: - resolution: {integrity: sha512-CNd4jim9RFPkObHSjVHlVrxoVQXz7quwNFpz7RY1okNNme49+sVyiTvTRobiLV548Hx/hb1BG+iE7h9493WzFw==} - engines: {node: '>= 6'} - - postgres-date@1.0.7: - resolution: {integrity: sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q==} - engines: {node: '>=0.10.0'} - - postgres-date@2.1.0: - resolution: {integrity: sha512-K7Juri8gtgXVcDfZttFKVmhglp7epKb1K4pgrkLxehjqkrgPhfG6OO8LHLkfaqkbpjNRnra018XwAr1yQFWGcA==} - engines: {node: '>=12'} - - postgres-interval@1.2.0: - resolution: {integrity: sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==} - engines: {node: '>=0.10.0'} - - postgres-interval@3.0.0: - resolution: {integrity: sha512-BSNDnbyZCXSxgA+1f5UU2GmwhoI0aU5yMxRGO8CdFEcY2BQF9xm/7MqKnYoM1nJDk8nONNWDk9WeSmePFhQdlw==} - engines: {node: '>=12'} - - postgres-range@1.1.4: - resolution: {integrity: sha512-i/hbxIE9803Alj/6ytL7UHQxRvZkI9O4Sy+J3HGc4F4oo/2eQAjTSNJ0bfxyse3bH0nuVesCk+3IRLaMtG3H6w==} - - process-warning@3.0.0: - resolution: {integrity: sha512-mqn0kFRl0EoqhnL0GQ0veqFHyIN1yig9RHh/InzORTUiZHFRAur+aMtRkELNwGs9aNwKS6tg/An4NYBPGwvtzQ==} - - process-warning@4.0.0: - resolution: {integrity: sha512-/MyYDxttz7DfGMMHiysAsFE4qF+pQYAA8ziO/3NcRVrQ5fSk+Mns4QZA/oRPFzvcqNoVJXQNWNAsdwBXLUkQKw==} - - process@0.11.10: - resolution: {integrity: sha512-cdGef/drWFoydD1JsMzuFf8100nZl+GT+yacc2bEced5f9Rjk4z+WtFUTBu9PhOi9j/jfmBPu0mMEY4wIdAF8A==} - engines: {node: '>= 0.6.0'} - - proxy-addr@2.0.7: - resolution: {integrity: sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==} - engines: {node: '>= 0.10'} - - punycode.js@2.3.1: - resolution: {integrity: sha512-uxFIHU0YlHYhDQtV4R9J6a52SLx28BCjT+4ieh7IGbgwVJWO+km431c4yRlREUAsAmt/uMjQUyQHNEPf0M39CA==} - engines: {node: '>=6'} - - punycode@2.3.1: - resolution: {integrity: sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==} - engines: {node: '>=6'} - - qs@6.13.0: - resolution: {integrity: sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==} - engines: {node: '>=0.6'} - - queue-microtask@1.2.3: - resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==} - - quick-format-unescaped@4.0.4: - resolution: {integrity: sha512-tYC1Q1hgyRuHgloV/YXs2w15unPVh8qfu/qCTfhTYamaw7fyhumKa2yGpdSo87vY32rIclj+4fWYQXUMs9EHvg==} - - randombytes@2.1.0: - resolution: {integrity: sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==} - - readable-stream@4.5.2: - resolution: {integrity: sha512-yjavECdqeZ3GLXNgRXgeQEdz9fvDDkNKyHnbHRFtOr7/LcfgBcmct7t/ET+HaCTqfh06OzoAxrkN/IfjJBVe+g==} - engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} - - readdirp@3.6.0: - resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==} - engines: {node: '>=8.10.0'} - - real-require@0.2.0: - resolution: {integrity: sha512-57frrGM/OCTLqLOAh0mhVA9VBMHd+9U7Zb2THMGdBUoZVOtGbJzjxsYGDJ3A9AYYCP4hn6y1TVbaOfzWtm5GFg==} - engines: {node: '>= 12.13.0'} - - require-directory@2.1.1: - resolution: {integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==} - engines: {node: '>=0.10.0'} - - require-from-string@2.0.2: - resolution: {integrity: sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==} - engines: {node: '>=0.10.0'} - - resolve-from@4.0.0: - resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==} - engines: {node: '>=4'} - - resolve-from@5.0.0: - resolution: {integrity: sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==} - engines: {node: '>=8'} - - ret@0.4.3: - resolution: {integrity: sha512-0f4Memo5QP7WQyUEAYUO3esD/XjOc3Zjjg5CPsAq1p8sIu0XPeMbHJemKA0BO7tV0X7+A0FoEpbmHXWxPyD3wQ==} - engines: {node: '>=10'} - - reusify@1.0.4: - resolution: {integrity: sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==} - engines: {iojs: '>=1.0.0', node: '>=0.10.0'} - - rfdc@1.4.1: - resolution: {integrity: sha512-q1b3N5QkRUWUl7iyylaaj3kOpIT0N2i9MqIEQXP73GVsN9cw3fdx8X63cEmWhJGi2PPCF23Ijp7ktmd39rawIA==} - - rollup@4.21.2: - resolution: {integrity: sha512-e3TapAgYf9xjdLvKQCkQTnbTKd4a6jwlpQSJJFokHGaX2IVjoEqkIIhiQfqsi0cdwlOD+tQGuOd5AJkc5RngBw==} - engines: {node: '>=18.0.0', npm: '>=8.0.0'} - hasBin: true - - run-parallel@1.2.0: - resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} - - safe-buffer@5.2.1: - resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} - - safe-regex2@3.1.0: - resolution: {integrity: sha512-RAAZAGbap2kBfbVhvmnTFv73NWLMvDGOITFYTZBAaY8eR+Ir4ef7Up/e7amo+y1+AH+3PtLkrt9mvcTsG9LXug==} - - safe-stable-stringify@2.5.0: - resolution: {integrity: sha512-b3rppTKm9T+PsVCBEOUR46GWI7fdOs00VKZ1+9c1EWDaDMvjQc6tUwuFyIprgGgTcWoVHSKrU8H31ZHA2e0RHA==} - engines: {node: '>=10'} - - safer-buffer@2.1.2: - resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==} - - secure-json-parse@2.7.0: - resolution: {integrity: sha512-6aU+Rwsezw7VR8/nyvKTx8QpWH9FrcYiXXlqC4z5d5XQBDRqtbfsRjnwGyqbi3gddNtWHuEk9OANUotL26qKUw==} - - selderee@0.11.0: - resolution: {integrity: sha512-5TF+l7p4+OsnP8BCCvSyZiSPc4x4//p5uPwK8TCnVPJYRmU2aYKMpOXvw8zM5a5JvuuCGN1jmsMwuU2W02ukfA==} - - semver@7.6.3: - resolution: {integrity: sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==} - engines: {node: '>=10'} - hasBin: true - - serialize-javascript@6.0.2: - resolution: {integrity: sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g==} - - set-cookie-parser@2.7.0: - resolution: {integrity: sha512-lXLOiqpkUumhRdFF3k1osNXCy9akgx/dyPZ5p8qAg9seJzXr5ZrlqZuWIMuY6ejOsVLE6flJ5/h3lsn57fQ/PQ==} - - set-function-length@1.2.2: - resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==} - engines: {node: '>= 0.4'} - - shebang-command@2.0.0: - resolution: {integrity: sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==} - engines: {node: '>=8'} - - shebang-regex@3.0.0: - resolution: {integrity: sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==} - engines: {node: '>=8'} - - side-channel@1.0.6: - resolution: {integrity: sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==} - engines: {node: '>= 0.4'} - - signal-exit@3.0.7: - resolution: {integrity: sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==} - - signal-exit@4.1.0: - resolution: {integrity: sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==} - engines: {node: '>=14'} - - slash@3.0.0: - resolution: {integrity: sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==} - engines: {node: '>=8'} - - smart-buffer@4.2.0: - resolution: {integrity: sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg==} - engines: {node: '>= 6.0.0', npm: '>= 3.0.0'} - - socks@2.8.3: - resolution: {integrity: sha512-l5x7VUUWbjVFbafGLxPWkYsHIhEvmF85tbIeFZWc8ZPtoMyybuEhL7Jye/ooC4/d48FgOjSJXgsF/AJPYCW8Zw==} - engines: {node: '>= 10.0.0', npm: '>= 3.0.0'} - - sonic-boom@4.1.0: - resolution: {integrity: sha512-NGipjjRicyJJ03rPiZCJYjwlsuP2d1/5QUviozRXC7S3WdVWNK5e3Ojieb9CCyfhq2UC+3+SRd9nG3I2lPRvUw==} - - source-map@0.8.0-beta.0: - resolution: {integrity: sha512-2ymg6oRBpebeZi9UUNsgQ89bhx01TcTkmNTGnNO88imTmbSgy4nfujrgVEFKWpMTEGA11EDkTt7mqObTPdigIA==} - engines: {node: '>= 8'} - - split2@4.2.0: - resolution: {integrity: sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==} - engines: {node: '>= 10.x'} - - sprintf-js@1.1.3: - resolution: {integrity: sha512-Oo+0REFV59/rz3gfJNKQiBlwfHaSESl1pcGyABQsnnIfWOFt6JNj5gCog2U6MLZ//IGYD+nA8nI+mTShREReaA==} - - string-width@4.2.3: - resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} - engines: {node: '>=8'} - - string-width@5.1.2: - resolution: {integrity: sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==} - engines: {node: '>=12'} - - string_decoder@1.3.0: - resolution: {integrity: sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==} - - strip-ansi@6.0.1: - resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} - engines: {node: '>=8'} - - strip-ansi@7.1.0: - resolution: {integrity: sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==} - engines: {node: '>=12'} - - strip-final-newline@2.0.0: - resolution: {integrity: sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==} - engines: {node: '>=6'} - - strip-json-comments@3.1.1: - resolution: {integrity: sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==} - engines: {node: '>=8'} - - sucrase@3.35.0: - resolution: {integrity: sha512-8EbVDiu9iN/nESwxeSxDKe0dunta1GOlHufmSSXxMD2z2/tMZpDMpvXQGsc+ajGo8y2uYUmixaSRUc/QPoQ0GA==} - engines: {node: '>=16 || 14 >=14.17'} - hasBin: true - - supports-color@5.5.0: - resolution: {integrity: sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==} - engines: {node: '>=4'} - - supports-color@7.2.0: - resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==} - engines: {node: '>=8'} - - supports-color@8.1.1: - resolution: {integrity: sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==} - engines: {node: '>=10'} - - thenify-all@1.6.0: - resolution: {integrity: sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==} - engines: {node: '>=0.8'} - - thenify@3.3.1: - resolution: {integrity: sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==} - - thread-stream@3.1.0: - resolution: {integrity: sha512-OqyPZ9u96VohAyMfJykzmivOrY2wfMSf3C5TtFJVgN+Hm6aj+voFhlK+kZEIv2FBh1X6Xp3DlnCOfEQ3B2J86A==} - - tlds@1.252.0: - resolution: {integrity: sha512-GA16+8HXvqtfEnw/DTcwB0UU354QE1n3+wh08oFjr6Znl7ZLAeUgYzCcK+/CCrOyE0vnHR8/pu3XXG3vDijXpQ==} - hasBin: true - - to-regex-range@5.0.1: - resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} - engines: {node: '>=8.0'} - - toad-cache@3.7.0: - resolution: {integrity: sha512-/m8M+2BJUpoJdgAHoG+baCwBT+tf2VraSfkBgl0Y00qIWt41DJ8R5B8nsEw0I58YwF5IZH6z24/2TobDKnqSWw==} - engines: {node: '>=12'} - - tr46@1.0.1: - resolution: {integrity: sha512-dTpowEjclQ7Kgx5SdBkqRzVhERQXov8/l9Ft9dVM9fmg0W0KQSVaXX9T4i6twCPNtYiZM53lpSSUAwJbFPOHxA==} - - tree-kill@1.2.2: - resolution: {integrity: sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==} - hasBin: true - - ts-interface-checker@0.1.13: - resolution: {integrity: sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==} - - tslib@2.7.0: - resolution: {integrity: sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA==} - - tsup@8.2.4: - resolution: {integrity: sha512-akpCPePnBnC/CXgRrcy72ZSntgIEUa1jN0oJbbvpALWKNOz1B7aM+UVDWGRGIO/T/PZugAESWDJUAb5FD48o8Q==} - engines: {node: '>=18'} - hasBin: true - peerDependencies: - '@microsoft/api-extractor': ^7.36.0 - '@swc/core': ^1 - postcss: ^8.4.12 - typescript: '>=4.5.0' - peerDependenciesMeta: - '@microsoft/api-extractor': - optional: true - '@swc/core': - optional: true - postcss: - optional: true - typescript: - optional: true - - typescript@5.5.4: - resolution: {integrity: sha512-Mtq29sKDAEYP7aljRgtPOpTvOfbwRWlS6dPRzwjdE+C0R4brX/GUyhHSecbHMFLNBLcJIPt9nl9yG5TZ1weH+Q==} - engines: {node: '>=14.17'} - hasBin: true - - uc.micro@2.1.0: - resolution: {integrity: sha512-ARDJmphmdvUk6Glw7y9DQ2bFkKBHwQHLi2lsaH6PPmz/Ka9sFOBsBluozhDltWmnv9u/cF6Rt87znRTPV+yp/A==} - - undici-types@6.19.8: - resolution: {integrity: sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==} - - undici@6.19.8: - resolution: {integrity: sha512-U8uCCl2x9TK3WANvmBavymRzxbfFYG+tAu+fgx3zxQy3qdagQqBLwJVrdyO1TBfUXvfKveMKJZhpvUYoOjM+4g==} - engines: {node: '>=18.17'} - - webidl-conversions@4.0.2: - resolution: {integrity: sha512-YQ+BmxuTgd6UXZW3+ICGfyqRyHXVlD5GtQr5+qjiNW7bF0cqrzX500HVXPBOvgXb5YnzDd+h0zqyv61KUD7+Sg==} - - whatwg-encoding@3.1.1: - resolution: {integrity: sha512-6qN4hJdMwfYBtE3YBTTHhoeuUrDBPZmbQaxWAqSALV/MeEnR5z1xd8UKud2RAkFoPkmB+hli1TZSnyi84xz1vQ==} - engines: {node: '>=18'} - - whatwg-mimetype@4.0.0: - resolution: {integrity: sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg==} - engines: {node: '>=18'} - - whatwg-url@7.1.0: - resolution: {integrity: sha512-WUu7Rg1DroM7oQvGWfOiAK21n74Gg+T4elXEQYkOhtyLeWiJFoOGLXPKI/9gzIie9CtwVLm8wtw6YJdKyxSjeg==} - - which@2.0.2: - resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==} - engines: {node: '>= 8'} - hasBin: true - - workerpool@6.5.1: - resolution: {integrity: sha512-Fs4dNYcsdpYSAfVxhnl1L5zTksjvOJxtC5hzMNl+1t9B8hTJTdKDyZ5ju7ztgPy+ft9tBFXoOlDNiOT9WUXZlA==} - - wrap-ansi@7.0.0: - resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} - engines: {node: '>=10'} - - wrap-ansi@8.1.0: - resolution: {integrity: sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==} - engines: {node: '>=12'} - - wrappy@1.0.2: - resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==} - - xtend@4.0.2: - resolution: {integrity: sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==} - engines: {node: '>=0.4'} - - y18n@5.0.8: - resolution: {integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==} - engines: {node: '>=10'} - - yargs-parser@20.2.9: - resolution: {integrity: sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==} - engines: {node: '>=10'} - - yargs-parser@21.1.1: - resolution: {integrity: sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==} - engines: {node: '>=12'} - - yargs-unparser@2.0.0: - resolution: {integrity: sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==} - engines: {node: '>=10'} - - yargs@16.2.0: - resolution: {integrity: sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==} - engines: {node: '>=10'} - - yargs@17.7.2: - resolution: {integrity: sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==} - engines: {node: '>=12'} - - yocto-queue@0.1.0: - resolution: {integrity: sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==} - engines: {node: '>=10'} - -snapshots: - - '@babel/code-frame@7.24.7': - dependencies: - '@babel/highlight': 7.24.7 - picocolors: 1.1.0 - - '@babel/helper-validator-identifier@7.24.7': {} - - '@babel/highlight@7.24.7': - dependencies: - '@babel/helper-validator-identifier': 7.24.7 - chalk: 2.4.2 - js-tokens: 4.0.0 - picocolors: 1.1.0 - - '@esbuild-plugins/esm-externals@0.1.2(esbuild@0.23.1)': - dependencies: - debug: 4.3.6(supports-color@8.1.1) - esbuild: 0.23.1 - escape-string-regexp: 4.0.0 - transitivePeerDependencies: - - supports-color - - '@esbuild/aix-ppc64@0.23.1': - optional: true - - '@esbuild/android-arm64@0.23.1': - optional: true - - '@esbuild/android-arm@0.23.1': - optional: true - - '@esbuild/android-x64@0.23.1': - optional: true - - '@esbuild/darwin-arm64@0.23.1': - optional: true - - '@esbuild/darwin-x64@0.23.1': - optional: true - - '@esbuild/freebsd-arm64@0.23.1': - optional: true - - '@esbuild/freebsd-x64@0.23.1': - optional: true - - '@esbuild/linux-arm64@0.23.1': - optional: true - - '@esbuild/linux-arm@0.23.1': - optional: true - - '@esbuild/linux-ia32@0.23.1': - optional: true - - '@esbuild/linux-loong64@0.23.1': - optional: true - - '@esbuild/linux-mips64el@0.23.1': - optional: true - - '@esbuild/linux-ppc64@0.23.1': - optional: true - - '@esbuild/linux-riscv64@0.23.1': - optional: true - - '@esbuild/linux-s390x@0.23.1': - optional: true - - '@esbuild/linux-x64@0.23.1': - optional: true - - '@esbuild/netbsd-x64@0.23.1': - optional: true - - '@esbuild/openbsd-arm64@0.23.1': - optional: true - - '@esbuild/openbsd-x64@0.23.1': - optional: true - - '@esbuild/sunos-x64@0.23.1': - optional: true - - '@esbuild/win32-arm64@0.23.1': - optional: true - - '@esbuild/win32-ia32@0.23.1': - optional: true - - '@esbuild/win32-x64@0.23.1': - optional: true - - '@fastify/ajv-compiler@3.6.0': - dependencies: - ajv: 8.17.1 - ajv-formats: 2.1.1(ajv@8.17.1) - fast-uri: 2.4.0 - - '@fastify/error@3.4.1': {} - - '@fastify/fast-json-stringify-compiler@4.3.0': - dependencies: - fast-json-stringify: 5.16.1 - - '@fastify/merge-json-schemas@0.1.1': - dependencies: - fast-deep-equal: 3.1.3 - - '@graphile/logger@0.2.0': {} - - '@isaacs/cliui@8.0.2': - dependencies: - string-width: 5.1.2 - string-width-cjs: string-width@4.2.3 - strip-ansi: 7.1.0 - strip-ansi-cjs: strip-ansi@6.0.1 - wrap-ansi: 8.1.0 - wrap-ansi-cjs: wrap-ansi@7.0.0 - - '@jridgewell/gen-mapping@0.3.5': - dependencies: - '@jridgewell/set-array': 1.2.1 - '@jridgewell/sourcemap-codec': 1.5.0 - '@jridgewell/trace-mapping': 0.3.25 - - '@jridgewell/resolve-uri@3.1.2': {} - - '@jridgewell/set-array@1.2.1': {} - - '@jridgewell/sourcemap-codec@1.5.0': {} - - '@jridgewell/trace-mapping@0.3.25': - dependencies: - '@jridgewell/resolve-uri': 3.1.2 - '@jridgewell/sourcemap-codec': 1.5.0 - - '@nodelib/fs.scandir@2.1.5': - dependencies: - '@nodelib/fs.stat': 2.0.5 - run-parallel: 1.2.0 - - '@nodelib/fs.stat@2.0.5': {} - - '@nodelib/fs.walk@1.2.8': - dependencies: - '@nodelib/fs.scandir': 2.1.5 - fastq: 1.17.1 - - '@pkgjs/parseargs@0.11.0': - optional: true - - '@rollup/rollup-android-arm-eabi@4.21.2': - optional: true - - '@rollup/rollup-android-arm64@4.21.2': - optional: true - - '@rollup/rollup-darwin-arm64@4.21.2': - optional: true - - '@rollup/rollup-darwin-x64@4.21.2': - optional: true - - '@rollup/rollup-linux-arm-gnueabihf@4.21.2': - optional: true - - '@rollup/rollup-linux-arm-musleabihf@4.21.2': - optional: true - - '@rollup/rollup-linux-arm64-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-arm64-musl@4.21.2': - optional: true - - '@rollup/rollup-linux-powerpc64le-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-riscv64-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-s390x-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-x64-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-x64-musl@4.21.2': - optional: true - - '@rollup/rollup-win32-arm64-msvc@4.21.2': - optional: true - - '@rollup/rollup-win32-ia32-msvc@4.21.2': - optional: true - - '@rollup/rollup-win32-x64-msvc@4.21.2': - optional: true - - '@selderee/plugin-htmlparser2@0.11.0': - dependencies: - domhandler: 5.0.3 - selderee: 0.11.0 - - '@types/chai@4.3.19': {} - - '@types/debug@4.1.12': - dependencies: - '@types/ms': 0.7.34 - - '@types/estree@1.0.5': {} - - '@types/imapflow@1.0.19': - dependencies: - '@types/node': 22.5.2 - - '@types/interpret@1.1.3': - dependencies: - '@types/node': 22.5.2 - - '@types/mailparser@3.4.4': - dependencies: - '@types/node': 22.5.2 - iconv-lite: 0.6.3 - - '@types/mocha@10.0.7': {} - - '@types/ms@0.7.34': {} - - '@types/node@20.16.3': - dependencies: - undici-types: 6.19.8 - - '@types/node@22.5.2': - dependencies: - undici-types: 6.19.8 - - '@types/pg@8.11.8': - dependencies: - '@types/node': 22.5.2 - pg-protocol: 1.6.1 - pg-types: 4.0.2 - - '@types/qs@6.9.15': {} - - '@types/semver@7.5.8': {} - - abort-controller@3.0.0: - dependencies: - event-target-shim: 5.0.1 - - abstract-logging@2.0.1: {} - - ajv-formats@2.1.1(ajv@8.17.1): - optionalDependencies: - ajv: 8.17.1 - - ajv-formats@3.0.1(ajv@8.17.1): - optionalDependencies: - ajv: 8.17.1 - - ajv@8.17.1: - dependencies: - fast-deep-equal: 3.1.3 - fast-uri: 3.0.1 - json-schema-traverse: 1.0.0 - require-from-string: 2.0.2 - - ansi-colors@4.1.3: {} - - ansi-regex@5.0.1: {} - - ansi-regex@6.0.1: {} - - ansi-styles@3.2.1: - dependencies: - color-convert: 1.9.3 - - ansi-styles@4.3.0: - dependencies: - color-convert: 2.0.1 - - ansi-styles@6.2.1: {} - - any-promise@1.3.0: {} - - anymatch@3.1.3: - dependencies: - normalize-path: 3.0.0 - picomatch: 2.3.1 - - argparse@2.0.1: {} - - array-union@2.1.0: {} - - assertion-error@2.0.1: {} - - atomic-sleep@1.0.0: {} - - avvio@8.4.0: - dependencies: - '@fastify/error': 3.4.1 - fastq: 1.17.1 - - balanced-match@1.0.2: {} - - base64-js@1.5.1: {} - - binary-extensions@2.3.0: {} - - boolbase@1.0.0: {} - - brace-expansion@2.0.1: - dependencies: - balanced-match: 1.0.2 - - braces@3.0.3: - dependencies: - fill-range: 7.1.1 - - browser-stdout@1.3.1: {} - - buffer@6.0.3: - dependencies: - base64-js: 1.5.1 - ieee754: 1.2.1 - - bundle-require@5.0.0(esbuild@0.23.1): - dependencies: - esbuild: 0.23.1 - load-tsconfig: 0.2.5 - - cac@6.7.14: {} - - call-bind@1.0.7: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - set-function-length: 1.2.2 - - callsites@3.1.0: {} - - camelcase@6.3.0: {} - - chai@5.1.1: - dependencies: - assertion-error: 2.0.1 - check-error: 2.1.1 - deep-eql: 5.0.2 - loupe: 3.1.1 - pathval: 2.0.0 - - chalk@2.4.2: - dependencies: - ansi-styles: 3.2.1 - escape-string-regexp: 1.0.5 - supports-color: 5.5.0 - - chalk@4.1.2: - dependencies: - ansi-styles: 4.3.0 - supports-color: 7.2.0 - - check-error@2.1.1: {} - - cheerio-select@2.1.0: - dependencies: - boolbase: 1.0.0 - css-select: 5.1.0 - css-what: 6.1.0 - domelementtype: 2.3.0 - domhandler: 5.0.3 - domutils: 3.1.0 - - cheerio@1.0.0: - dependencies: - cheerio-select: 2.1.0 - dom-serializer: 2.0.0 - domhandler: 5.0.3 - domutils: 3.1.0 - encoding-sniffer: 0.2.0 - htmlparser2: 9.1.0 - parse5: 7.1.2 - parse5-htmlparser2-tree-adapter: 7.0.0 - parse5-parser-stream: 7.1.2 - undici: 6.19.8 - whatwg-mimetype: 4.0.0 - - chokidar@3.6.0: - dependencies: - anymatch: 3.1.3 - braces: 3.0.3 - glob-parent: 5.1.2 - is-binary-path: 2.1.0 - is-glob: 4.0.3 - normalize-path: 3.0.0 - readdirp: 3.6.0 - optionalDependencies: - fsevents: 2.3.3 - - cliui@7.0.4: - dependencies: - string-width: 4.2.3 - strip-ansi: 6.0.1 - wrap-ansi: 7.0.0 - - cliui@8.0.1: - dependencies: - string-width: 4.2.3 - strip-ansi: 6.0.1 - wrap-ansi: 7.0.0 - - color-convert@1.9.3: - dependencies: - color-name: 1.1.3 - - color-convert@2.0.1: - dependencies: - color-name: 1.1.4 - - color-name@1.1.3: {} - - color-name@1.1.4: {} - - commander@4.1.1: {} - - consola@3.2.3: {} - - cookie@0.6.0: {} - - cosmiconfig@8.3.6(typescript@5.5.4): - dependencies: - import-fresh: 3.3.0 - js-yaml: 4.1.0 - parse-json: 5.2.0 - path-type: 4.0.0 - optionalDependencies: - typescript: 5.5.4 - - cross-spawn@7.0.3: - dependencies: - path-key: 3.1.1 - shebang-command: 2.0.0 - which: 2.0.2 - - css-select@5.1.0: - dependencies: - boolbase: 1.0.0 - css-what: 6.1.0 - domhandler: 5.0.3 - domutils: 3.1.0 - nth-check: 2.1.1 - - css-what@6.1.0: {} - - date-fns@3.6.0: {} - - debug@4.3.6(supports-color@8.1.1): - dependencies: - ms: 2.1.2 - optionalDependencies: - supports-color: 8.1.1 - - decamelize@4.0.0: {} - - deep-eql@5.0.2: {} - - deepmerge@4.3.1: {} - - define-data-property@1.1.4: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - gopd: 1.0.1 - - diff@5.2.0: {} - - dir-glob@3.0.1: - dependencies: - path-type: 4.0.0 - - dom-serializer@2.0.0: - dependencies: - domelementtype: 2.3.0 - domhandler: 5.0.3 - entities: 4.5.0 - - domelementtype@2.3.0: {} - - domhandler@5.0.3: - dependencies: - domelementtype: 2.3.0 - - domutils@3.1.0: - dependencies: - dom-serializer: 2.0.0 - domelementtype: 2.3.0 - domhandler: 5.0.3 - - dotenv@16.4.5: {} - - eastasianwidth@0.2.0: {} - - emoji-regex@8.0.0: {} - - emoji-regex@9.2.2: {} - - encoding-japanese@2.0.0: {} - - encoding-japanese@2.1.0: {} - - encoding-japanese@2.2.0: {} - - encoding-sniffer@0.2.0: - dependencies: - iconv-lite: 0.6.3 - whatwg-encoding: 3.1.1 - - entities@4.5.0: {} - - error-ex@1.3.2: - dependencies: - is-arrayish: 0.2.1 - - es-define-property@1.0.0: - dependencies: - get-intrinsic: 1.2.4 - - es-errors@1.3.0: {} - - esbuild@0.23.1: - optionalDependencies: - '@esbuild/aix-ppc64': 0.23.1 - '@esbuild/android-arm': 0.23.1 - '@esbuild/android-arm64': 0.23.1 - '@esbuild/android-x64': 0.23.1 - '@esbuild/darwin-arm64': 0.23.1 - '@esbuild/darwin-x64': 0.23.1 - '@esbuild/freebsd-arm64': 0.23.1 - '@esbuild/freebsd-x64': 0.23.1 - '@esbuild/linux-arm': 0.23.1 - '@esbuild/linux-arm64': 0.23.1 - '@esbuild/linux-ia32': 0.23.1 - '@esbuild/linux-loong64': 0.23.1 - '@esbuild/linux-mips64el': 0.23.1 - '@esbuild/linux-ppc64': 0.23.1 - '@esbuild/linux-riscv64': 0.23.1 - '@esbuild/linux-s390x': 0.23.1 - '@esbuild/linux-x64': 0.23.1 - '@esbuild/netbsd-x64': 0.23.1 - '@esbuild/openbsd-arm64': 0.23.1 - '@esbuild/openbsd-x64': 0.23.1 - '@esbuild/sunos-x64': 0.23.1 - '@esbuild/win32-arm64': 0.23.1 - '@esbuild/win32-ia32': 0.23.1 - '@esbuild/win32-x64': 0.23.1 - - escalade@3.2.0: {} - - escape-string-regexp@1.0.5: {} - - escape-string-regexp@4.0.0: {} - - event-target-shim@5.0.1: {} - - events@3.3.0: {} - - execa@5.1.1: - dependencies: - cross-spawn: 7.0.3 - get-stream: 6.0.1 - human-signals: 2.1.0 - is-stream: 2.0.1 - merge-stream: 2.0.0 - npm-run-path: 4.0.1 - onetime: 5.1.2 - signal-exit: 3.0.7 - strip-final-newline: 2.0.0 - - fast-content-type-parse@1.1.0: {} - - fast-decode-uri-component@1.0.1: {} - - fast-deep-equal@3.1.3: {} - - fast-glob@3.3.2: - dependencies: - '@nodelib/fs.stat': 2.0.5 - '@nodelib/fs.walk': 1.2.8 - glob-parent: 5.1.2 - merge2: 1.4.1 - micromatch: 4.0.8 - - fast-json-stringify@5.16.1: - dependencies: - '@fastify/merge-json-schemas': 0.1.1 - ajv: 8.17.1 - ajv-formats: 3.0.1(ajv@8.17.1) - fast-deep-equal: 3.1.3 - fast-uri: 2.4.0 - json-schema-ref-resolver: 1.0.1 - rfdc: 1.4.1 - - fast-querystring@1.1.2: - dependencies: - fast-decode-uri-component: 1.0.1 - - fast-redact@3.5.0: {} - - fast-uri@2.4.0: {} - - fast-uri@3.0.1: {} - - fastify@4.28.1: - dependencies: - '@fastify/ajv-compiler': 3.6.0 - '@fastify/error': 3.4.1 - '@fastify/fast-json-stringify-compiler': 4.3.0 - abstract-logging: 2.0.1 - avvio: 8.4.0 - fast-content-type-parse: 1.1.0 - fast-json-stringify: 5.16.1 - find-my-way: 8.2.0 - light-my-request: 5.13.0 - pino: 9.4.0 - process-warning: 3.0.0 - proxy-addr: 2.0.7 - rfdc: 1.4.1 - secure-json-parse: 2.7.0 - semver: 7.6.3 - toad-cache: 3.7.0 - - fastq@1.17.1: - dependencies: - reusify: 1.0.4 - - fill-range@7.1.1: - dependencies: - to-regex-range: 5.0.1 - - find-my-way@8.2.0: - dependencies: - fast-deep-equal: 3.1.3 - fast-querystring: 1.1.2 - safe-regex2: 3.1.0 - - find-up@5.0.0: - dependencies: - locate-path: 6.0.0 - path-exists: 4.0.0 - - flat@5.0.2: {} - - foreground-child@3.3.0: - dependencies: - cross-spawn: 7.0.3 - signal-exit: 4.1.0 - - forwarded@0.2.0: {} - - fs.realpath@1.0.0: {} - - fsevents@2.3.3: - optional: true - - function-bind@1.1.2: {} - - get-caller-file@2.0.5: {} - - get-func-name@2.0.2: {} - - get-intrinsic@1.2.4: - dependencies: - es-errors: 1.3.0 - function-bind: 1.1.2 - has-proto: 1.0.3 - has-symbols: 1.0.3 - hasown: 2.0.2 - - get-stream@6.0.1: {} - - glob-parent@5.1.2: - dependencies: - is-glob: 4.0.3 - - glob@10.4.5: - dependencies: - foreground-child: 3.3.0 - jackspeak: 3.4.3 - minimatch: 9.0.5 - minipass: 7.1.2 - package-json-from-dist: 1.0.0 - path-scurry: 1.11.1 - - glob@8.1.0: - dependencies: - fs.realpath: 1.0.0 - inflight: 1.0.6 - inherits: 2.0.4 - minimatch: 5.1.6 - once: 1.4.0 - - globby@11.1.0: - dependencies: - array-union: 2.1.0 - dir-glob: 3.0.1 - fast-glob: 3.3.2 - ignore: 5.3.2 - merge2: 1.4.1 - slash: 3.0.0 - - gopd@1.0.1: - dependencies: - get-intrinsic: 1.2.4 - - graphile-config@0.0.1-beta.9: - dependencies: - '@types/interpret': 1.1.3 - '@types/node': 20.16.3 - '@types/semver': 7.5.8 - chalk: 4.1.2 - debug: 4.3.6(supports-color@8.1.1) - interpret: 3.1.1 - semver: 7.6.3 - tslib: 2.7.0 - yargs: 17.7.2 - transitivePeerDependencies: - - supports-color - - graphile-worker@0.16.6(typescript@5.5.4): - dependencies: - '@graphile/logger': 0.2.0 - '@types/debug': 4.1.12 - '@types/pg': 8.11.8 - cosmiconfig: 8.3.6(typescript@5.5.4) - graphile-config: 0.0.1-beta.9 - json5: 2.2.3 - pg: 8.12.0 - tslib: 2.7.0 - yargs: 17.7.2 - transitivePeerDependencies: - - pg-native - - supports-color - - typescript - - has-flag@3.0.0: {} - - has-flag@4.0.0: {} - - has-property-descriptors@1.0.2: - dependencies: - es-define-property: 1.0.0 - - has-proto@1.0.3: {} - - has-symbols@1.0.3: {} - - hasown@2.0.2: - dependencies: - function-bind: 1.1.2 - - he@1.2.0: {} - - html-to-text@9.0.5: - dependencies: - '@selderee/plugin-htmlparser2': 0.11.0 - deepmerge: 4.3.1 - dom-serializer: 2.0.0 - htmlparser2: 8.0.2 - selderee: 0.11.0 - - htmlparser2@8.0.2: - dependencies: - domelementtype: 2.3.0 - domhandler: 5.0.3 - domutils: 3.1.0 - entities: 4.5.0 - - htmlparser2@9.1.0: - dependencies: - domelementtype: 2.3.0 - domhandler: 5.0.3 - domutils: 3.1.0 - entities: 4.5.0 - - human-signals@2.1.0: {} - - iconv-lite@0.6.3: - dependencies: - safer-buffer: 2.1.2 - - ieee754@1.2.1: {} - - ignore@5.3.2: {} - - imapflow@1.0.164: - dependencies: - encoding-japanese: 2.2.0 - iconv-lite: 0.6.3 - libbase64: 1.3.0 - libmime: 5.3.5 - libqp: 2.1.0 - mailsplit: 5.4.0 - nodemailer: 6.9.14 - pino: 9.2.0 - socks: 2.8.3 - - import-fresh@3.3.0: - dependencies: - parent-module: 1.0.1 - resolve-from: 4.0.0 - - inflight@1.0.6: - dependencies: - once: 1.4.0 - wrappy: 1.0.2 - - inherits@2.0.4: {} - - interpret@3.1.1: {} - - ip-address@9.0.5: - dependencies: - jsbn: 1.1.0 - sprintf-js: 1.1.3 - - ipaddr.js@1.9.1: {} - - is-arrayish@0.2.1: {} - - is-binary-path@2.1.0: - dependencies: - binary-extensions: 2.3.0 - - is-extglob@2.1.1: {} - - is-fullwidth-code-point@3.0.0: {} - - is-glob@4.0.3: - dependencies: - is-extglob: 2.1.1 - - is-number@7.0.0: {} - - is-plain-obj@2.1.0: {} - - is-stream@2.0.1: {} - - is-unicode-supported@0.1.0: {} - - isexe@2.0.0: {} - - jackspeak@3.4.3: - dependencies: - '@isaacs/cliui': 8.0.2 - optionalDependencies: - '@pkgjs/parseargs': 0.11.0 - - joycon@3.1.1: {} - - js-tokens@4.0.0: {} - - js-yaml@4.1.0: - dependencies: - argparse: 2.0.1 - - jsbn@1.1.0: {} - - json-parse-even-better-errors@2.3.1: {} - - json-schema-ref-resolver@1.0.1: - dependencies: - fast-deep-equal: 3.1.3 - - json-schema-traverse@1.0.0: {} - - json5@2.2.3: {} - - leac@0.6.0: {} - - libbase64@1.2.1: {} - - libbase64@1.3.0: {} - - libmime@5.2.0: - dependencies: - encoding-japanese: 2.0.0 - iconv-lite: 0.6.3 - libbase64: 1.2.1 - libqp: 2.0.1 - - libmime@5.3.5: - dependencies: - encoding-japanese: 2.1.0 - iconv-lite: 0.6.3 - libbase64: 1.3.0 - libqp: 2.1.0 - - libqp@2.0.1: {} - - libqp@2.1.0: {} - - light-my-request@5.13.0: - dependencies: - cookie: 0.6.0 - process-warning: 3.0.0 - set-cookie-parser: 2.7.0 - - lilconfig@3.1.2: {} - - lines-and-columns@1.2.4: {} - - linkify-it@5.0.0: - dependencies: - uc.micro: 2.1.0 - - load-tsconfig@0.2.5: {} - - locate-path@6.0.0: - dependencies: - p-locate: 5.0.0 - - lodash.sortby@4.7.0: {} - - log-symbols@4.1.0: - dependencies: - chalk: 4.1.2 - is-unicode-supported: 0.1.0 - - loupe@3.1.1: - dependencies: - get-func-name: 2.0.2 - - lru-cache@10.4.3: {} - - mailparser@3.7.1: - dependencies: - encoding-japanese: 2.1.0 - he: 1.2.0 - html-to-text: 9.0.5 - iconv-lite: 0.6.3 - libmime: 5.3.5 - linkify-it: 5.0.0 - mailsplit: 5.4.0 - nodemailer: 6.9.13 - punycode.js: 2.3.1 - tlds: 1.252.0 - - mailsplit@5.4.0: - dependencies: - libbase64: 1.2.1 - libmime: 5.2.0 - libqp: 2.0.1 - - merge-stream@2.0.0: {} - - merge2@1.4.1: {} - - micromatch@4.0.8: - dependencies: - braces: 3.0.3 - picomatch: 2.3.1 - - mimic-fn@2.1.0: {} - - minimatch@5.1.6: - dependencies: - brace-expansion: 2.0.1 - - minimatch@9.0.5: - dependencies: - brace-expansion: 2.0.1 - - minipass@7.1.2: {} - - mocha@10.7.3: - dependencies: - ansi-colors: 4.1.3 - browser-stdout: 1.3.1 - chokidar: 3.6.0 - debug: 4.3.6(supports-color@8.1.1) - diff: 5.2.0 - escape-string-regexp: 4.0.0 - find-up: 5.0.0 - glob: 8.1.0 - he: 1.2.0 - js-yaml: 4.1.0 - log-symbols: 4.1.0 - minimatch: 5.1.6 - ms: 2.1.3 - serialize-javascript: 6.0.2 - strip-json-comments: 3.1.1 - supports-color: 8.1.1 - workerpool: 6.5.1 - yargs: 16.2.0 - yargs-parser: 20.2.9 - yargs-unparser: 2.0.0 - - ms@2.1.2: {} - - ms@2.1.3: {} - - mz@2.7.0: - dependencies: - any-promise: 1.3.0 - object-assign: 4.1.1 - thenify-all: 1.6.0 - - nodemailer@6.9.13: {} - - nodemailer@6.9.14: {} - - normalize-path@3.0.0: {} - - npm-run-path@4.0.1: - dependencies: - path-key: 3.1.1 - - nth-check@2.1.1: - dependencies: - boolbase: 1.0.0 - - object-assign@4.1.1: {} - - object-inspect@1.13.2: {} - - obuf@1.1.2: {} - - on-exit-leak-free@2.1.2: {} - - once@1.4.0: - dependencies: - wrappy: 1.0.2 - - onetime@5.1.2: - dependencies: - mimic-fn: 2.1.0 - - p-limit@3.1.0: - dependencies: - yocto-queue: 0.1.0 - - p-locate@5.0.0: - dependencies: - p-limit: 3.1.0 - - package-json-from-dist@1.0.0: {} - - parent-module@1.0.1: - dependencies: - callsites: 3.1.0 - - parse-json@5.2.0: - dependencies: - '@babel/code-frame': 7.24.7 - error-ex: 1.3.2 - json-parse-even-better-errors: 2.3.1 - lines-and-columns: 1.2.4 - - parse5-htmlparser2-tree-adapter@7.0.0: - dependencies: - domhandler: 5.0.3 - parse5: 7.1.2 - - parse5-parser-stream@7.1.2: - dependencies: - parse5: 7.1.2 - - parse5@7.1.2: - dependencies: - entities: 4.5.0 - - parseley@0.12.1: - dependencies: - leac: 0.6.0 - peberminta: 0.9.0 - - path-exists@4.0.0: {} - - path-key@3.1.1: {} - - path-scurry@1.11.1: - dependencies: - lru-cache: 10.4.3 - minipass: 7.1.2 - - path-type@4.0.0: {} - - pathval@2.0.0: {} - - peberminta@0.9.0: {} - - pg-cloudflare@1.1.1: - optional: true - - pg-connection-string@2.6.4: {} - - pg-int8@1.0.1: {} - - pg-numeric@1.0.2: {} - - pg-pool@3.6.2(pg@8.12.0): - dependencies: - pg: 8.12.0 - - pg-protocol@1.6.1: {} - - pg-types@2.2.0: - dependencies: - pg-int8: 1.0.1 - postgres-array: 2.0.0 - postgres-bytea: 1.0.0 - postgres-date: 1.0.7 - postgres-interval: 1.2.0 - - pg-types@4.0.2: - dependencies: - pg-int8: 1.0.1 - pg-numeric: 1.0.2 - postgres-array: 3.0.2 - postgres-bytea: 3.0.0 - postgres-date: 2.1.0 - postgres-interval: 3.0.0 - postgres-range: 1.1.4 - - pg@8.12.0: - dependencies: - pg-connection-string: 2.6.4 - pg-pool: 3.6.2(pg@8.12.0) - pg-protocol: 1.6.1 - pg-types: 2.2.0 - pgpass: 1.0.5 - optionalDependencies: - pg-cloudflare: 1.1.1 - - pgpass@1.0.5: - dependencies: - split2: 4.2.0 - - picocolors@1.1.0: {} - - picomatch@2.3.1: {} - - pino-abstract-transport@1.2.0: - dependencies: - readable-stream: 4.5.2 - split2: 4.2.0 - - pino-std-serializers@7.0.0: {} - - pino@9.2.0: - dependencies: - atomic-sleep: 1.0.0 - fast-redact: 3.5.0 - on-exit-leak-free: 2.1.2 - pino-abstract-transport: 1.2.0 - pino-std-serializers: 7.0.0 - process-warning: 3.0.0 - quick-format-unescaped: 4.0.4 - real-require: 0.2.0 - safe-stable-stringify: 2.5.0 - sonic-boom: 4.1.0 - thread-stream: 3.1.0 - - pino@9.4.0: - dependencies: - atomic-sleep: 1.0.0 - fast-redact: 3.5.0 - on-exit-leak-free: 2.1.2 - pino-abstract-transport: 1.2.0 - pino-std-serializers: 7.0.0 - process-warning: 4.0.0 - quick-format-unescaped: 4.0.4 - real-require: 0.2.0 - safe-stable-stringify: 2.5.0 - sonic-boom: 4.1.0 - thread-stream: 3.1.0 - - pirates@4.0.6: {} - - postcss-load-config@6.0.1: - dependencies: - lilconfig: 3.1.2 - - postgres-array@2.0.0: {} - - postgres-array@3.0.2: {} - - postgres-bytea@1.0.0: {} - - postgres-bytea@3.0.0: - dependencies: - obuf: 1.1.2 - - postgres-date@1.0.7: {} - - postgres-date@2.1.0: {} - - postgres-interval@1.2.0: - dependencies: - xtend: 4.0.2 - - postgres-interval@3.0.0: {} - - postgres-range@1.1.4: {} - - process-warning@3.0.0: {} - - process-warning@4.0.0: {} - - process@0.11.10: {} - - proxy-addr@2.0.7: - dependencies: - forwarded: 0.2.0 - ipaddr.js: 1.9.1 - - punycode.js@2.3.1: {} - - punycode@2.3.1: {} - - qs@6.13.0: - dependencies: - side-channel: 1.0.6 - - queue-microtask@1.2.3: {} - - quick-format-unescaped@4.0.4: {} - - randombytes@2.1.0: - dependencies: - safe-buffer: 5.2.1 - - readable-stream@4.5.2: - dependencies: - abort-controller: 3.0.0 - buffer: 6.0.3 - events: 3.3.0 - process: 0.11.10 - string_decoder: 1.3.0 - - readdirp@3.6.0: - dependencies: - picomatch: 2.3.1 - - real-require@0.2.0: {} - - require-directory@2.1.1: {} - - require-from-string@2.0.2: {} - - resolve-from@4.0.0: {} - - resolve-from@5.0.0: {} - - ret@0.4.3: {} - - reusify@1.0.4: {} - - rfdc@1.4.1: {} - - rollup@4.21.2: - dependencies: - '@types/estree': 1.0.5 - optionalDependencies: - '@rollup/rollup-android-arm-eabi': 4.21.2 - '@rollup/rollup-android-arm64': 4.21.2 - '@rollup/rollup-darwin-arm64': 4.21.2 - '@rollup/rollup-darwin-x64': 4.21.2 - '@rollup/rollup-linux-arm-gnueabihf': 4.21.2 - '@rollup/rollup-linux-arm-musleabihf': 4.21.2 - '@rollup/rollup-linux-arm64-gnu': 4.21.2 - '@rollup/rollup-linux-arm64-musl': 4.21.2 - '@rollup/rollup-linux-powerpc64le-gnu': 4.21.2 - '@rollup/rollup-linux-riscv64-gnu': 4.21.2 - '@rollup/rollup-linux-s390x-gnu': 4.21.2 - '@rollup/rollup-linux-x64-gnu': 4.21.2 - '@rollup/rollup-linux-x64-musl': 4.21.2 - '@rollup/rollup-win32-arm64-msvc': 4.21.2 - '@rollup/rollup-win32-ia32-msvc': 4.21.2 - '@rollup/rollup-win32-x64-msvc': 4.21.2 - fsevents: 2.3.3 - - run-parallel@1.2.0: - dependencies: - queue-microtask: 1.2.3 - - safe-buffer@5.2.1: {} - - safe-regex2@3.1.0: - dependencies: - ret: 0.4.3 - - safe-stable-stringify@2.5.0: {} - - safer-buffer@2.1.2: {} - - secure-json-parse@2.7.0: {} - - selderee@0.11.0: - dependencies: - parseley: 0.12.1 - - semver@7.6.3: {} - - serialize-javascript@6.0.2: - dependencies: - randombytes: 2.1.0 - - set-cookie-parser@2.7.0: {} - - set-function-length@1.2.2: - dependencies: - define-data-property: 1.1.4 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - gopd: 1.0.1 - has-property-descriptors: 1.0.2 - - shebang-command@2.0.0: - dependencies: - shebang-regex: 3.0.0 - - shebang-regex@3.0.0: {} - - side-channel@1.0.6: - dependencies: - call-bind: 1.0.7 - es-errors: 1.3.0 - get-intrinsic: 1.2.4 - object-inspect: 1.13.2 - - signal-exit@3.0.7: {} - - signal-exit@4.1.0: {} - - slash@3.0.0: {} - - smart-buffer@4.2.0: {} - - socks@2.8.3: - dependencies: - ip-address: 9.0.5 - smart-buffer: 4.2.0 - - sonic-boom@4.1.0: - dependencies: - atomic-sleep: 1.0.0 - - source-map@0.8.0-beta.0: - dependencies: - whatwg-url: 7.1.0 - - split2@4.2.0: {} - - sprintf-js@1.1.3: {} - - string-width@4.2.3: - dependencies: - emoji-regex: 8.0.0 - is-fullwidth-code-point: 3.0.0 - strip-ansi: 6.0.1 - - string-width@5.1.2: - dependencies: - eastasianwidth: 0.2.0 - emoji-regex: 9.2.2 - strip-ansi: 7.1.0 - - string_decoder@1.3.0: - dependencies: - safe-buffer: 5.2.1 - - strip-ansi@6.0.1: - dependencies: - ansi-regex: 5.0.1 - - strip-ansi@7.1.0: - dependencies: - ansi-regex: 6.0.1 - - strip-final-newline@2.0.0: {} - - strip-json-comments@3.1.1: {} - - sucrase@3.35.0: - dependencies: - '@jridgewell/gen-mapping': 0.3.5 - commander: 4.1.1 - glob: 10.4.5 - lines-and-columns: 1.2.4 - mz: 2.7.0 - pirates: 4.0.6 - ts-interface-checker: 0.1.13 - - supports-color@5.5.0: - dependencies: - has-flag: 3.0.0 - - supports-color@7.2.0: - dependencies: - has-flag: 4.0.0 - - supports-color@8.1.1: - dependencies: - has-flag: 4.0.0 - - thenify-all@1.6.0: - dependencies: - thenify: 3.3.1 - - thenify@3.3.1: - dependencies: - any-promise: 1.3.0 - - thread-stream@3.1.0: - dependencies: - real-require: 0.2.0 - - tlds@1.252.0: {} - - to-regex-range@5.0.1: - dependencies: - is-number: 7.0.0 - - toad-cache@3.7.0: {} - - tr46@1.0.1: - dependencies: - punycode: 2.3.1 - - tree-kill@1.2.2: {} - - ts-interface-checker@0.1.13: {} - - tslib@2.7.0: {} - - tsup@8.2.4(typescript@5.5.4): - dependencies: - bundle-require: 5.0.0(esbuild@0.23.1) - cac: 6.7.14 - chokidar: 3.6.0 - consola: 3.2.3 - debug: 4.3.6(supports-color@8.1.1) - esbuild: 0.23.1 - execa: 5.1.1 - globby: 11.1.0 - joycon: 3.1.1 - picocolors: 1.1.0 - postcss-load-config: 6.0.1 - resolve-from: 5.0.0 - rollup: 4.21.2 - source-map: 0.8.0-beta.0 - sucrase: 3.35.0 - tree-kill: 1.2.2 - optionalDependencies: - typescript: 5.5.4 - transitivePeerDependencies: - - jiti - - supports-color - - tsx - - yaml - - typescript@5.5.4: {} - - uc.micro@2.1.0: {} - - undici-types@6.19.8: {} - - undici@6.19.8: {} - - webidl-conversions@4.0.2: {} - - whatwg-encoding@3.1.1: - dependencies: - iconv-lite: 0.6.3 - - whatwg-mimetype@4.0.0: {} - - whatwg-url@7.1.0: - dependencies: - lodash.sortby: 4.7.0 - tr46: 1.0.1 - webidl-conversions: 4.0.2 - - which@2.0.2: - dependencies: - isexe: 2.0.0 - - workerpool@6.5.1: {} - - wrap-ansi@7.0.0: - dependencies: - ansi-styles: 4.3.0 - string-width: 4.2.3 - strip-ansi: 6.0.1 - - wrap-ansi@8.1.0: - dependencies: - ansi-styles: 6.2.1 - string-width: 5.1.2 - strip-ansi: 7.1.0 - - wrappy@1.0.2: {} - - xtend@4.0.2: {} - - y18n@5.0.8: {} - - yargs-parser@20.2.9: {} - - yargs-parser@21.1.1: {} - - yargs-unparser@2.0.0: - dependencies: - camelcase: 6.3.0 - decamelize: 4.0.0 - flat: 5.0.2 - is-plain-obj: 2.1.0 - - yargs@16.2.0: - dependencies: - cliui: 7.0.4 - escalade: 3.2.0 - get-caller-file: 2.0.5 - require-directory: 2.1.1 - string-width: 4.2.3 - y18n: 5.0.8 - yargs-parser: 20.2.9 - - yargs@17.7.2: - dependencies: - cliui: 8.0.1 - escalade: 3.2.0 - get-caller-file: 2.0.5 - require-directory: 2.1.1 - string-width: 4.2.3 - y18n: 5.0.8 - yargs-parser: 21.1.1 - - yocto-queue@0.1.0: {} diff --git a/services/mailbox/src/imap.ts b/services/mailbox/src/imap.ts deleted file mode 100644 index 7c376a4..0000000 --- a/services/mailbox/src/imap.ts +++ /dev/null @@ -1,131 +0,0 @@ - -import { ImapFlow } from 'imapflow'; -import EventEmitter from 'node:events'; -import 'dotenv/config'; - - - -if (!process.env.IMAP_SERVER) throw new Error('IMAP_SERVER is missing from env'); -if (!process.env.IMAP_PORT) throw new Error('IMAP_PORT is missing from env'); -if (!process.env.IMAP_USERNAME) throw new Error('IMAP_USERNAME is missing from env'); -if (!process.env.IMAP_PASSWORD) throw new Error('IMAP_PASSWORD is missing from env'); - -// https://stackoverflow.com/a/49428486/1004931 -function streamToString(stream: any) { - const chunks: any = []; - return new Promise((resolve, reject) => { - stream.on('data', (chunk: any) => chunks.push(Buffer.from(chunk))); - stream.on('error', (err: any) => reject(err)); - stream.on('end', () => resolve(Buffer.concat(chunks).toString('utf8'))); - }) -} - - -export class Email extends EventEmitter { - private client?: ImapFlow | null = null - - constructor() { - super() - } - - async archiveMessage(uid: any) { - if (!this.client) throw new Error('archiveMessage was called before this.client was instantiated.'); - await this.client.messageDelete(uid, { uid: true }) - } - - async connect() { - this.client = new ImapFlow({ - host: process.env.IMAP_SERVER!, - port: parseInt(process.env.IMAP_PORT!), - secure: true, - auth: { - user: process.env.IMAP_USERNAME!, - pass: process.env.IMAP_PASSWORD! - }, - logger: false - }); - - this.registerEventListeners() - await this.client.connect() - const stat = await this.getStatus() - if (stat?.messages && stat.messages > 0) { - await this.emitAllMessages() - } - } - - async reconnect() { - console.log(' RECONNECTING...') - delete this.client - await this.connect() - } - - async getStatus() { - if (!this.client) throw new Error('getStatus was called before this.client was instantiated.'); - let lock = await this.client.getMailboxLock('INBOX'); - let status; - try { - status = await this.client.status('INBOX', { messages: true }); - } finally { - lock.release() - } - return status - } - - async loadMessage(uid: any) { - if (!this.client) throw new Error('loadMessage was called before this.client was instantiated.'); - console.log(` 💾 loading message uid=${uid}`) - let lock = await this.client.getMailboxLock('INBOX') - let dl, body - try { - dl = await this.client.download(uid, undefined, { uid: true }) - body = await streamToString(dl.content) - } finally { - lock.release() - } - return body - } - - async emitAllMessages() { - if (!this.client) throw new Error('emitAllMessages was called before this.client was instantiated.'); - console.log('emitAllMessages is running') - let lock = await this.client.getMailboxLock('INBOX'); - try { - for await (let message of this.client.fetch('1:*', { envelope: true })) { - // it is tempting to call this.client.download here, but that is not possible while the mailbox is locked. - // client.download must be called outside of this lock - // commenting these out because it lags Tilt - // console.log('here is a message') - // console.log(JSON.stringify(message, null, 2)) - this.emit('message', message) - } - } finally { - lock.release(); - } - } - - registerEventListeners() { - if (!this.client) throw new Error('registerEventListeners was called before this.client was instantiated.'); - console.log(` > REGISTERING EVENT LISTENERS <`) - this.client.once('end', () => this.reconnect()) - this.client.on('exists', (evt: any) => { - console.log(`exists event! count=${evt.count} prevCount=${evt.prevCount}`) - // console.log(evt) - if (evt.path === 'INBOX') { - this.emitAllMessages() - } - }) - } - -} - - - - - - -// // Select and lock a mailbox. Throws if mailbox does not exist -// console.log('get lock') - - -// // log out and close connection -// // await client.logout(); \ No newline at end of file diff --git a/services/mailbox/src/index.ts b/services/mailbox/src/index.ts deleted file mode 100644 index e2e123c..0000000 --- a/services/mailbox/src/index.ts +++ /dev/null @@ -1,70 +0,0 @@ -'use strict' - -/** - * watches an e-mail inbox for going live notifications - */ - -import { checkEmail } from './parsers.js' -import { Email } from './imap.js' -import { makeWorkerUtils, WorkerUtils } from 'graphile-worker' -import { type NotificationData } from '@futureporn/types' -import { type FetchMessageObject } from 'imapflow' - - -if (!process.env.DATABASE_URL) throw new Error('DATABASE_URL is undefined in env'); -if (!process.env.PORT) throw new Error('PORT is undefined in env'); -const connectionString = process.env.DATABASE_URL -console.log(`process.env.DATABASE_URL=${connectionString}`) - - -async function handleMessage({ workerUtils, email, msg }: { workerUtils: WorkerUtils, email: Email, msg: FetchMessageObject }) { - try { - console.log(' ✏️ loading message') - const body = await email.loadMessage(msg.uid) as string - - console.log(' ✏️ checking e-mail') - const { isMatch, url, platform, channel, displayName, date, userId, avatar }: NotificationData = (await checkEmail(body) ) - - if (isMatch) { - console.log(' ✏️✏️ adding process_notif_email job to queue') - workerUtils.addJob('process_notif_email', { isMatch, url, platform, channel, displayName, date, userId, avatar }, { maxAttempts: 3 }) - } - - console.log(' ✏️ archiving e-mail') - await email.archiveMessage(msg.uid) - - } catch (e) { - console.error(`An error was encountered while handling the following e-mail message.\n${JSON.stringify(msg, null, 2)}\nError as follows.`) - console.error(e) - } -} - - - - -async function main() { - - - const workerUtils = await makeWorkerUtils({ - connectionString - }) - - // demonstrate that we are connected @todo remove this - workerUtils.addJob('hello', { name: 'worker' }, { maxAttempts: 3 }) - workerUtils.addJob('identify_image_color', { url: 'https://futureporn-b2.b-cdn.net/ti8ht9bgwj6k783j7hglfg8j_projektmelody-chaturbate-2024-07-18.png' }, { maxAttempts: 3 }) - - // connect to IMAP inbox and wait for new e-mails - const email = new Email() - email.once('message', (msg: FetchMessageObject) => handleMessage({ workerUtils, email, msg })) - console.log('mailbox is starting...') - await email.connect() - -} - -main().catch((err) => { - console.error(err); - process.exit(1); -}); - - - diff --git a/services/mailbox/src/parsers.spec.ts b/services/mailbox/src/parsers.spec.ts deleted file mode 100644 index faffd58..0000000 --- a/services/mailbox/src/parsers.spec.ts +++ /dev/null @@ -1,34 +0,0 @@ -'use strict' - -import { expect } from 'chai' -import { checkEmail } from './parsers.js' -import fs from 'node:fs/promises' -import path from 'node:path' -const __dirname = import.meta.dirname; - -describe('parsers', function () { - describe('checkEmail', function () { - it('should detect fansly e-mails and return channel data', async function () { - const mailBody = await fs.readFile(path.join(__dirname, './fixtures/fansly.fixture.txt'), { encoding: 'utf8' }) - const { isMatch, channel, platform, url, date, userId, avatar } = await checkEmail(mailBody) - expect(isMatch).to.equal(true, 'a Fansly heuristic was not found') - expect(platform).to.equal('fansly') - expect(channel).to.equal('SkiaObsidian') - expect(url).to.equal('https://fansly.com/SkiaObsidian') - expect(date).to.equal('2024-05-05T03:04:33.000Z') - expect(userId).to.equal('555722198917066752') - expect(avatar).to.equal('https://api.fansly.com/api/v1/account/555722198917066752/avatar') - }) - it('should detect cb e-mails and return channel data', async function () { - const mailBody = await fs.readFile(path.join(__dirname, './fixtures/chaturbate.fixture.txt'), { encoding: 'utf8' }) - const { isMatch, channel, platform, url, date, userId, avatar } = await checkEmail(mailBody) - expect(isMatch).to.equal(true, 'a CB heuristic was not found') - expect(platform).to.equal('chaturbate') - expect(channel).to.equal('skyeanette') - expect(url).to.equal('https://chaturbate.com/skyeanette') - expect(date).to.equal('2023-07-24T01:08:28.000Z') - expect(userId).to.equal(null) // this info is not in the CB e-mail - expect(avatar).to.equal(null) - }) - }) -}) \ No newline at end of file diff --git a/services/mailbox/src/parsers.ts b/services/mailbox/src/parsers.ts deleted file mode 100644 index 1eb0380..0000000 --- a/services/mailbox/src/parsers.ts +++ /dev/null @@ -1,99 +0,0 @@ - -import { simpleParser } from 'mailparser'; -import { load } from 'cheerio' -import { type NotificationData } from '@futureporn/types' - -const definitions = [ - { - platform: 'chaturbate', - selectors: { - channel: 'td[id*="onlinemessage"] a:nth-child(1)' - }, - from: 'follownotify@bk.chaturbate.com', - template: 'https://chaturbate.com/:channel' - }, - { - platform: 'fansly', - selectors: { - channel: ($: any) => $("a[href*='/live/']").attr('href').toString().split('/').at(-1), - displayName: 'div[class*="message-col"] div:nth-child(5)', - userId: ($: any) => $("img[src*='/api/v1/account/']").attr('src').toString().split('/').at(-2), - avatar: ($: any) => $("img[src*='/api/v1/account/']").attr('src').toString() - }, - from: 'no-reply@fansly.com', - template: 'https://fansly.com/:channel', - regex: /https:\/\/fansly.com\/live\/([a-zA-Z0-9_]+)/ - } -] - -function render(template: string, values: any) { - // console.log(`values=${values}`) - // console.log(values) - return template.replace(/:([a-zA-Z0-9_]+)/g, (match, key) => { - // Replace :channel with the corresponding property from the values object - return values[key] || match; - }); -} - - -/** - * checkEmail - * - * Check an e-mail for go-live notification content. - * - * { isMatch, url, platform, channel, displayName, date } - * - * @param {String} body raw mail body - * @returns {Object} result - * @returns {Boolean} result.isMatch true if e-mail contains a go-live notification - * @returns {String} result.url example: https://fansly.com/projektmelody - * @returns {String} result.platform example: fansly - * @returns {String} result.channel example: projektmelody - * @returns {String} result.displayName example: ProjektMelody - * @returns {String} result.date example: 2024-05-31T01:02:00.000Z - * @returns {String|null} result.userId Varies by platform. Some platforms don't have the userId in the e-mail, so it's null. - * fansly example: '555722198917066752' - */ -export async function checkEmail (body: string): Promise { - - const mail = await simpleParser(body) - if (!mail) throw new Error('mail object was unable to be gathered from simpleParser'); - if (!mail.from) throw new Error('mail.from was empty'); - if (!mail.from.value) throw new Error('mail.from.value was empty'); - if (!mail.from.value[0]) throw new Error('mail.from.value[0] was empty'); - if (!mail.date) throw new Error('mail.date was empty'); - if (!mail?.html) { - console.log(`mail.html was not truthy. This means the e-mail was text-only mode. This also means the e-mail is not a go-live notification.`); - return { isMatch: false } - } - - let res: Record = {} - let def: any = definitions.find((def) => def.from === mail.from!.value[0]!.address) - if (!def) return { isMatch: false, channel: undefined, platform: undefined, url: undefined }; - res.isMatch = true - - // Step 0, get values from e-mail metadata - res.platform = def.platform - res.date = new Date(mail.date).toISOString() - - // Step 1, get values using CSS selectors - const $ = load(mail.html) - for (const s in def.selectors) { - res[s] = (def.selectors[s] instanceof Object) ? def.selectors[s]($) : $(def.selectors[s]).text() - } - - // console.log(`res.url=${res.url} res.userId=${res.userId}`) - - // Step 2, get values using regex & templates - res.channel = (() => { - if (res.channel) return res.channel; - if (def.regex && res.url) return def.regex.exec(res.url).at(1); - })() - - res.userId = res.userId || undefined - res.avatar = res.avatar || undefined - - res.url = res.url || render(def.template, { channel: res.channel }) - - return res as NotificationData -} \ No newline at end of file diff --git a/services/mailbox/tsconfig.json b/services/mailbox/tsconfig.json deleted file mode 100644 index f09143d..0000000 --- a/services/mailbox/tsconfig.json +++ /dev/null @@ -1,29 +0,0 @@ -{ - "compilerOptions": { - // Base Options recommended for all projects - "esModuleInterop": true, - "skipLibCheck": true, - "target": "es2022", - "allowJs": true, - "resolveJsonModule": true, - "moduleDetection": "force", - "isolatedModules": true, - // Enable strict type checking so you can catch bugs early - "strict": true, - "noUncheckedIndexedAccess": true, - "noImplicitOverride": true, - // Transpile our TypeScript code to JavaScript - "module": "NodeNext", - "outDir": "dist", - "lib": [ - "es2022" - ] - }, - // Include the necessary files for your project - "include": [ - "**/*.ts" - ], - "exclude": [ - "node_modules" - ] -} \ No newline at end of file diff --git a/services/mailbox/tsup.config.js b/services/mailbox/tsup.config.js deleted file mode 100644 index 900d0f6..0000000 --- a/services/mailbox/tsup.config.js +++ /dev/null @@ -1,38 +0,0 @@ -import { defineConfig } from "tsup" -import { exec } from 'node:child_process' - -export default defineConfig({ - entry: [ - "src/index.ts", - "src/tasks/**.ts" - ], - /** - * lots of compatibility issues outputting to ESM. So we output to CJS to get things working. - */ - format: ["cjs"], - target: "es2022", - sourcemap: true, - /** - * The common package is using the internal packages approach, so it needs to - * be transpiled / bundled together with the deployed code. - */ - noExternal: [ - "@futureporn/utils", - "@futureporn/scout", - "@futureporn/storage", - "@futureporn/types", - ], - /** - * Do not use tsup for generating d.ts files because it can not generate type - * the definition maps required for go-to-definition to work in our IDE. We - * use tsc for that. - */ - splitting: false, - treeshake: true, - clean: true, - outDir: 'dist', // Where you want your compiled files to live - shims: true, - onSuccess: async () => { - exec('tsc --emitDeclarationOnly'); - } -}); \ No newline at end of file diff --git a/services/migrations-data/README.md b/services/migrations-data/README.md deleted file mode 100644 index 094ae89..0000000 --- a/services/migrations-data/README.md +++ /dev/null @@ -1,27 +0,0 @@ -# @futureporn/migrations-data - -Here we handle data migrations for the postgrest database. - -@see https://github.com/zakpatterson/postgres-schema-migrations - -Reminder: only write migrations that affect data. (don't write migrations that affect schema) - -## K.I.S.S. - -Keep It Stupidly Simple. - -We are keeping this module as simple as possible. This means pure JS (no typescript!) - - -## troubleshooting - -If you see the following error, graphile_worker likely hasn't had a chance to create it's functions. Make sure that a graphile_worker is running, so it can automatically create the necessary functions. - -```json -{ - "code": "42883", - "details": null, - "hint": "No function matches the given name and argument types. You might need to add explicit type casts.", - "message": "function graphile_worker.add_job(text, json, max_attempts => integer) does not exist" -} -``` \ No newline at end of file diff --git a/services/migrations-data/index.js b/services/migrations-data/index.js deleted file mode 100644 index 2b8ac31..0000000 --- a/services/migrations-data/index.js +++ /dev/null @@ -1,68 +0,0 @@ -import pg from 'pg' -import { migrate } from 'postgres-schema-migrations'; -import path, { dirname } from 'node:path'; -import { fileURLToPath } from 'url'; -import 'dotenv/config'; - -const { Client } = pg -const __dirname = dirname(fileURLToPath(import.meta.url)); - -if (!process.env.DATABASE_PASSWORD) throw new Error('DATABASE_PASSWORD is missing in env'); - -/* - * Here we set up a Foreign Data Wrapper which connects us to the old strapi database. - * From this Strapi db futureporn-old, we migrate data to futureporn database. - */ - -async function setupForeignDataWrapper(client) { - - - - // Run the SQL commands - const sql = ` - BEGIN; - - CREATE EXTENSION IF NOT EXISTS postgres_fdw; - - CREATE SERVER IF NOT EXISTS futureporn_old - FOREIGN DATA WRAPPER postgres_fdw - OPTIONS (dbname 'futureporn_old'); - - CREATE USER MAPPING IF NOT EXISTS FOR postgres - SERVER futureporn_old - OPTIONS (password_required 'true', password '${process.env.DATABASE_PASSWORD}'); - - COMMIT; - `; - - await client.query(sql); - console.log('Foreign Data Wrapper setup completed successfully.'); - -} - -async function main() { - const dbConfig = { - database: "futureporn", - user: "postgres", - password: process.env.DATABASE_PASSWORD, - host: 'postgresql-primary.futureporn.svc.cluster.local', - port: 5432, - } - const client = new Client(dbConfig) - await client.connect() - const migrateConfig = { - client, - ensureDatabaseExists: false, - defaultDatabase: 'postgres' - } - - try { - await setupForeignDataWrapper(client) - await migrate(migrateConfig, path.join(__dirname, "./migrations/"), { logger: console.log, schema: 'migrations_data' }) - } finally { - await client.end() - } -} - - -await main() \ No newline at end of file diff --git a/services/migrations-data/migrations/0001_from-strapi-to-postgrest-mk2.sql b/services/migrations-data/migrations/0001_from-strapi-to-postgrest-mk2.sql deleted file mode 100644 index af96bee..0000000 --- a/services/migrations-data/migrations/0001_from-strapi-to-postgrest-mk2.sql +++ /dev/null @@ -1,980 +0,0 @@ --- 2024-10-25 - -/** - * - * - * - * - * - * - * - * - * - * - * FOREIGN TABLES CREATION - * - * - * - * - [x] external_b2_files - * - [x] external_mux_assets - * - [x] external_streams - * - [x] external_streams_vtuber_links - * - [x] external_tag_vod_relations - * - [x] external_tag_vod_relations_tag_links - * - [x] external_tag_vod_relations_vod_links - * - [x] external_tags - * - [x] external_tags_toy_links - * - [x] external_tags_vods_links - * - [x] external_timestamps - * - [x] external_timestamps_tag_links - * - [x] external_timestamps_vod_links - * - [x] external_toys - * - [x] external_toys_link_tag_links - * - [x] external_vods - * - [x] external_vods_mux_asset_links - * - [x] external_vods_stream_links - * - [x] external_vods_video_src_b_2_links - * - [x] external_vods_vtuber_links - * - [x] external_vods_thumbnail_links - * - [x] external_vods_uploader_links - * - [x] external_vtubers - * - [x] external_vtubers_toy_links - * - [x] external_vtubers_toys_links - * - * - * - */ - -SET search_path TO 'public'; - -CREATE FOREIGN TABLE external_b2_files -( - id INT, - url CHARACTER VARYING(255), - key CHARACTER VARYING(255), - upload_id CHARACTER VARYING(255), - created_at TIMESTAMP(6) WITHOUT TIME ZONE, - updated_at TIMESTAMP(6) WITHOUT TIME ZONE, - created_by_id INT, - updated_by_id INT, - cdn_url CHARACTER VARYING(255) -) -SERVER futureporn_old -OPTIONS (table_name 'b2_files'); - - - -CREATE FOREIGN TABLE external_mux_assets -( - id INT, - playback_id CHARACTER VARYING(255), - asset_id CHARACTER VARYING(255), - created_at TIMESTAMP(6) WITHOUT TIME ZONE, - updated_at TIMESTAMP(6) WITHOUT TIME ZONE, - created_by_id INT, - updated_by_id INT -) -SERVER futureporn_old -OPTIONS (table_name 'mux_assets'); - - - -CREATE FOREIGN TABLE external_streams -( - id INT, - date_str CHARACTER VARYING(255), - date TIMESTAMP WITHOUT TIME ZONE, - created_by_id INT, - updated_by_id INT, - created_at TIMESTAMP WITHOUT TIME ZONE, - updated_at TIMESTAMP WITHOUT TIME ZONE, - cuid CHARACTER VARYING(255), - date_2 CHARACTER VARYING(255), - archive_status CHARACTER VARYING(255), - is_chaturbate_stream BOOLEAN, - is_fansly_stream BOOLEAN -) -SERVER futureporn_old -OPTIONS (table_name 'streams'); - - - -CREATE FOREIGN TABLE external_streams_vtuber_links -( - id INT, - stream_id INT, - vtuber_id INT, - stream_order DOUBLE PRECISION -) -SERVER futureporn_old -OPTIONS (table_name 'streams_vtuber_links'); - - - -CREATE FOREIGN TABLE external_tag_vod_relations -( - id INT, - votes INT, - creator_id INT, - created_at TIMESTAMP(6) WITHOUT TIME ZONE, - updated_at TIMESTAMP(6) WITHOUT TIME ZONE, - created_by_id INT, - updated_by_id INT -) -SERVER futureporn_old -OPTIONS (table_name 'tag_vod_relations'); - - - -CREATE FOREIGN TABLE external_tag_vod_relations_tag_links -( - id INT, - tag_vod_relation_id INT, - tag_id INT -) -SERVER futureporn_old -OPTIONS (table_name 'tag_vod_relations_tag_links'); - - - -CREATE FOREIGN TABLE external_tag_vod_relations_vod_links -( - id INT, - tag_vod_relation_id INT, - vod_id INT, - tag_vod_relation_order DOUBLE PRECISION -) -SERVER futureporn_old -OPTIONS (table_name 'tag_vod_relations_vod_links'); - - - -CREATE FOREIGN TABLE external_tags -( - id INT, - name CHARACTER VARYING(255), - created_at TIMESTAMP WITHOUT TIME ZONE, - updated_at TIMESTAMP WITHOUT TIME ZONE, - published_at TIMESTAMP WITHOUT TIME ZONE, - created_by_id INT, - updated_by_id INT -) -SERVER futureporn_old -OPTIONS (table_name 'tags'); - - -CREATE FOREIGN TABLE external_tags_toy_links -( - id INT, - tag_id INT, - toy_id INT, - tag_order DOUBLE PRECISION -) -SERVER futureporn_old -OPTIONS (table_name 'tags_toy_links'); - - - -CREATE FOREIGN TABLE external_tags_vods_links -( - id INT, - tag_id INT, - vod_id INT, - tag_order DOUBLE PRECISION, - vod_order DOUBLE PRECISION -) -SERVER futureporn_old -OPTIONS (table_name 'tags_vods_links'); - - - -CREATE FOREIGN TABLE external_timestamps -( - id INT, - time INT, - creator_id INT, - created_at TIMESTAMP(6) WITHOUT TIME ZONE, - updated_at TIMESTAMP(6) WITHOUT TIME ZONE, - created_by_id INT, - updated_by_id INT -) -SERVER futureporn_old -OPTIONS (table_name 'timestamps'); - - - -CREATE FOREIGN TABLE external_timestamps_tag_links -( - id INT, - timestamp_id INT, - tag_id INT -) -SERVER futureporn_old -OPTIONS (table_name 'timestamps_tag_links'); - - - -CREATE FOREIGN TABLE external_timestamps_vod_links -( - id INT, - timestamp_id INT, - vod_id INT, - timestamp_order DOUBLE PRECISION -) -SERVER futureporn_old -OPTIONS (table_name 'timestamps_vod_links'); - - -CREATE FOREIGN TABLE external_toys -( - id INT, - make CHARACTER VARYING(255), - model CHARACTER VARYING(255), - created_at TIMESTAMP WITHOUT TIME ZONE, - updated_at TIMESTAMP WITHOUT TIME ZONE, - created_by_id INT, - updated_by_id INT, - image_2 CHARACTER VARYING(255) -) -SERVER futureporn_old -OPTIONS (table_name 'toys'); - - - -CREATE FOREIGN TABLE external_toys_link_tag_links -( - id INT, - toy_id INT, - tag_id INT -) -SERVER futureporn_old -OPTIONS (table_name 'toys_link_tag_links'); - - - - - - -CREATE FOREIGN TABLE external_vods -( - id integer, - created_at timestamp without time zone, - updated_at timestamp without time zone, - published_at date, - title text, - date date NOT NULL, - date_2 timestamp without time zone, - mux_asset INT, - thumbnail INT, - vtuber INT, - ipfs_cid text, - video_src_hash CHARACTER VARYING(255), - s3_file INT, - torrent text, - announce_title text, - announce_url text, - note text, - url text, - discord_message_id text, - status text COLLATE pg_catalog."default" DEFAULT 'pending_recording'::text, - recording_id INT, - stream_id INT -) -SERVER futureporn_old -OPTIONS (table_name 'vods'); - - - -CREATE FOREIGN TABLE external_vods_mux_asset_links -( - id INT, - vod_id INT, - mux_asset_id INT -) -SERVER futureporn_old -OPTIONS (table_name 'vods_mux_asset_links'); - - - - - -CREATE FOREIGN TABLE external_vods_stream_links -( - id INT, - vod_id INT, - stream_id INT, - vod_order DOUBLE PRECISION -) -SERVER futureporn_old -OPTIONS (table_name 'vods_stream_links'); - - -CREATE FOREIGN TABLE external_vods_vtuber_links -( - id INT, - vod_id INT, - vtuber_id INT, - vod_order DOUBLE PRECISION -) -SERVER futureporn_old -OPTIONS (table_name 'vods_vtuber_links'); - - - -CREATE FOREIGN TABLE external_vods_thumbnail_links -( - id INT, - vod_id INT, - b_2_file_id INT -) -SERVER futureporn_old -OPTIONS (table_name 'vods_thumbnail_links'); - - -CREATE FOREIGN TABLE external_vods_uploader_links -( - id INT, - vod_id INT, - user_id INT -) -SERVER futureporn_old -OPTIONS (table_name 'vods_uploader_links'); - - - -CREATE FOREIGN TABLE external_vods_video_src_b_2_links -( - id INT, - vod_id INT, - b_2_file_id INT -) -SERVER futureporn_old -OPTIONS (table_name 'vods_video_src_b_2_links'); - - - -CREATE FOREIGN TABLE external_vtubers -( - id INT, - chaturbate CHARACTER VARYING(255), - twitter CHARACTER VARYING(255), - patreon CHARACTER VARYING(255), - twitch CHARACTER VARYING(255), - tiktok CHARACTER VARYING(255), - onlyfans CHARACTER VARYING(255), - youtube CHARACTER VARYING(255), - linktree CHARACTER VARYING(255), - carrd CHARACTER VARYING(255), - fansly CHARACTER VARYING(255), - pornhub CHARACTER VARYING(255), - discord CHARACTER VARYING(255), - reddit CHARACTER VARYING(255), - throne CHARACTER VARYING(255), - instagram CHARACTER VARYING(255), - facebook CHARACTER VARYING(255), - merch CHARACTER VARYING(255), - slug CHARACTER VARYING(255), - image CHARACTER VARYING(255), - description_1 TEXT, - description_2 TEXT, - created_at TIMESTAMP(6) WITHOUT TIME ZONE, - updated_at TIMESTAMP(6) WITHOUT TIME ZONE, - published_at TIMESTAMP(6) WITHOUT TIME ZONE, - created_by_id INT, - updated_by_id INT, - theme_color CHARACTER VARYING(255), - display_name CHARACTER VARYING(255), - image_blur CHARACTER VARYING(255) -) -SERVER futureporn_old -OPTIONS (table_name 'vtubers'); - - -CREATE FOREIGN TABLE external_vtubers_toy_links -( - id INT, - vtuber_id INT, - toy_id INT, - vtuber_order DOUBLE PRECISION -) -SERVER futureporn_old -OPTIONS (table_name 'vtubers_toy_links'); - - - -CREATE FOREIGN TABLE external_vtubers_toys_links -( - id INT, - vtuber_id INT, - toy_id INT, - toy_order DOUBLE PRECISION -) -SERVER futureporn_old -OPTIONS (table_name 'vtubers_toys_links'); - - - -/** - * - * - * - * - * - * - * - * Migrations - * - * - * - * - * The tables we need to copy are as follows. - * Note: this is not alphabetical on purpose, because some tables depend on others. - * - * - [x] vtubers - * - [x] tags - * - [x] b2_files - * - [x] mux_assets - * - [x] streams - * - [x] streams_vtuber_links - * - [x] vods - * - [x] toys - * - [x] tag_vod_relations - * - [x] tag_vod_relations_tag_links - * - [x] tag_vod_relations_vod_links - * - [x] tags_toy_links - * - [x] tags_vods_links - * - [x] timestamps - * - [x] timestamps_tag_links - * - [x] timestamps_vod_links - * - [x] toys_link_tag_links - * - [x] vods_mux_asset_links - * - [x] vods_stream_links - * - [x] vods_thumbnail_links - * - [x] vods_uploader_links - * - [x] vods_video_src_b_2_links - * - [x] vods_vtuber_links - * - [x] vtubers_toy_links - * - [x] vtubers_toys_links - * - * - * - */ - - -INSERT INTO api.vtubers -( - id, - display_name, - chaturbate, - twitter, - patreon, - twitch, - tiktok, - onlyfans, - youtube, - linktree, - carrd, - fansly, - pornhub, - discord, - reddit, - throne, - instagram, - facebook, - merch, - slug, - description_1, - description_2, - image, - theme_color, - image_blur, - created_at, - updated_at -) -OVERRIDING SYSTEM VALUE -SELECT - vtubers.id, - display_name, - chaturbate, - twitter, - patreon, - twitch, - tiktok, - onlyfans, - youtube, - linktree, - carrd, - fansly, - pornhub, - discord, - reddit, - throne, - instagram, - facebook, - merch, - slug, - description_1, - description_2, - image, - theme_color, - image_blur, - created_at, - updated_at -FROM public.external_vtubers AS vtubers; - - - - - - -INSERT INTO api.b2_files ( - id, - url, - key, - upload_id, - created_at, - updated_at, - created_by_id, - updated_by_id, - cdn_url -) -OVERRIDING SYSTEM VALUE -SELECT - id, - url, - key, - upload_id, - created_at, - updated_at, - created_by_id, - updated_by_id, - cdn_url -FROM public.external_b2_files; - - -INSERT INTO api.mux_assets ( - id, - asset_id, - playback_id, - created_at, - updated_at -) -OVERRIDING SYSTEM VALUE -SELECT - id, - asset_id, - playback_id, - created_at, - updated_at -FROM public.external_mux_assets; - - - -INSERT INTO api.streams ( - id, - platform_notification_type, - date, - date_2, - created_at, - updated_at, - vtuber_num, - archive_status, - is_chaturbate_stream, - is_fansly_stream -) -OVERRIDING SYSTEM VALUE -SELECT - streams.id, - NULL AS platform_notification_type, -- Modify if necessary - streams.date_2::TIMESTAMP WITH TIME ZONE AS date, - streams.date_2::TIMESTAMPTZ date_2, - streams.created_at, - streams.updated_at, - links.vtuber_id AS vtuber_num, - streams.archive_status, - streams.is_chaturbate_stream, - streams.is_fansly_stream -FROM public.external_streams AS streams -LEFT JOIN public.external_streams_vtuber_links AS links - ON streams.id = links.stream_id; - - - -INSERT INTO api.streams_vtuber_links ( - id, - stream_id, - vtuber_id, - stream_order -) -OVERRIDING SYSTEM VALUE -SELECT - id, - stream_id, - vtuber_id, - stream_order -FROM public.external_streams_vtuber_links; - - - -INSERT INTO api.vods ( - id, - created_at, - updated_at, - published_at, - title, - date, - date_2, - note, - ipfs_cid, - announce_title, - announce_url, - status -) -OVERRIDING SYSTEM VALUE -SELECT - vods.id, - vods.created_at, - vods.updated_at, - vods.published_at, - vods.title, - vods.date::TIMESTAMP WITH TIME ZONE, - vods.date_2::TIMESTAMPTZ AS date_2, - vods.note, - vods.video_src_hash AS ipfs_cid, - vods.announce_title, - vods.announce_url, - 'pending_recording' -FROM public.external_vods AS vods; - - - -INSERT INTO api.toys ( - id, - make, - model, - image, - created_at, - updated_at -) -OVERRIDING SYSTEM VALUE -SELECT - toys.id, - toys.make, - toys.model, - toys.image_2 AS image, - toys.created_at, - toys.updated_at -FROM public.external_toys AS toys; - - - -INSERT INTO api.tags ( - id, - name, - created_at, - updated_at, - created_by_id, - updated_by_id -) -OVERRIDING SYSTEM VALUE -SELECT - id, - name, - created_at, - updated_at, - created_by_id, - updated_by_id -FROM public.external_tags; - - - -INSERT INTO api.tag_vod_relations ( - id, - votes, - creator_id, - created_at, - updated_at, - created_by_id, - updated_by_id -) -SELECT - id, - votes, - creator_id, - created_at, - updated_at, - created_by_id, - updated_by_id -FROM public.external_tag_vod_relations; - - - -INSERT INTO api.tag_vod_relations_tag_links ( - id, - tag_vod_relation_id, - tag_id -) -SELECT - id, - tag_vod_relation_id, - tag_id -FROM public.external_tag_vod_relations_tag_links; - - - -INSERT INTO api.tag_vod_relations_vod_links -( - id, - tag_vod_relation_id, - vod_id, - tag_vod_relation_order -) -SELECT - id, - tag_vod_relation_id, - vod_id, - tag_vod_relation_order -FROM public.external_tag_vod_relations_vod_links; - - - -INSERT INTO api.tags_toy_links -( - id, - tag_id, - toy_id, - tag_order -) -SELECT - id, - tag_id, - toy_id, - tag_order -FROM public.external_tags_toy_links; - - - -INSERT INTO api.tags_vods_links -( - id, - tag_id, - vod_id, - tag_order, - vod_order -) -SELECT - id, - tag_id, - vod_id, - tag_order, - vod_order -FROM public.external_tags_vods_links; - - - -INSERT INTO api.timestamps -( - id, - time, - creator_id, - created_at, - updated_at, - created_by_id, - updated_by_id -) -SELECT - id, - time, - creator_id, - created_at, - updated_at, - created_by_id, - updated_by_id -FROM public.external_timestamps; - - - -INSERT INTO api.timestamps_tag_links -( - id, - timestamp_id, - tag_id -) -SELECT - id, - timestamp_id, - tag_id -FROM public.external_timestamps_tag_links; - - - -INSERT INTO api.timestamps_vod_links -( - id, - timestamp_id, - vod_id, - timestamp_order -) -SELECT - id, - timestamp_id, - vod_id, - timestamp_order -FROM public.external_timestamps_vod_links; - - - -INSERT INTO api.toys_link_tag_links -( - id, - toy_id, - tag_id -) -SELECT - id, - toy_id, - tag_id -FROM public.external_toys_link_tag_links; - - - -INSERT INTO api.vods_mux_asset_links ( - id, - vod_id, - mux_asset_id -) -SELECT - vmal.id, - vmal.vod_id, - vmal.mux_asset_id -FROM public.external_vods_mux_asset_links AS vmal; - - - -INSERT INTO api.vods_stream_links ( - id, - vod_id, - stream_id, - vod_order -) -SELECT - vsl.id, - vsl.vod_id, - vsl.stream_id, - vsl.vod_order -FROM public.external_vods_stream_links AS vsl; - - - -INSERT INTO api.vods_thumbnail_links ( - id, - vod_id, - b_2_file_id -) -SELECT - vtl.id, - vtl.vod_id, - vtl.b_2_file_id -FROM public.external_vods_thumbnail_links AS vtl; - - - -INSERT INTO api.vods_uploader_links ( - id, - vod_id, - user_id -) -SELECT - vul.id, - vul.vod_id, - vul.user_id -FROM public.external_vods_uploader_links AS vul; - - - -INSERT INTO api.vods_video_src_b_2_links ( - id, - vod_id, - b_2_file_id -) -SELECT - v.id, - v.vod_id, - v.b_2_file_id -FROM public.external_vods_video_src_b_2_links AS v; - - - -INSERT INTO api.vods_vtuber_links ( - id, - vod_id, - vtuber_id, - vod_order -) -SELECT - v.id, - v.vod_id, - v.vtuber_id, - v.vod_order -FROM public.external_vods_vtuber_links AS v; - - - -INSERT INTO api.vtubers_toy_links -( - id, - vtuber_id, - toy_id, - vtuber_order -) -SELECT - id, - vtuber_id, - toy_id, - vtuber_order -FROM public.external_vtubers_toy_links; - - -INSERT INTO api.vtubers_toys_links -( - id, - vtuber_id, - toy_id, - toy_order -) -SELECT - id, - vtuber_id, - toy_id, - toy_order -FROM public.external_vtubers_toys_links; - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/services/migrations-data/migrations/0002_relate-vtubers-to-vods.sql b/services/migrations-data/migrations/0002_relate-vtubers-to-vods.sql deleted file mode 100644 index 801e0e8..0000000 --- a/services/migrations-data/migrations/0002_relate-vtubers-to-vods.sql +++ /dev/null @@ -1,17 +0,0 @@ --- 2024-11-21 - --- SELECT * FROM api.vods AS vods --- INNER JOIN api.vods_vtuber_links AS links --- ON vods.id = links.vtuber_id - --- INSERT INTO api.vods (vtuber) --- SELECT vtuber --- FROM api.vods_vtuber_links --- WHERE api.vods.id = api.vods_vtuber_links.vtuber_id; - -UPDATE api.vods -SET vtuber_id = links.vtuber_id -FROM api.vods_vtuber_links links -WHERE api.vods.id = links.vod_id; - - diff --git a/services/migrations-data/migrations/0003_relate-mux_asset_id-to-vods.sql b/services/migrations-data/migrations/0003_relate-mux_asset_id-to-vods.sql deleted file mode 100644 index c118eba..0000000 --- a/services/migrations-data/migrations/0003_relate-mux_asset_id-to-vods.sql +++ /dev/null @@ -1,9 +0,0 @@ --- 2024-11-22 - -UPDATE api.vods -SET mux_asset_id = links.mux_asset_id -FROM api.vods_mux_asset_links links -WHERE api.vods.id = links.vod_id; - - - diff --git a/services/migrations-data/migrations/0004_relate-thumbnails-to-vods.sql b/services/migrations-data/migrations/0004_relate-thumbnails-to-vods.sql deleted file mode 100644 index d812c90..0000000 --- a/services/migrations-data/migrations/0004_relate-thumbnails-to-vods.sql +++ /dev/null @@ -1,9 +0,0 @@ --- 2024-11-22 - -UPDATE api.vods -SET thumbnail_id = links.b_2_file_id -FROM api.vods_thumbnail_links links -WHERE api.vods.id = links.vod_id; - - - diff --git a/services/migrations-data/migrations/0005_rename-vtuber_num-to-vtuber_id.sql b/services/migrations-data/migrations/0005_rename-vtuber_num-to-vtuber_id.sql deleted file mode 100644 index cefe465..0000000 --- a/services/migrations-data/migrations/0005_rename-vtuber_num-to-vtuber_id.sql +++ /dev/null @@ -1,9 +0,0 @@ --- 2024-11-22 - -UPDATE api.streams -SET vtuber_id = vtuber_num -WHERE vtuber_num IS NOT NULL; - - - --- @TODO api.streams.vtuber_num is deprecated in favor of api.streams.vtuber_Id \ No newline at end of file diff --git a/services/migrations-data/migrations/0006_relate-vods-to-streams.sql b/services/migrations-data/migrations/0006_relate-vods-to-streams.sql deleted file mode 100644 index e9a7695..0000000 --- a/services/migrations-data/migrations/0006_relate-vods-to-streams.sql +++ /dev/null @@ -1,9 +0,0 @@ --- 2024-12-16 - --- Relate VODs to streams by matching the same date - --- Update existing VODs to associate them with the corresponding stream -UPDATE api.vods -SET stream_id = streams.id -FROM api.streams -WHERE vods.date = streams.date; \ No newline at end of file diff --git a/services/migrations-data/migrations/README.md b/services/migrations-data/migrations/README.md deleted file mode 100644 index 1d6b1ca..0000000 --- a/services/migrations-data/migrations/README.md +++ /dev/null @@ -1,17 +0,0 @@ -# Futureporn data migrations - -This directory is for data migrations ONLY. -For schema migrations, see ./migrations node package - - -## Usage - - $ a.migration.sh - -### Real-world Example - -Here we use dotenvx to load environment variables (specifically POSTGRES_PASSWORD is needed) and run the migration inside the postgres pod - - dotenvx run -f ./.env.development -- bash -x ./scripts/data-migrations/a.migration.sh ./scripts/data-migrations/2024-10-07-transfer-streams.sql - - diff --git a/services/migrations-data/package.json b/services/migrations-data/package.json deleted file mode 100644 index 3925948..0000000 --- a/services/migrations-data/package.json +++ /dev/null @@ -1,20 +0,0 @@ -{ - "name": "@futureporn/migrations-data", - "type": "module", - "version": "0.6.0", - "description": "", - "main": "index.js", - "scripts": { - "test": "echo \"Error: no test specified\" && exit 0", - "start": "node index.js" - }, - "packageManager": "pnpm@9.6.0", - "keywords": [], - "author": "@CJ_Clippy", - "license": "Unlicense", - "dependencies": { - "dotenv": "^16.4.5", - "pg": "8.12.0", - "postgres-schema-migrations": "^6.1.0" - } -} diff --git a/services/migrations-data/pnpm-lock.yaml b/services/migrations-data/pnpm-lock.yaml deleted file mode 100644 index 3fe9609..0000000 --- a/services/migrations-data/pnpm-lock.yaml +++ /dev/null @@ -1,186 +0,0 @@ -lockfileVersion: '9.0' - -settings: - autoInstallPeers: true - excludeLinksFromLockfile: false - -importers: - - .: - dependencies: - dotenv: - specifier: ^16.4.5 - version: 16.4.5 - pg: - specifier: 8.12.0 - version: 8.12.0 - postgres-schema-migrations: - specifier: ^6.1.0 - version: 6.1.0 - - ../..: {} - - ../../packages/fetchers: {} - - ../../packages/infra: {} - - ../../packages/storage: {} - - ../../packages/types: {} - - ../../packages/utils: {} - - ../bot: {} - - ../capture: {} - - ../factory: {} - - ../htmx: {} - - ../mailbox: {} - - ../migrations-schema: {} - - ../next: {} - - ../scout: {} - - ../strapi: {} - - ../uppy: {} - -packages: - - dotenv@16.4.5: - resolution: {integrity: sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==} - engines: {node: '>=12'} - - pg-cloudflare@1.1.1: - resolution: {integrity: sha512-xWPagP/4B6BgFO+EKz3JONXv3YDgvkbVrGw2mTo3D6tVDQRh1e7cqVGvyR3BE+eQgAvx1XhW/iEASj4/jCWl3Q==} - - pg-connection-string@2.6.4: - resolution: {integrity: sha512-v+Z7W/0EO707aNMaAEfiGnGL9sxxumwLl2fJvCQtMn9Fxsg+lPpPkdcyBSv/KFgpGdYkMfn+EI1Or2EHjpgLCA==} - - pg-int8@1.0.1: - resolution: {integrity: sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw==} - engines: {node: '>=4.0.0'} - - pg-pool@3.6.2: - resolution: {integrity: sha512-Htjbg8BlwXqSBQ9V8Vjtc+vzf/6fVUuak/3/XXKA9oxZprwW3IMDQTGHP+KDmVL7rtd+R1QjbnCFPuTHm3G4hg==} - peerDependencies: - pg: '>=8.0' - - pg-protocol@1.6.1: - resolution: {integrity: sha512-jPIlvgoD63hrEuihvIg+tJhoGjUsLPn6poJY9N5CnlPd91c2T18T/9zBtLxZSb1EhYxBRoZJtzScCaWlYLtktg==} - - pg-types@2.2.0: - resolution: {integrity: sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==} - engines: {node: '>=4'} - - pg@8.12.0: - resolution: {integrity: sha512-A+LHUSnwnxrnL/tZ+OLfqR1SxLN3c/pgDztZ47Rpbsd4jUytsTtwQo/TLPRzPJMp/1pbhYVhH9cuSZLAajNfjQ==} - engines: {node: '>= 8.0.0'} - peerDependencies: - pg-native: '>=3.0.1' - peerDependenciesMeta: - pg-native: - optional: true - - pgpass@1.0.5: - resolution: {integrity: sha512-FdW9r/jQZhSeohs1Z3sI1yxFQNFvMcnmfuj4WBMUTxOrAyLMaTcE1aAMBiTlbMNaXvBCQuVi0R7hd8udDSP7ug==} - - postgres-array@2.0.0: - resolution: {integrity: sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA==} - engines: {node: '>=4'} - - postgres-bytea@1.0.0: - resolution: {integrity: sha512-xy3pmLuQqRBZBXDULy7KbaitYqLcmxigw14Q5sj8QBVLqEwXfeybIKVWiqAXTlcvdvb0+xkOtDbfQMOf4lST1w==} - engines: {node: '>=0.10.0'} - - postgres-date@1.0.7: - resolution: {integrity: sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q==} - engines: {node: '>=0.10.0'} - - postgres-interval@1.2.0: - resolution: {integrity: sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==} - engines: {node: '>=0.10.0'} - - postgres-schema-migrations@6.1.0: - resolution: {integrity: sha512-d1LJ+A9Lg4kAwuh91S8ozF8q3adFNJlStbpUF/sbjMTzSIzJClpmg4D6qyd9nvKt2el0rnZJjXZQ2r01Y5OpzA==} - engines: {node: '>10.17.0'} - hasBin: true - - split2@4.2.0: - resolution: {integrity: sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==} - engines: {node: '>= 10.x'} - - sql-template-strings@2.2.2: - resolution: {integrity: sha512-UXhXR2869FQaD+GMly8jAMCRZ94nU5KcrFetZfWEMd+LVVG6y0ExgHAhatEcKZ/wk8YcKPdi+hiD2wm75lq3/Q==} - engines: {node: '>=4.0.0'} - - xtend@4.0.2: - resolution: {integrity: sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==} - engines: {node: '>=0.4'} - -snapshots: - - dotenv@16.4.5: {} - - pg-cloudflare@1.1.1: - optional: true - - pg-connection-string@2.6.4: {} - - pg-int8@1.0.1: {} - - pg-pool@3.6.2(pg@8.12.0): - dependencies: - pg: 8.12.0 - - pg-protocol@1.6.1: {} - - pg-types@2.2.0: - dependencies: - pg-int8: 1.0.1 - postgres-array: 2.0.0 - postgres-bytea: 1.0.0 - postgres-date: 1.0.7 - postgres-interval: 1.2.0 - - pg@8.12.0: - dependencies: - pg-connection-string: 2.6.4 - pg-pool: 3.6.2(pg@8.12.0) - pg-protocol: 1.6.1 - pg-types: 2.2.0 - pgpass: 1.0.5 - optionalDependencies: - pg-cloudflare: 1.1.1 - - pgpass@1.0.5: - dependencies: - split2: 4.2.0 - - postgres-array@2.0.0: {} - - postgres-bytea@1.0.0: {} - - postgres-date@1.0.7: {} - - postgres-interval@1.2.0: - dependencies: - xtend: 4.0.2 - - postgres-schema-migrations@6.1.0: - dependencies: - pg: 8.12.0 - sql-template-strings: 2.2.2 - transitivePeerDependencies: - - pg-native - - split2@4.2.0: {} - - sql-template-strings@2.2.2: {} - - xtend@4.0.2: {} diff --git a/services/migrations-schema/README.md b/services/migrations-schema/README.md deleted file mode 100644 index 4fdd8da..0000000 --- a/services/migrations-schema/README.md +++ /dev/null @@ -1,27 +0,0 @@ -# @futureporn/migrations - -Here we handle migrations for the postgrest database. - -@see https://github.com/zakpatterson/postgres-schema-migrations - -Reminder: only write migrations that affect schema. (don't write migrations that affect data) - -## K.I.S.S. - -Keep It Stupidly Simple. - -We are keeping this module as simple as possible. This means pure JS (no typescript!) - - -## troubleshooting - -If you see the following error, graphile_worker likely hasn't had a chance to create it's functions. Make sure that a graphile_worker is running, so it can automatically create the necessary functions. - -```json -{ - "code": "42883", - "details": null, - "hint": "No function matches the given name and argument types. You might need to add explicit type casts.", - "message": "function graphile_worker.add_job(text, json, max_attempts => integer) does not exist" -} -``` \ No newline at end of file diff --git a/services/migrations-schema/index.js b/services/migrations-schema/index.js deleted file mode 100644 index 435b172..0000000 --- a/services/migrations-schema/index.js +++ /dev/null @@ -1,32 +0,0 @@ -import {migrate} from 'postgres-schema-migrations' -import path, { dirname } from 'node:path' -import { fileURLToPath } from 'url'; -import 'dotenv/config' - - -const __dirname = dirname(fileURLToPath(import.meta.url)); - -if (!process.env.DATABASE_PASSWORD) throw new Error('DATABASE_PASSWORD is missing in env'); - -async function main() { - const dbConfig = { - database: "futureporn", - user: "postgres", - password: process.env.DATABASE_PASSWORD, - host: 'postgresql-primary.futureporn.svc.cluster.local', - port: 5432, - - // Default: false for backwards-compatibility - // This might change! - ensureDatabaseExists: true, - - // Default: "postgres" - // Used when checking/creating "database-name" - defaultDatabase: "postgres" - } - - await migrate(dbConfig, path.join(__dirname, "./migrations/"), { schema: 'migrations_schema', logger: console.log }) -} - - -await main() \ No newline at end of file diff --git a/services/migrations-schema/migrations/00001_create.sql b/services/migrations-schema/migrations/00001_create.sql deleted file mode 100644 index 344918c..0000000 --- a/services/migrations-schema/migrations/00001_create.sql +++ /dev/null @@ -1,26 +0,0 @@ --- api schema, which houses all the tables for api endpoints --- example: api.discord_interactions becomes accessible at localhost:9000/discord_interactions -CREATE schema api; - --- authenticator is the role which can "impersonate" other users. -CREATE ROLE authenticator LOGIN NOINHERIT NOCREATEDB NOCREATEROLE NOSUPERUSER; --- web_anon is the role assigned to anonymous web requests -CREATE ROLE web_anon NOLOGIN; - --- schema for @futureporn/capture and @futureporn/bot -CREATE TABLE api.discord_interactions ( - id int PRIMARY KEY GENERATED ALWAYS AS IDENTITY, - discord_message_id text NOT NULL, - capture_job_id text NOT NULL -); - - --- roles & permissions for our backend automation user -CREATE ROLE automation NOLOGIN; -GRANT automation TO authenticator; -GRANT usage ON SCHEMA api TO automation; -GRANT all ON api.discord_interactions TO automation; - --- role & permissions for web_anon web user -GRANT usage on schema api TO web_anon; -GRANT SELECT ON api.discord_interactions TO web_anon; diff --git a/services/migrations-schema/migrations/00002_add-records-table.sql b/services/migrations-schema/migrations/00002_add-records-table.sql deleted file mode 100644 index b9cd2e3..0000000 --- a/services/migrations-schema/migrations/00002_add-records-table.sql +++ /dev/null @@ -1,13 +0,0 @@ --- records table schema -CREATE TABLE api.records ( - id INT PRIMARY KEY GENERATED ALWAYS AS IDENTITY, - url TEXT NOT NULL, - discord_message_id TEXT NOT NULL, - recording_state TEXT NOT NULL DEFAULT 'pending', - file_size BIGINT NOT NULL DEFAULT 0, - is_aborted BOOLEAN NOT NULL DEFAULT FALSE -); - --- roles & permissions for our backend automation user -GRANT all ON api.records TO automation; -GRANT SELECT ON api.records TO web_anon; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00003_create-graphile-worker-schema.sql b/services/migrations-schema/migrations/00003_create-graphile-worker-schema.sql deleted file mode 100644 index 173025a..0000000 --- a/services/migrations-schema/migrations/00003_create-graphile-worker-schema.sql +++ /dev/null @@ -1,4 +0,0 @@ -CREATE schema graphile_worker; -GRANT all ON SCHEMA graphile_worker TO postgres; -GRANT all ON SCHEMA graphile_worker TO automation; - diff --git a/services/migrations-schema/migrations/00004_create-trigger-function.sql b/services/migrations-schema/migrations/00004_create-trigger-function.sql deleted file mode 100644 index f3e42f8..0000000 --- a/services/migrations-schema/migrations/00004_create-trigger-function.sql +++ /dev/null @@ -1,29 +0,0 @@ - --- We create a function which lets Postgrest's automation user create jobs in Graphile Worker. --- Normally only the database owner, in our case `postgres`, can add jobs due to RLS in graphile_worker tables. --- Under the advice of graphile_worker author, we can use a SECURITY DEFINER wrapper function. --- @see https://worker.graphile.org/docs/sql-add-job#graphile_workeradd_job:~:text=graphile_worker.add_job(...),that%20are%20necessary.) --- @see https://discord.com/channels/489127045289476126/1179293106336694333/1179605043729670306 --- @see https://discord.com/channels/489127045289476126/498852330754801666/1067707497235873822 - - - -CREATE FUNCTION public.tg__add_job() RETURNS trigger - LANGUAGE plpgsql SECURITY DEFINER - SET search_path TO 'pg_catalog', 'public', 'pg_temp' - AS $$ - begin - PERFORM graphile_worker.add_job(tg_argv[0], json_build_object( - 'url', NEW.url, - 'record_id', NEW.id - ), max_attempts := 12); - return NEW; - end; - $$; - - -CREATE TRIGGER record - AFTER INSERT ON api.records - FOR EACH ROW - EXECUTE PROCEDURE public.tg__add_job('record'); - diff --git a/services/migrations-schema/migrations/00005_add-trigger-for-record-update.sql b/services/migrations-schema/migrations/00005_add-trigger-for-record-update.sql deleted file mode 100644 index 5880cc4..0000000 --- a/services/migrations-schema/migrations/00005_add-trigger-for-record-update.sql +++ /dev/null @@ -1,21 +0,0 @@ - -CREATE FUNCTION public.tg__update_discord_message() RETURNS trigger - LANGUAGE plpgsql SECURITY DEFINER - SET search_path TO 'pg_catalog', 'public', 'pg_temp' - AS $$ - begin - PERFORM graphile_worker.add_job('update_discord_message', json_build_object( - 'record_id', NEW.id - ), max_attempts := 3); - return NEW; - end; - $$; - - - --- when a record is updated, we add a job in graphile to update_discord_message -CREATE TRIGGER record_update - AFTER UPDATE ON api.records - FOR EACH ROW - EXECUTE PROCEDURE public.tg__update_discord_message('update_discord_message'); - diff --git a/services/migrations-schema/migrations/00006_add-updated-at-to-records.sql b/services/migrations-schema/migrations/00006_add-updated-at-to-records.sql deleted file mode 100644 index 6673808..0000000 --- a/services/migrations-schema/migrations/00006_add-updated-at-to-records.sql +++ /dev/null @@ -1,5 +0,0 @@ -ALTER TABLE IF EXISTS api.records - ADD COLUMN created_at timestamp(6) without time zone; - -ALTER TABLE IF EXISTS api.records - ADD COLUMN updated_at timestamp(6) without time zone; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00007_add-default-records-timestamps.sql b/services/migrations-schema/migrations/00007_add-default-records-timestamps.sql deleted file mode 100644 index 17caebf..0000000 --- a/services/migrations-schema/migrations/00007_add-default-records-timestamps.sql +++ /dev/null @@ -1,7 +0,0 @@ -ALTER TABLE IF EXISTS api.records - ADD CONSTRAINT created_at_not_null - CHECK (created_at IS NOT NULL) NOT VALID; - -ALTER TABLE IF EXISTS api.records - ADD CONSTRAINT updated_at_not_null - CHECK (updated_at IS NOT NULL) NOT VALID; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00008_add-default-records-timestamp.sql b/services/migrations-schema/migrations/00008_add-default-records-timestamp.sql deleted file mode 100644 index 1005698..0000000 --- a/services/migrations-schema/migrations/00008_add-default-records-timestamp.sql +++ /dev/null @@ -1,26 +0,0 @@ --- In the prev. migration I added a CHECK, but I forgot to add the default - - -ALTER TABLE IF EXISTS api.records - ALTER COLUMN created_at SET DEFAULT now(); - -ALTER TABLE IF EXISTS api.records - ALTER COLUMN updated_at SET DEFAULT now(); - - --- create a function which updates the row's updated_at -CREATE FUNCTION public.tg__updated_at() RETURNS trigger - LANGUAGE plpgsql - SET search_path TO 'pg_catalog', 'public', 'pg_temp' - AS $$ - BEGIN - NEW.updated_at = now(); - RETURN NEW; - END; - $$; - --- create a trigger which runs the above function when a /record is updated -CREATE TRIGGER record_updated_at - AFTER UPDATE ON api.records - FOR EACH ROW - EXECUTE PROCEDURE public.tg__updated_at(); diff --git a/services/migrations-schema/migrations/00009_add-streams-vods-vtubers.sql b/services/migrations-schema/migrations/00009_add-streams-vods-vtubers.sql deleted file mode 100644 index 134fc45..0000000 --- a/services/migrations-schema/migrations/00009_add-streams-vods-vtubers.sql +++ /dev/null @@ -1,140 +0,0 @@ - --- vtubers table -CREATE TABLE api.vtubers ( - id uuid PRIMARY KEY DEFAULT gen_random_uuid(), - display_name TEXT NOT NULL, - chaturbate TEXT, - twitter TEXT, - patreon TEXT, - twitch TEXT, - tiktok TEXT, - onlyfans TEXT, - youtube TEXT, - linktree TEXT, - carrd TEXT, - fansly TEXT, - pornhub TEXT, - discord TEXT, - reddit TEXT, - throne TEXT, - instagram TEXT, - facebook TEXT, - merch TEXT, - slug TEXT NOT NULL, - description1 TEXT, - description2 TEXT, - image TEXT NOT NULL, - theme_color VARCHAR(7) NOT NULL, - image_blur TEXT DEFAULT '', - fansly_id TEXT, - chaturbate_id TEXT, - twitter_id TEXT - -- F.Y.I., relations as follows - -- toys (one-to-many) - -- vods (one-to-many) - -- streams (one-to-many) -); -GRANT all ON api.vtubers TO automation; -GRANT SELECT ON api.vtubers TO web_anon; - - --- streams table -CREATE TABLE api.streams ( - id uuid PRIMARY KEY DEFAULT gen_random_uuid(), - url TEXT NOT NULL, - platform_notification_type TEXT, - date timestamp(6) without time zone, - created_at timestamp(6) without time zone, - vtuber uuid, - FOREIGN KEY (vtuber) REFERENCES api.vtubers(id), - tweet TEXT, - archive_status TEXT, - is_chaturbate_stream BOOLEAN, - is_fansly_stream BOOLEAN -); -GRANT all ON api.streams TO automation; -GRANT SELECT ON api.streams TO web_anon; - --- toys table -CREATE TABLE api.toys ( - id uuid PRIMARY KEY DEFAULT gen_random_uuid(), - -- relation. one toy to many tags - -- relation. one toy to many vtubers - make TEXT NOT NULL, - model TEXT NOT NULL, - image TEXT NOT NULL DEFAULT 'https://futureporn-b2.b-cdn.net/default-thumbnail.webp' -); -GRANT all ON api.toys TO automation; -GRANT SELECT ON api.toys TO web_anon; - - --- tags table -CREATE TABLE api.tags ( - id uuid PRIMARY KEY DEFAULT gen_random_uuid(), - name TEXT NOT NULL UNIQUE, - toy_id uuid, - FOREIGN KEY (toy_id) REFERENCES api.toys -); -GRANT all ON api.tags TO automation; -GRANT SELECT ON api.tags TO web_anon; - --- toys-tags junction table -CREATE TABLE api.toys_tags( - id uuid PRIMARY KEY DEFAULT gen_random_uuid(), - toy_id uuid, - tag_id uuid, - CONSTRAINT fk_toys FOREIGN KEY(toy_id) REFERENCES api.toys(id), - CONSTRAINT fk_tags FOREIGN KEY(tag_id) REFERENCES api.tags(id) -); -GRANT all ON api.toys_tags TO automation; -GRANT SELECT ON api.toys_tags TO web_anon; - --- tags-vods junction table --- toys-vtubers junction table -CREATE TABLE api.toys_vtubers( - id uuid PRIMARY KEY DEFAULT gen_random_uuid(), - toy_id uuid, - vtuber_id uuid, - CONSTRAINT fk_toys FOREIGN KEY(toy_id) REFERENCES api.toys(id), - CONSTRAINT fk_vtubers FOREIGN KEY(vtuber_id) REFERENCES api.vtubers(id) -); -GRANT all ON api.toys_vtubers TO automation; -GRANT SELECT ON api.toys_vtubers TO web_anon; - - - - --- vods table -CREATE TABLE api.vods ( - id uuid PRIMARY KEY DEFAULT gen_random_uuid(), - stream_id uuid NOT NULL, - FOREIGN KEY (stream_id) REFERENCES api.streams(id), - video_cid TEXT UNIQUE, - CONSTRAINT check_video_cid CHECK (video_cid ~ 'Qm[1-9A-HJ-NP-Za-km-z]{44,}|b[A-Za-z2-7]{58,}|B[A-Z2-7]{58,}|z[1-9A-HJ-NP-Za-km-z]{48,}|F[0-9A-F]{50,}'), - announce_title TEXT, - announce_url TEXT, - note TEXT, - date timestamp(6) without time zone, - spoilers TEXT, - title TEXT, - uploader uuid, - mux_asset_id TEXT, - mux_playback_id TEXT, - s3_key TEXT, - s3_id TEXT, - thumbnail TEXT -); -GRANT all ON api.vods TO automation; -GRANT SELECT ON api.vods TO web_anon; - - --- tags-vods junction table -CREATE TABLE api.tags_vods( - id uuid PRIMARY KEY DEFAULT gen_random_uuid(), - tag_id uuid, - vod_id uuid, - CONSTRAINT fk_tags FOREIGN KEY(tag_id) REFERENCES api.tags(id), - CONSTRAINT fk_vods FOREIGN KEY(vod_id) REFERENCES api.vods(id) -); -GRANT all ON api.tags_vods TO automation; -GRANT SELECT ON api.tags_vods TO web_anon; diff --git a/services/migrations-schema/migrations/00010_record-segments.sql b/services/migrations-schema/migrations/00010_record-segments.sql deleted file mode 100644 index 0e8105c..0000000 --- a/services/migrations-schema/migrations/00010_record-segments.sql +++ /dev/null @@ -1,7 +0,0 @@ --- we add the concept of segments to api.records --- implemented as a multidimensional text array, s3_segments. --- the first value is the s3 id, the second value is the s3 key --- [id, key] - -ALTER TABLE IF EXISTS api.records - ADD COLUMN s3_segments text[][]; diff --git a/services/migrations-schema/migrations/00011_use-composite-primary-keys.sql b/services/migrations-schema/migrations/00011_use-composite-primary-keys.sql deleted file mode 100644 index 025f54d..0000000 --- a/services/migrations-schema/migrations/00011_use-composite-primary-keys.sql +++ /dev/null @@ -1,28 +0,0 @@ --- we don't need s3_segments multidimential array. we're moving it's functionality to a new table -ALTER TABLE IF EXISTS api.records - DROP COLUMN s3_segments; - - - --- segments table -CREATE TABLE api.segments ( - id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY, - s3_key TEXT NOT NULL, - s3_id TEXT NOT NULL, - bytes bigint DEFAULT 0 -); -GRANT all ON api.segments TO automation; -GRANT SELECT ON api.segments TO web_anon; - - --- records-segments join table -CREATE TABLE api.records_segments( - id INT GENERATED ALWAYS AS IDENTITY, - record_id INT NOT NULL, - segment_id INT NOT NULL, - CONSTRAINT fk_record FOREIGN KEY(record_id) REFERENCES api.records(id), - CONSTRAINT fk_segment FOREIGN KEY(segment_id) REFERENCES api.segments(id), - PRIMARY KEY(id, record_id, segment_id) -); -GRANT all ON api.records_segments TO automation; -GRANT SELECT ON api.records_segments TO web_anon; diff --git a/services/migrations-schema/migrations/00012_order-records_segments.sql b/services/migrations-schema/migrations/00012_order-records_segments.sql deleted file mode 100644 index daec178..0000000 --- a/services/migrations-schema/migrations/00012_order-records_segments.sql +++ /dev/null @@ -1,2 +0,0 @@ -ALTER TABLE IF EXISTS api.records_segments - ADD COLUMN segments_order INT NOT NULL DEFAULT 0; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00013_rename-segments-order.sql b/services/migrations-schema/migrations/00013_rename-segments-order.sql deleted file mode 100644 index 1f3f812..0000000 --- a/services/migrations-schema/migrations/00013_rename-segments-order.sql +++ /dev/null @@ -1,5 +0,0 @@ -ALTER TABLE IF EXISTS api.records_segments - DROP COLUMN segments_order; - -ALTER TABLE IF EXISTS api.records_segments - ADD COLUMN segment_order INT NOT NULL DEFAULT 0; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00014_create-segments-stream-links.sql b/services/migrations-schema/migrations/00014_create-segments-stream-links.sql deleted file mode 100644 index a28df1f..0000000 --- a/services/migrations-schema/migrations/00014_create-segments-stream-links.sql +++ /dev/null @@ -1,2 +0,0 @@ -DROP TABLE IF EXISTS api.records CASCADE; -DROP TABLE IF EXISTS api.records_segments CASCADE; diff --git a/services/migrations-schema/migrations/00015_create-segments-stream-links-2.sql b/services/migrations-schema/migrations/00015_create-segments-stream-links-2.sql deleted file mode 100644 index c28e45b..0000000 --- a/services/migrations-schema/migrations/00015_create-segments-stream-links-2.sql +++ /dev/null @@ -1,16 +0,0 @@ --- I forgot to actually create the new table -CREATE TABLE api.segments_stream_links ( - id int PRIMARY KEY GENERATED ALWAYS AS IDENTITY, - discord_message_id text NOT NULL, - capture_job_id text NOT NULL -); - - --- roles & permissions -GRANT all ON api.segments_stream_links TO automation; -GRANT SELECT ON api.segments_stream_links TO web_anon; - - --- there is no s3_id in the segments run context so we don't need a column for it -ALTER TABLE IF EXISTS api.segments - DROP COLUMN s3_id; diff --git a/services/migrations-schema/migrations/00016_remove-unecessary-columns.sql b/services/migrations-schema/migrations/00016_remove-unecessary-columns.sql deleted file mode 100644 index 92a483f..0000000 --- a/services/migrations-schema/migrations/00016_remove-unecessary-columns.sql +++ /dev/null @@ -1,8 +0,0 @@ --- oops. bit by unfinished copy-paste - --- there is no s3_id in the segments run context so we don't need a column for it -ALTER TABLE IF EXISTS api.segments_stream_links - DROP COLUMN discord_message_id; - -ALTER TABLE IF EXISTS api.segments_stream_links - DROP COLUMN capture_job_id; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00017_add-stream-status-col.sql b/services/migrations-schema/migrations/00017_add-stream-status-col.sql deleted file mode 100644 index de2d908..0000000 --- a/services/migrations-schema/migrations/00017_add-stream-status-col.sql +++ /dev/null @@ -1,5 +0,0 @@ -ALTER TABLE IF EXISTS api.streams - ADD COLUMN updated_at timestamp(6) without time zone; - -ALTER TABLE IF EXISTS api.streams - ADD COLUMN status TEXT NOT NULL; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00018_add-stream-status-default.sql b/services/migrations-schema/migrations/00018_add-stream-status-default.sql deleted file mode 100644 index 15cb0b6..0000000 --- a/services/migrations-schema/migrations/00018_add-stream-status-default.sql +++ /dev/null @@ -1,5 +0,0 @@ -ALTER TABLE IF EXISTS api.streams - DROP COLUMN IF EXISTS status; - -ALTER TABLE api.streams - ADD COLUMN status TEXT NOT NULL DEFAULT 'pending_recording'; diff --git a/services/migrations-schema/migrations/00019_drop-discord-interactions.sql b/services/migrations-schema/migrations/00019_drop-discord-interactions.sql deleted file mode 100644 index cd9e1cb..0000000 --- a/services/migrations-schema/migrations/00019_drop-discord-interactions.sql +++ /dev/null @@ -1 +0,0 @@ -DROP TABLE IF EXISTS api.discord_interactions CASCADE; diff --git a/services/migrations-schema/migrations/00020_add-streams-update-trigger.sql b/services/migrations-schema/migrations/00020_add-streams-update-trigger.sql deleted file mode 100644 index 27848de..0000000 --- a/services/migrations-schema/migrations/00020_add-streams-update-trigger.sql +++ /dev/null @@ -1,38 +0,0 @@ --- delete outdated -DROP FUNCTION IF EXISTS public.tg__add_job(); - - --- We create a function which lets Postgrest's automation user create jobs in Graphile Worker. --- Normally only the database owner, in our case `postgres`, can add jobs due to RLS in graphile_worker tables. --- Under the advice of graphile_worker author, we can use a SECURITY DEFINER wrapper function. --- @see https://worker.graphile.org/docs/sql-add-job#graphile_workeradd_job:~:text=graphile_worker.add_job(...),that%20are%20necessary.) --- @see https://discord.com/channels/489127045289476126/1179293106336694333/1179605043729670306 --- @see https://discord.com/channels/489127045289476126/498852330754801666/1067707497235873822 -CREATE FUNCTION public.tg__add_record_job() RETURNS trigger - LANGUAGE plpgsql SECURITY DEFINER - SET search_path TO 'pg_catalog', 'public', 'pg_temp' - AS $$ - begin - PERFORM graphile_worker.add_job('record', json_build_object( - 'url', NEW.url, - 'stream_id', NEW.id - ), max_attempts := 12); - return NEW; - end; - $$; - - - - --- when a stream is updated, we add a job in graphile to update_discord_message -CREATE TRIGGER stream_update - AFTER UPDATE ON api.streams - FOR EACH ROW - EXECUTE PROCEDURE public.tg__update_discord_message('update_discord_message'); - --- when a stream is created, we add a 'record' job in graphile-worker -CREATE TRIGGER stream_create - AFTER INSERT ON api.streams - FOR EACH ROW - EXECUTE PROCEDURE public.tg__add_record_job('record'); - diff --git a/services/migrations-schema/migrations/00021_add-foreign-key-to-segments_stream.sql b/services/migrations-schema/migrations/00021_add-foreign-key-to-segments_stream.sql deleted file mode 100644 index d298003..0000000 --- a/services/migrations-schema/migrations/00021_add-foreign-key-to-segments_stream.sql +++ /dev/null @@ -1,9 +0,0 @@ -DROP TABLE api.segments_stream_links; - -CREATE TABLE api.segments_stream_links ( - id int GENERATED ALWAYS AS IDENTITY, - stream_id UUID NOT NULL REFERENCES api.streams(id), - segment_id INT NOT NULL REFERENCES api.segments(id), - capture_job_id text NOT NULL, - PRIMARY KEY(id, stream_id, segment_id) -); diff --git a/services/migrations-schema/migrations/00022_add-permissions-for-segments_stream_links.sql b/services/migrations-schema/migrations/00022_add-permissions-for-segments_stream_links.sql deleted file mode 100644 index a089e1b..0000000 --- a/services/migrations-schema/migrations/00022_add-permissions-for-segments_stream_links.sql +++ /dev/null @@ -1,2 +0,0 @@ -GRANT all ON api.segments_stream_links TO automation; -GRANT SELECT ON api.segments_stream_links TO web_anon; diff --git a/services/migrations-schema/migrations/00023_drop-capture_job_id-column.sql b/services/migrations-schema/migrations/00023_drop-capture_job_id-column.sql deleted file mode 100644 index 98fe03d..0000000 --- a/services/migrations-schema/migrations/00023_drop-capture_job_id-column.sql +++ /dev/null @@ -1,3 +0,0 @@ - -ALTER TABLE IF EXISTS api.segments_stream_links - DROP COLUMN IF EXISTS capture_job_id; diff --git a/services/migrations-schema/migrations/00024_add-updated_at-for-segments.sql b/services/migrations-schema/migrations/00024_add-updated_at-for-segments.sql deleted file mode 100644 index 9143a87..0000000 --- a/services/migrations-schema/migrations/00024_add-updated_at-for-segments.sql +++ /dev/null @@ -1,39 +0,0 @@ - -ALTER TABLE api.segments - ADD COLUMN created_at TIMESTAMP(6) WITHOUT TIME ZONE; - -ALTER TABLE api.segments - ADD COLUMN updated_at TIMESTAMP(6) WITHOUT TIME ZONE; - - - --- in migration 8, we already created tg__updated_at() so we don't need to create that, --- but we do need to create a function which will the row's created_at -CREATE FUNCTION public.tg__created_at() RETURNS trigger - LANGUAGE plpgsql - SET search_path TO 'pg_catalog', 'public', 'pg_temp' - AS $$ - BEGIN - NEW.created_at = now(); - RETURN NEW; - END; - $$; - --- create a trigger which runs the tg__updated_at() function when a /segment is updated -CREATE TRIGGER segment_updated_at - AFTER UPDATE ON api.segments - FOR EACH ROW - EXECUTE PROCEDURE public.tg__updated_at(); - - --- create a trigger which runs the tg__created_at() function when a /segment is created -CREATE TRIGGER segment_created_at - AFTER INSERT ON api.segments - FOR EACH ROW - EXECUTE PROCEDURE public.tg__created_at(); - --- create a trigger which runs the tg__created_at() function when a /stream is created -CREATE TRIGGER stream_created_at - AFTER INSERT ON api.streams - FOR EACH ROW - EXECUTE PROCEDURE public.tg__created_at(); diff --git a/services/migrations-schema/migrations/00025_add-is_recording_aborted-to-streams.sql b/services/migrations-schema/migrations/00025_add-is_recording_aborted-to-streams.sql deleted file mode 100644 index ce10bc3..0000000 --- a/services/migrations-schema/migrations/00025_add-is_recording_aborted-to-streams.sql +++ /dev/null @@ -1,2 +0,0 @@ -ALTER TABLE api.streams - ADD COLUMN is_recording_aborted BOOLEAN DEFAULT FALSE; diff --git a/services/migrations-schema/migrations/00026_use-moddatetime.sql b/services/migrations-schema/migrations/00026_use-moddatetime.sql deleted file mode 100644 index f2929f9..0000000 --- a/services/migrations-schema/migrations/00026_use-moddatetime.sql +++ /dev/null @@ -1,2 +0,0 @@ -CREATE EXTENSION moddatetime; - diff --git a/services/migrations-schema/migrations/00027_create-triggers-for-moddatetime.sql b/services/migrations-schema/migrations/00027_create-triggers-for-moddatetime.sql deleted file mode 100644 index 74a4eb3..0000000 --- a/services/migrations-schema/migrations/00027_create-triggers-for-moddatetime.sql +++ /dev/null @@ -1,49 +0,0 @@ - - --- now we set up the triggers - --- streams created_at -ALTER TABLE api.streams - ALTER created_at SET DEFAULT now(); - -DROP TRIGGER stream_created_at ON api.streams; - -CREATE TRIGGER stream_created_at - BEFORE INSERT ON api.streams - FOR EACH ROW - EXECUTE PROCEDURE moddatetime (created_at); - - --- streams updated_at -ALTER TABLE api.streams - ALTER updated_at SET DEFAULT now(); - -CREATE TRIGGER stream_updated_at - BEFORE UPDATE ON api.streams - FOR EACH ROW - EXECUTE PROCEDURE moddatetime (updated_at); - - --- segments created_at -ALTER TABLE api.segments - ALTER created_at SET DEFAULT now(); - -DROP TRIGGER segment_created_at ON api.segments; - -CREATE TRIGGER segment_created_at - BEFORE INSERT ON api.segments - FOR EACH ROW - EXECUTE PROCEDURE moddatetime(created_at); - - --- segments updated_at -ALTER TABLE api.segments - ALTER updated_at SET DEFAULT now(); - -DROP TRIGGER segment_updated_at ON api.segments; - -CREATE TRIGGER segment_updated_at - BEFORE UPDATE ON api.segments - FOR EACH ROW - EXECUTE PROCEDURE moddatetime(updated_at); - diff --git a/services/migrations-schema/migrations/00028_remove-moddate-on-insert.sql b/services/migrations-schema/migrations/00028_remove-moddate-on-insert.sql deleted file mode 100644 index 3cb45ac..0000000 --- a/services/migrations-schema/migrations/00028_remove-moddate-on-insert.sql +++ /dev/null @@ -1,9 +0,0 @@ --- A fix for the following error --- moddatetime: cannot process INSERT events --- --- We don't need moddatetime for INSERT events because we have column defaults set the time when the row is created. - - -DROP TRIGGER segment_created_at ON api.segments; -DROP TRIGGER stream_created_at ON api.streams; - diff --git a/services/migrations-schema/migrations/00029_add-discord-message-id.sql b/services/migrations-schema/migrations/00029_add-discord-message-id.sql deleted file mode 100644 index 7a7fb4e..0000000 --- a/services/migrations-schema/migrations/00029_add-discord-message-id.sql +++ /dev/null @@ -1,5 +0,0 @@ - --- streams needs discord_message_id for chatops -ALTER TABLE api.streams - ADD COLUMN discord_message_id TEXT; - diff --git a/services/migrations-schema/migrations/00030_update-update_discord_message.sql b/services/migrations-schema/migrations/00030_update-update_discord_message.sql deleted file mode 100644 index 835cd63..0000000 --- a/services/migrations-schema/migrations/00030_update-update_discord_message.sql +++ /dev/null @@ -1,14 +0,0 @@ --- instead of using record_id, we need to use stream_id -DROP FUNCTION public.tg__update_discord_message CASCADE; - -CREATE FUNCTION public.tg__update_discord_message() RETURNS trigger - LANGUAGE plpgsql SECURITY DEFINER - SET search_path TO 'pg_catalog', 'public', 'pg_temp' - AS $$ - begin - PERFORM graphile_worker.add_job('update_discord_message', json_build_object( - 'stream_id', NEW.id - ), max_attempts := 3); - return NEW; - end; - $$; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00031_recreate-update_stream-trigger.sql b/services/migrations-schema/migrations/00031_recreate-update_stream-trigger.sql deleted file mode 100644 index ddb6957..0000000 --- a/services/migrations-schema/migrations/00031_recreate-update_stream-trigger.sql +++ /dev/null @@ -1,5 +0,0 @@ --- when a stream is updated, we add a job in graphile to update_discord_message -CREATE TRIGGER stream_update - AFTER UPDATE ON api.streams - FOR EACH ROW - EXECUTE PROCEDURE public.tg__update_discord_message('update_discord_message'); diff --git a/services/migrations-schema/migrations/00032_update-stream-when-segment-updates.sql b/services/migrations-schema/migrations/00032_update-stream-when-segment-updates.sql deleted file mode 100644 index 18578ff..0000000 --- a/services/migrations-schema/migrations/00032_update-stream-when-segment-updates.sql +++ /dev/null @@ -1,23 +0,0 @@ --- in order for discord chatops messages to be updated when a segment is updated, --- we need to have postgres update the related stream timestamp when a segment is updated. - -CREATE OR REPLACE FUNCTION update_stream_on_segment_update() -RETURNS TRIGGER AS $$ -BEGIN - UPDATE api.streams - SET updated_at = NOW() - WHERE id IN ( - SELECT stream_id - FROM segments_stream_links - WHERE segment_id = NEW.id - ); - - RETURN NEW; -END; -$$ LANGUAGE plpgsql; - - -CREATE TRIGGER trigger_update_stream - AFTER UPDATE ON api.segments - FOR EACH ROW - EXECUTE FUNCTION update_stream_on_segment_update(); diff --git a/services/migrations-schema/migrations/00033_create-vods.sql b/services/migrations-schema/migrations/00033_create-vods.sql deleted file mode 100644 index 24fa113..0000000 --- a/services/migrations-schema/migrations/00033_create-vods.sql +++ /dev/null @@ -1,46 +0,0 @@ --- create s3_files table -CREATE TABLE api.s3_files ( - id uuid PRIMARY KEY DEFAULT gen_random_uuid(), - s3_id TEXT NOT NULL, - s3_key TEXT NOT NULL, - bucket TEXT NOT NULL -); -GRANT all ON api.vods TO automation; -GRANT SELECT ON api.vods TO web_anon; - - --- create mux_assets table -CREATE TABLE api.mux_assets ( - id uuid PRIMARY KEY DEFAULT gen_random_uuid(), - asset_id TEXT NOT NULL, - playback_id TEXT NOT NULL, - created_at TIMESTAMP DEFAULT NOW(), - updated_at TIMESTAMP DEFAULT NOW() -); -GRANT all ON api.mux_assets TO automation; --- web_anon is intentionally not given privs to this table - - --- re-create vods table -DROP TABLE api.vods CASCADE; -CREATE TABLE api.vods ( - id uuid PRIMARY KEY DEFAULT gen_random_uuid(), - stream_id UUID NOT NULL REFERENCES api.streams(id), - created_at TIMESTAMP DEFAULT NOW(), - updated_at TIMESTAMP DEFAULT NOW(), - published_at DATE, - title TEXT, - date DATE NOT NULL, - mux_asset UUID REFERENCES api.mux_assets(id), - thumbnail UUID REFERENCES api.s3_files(id), - vtuber UUID REFERENCES api.vtubers(id), - ipfs_cid TEXT, - s3_file UUID REFERENCES api.s3_files(id), - torrent TEXT, - announce_title TEXT, - announce_url TEXT, - note TEXT -); -GRANT all ON api.vods TO automation; -GRANT SELECT ON api.vods TO web_anon; - diff --git a/services/migrations-schema/migrations/00034_move-segments-to-vods.sql b/services/migrations-schema/migrations/00034_move-segments-to-vods.sql deleted file mode 100644 index 39e78d4..0000000 --- a/services/migrations-schema/migrations/00034_move-segments-to-vods.sql +++ /dev/null @@ -1,24 +0,0 @@ --- segments get moved to vods -DROP TABLE api.segments_stream_links CASCADE; - - --- segments to vod, many-to-one -CREATE TABLE api.segments_vod_links ( - id int PRIMARY KEY GENERATED ALWAYS AS IDENTITY, - vod_id text NOT NULL, - segment_id text NOT NULL -); - - --- roles & permissions -GRANT all ON api.segments_vod_links TO automation; -GRANT SELECT ON api.segments_vod_links TO web_anon; - - --- establish many-to-one relationship by adding a foreign key to segments -ALTER TABLE api.segments - ADD COLUMN vod_id TEXT; - -ALTER TABLE api.segments - ADD CONSTRAINT vod_id_not_null - CHECK (vod_id IS NOT NULL) NOT VALID; diff --git a/services/migrations-schema/migrations/00035_seed-vtubers.sql b/services/migrations-schema/migrations/00035_seed-vtubers.sql deleted file mode 100644 index 1537a9d..0000000 --- a/services/migrations-schema/migrations/00035_seed-vtubers.sql +++ /dev/null @@ -1,159 +0,0 @@ --- seeding some test vtubers for dev environment -INSERT INTO api.vtubers ( - display_name, - slug, - image, - theme_color, - chaturbate, - twitter, - patreon, - twitch, - tiktok, - onlyfans, - youtube, - linktree, - carrd, - fansly, - pornhub, - discord, - reddit, - throne, - instagram, - facebook, - merch, - description1, - description2, - image_blur -) VALUES ( - 'ProjektMelody', -- display_name - 'projektmelody', -- slug - 'https://futureporn-b2.b-cdn.net/projekt-melody.jpg', -- image - '#5C23C0', -- theme_color - 'https://chaturbate.com/projektmelody', -- chaturbate - 'https://twitter.com/projektmelody', -- twitter - 'https://www.patreon.com/projektmelody', -- patreon - 'https://twitch.tv/projektmelody', -- twitch - 'https://www.tiktok.com/@realprojektmelody', -- tiktok - 'https://onlyfans.com/projektbutt', -- onlyfans - 'https://www.youtube.com/projektmelodyofficial', -- youtube - 'https://linktr.ee/projektmelody', -- linktree - NULL, -- carrd (no data provided) - 'https://fansly.com/r/scienceteam', -- fansly - 'https://www.pornhub.com/model/projekt-melody', -- pornhub - NULL, -- discord (no data provided) - 'https://www.reddit.com/r/projektmelody/', -- reddit - 'https://throne.com/realprojektmelody', -- throne - NULL, -- instagram (no data provided) - NULL, -- facebook (no data provided) - 'https://melody.vshojo.com/', -- merch - 'Also known as, ''Daddy''s little grandpa,'' ProjektMelody is the pioneering hentai cam model who embodies a unique blend of sweetness and perversion. She engages with her audience with remarkable patience and politeness, demonstrating her commitment to fostering an inclusive environment.', -- description1 - 'On her livestreams, ProjektMelody''s demeanor can range from overtly sexual to adorably reserved and self-censored. Alongside her captivating presence, she actively promotes pro-social behavior, advocates for sexual education, and emphasizes the importance of kindness.', -- description2 - '' -- image_blur -); - - - - -INSERT INTO api.vtubers ( - display_name, - slug, - image, - theme_color, - chaturbate, - twitter, - patreon, - twitch, - tiktok, - onlyfans, - youtube, - linktree, - carrd, - fansly, - pornhub, - discord, - reddit, - throne, - instagram, - facebook, - merch, - description1, - description2, - image_blur -) VALUES ( - 'el_XoX', -- display_name - 'el_xox', -- slug - 'https://futureporn-b2.b-cdn.net/el_xox.jpg', -- image - '#353FFF', -- theme_color - 'https://chaturbate.com/el_xox/', -- chaturbate - 'https://twitter.com/el_XoX34', -- twitter - NULL, -- patreon (no data provided) - 'https://www.twitch.tv/el_xox', -- twitch - NULL, -- tiktok (no data provided) - NULL, -- onlyfans (no data provided) - NULL, -- youtube (no data provided) - NULL, -- linktree (no data provided) - 'https://elxox.carrd.co/', -- carrd - NULL, -- fansly (no data provided) - NULL, -- pornhub (no data provided) - NULL, -- discord (no data provided) - NULL, -- reddit (no data provided) - NULL, -- throne (no data provided) - NULL, -- instagram (no data provided) - NULL, -- facebook (no data provided) - 'https://elxox34.com/pages/limited-merch', -- merch - ' ', -- description1 (empty string) - NULL, -- description2 (no data provided) - '' -- image_blur -); - -INSERT INTO api.vtubers ( - display_name, - slug, - image, - theme_color, - chaturbate, - twitter, - patreon, - twitch, - tiktok, - onlyfans, - youtube, - linktree, - carrd, - fansly, - pornhub, - discord, - reddit, - throne, - instagram, - facebook, - merch, - description1, - description2, - image_blur -) VALUES ( - 'Vexruby', -- display_name - 'vexruby', -- slug - 'https://futureporn-b2.b-cdn.net/vexruby.jpg', -- image - '#f882f5', -- theme_color - 'https://chaturbate.com/vexruby', -- chaturbate - 'https://x.com/vexxxruby', -- twitter - 'https://www.patreon.com/ViRoClub', -- patreon - NULL, -- twitch (no data provided) - NULL, -- tiktok (no data provided) - NULL, -- onlyfans (no data provided) - NULL, -- youtube (no data provided) - NULL, -- linktree (no data provided) - NULL, -- carrd (no data provided) - NULL, -- fansly (no data provided) - NULL, -- pornhub (no data provided) - NULL, -- discord (no data provided) - NULL, -- reddit (no data provided) - NULL, -- throne (no data provided) - NULL, -- instagram (no data provided) - NULL, -- facebook (no data provided) - NULL, -- merch (no data provided) - ' ', -- description1 (empty string) - NULL, -- description2 (no data provided) - '' -- image_blur -); diff --git a/services/migrations-schema/migrations/00036_update_discord_message-use-vod_id.sql b/services/migrations-schema/migrations/00036_update_discord_message-use-vod_id.sql deleted file mode 100644 index 1adc464..0000000 --- a/services/migrations-schema/migrations/00036_update_discord_message-use-vod_id.sql +++ /dev/null @@ -1,14 +0,0 @@ --- instead of using record_id, we need to use stream_id -DROP FUNCTION public.tg__update_discord_message CASCADE; - -CREATE FUNCTION public.tg__update_discord_message() RETURNS trigger - LANGUAGE plpgsql SECURITY DEFINER - SET search_path TO 'pg_catalog', 'public', 'pg_temp' - AS $$ - begin - PERFORM graphile_worker.add_job('update_discord_message', json_build_object( - 'vod_id', NEW.id - ), max_attempts := 3); - return NEW; - end; - $$; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00037_drop-oudated-streamscolumns.sql b/services/migrations-schema/migrations/00037_drop-oudated-streamscolumns.sql deleted file mode 100644 index 7c1cffe..0000000 --- a/services/migrations-schema/migrations/00037_drop-oudated-streamscolumns.sql +++ /dev/null @@ -1,16 +0,0 @@ --- we are moving url column from streams to vods -ALTER TABLE api.streams - DROP COLUMN url; - -ALTER TABLE api.vods - ADD COLUMN url TEXT; - --- remove unused things -ALTER TABLE api.streams - DROP COLUMN status; - -ALTER TABLE api.streams - DROP COLUMN is_recording_aborted; - -ALTER TABLE api.streams - DROP COLUMN discord_message_id; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00038_add-vods-update-trigger.sql b/services/migrations-schema/migrations/00038_add-vods-update-trigger.sql deleted file mode 100644 index 5797a87..0000000 --- a/services/migrations-schema/migrations/00038_add-vods-update-trigger.sql +++ /dev/null @@ -1,41 +0,0 @@ --- we are moving recording functionality from streams to vods - --- delete outdated -DROP FUNCTION IF EXISTS public.tg__add_record_job CASCADE; -DROP TRIGGER IF EXISTS stream_update ON api.streams; -DROP TRIGGER IF EXISTS stream_create ON api.streams; - --- We create a function which lets Postgrest's automation user create jobs in Graphile Worker. --- Normally only the database owner, in our case `postgres`, can add jobs due to RLS in graphile_worker tables. --- Under the advice of graphile_worker author, we can use a SECURITY DEFINER wrapper function. --- @see https://worker.graphile.org/docs/sql-add-job#graphile_workeradd_job:~:text=graphile_worker.add_job(...),that%20are%20necessary.) --- @see https://discord.com/channels/489127045289476126/1179293106336694333/1179605043729670306 --- @see https://discord.com/channels/489127045289476126/498852330754801666/1067707497235873822 -CREATE FUNCTION public.tg__add_record_job() RETURNS trigger - LANGUAGE plpgsql SECURITY DEFINER - SET search_path TO 'pg_catalog', 'public', 'pg_temp' - AS $$ - begin - PERFORM graphile_worker.add_job('record', json_build_object( - 'url', NEW.url, - 'vod_id', NEW.id - ), max_attempts := 12); - return NEW; - end; - $$; - - - - --- when a vod is updated, we add a job in graphile to update_discord_message -CREATE TRIGGER vod_update - AFTER UPDATE ON api.vods - FOR EACH ROW - EXECUTE PROCEDURE public.tg__update_discord_message('update_discord_message'); - --- when a vod is created, we add a 'record' job in graphile-worker -CREATE TRIGGER vod_create - AFTER INSERT ON api.vods - FOR EACH ROW - EXECUTE PROCEDURE public.tg__add_record_job('record'); - diff --git a/services/migrations-schema/migrations/00039_create-limiter-table.sql b/services/migrations-schema/migrations/00039_create-limiter-table.sql deleted file mode 100644 index f7ec36e..0000000 --- a/services/migrations-schema/migrations/00039_create-limiter-table.sql +++ /dev/null @@ -1,7 +0,0 @@ --- schema for rate limiter (https://github.com/animir/node-rate-limiter-flexible) --- @see https://github.com/animir/node-rate-limiter-flexible/blob/661d794212441f104a6941092c28805b3bd76537/lib/RateLimiterPostgres.js#L161 -CREATE TABLE public.limiter ( - key varchar(255) PRIMARY KEY, - points integer NOT NULL DEFAULT 0, - expire bigint -); diff --git a/services/migrations-schema/migrations/00040_theme_color-is-optional.sql b/services/migrations-schema/migrations/00040_theme_color-is-optional.sql deleted file mode 100644 index d4465ae..0000000 --- a/services/migrations-schema/migrations/00040_theme_color-is-optional.sql +++ /dev/null @@ -1,9 +0,0 @@ --- theme_color, and image are made optional, because they require extra fetches. --- the intention is to have a graphile-worker populate these columns in a parallel workflow, --- rather than force the onboarding process to gather this data which increases the runtime of that process. - -ALTER TABLE IF EXISTS api.vtubers - ALTER COLUMN image DROP NOT NULL; - -ALTER TABLE IF EXISTS api.vtubers - ALTER COLUMN theme_color DROP NOT NULL; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00041_add-is_recording_aborted-to-vods.sql b/services/migrations-schema/migrations/00041_add-is_recording_aborted-to-vods.sql deleted file mode 100644 index 22ad593..0000000 --- a/services/migrations-schema/migrations/00041_add-is_recording_aborted-to-vods.sql +++ /dev/null @@ -1,3 +0,0 @@ - -ALTER TABLE api.vods - ADD COLUMN is_recording_aborted BOOLEAN; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00042_segments-vods-fk.sql b/services/migrations-schema/migrations/00042_segments-vods-fk.sql deleted file mode 100644 index fc1d528..0000000 --- a/services/migrations-schema/migrations/00042_segments-vods-fk.sql +++ /dev/null @@ -1,6 +0,0 @@ -ALTER TABLE api.segments - DROP COLUMN vod_id; - - -ALTER TABLE api.segments - ADD COLUMN vod_id UUID REFERENCES api.vods(id); diff --git a/services/migrations-schema/migrations/00043_update_vod_on_segment_update.sql b/services/migrations-schema/migrations/00043_update_vod_on_segment_update.sql deleted file mode 100644 index 9027379..0000000 --- a/services/migrations-schema/migrations/00043_update_vod_on_segment_update.sql +++ /dev/null @@ -1,23 +0,0 @@ -DROP FUNCTION update_stream_on_segment_update CASCADE; - - -CREATE OR REPLACE FUNCTION update_vod_on_segment_update() -RETURNS TRIGGER AS $$ -BEGIN - UPDATE api.vods - SET updated_at = NOW() - WHERE id IN ( - SELECT vod_id - FROM segments_vod_links - WHERE segment_id = NEW.id - ); - - RETURN NEW; -END; -$$ LANGUAGE plpgsql; - - -CREATE TRIGGER trigger_update_vod - AFTER UPDATE ON api.segments - FOR EACH ROW - EXECUTE FUNCTION update_vod_on_segment_update(); diff --git a/services/migrations-schema/migrations/00044_segments-uuid.sql b/services/migrations-schema/migrations/00044_segments-uuid.sql deleted file mode 100644 index dcae3d4..0000000 --- a/services/migrations-schema/migrations/00044_segments-uuid.sql +++ /dev/null @@ -1,6 +0,0 @@ - -ALTER TABLE api.segments - DROP COLUMN id; - -ALTER TABLE api.segments - ADD COLUMN id uuid PRIMARY KEY DEFAULT gen_random_uuid(); diff --git a/services/migrations-schema/migrations/00045_recreate-segments_vod_links.sql b/services/migrations-schema/migrations/00045_recreate-segments_vod_links.sql deleted file mode 100644 index f69346b..0000000 --- a/services/migrations-schema/migrations/00045_recreate-segments_vod_links.sql +++ /dev/null @@ -1,10 +0,0 @@ --- we need vod_id and segment_id to be uuid, not text - -DROP TABLE api.segments_vod_links; - -CREATE TABLE api.segments_vod_links ( - id UUID DEFAULT gen_random_uuid(), - vod_id UUID NOT NULL REFERENCES api.vods(id), - segment_id UUID NOT NULL REFERENCES api.segments(id), - PRIMARY KEY(id, vod_id, segment_id) -); diff --git a/services/migrations-schema/migrations/00046_segments_vod_links-many-to-one.sql b/services/migrations-schema/migrations/00046_segments_vod_links-many-to-one.sql deleted file mode 100644 index eb0bceb..0000000 --- a/services/migrations-schema/migrations/00046_segments_vod_links-many-to-one.sql +++ /dev/null @@ -1,10 +0,0 @@ --- In the last migration, I accidentally created a many-to-many relationship. --- What I actually need is a many-to-one relationship. - -DROP TABLE api.segments_vod_links; - -CREATE TABLE api.segments_vod_links ( - id UUID PRIMARY KEY DEFAULT gen_random_uuid(), - vod_id UUID NOT NULL REFERENCES api.vods(id), - segment_id UUID NOT NULL REFERENCES api.segments(id) -); diff --git a/services/migrations-schema/migrations/00047_add-is_recording_aborted-to-vods-default-false.sql b/services/migrations-schema/migrations/00047_add-is_recording_aborted-to-vods-default-false.sql deleted file mode 100644 index a5b8b25..0000000 --- a/services/migrations-schema/migrations/00047_add-is_recording_aborted-to-vods-default-false.sql +++ /dev/null @@ -1,5 +0,0 @@ -ALTER TABLE api.vods - DROP COLUMN is_recording_aborted; - -ALTER TABLE api.vods - ADD COLUMN is_recording_aborted BOOLEAN DEFAULT FALSE; diff --git a/services/migrations-schema/migrations/00048_segments-vod-links-perms.sql b/services/migrations-schema/migrations/00048_segments-vod-links-perms.sql deleted file mode 100644 index c9cce2f..0000000 --- a/services/migrations-schema/migrations/00048_segments-vod-links-perms.sql +++ /dev/null @@ -1,3 +0,0 @@ --- roles & permissions -GRANT all ON api.segments_vod_links TO automation; -GRANT SELECT ON api.segments_vod_links TO web_anon; diff --git a/services/migrations-schema/migrations/00049_add-discord_message_id-on-vods.sql b/services/migrations-schema/migrations/00049_add-discord_message_id-on-vods.sql deleted file mode 100644 index 7960a26..0000000 --- a/services/migrations-schema/migrations/00049_add-discord_message_id-on-vods.sql +++ /dev/null @@ -1,4 +0,0 @@ --- vods needs discord_message_id for chatops -ALTER TABLE api.vods - ADD COLUMN discord_message_id TEXT; - diff --git a/services/migrations-schema/migrations/00050_add-status-to-vod.sql b/services/migrations-schema/migrations/00050_add-status-to-vod.sql deleted file mode 100644 index 75d78b6..0000000 --- a/services/migrations-schema/migrations/00050_add-status-to-vod.sql +++ /dev/null @@ -1,3 +0,0 @@ - -ALTER TABLE api.vods - ADD COLUMN status TEXT; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00051_set-vod-status.sql b/services/migrations-schema/migrations/00051_set-vod-status.sql deleted file mode 100644 index 4336e45..0000000 --- a/services/migrations-schema/migrations/00051_set-vod-status.sql +++ /dev/null @@ -1,21 +0,0 @@ --- 'NEW' in this context is a segment row. --- we update this function to also set the vod status to 'recording' if applicable. - -CREATE OR REPLACE FUNCTION update_vod_on_segment_update() -RETURNS TRIGGER AS $$ -BEGIN - UPDATE api.vods - SET - updated_at = NOW(), - status = CASE - WHEN NEW.filesize > OLD.filesize THEN 'recording' - ELSE status - END - WHERE id IN ( - SELECT vod_id - FROM segments_vod_links - WHERE segment_id = NEW.id - ); - RETURN NEW; -END; -$$ LANGUAGE plpgsql; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00052_correct-typo-filesize-to-bytes.sql b/services/migrations-schema/migrations/00052_correct-typo-filesize-to-bytes.sql deleted file mode 100644 index 419df34..0000000 --- a/services/migrations-schema/migrations/00052_correct-typo-filesize-to-bytes.sql +++ /dev/null @@ -1,21 +0,0 @@ --- 'NEW' in this context is a segment row. --- we update this function to also set the vod status to 'recording' if applicable. - -CREATE OR REPLACE FUNCTION update_vod_on_segment_update() -RETURNS TRIGGER AS $$ -BEGIN - UPDATE api.vods - SET - updated_at = NOW(), - status = CASE - WHEN NEW.bytes > OLD.bytes THEN 'recording' - ELSE status - END - WHERE id IN ( - SELECT vod_id - FROM segments_vod_links - WHERE segment_id = NEW.id - ); - RETURN NEW; -END; -$$ LANGUAGE plpgsql; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00053_add-default-status-on-vods.sql b/services/migrations-schema/migrations/00053_add-default-status-on-vods.sql deleted file mode 100644 index 3fd37a8..0000000 --- a/services/migrations-schema/migrations/00053_add-default-status-on-vods.sql +++ /dev/null @@ -1,5 +0,0 @@ -ALTER TABLE api.vods - DROP COLUMN status; - -ALTER TABLE api.vods - ADD COLUMN status TEXT DEFAULT 'pending_recording'; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00054_remove-bytes-from-vod-tg-function.sql b/services/migrations-schema/migrations/00054_remove-bytes-from-vod-tg-function.sql deleted file mode 100644 index 0671976..0000000 --- a/services/migrations-schema/migrations/00054_remove-bytes-from-vod-tg-function.sql +++ /dev/null @@ -1,27 +0,0 @@ - -DROP FUNCTION update_vod_on_segment_update CASCADE; - --- 'NEW' in this context is a segment row. --- we re-create this trigger function, but this time make it's name more consistent with others --- we also remove the conditional status column because it relies on the non-existant bytes column -CREATE OR REPLACE FUNCTION tg__update_vod_updated_at() -RETURNS TRIGGER AS $$ -BEGIN - UPDATE api.vods - SET - updated_at = NOW() - WHERE id IN ( - SELECT vod_id - FROM segments_vod_links - WHERE segment_id = NEW.id - ); - RETURN NEW; -END; -$$ LANGUAGE plpgsql; - - - -CREATE TRIGGER trigger_update_vod - AFTER UPDATE ON api.segments - FOR EACH ROW - EXECUTE FUNCTION tg__update_vod_updated_at(); diff --git a/services/migrations-schema/migrations/00055_trigger_update_vod-switch-to-before.sql b/services/migrations-schema/migrations/00055_trigger_update_vod-switch-to-before.sql deleted file mode 100644 index ed5c903..0000000 --- a/services/migrations-schema/migrations/00055_trigger_update_vod-switch-to-before.sql +++ /dev/null @@ -1,7 +0,0 @@ -DROP TRIGGER trigger_update_vod ON api.segments; - --- switch from `AFTER UPDATE` to `BEFORE UPDATE` -CREATE TRIGGER trigger_update_vod - BEFORE UPDATE ON api.segments - FOR EACH ROW - EXECUTE FUNCTION tg__update_vod_updated_at(); diff --git a/services/migrations-schema/migrations/00056_add_job-retry-only-6-times.sql b/services/migrations-schema/migrations/00056_add_job-retry-only-6-times.sql deleted file mode 100644 index 9965e6d..0000000 --- a/services/migrations-schema/migrations/00056_add_job-retry-only-6-times.sql +++ /dev/null @@ -1,15 +0,0 @@ - --- 12 times is too many, as it will retry the record task hours after a stream has gone offline. We're reducing retries to 6 -DROP FUNCTION public.tg__add_record_job CASCADE; -CREATE FUNCTION public.tg__add_record_job() RETURNS trigger - LANGUAGE plpgsql SECURITY DEFINER - SET search_path TO 'pg_catalog', 'public', 'pg_temp' - AS $$ - begin - PERFORM graphile_worker.add_job('record', json_build_object( - 'url', NEW.url, - 'vod_id', NEW.id - ), max_attempts := 6); - return NEW; - end; - $$; diff --git a/services/migrations-schema/migrations/00057_recreate-vod_create.sql b/services/migrations-schema/migrations/00057_recreate-vod_create.sql deleted file mode 100644 index 592690b..0000000 --- a/services/migrations-schema/migrations/00057_recreate-vod_create.sql +++ /dev/null @@ -1,8 +0,0 @@ - --- we need to re-create this because I think a CASCADE deleted it. --- when a vod is created, we add a 'record' job in graphile-worker -CREATE TRIGGER vod_create - AFTER INSERT ON api.vods - FOR EACH ROW - EXECUTE PROCEDURE public.tg__add_record_job('record'); - diff --git a/services/migrations-schema/migrations/00058_conditionally-change-vod-status.sql b/services/migrations-schema/migrations/00058_conditionally-change-vod-status.sql deleted file mode 100644 index 7ad02b2..0000000 --- a/services/migrations-schema/migrations/00058_conditionally-change-vod-status.sql +++ /dev/null @@ -1,19 +0,0 @@ --- we want this function to set the vod status to recording -CREATE OR REPLACE FUNCTION tg__update_vod_updated_at() -RETURNS TRIGGER AS $$ -BEGIN - UPDATE api.vods - SET - updated_at = NOW(), - status = CASE - WHEN NEW.filesize > OLD.filesize THEN 'recording' - ELSE status - END - WHERE id IN ( - SELECT vod_id - FROM segments_vod_links - WHERE segment_id = NEW.id - ); - RETURN NEW; -END; -$$ LANGUAGE plpgsql; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00059_fix-filesize-typo.sql b/services/migrations-schema/migrations/00059_fix-filesize-typo.sql deleted file mode 100644 index 3650c24..0000000 --- a/services/migrations-schema/migrations/00059_fix-filesize-typo.sql +++ /dev/null @@ -1,20 +0,0 @@ --- we want this function to set the vod status to recording --- the correct row is bytes, not filesize. -CREATE OR REPLACE FUNCTION tg__update_vod_updated_at() -RETURNS TRIGGER AS $$ -BEGIN - UPDATE api.vods - SET - updated_at = NOW(), - status = CASE - WHEN NEW.bytes > OLD.bytes THEN 'recording' - ELSE status - END - WHERE id IN ( - SELECT vod_id - FROM segments_vod_links - WHERE segment_id = NEW.id - ); - RETURN NEW; -END; -$$ LANGUAGE plpgsql; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00060_create-recordings-table.sql b/services/migrations-schema/migrations/00060_create-recordings-table.sql deleted file mode 100644 index 0691e22..0000000 --- a/services/migrations-schema/migrations/00060_create-recordings-table.sql +++ /dev/null @@ -1,35 +0,0 @@ --- recordings table schema -CREATE TABLE api.recordings ( - id UUID PRIMARY KEY DEFAULT gen_random_uuid(), - url TEXT NOT NULL, - created_at timestamp(6) without time zone, - updated_at timestamp(6) without time zone, - discord_message_id TEXT -); - --- roles & permissions for our backend automation user -GRANT all ON api.recordings TO automation; -GRANT SELECT ON api.recordings TO web_anon; - - --- we re-create this function to use recording_id instead of vod_id -DROP FUNCTION public.tg__add_record_job CASCADE; -CREATE FUNCTION public.tg__add_record_job() RETURNS trigger - LANGUAGE plpgsql SECURITY DEFINER - SET search_path TO 'pg_catalog', 'public', 'pg_temp' - AS $$ - begin - PERFORM graphile_worker.add_job('record', json_build_object( - 'url', NEW.url, - 'recording_id', NEW.id - ), max_attempts := 6); - return NEW; - end; - $$; - - - -CREATE TRIGGER create_recording - AFTER UPDATE ON api.recordings - FOR EACH ROW - EXECUTE FUNCTION tg__add_record_job(); diff --git a/services/migrations-schema/migrations/00061_defaults-for-created_at-and-updated_at.sql b/services/migrations-schema/migrations/00061_defaults-for-created_at-and-updated_at.sql deleted file mode 100644 index 34e7eda..0000000 --- a/services/migrations-schema/migrations/00061_defaults-for-created_at-and-updated_at.sql +++ /dev/null @@ -1,6 +0,0 @@ - -ALTER TABLE IF EXISTS api.recordings - ALTER COLUMN created_at SET DEFAULT now(); - -ALTER TABLE IF EXISTS api.recordings - ALTER COLUMN updated_at SET DEFAULT now(); \ No newline at end of file diff --git a/services/migrations-schema/migrations/00062_create-discord_interactions.sql b/services/migrations-schema/migrations/00062_create-discord_interactions.sql deleted file mode 100644 index 86cfc57..0000000 --- a/services/migrations-schema/migrations/00062_create-discord_interactions.sql +++ /dev/null @@ -1,8 +0,0 @@ --- create discord_interactions table -CREATE TABLE api.discord_interactions ( - id UUID PRIMARY KEY DEFAULT gen_random_uuid(), - discord_message_id TEXT NOT NULL -); -GRANT all ON api.discord_interactions TO automation; -GRANT SELECT ON api.discord_interactions TO web_anon; - diff --git a/services/migrations-schema/migrations/00063_relate-discord_interactions-with-recordings.sql b/services/migrations-schema/migrations/00063_relate-discord_interactions-with-recordings.sql deleted file mode 100644 index 3c1b685..0000000 --- a/services/migrations-schema/migrations/00063_relate-discord_interactions-with-recordings.sql +++ /dev/null @@ -1,14 +0,0 @@ - --- add more cols to api.recordings -ALTER TABLE api.recordings - ADD COLUMN date TIMESTAMP(6) WITHOUT TIME ZONE; - -ALTER TABLE api.recordings - ADD COLUMN vod_id UUID REFERENCES api.vods(id); - -ALTER TABLE api.recordings - DROP COLUMN discord_message_id; - -ALTER TABLE api.recordings - ADD COLUMN discord_interaction_id UUID REFERENCES api.discord_interactions(id); - diff --git a/services/migrations-schema/migrations/00064_execute-procedure-instead-of-function.sql b/services/migrations-schema/migrations/00064_execute-procedure-instead-of-function.sql deleted file mode 100644 index 53e7ad2..0000000 --- a/services/migrations-schema/migrations/00064_execute-procedure-instead-of-function.sql +++ /dev/null @@ -1,8 +0,0 @@ - -DROP TRIGGER create_recording ON api.recordings; - -CREATE TRIGGER recording_create - AFTER INSERT ON api.recordings - FOR EACH ROW - EXECUTE PROCEDURE public.tg__add_record_job('record'); - diff --git a/services/migrations-schema/migrations/00065_move-is_aborted-to-recordings.sql b/services/migrations-schema/migrations/00065_move-is_aborted-to-recordings.sql deleted file mode 100644 index d8bcf37..0000000 --- a/services/migrations-schema/migrations/00065_move-is_aborted-to-recordings.sql +++ /dev/null @@ -1,6 +0,0 @@ --- move is_aborted COL from vods to recordings -ALTER TABLE api.vods - DROP COLUMN is_recording_aborted; - -ALTER TABLE api.recordings - ADD COLUMN is_aborted BOOLEAN NOT NULL DEFAULT FALSE; diff --git a/services/migrations-schema/migrations/00066_add-segments-fk-to-vods.sql b/services/migrations-schema/migrations/00066_add-segments-fk-to-vods.sql deleted file mode 100644 index 90c2010..0000000 --- a/services/migrations-schema/migrations/00066_add-segments-fk-to-vods.sql +++ /dev/null @@ -1,3 +0,0 @@ - -ALTER TABLE api.vods - ADD COLUMN segments UUID REFERENCES api.segments(id); \ No newline at end of file diff --git a/services/migrations-schema/migrations/00067_remove-duplicate-vods-segments.sql b/services/migrations-schema/migrations/00067_remove-duplicate-vods-segments.sql deleted file mode 100644 index b063584..0000000 --- a/services/migrations-schema/migrations/00067_remove-duplicate-vods-segments.sql +++ /dev/null @@ -1,29 +0,0 @@ -/* - -we already have a many-to-one relation. this col is creating a one-to-many relation, causing the following problem. - -{ - "code": "PGRST201", - "details": [ - { - "cardinality": "one-to-many", - "embedding": "segments with vods", - "relationship": "vods_segments_fkey using segments(id) and vods(segments)" - }, - { - "cardinality": "many-to-one", - "embedding": "segments with vods", - "relationship": "segments_vod_id_fkey using segments(vod_id) and vods(id)" - } - ], - "hint": "Try changing 'vods' to one of the following: 'vods!vods_segments_fkey', 'vods!segments_vod_id_fkey'. Find the desired relationship in the 'details' key.", - "message": "Could not embed because more than one relationship was found for 'segments' and 'vods'" -} - -*/ - - - - -ALTER TABLE api.vods - DROP COLUMN segments; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00068_remove-vod_id-from-recordings.sql b/services/migrations-schema/migrations/00068_remove-vod_id-from-recordings.sql deleted file mode 100644 index e3aa3f1..0000000 --- a/services/migrations-schema/migrations/00068_remove-vod_id-from-recordings.sql +++ /dev/null @@ -1,27 +0,0 @@ -/* - -we already have a many-to-one relation. this col is creating a one-to-many relation, causing the following problem. - -{ - "code": "PGRST201", - "details": [ - { - "cardinality": "one-to-many", - "embedding": "segments with vods", - "relationship": "vods_segments_fkey using segments(id) and vods(segments)" - }, - { - "cardinality": "many-to-one", - "embedding": "segments with vods", - "relationship": "segments_vod_id_fkey using segments(vod_id) and vods(id)" - } - ], - "hint": "Try changing 'vods' to one of the following: 'vods!vods_segments_fkey', 'vods!segments_vod_id_fkey'. Find the desired relationship in the 'details' key.", - "message": "Could not embed because more than one relationship was found for 'segments' and 'vods'" -} - -*/ - - -ALTER TABLE api.recordings - DROP COLUMN vod_id; diff --git a/services/migrations-schema/migrations/00069_recreate-recordings-vod-relationship.sql b/services/migrations-schema/migrations/00069_recreate-recordings-vod-relationship.sql deleted file mode 100644 index 7d8a60a..0000000 --- a/services/migrations-schema/migrations/00069_recreate-recordings-vod-relationship.sql +++ /dev/null @@ -1,3 +0,0 @@ - -ALTER TABLE api.recordings - ADD COLUMN vod_id UUID REFERENCES api.vods(id); diff --git a/services/migrations-schema/migrations/00070_relate-vod-to-recording.sql b/services/migrations-schema/migrations/00070_relate-vod-to-recording.sql deleted file mode 100644 index e8ad16a..0000000 --- a/services/migrations-schema/migrations/00070_relate-vod-to-recording.sql +++ /dev/null @@ -1,3 +0,0 @@ - -ALTER TABLE api.vods - ADD COLUMN recording_id UUID REFERENCES api.recordings(id); diff --git a/services/migrations-schema/migrations/00071_remove-recordings_vod_id_fkey.sql b/services/migrations-schema/migrations/00071_remove-recordings_vod_id_fkey.sql deleted file mode 100644 index d4f63da..0000000 --- a/services/migrations-schema/migrations/00071_remove-recordings_vod_id_fkey.sql +++ /dev/null @@ -1,4 +0,0 @@ - --- potentially unecessary fk, because api.vods has the relation too. -ALTER TABLE api.recordings - DROP COLUMN vod_id; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00072_create-builds-table.sql b/services/migrations-schema/migrations/00072_create-builds-table.sql deleted file mode 100644 index 92bd578..0000000 --- a/services/migrations-schema/migrations/00072_create-builds-table.sql +++ /dev/null @@ -1,33 +0,0 @@ --- builds table schema -CREATE TABLE api.builds ( - id UUID PRIMARY KEY DEFAULT gen_random_uuid(), - vod UUID NOT NULL REFERENCES api.vods(id), - task TEXT NOT NULL, - created_at timestamp(6) without time zone, - updated_at timestamp(6) without time zone -); - --- roles & permissions for our backend automation user -GRANT all ON api.builds TO automation; -GRANT SELECT ON api.builds TO web_anon; - - --- trigger function for starting the appropriate task when a new api.builds row is added -CREATE FUNCTION public.tg__add_build_job() RETURNS trigger - LANGUAGE plpgsql SECURITY DEFINER - SET search_path TO 'pg_catalog', 'public', 'pg_temp' - AS $$ - begin - PERFORM graphile_worker.add_job(NEW.task, json_build_object( - 'vod', NEW.vod - ), max_attempts := 6); - return NEW; - end; - $$; - - - -CREATE TRIGGER create_build - AFTER UPDATE ON api.builds - FOR EACH ROW - EXECUTE FUNCTION tg__add_build_job(); diff --git a/services/migrations-schema/migrations/00073_update-builds-table-timestamps.sql b/services/migrations-schema/migrations/00073_update-builds-table-timestamps.sql deleted file mode 100644 index eb3f060..0000000 --- a/services/migrations-schema/migrations/00073_update-builds-table-timestamps.sql +++ /dev/null @@ -1,14 +0,0 @@ - - - -ALTER TABLE api.builds - ALTER COLUMN created_at SET DEFAULT now(); - -ALTER TABLE api.builds - ALTER COLUMN updated_at SET DEFAULT now(); - -CREATE TRIGGER update_builds - BEFORE UPDATE ON api.builds - FOR EACH ROW - EXECUTE PROCEDURE moddatetime(updated_at); - diff --git a/services/migrations-schema/migrations/00074_switch-after-update-to-after-insert.sql b/services/migrations-schema/migrations/00074_switch-after-update-to-after-insert.sql deleted file mode 100644 index 30d0bfe..0000000 --- a/services/migrations-schema/migrations/00074_switch-after-update-to-after-insert.sql +++ /dev/null @@ -1,7 +0,0 @@ --- fixing a mistake. I accidentally did AFTER UPDATE when I wanted AFTER INSERT. -DROP TRIGGER create_build ON api.builds; - -CREATE TRIGGER create_build - AFTER INSERT ON api.builds - FOR EACH ROW - EXECUTE FUNCTION tg__add_build_job(); diff --git a/services/migrations-schema/migrations/00075_add-checksum-to-segments.sql b/services/migrations-schema/migrations/00075_add-checksum-to-segments.sql deleted file mode 100644 index 609c92d..0000000 --- a/services/migrations-schema/migrations/00075_add-checksum-to-segments.sql +++ /dev/null @@ -1,2 +0,0 @@ -ALTER TABLE api.segments - ADD COLUMN checksum TEXT; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00076_builds-use-vod_id-instead-of-vod.sql b/services/migrations-schema/migrations/00076_builds-use-vod_id-instead-of-vod.sql deleted file mode 100644 index e0ec59e..0000000 --- a/services/migrations-schema/migrations/00076_builds-use-vod_id-instead-of-vod.sql +++ /dev/null @@ -1,8 +0,0 @@ - --- for consistency, using vod_id instead of vod - -ALTER TABLE api.builds - DROP COLUMN vod; - -ALTER TABLE api.builds - ADD COLUMN vod_id UUID NOT NULL REFERENCES api.vods(id); \ No newline at end of file diff --git a/services/migrations-schema/migrations/00077_builds-trigger-change-vod-to-vod_id.sql b/services/migrations-schema/migrations/00077_builds-trigger-change-vod-to-vod_id.sql deleted file mode 100644 index d52cf50..0000000 --- a/services/migrations-schema/migrations/00077_builds-trigger-change-vod-to-vod_id.sql +++ /dev/null @@ -1,22 +0,0 @@ - --- we've renamed api.builds.vod to api.builds.vod_id -DROP FUNCTION public.tg__add_build_job CASCADE; - - -CREATE FUNCTION public.tg__add_build_job() RETURNS trigger - LANGUAGE plpgsql SECURITY DEFINER - SET search_path TO 'pg_catalog', 'public', 'pg_temp' - AS $$ - begin - PERFORM graphile_worker.add_job(NEW.task, json_build_object( - 'vod_id', NEW.vod_id - ), max_attempts := 6); - return NEW; - end; - $$; - - -CREATE TRIGGER create_build - AFTER UPDATE ON api.builds - FOR EACH ROW - EXECUTE FUNCTION tg__add_build_job(); diff --git a/services/migrations-schema/migrations/00078_add-bytes_uploaded-to-segments.sql b/services/migrations-schema/migrations/00078_add-bytes_uploaded-to-segments.sql deleted file mode 100644 index 32752b4..0000000 --- a/services/migrations-schema/migrations/00078_add-bytes_uploaded-to-segments.sql +++ /dev/null @@ -1,2 +0,0 @@ -ALTER TABLE api.segments - ADD COLUMN bytes_uploaded BIGINT DEFAULT 0; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00079_stream_id-optional.sql b/services/migrations-schema/migrations/00079_stream_id-optional.sql deleted file mode 100644 index 3d0a395..0000000 --- a/services/migrations-schema/migrations/00079_stream_id-optional.sql +++ /dev/null @@ -1,9 +0,0 @@ - --- data migrations are really difficult if api.vods.stream_id is required. Thus we make NULL allowed on stream_id - -ALTER TABLE api.vods - DROP COLUMN stream_id; - -ALTER TABLE api.vods - ADD COLUMN stream_id UUID REFERENCES api.streams(id); - diff --git a/services/migrations-schema/migrations/00080_add-created_at-to-vtubers.sql b/services/migrations-schema/migrations/00080_add-created_at-to-vtubers.sql deleted file mode 100644 index d43464c..0000000 --- a/services/migrations-schema/migrations/00080_add-created_at-to-vtubers.sql +++ /dev/null @@ -1,11 +0,0 @@ -ALTER TABLE api.vtubers - ADD COLUMN created_at TIMESTAMP(6) WITHOUT TIME ZONE; - -ALTER TABLE api.vtubers - ADD COLUMN updated_at TIMESTAMP(6) WITHOUT TIME ZONE; - -ALTER TABLE api.vtubers - ALTER COLUMN created_at SET DEFAULT now(); - -ALTER TABLE api.vtubers - ALTER COLUMN updated_at SET DEFAULT now(); \ No newline at end of file diff --git a/services/migrations-schema/migrations/00081_add-id_deprecated-to-vods.sql b/services/migrations-schema/migrations/00081_add-id_deprecated-to-vods.sql deleted file mode 100644 index e863384..0000000 --- a/services/migrations-schema/migrations/00081_add-id_deprecated-to-vods.sql +++ /dev/null @@ -1,3 +0,0 @@ --- for temporary data migration purposes, we add api.vods.id_deprecated which is an int and thus compatible with Strapi's IDs that we are importing -ALTER TABLE api.vtubers - ADD COLUMN id_deprecated int; diff --git a/services/migrations-schema/migrations/00082_add-id_deprecated-to-s3_files.sql b/services/migrations-schema/migrations/00082_add-id_deprecated-to-s3_files.sql deleted file mode 100644 index 4c778a3..0000000 --- a/services/migrations-schema/migrations/00082_add-id_deprecated-to-s3_files.sql +++ /dev/null @@ -1,3 +0,0 @@ --- for temporary data migration purposes, we add api.s3_files.id_deprecated which is an int and thus compatible with Strapi's IDs that we are importing -ALTER TABLE api.s3_files - ADD COLUMN id_deprecated int; diff --git a/services/migrations-schema/migrations/00083_create_vods_s3_join.sql b/services/migrations-schema/migrations/00083_create_vods_s3_join.sql deleted file mode 100644 index 735e8aa..0000000 --- a/services/migrations-schema/migrations/00083_create_vods_s3_join.sql +++ /dev/null @@ -1,5 +0,0 @@ -CREATE TABLE api.vods_s3_file_join ( - id uuid PRIMARY KEY DEFAULT gen_random_uuid(), - vod_id uuid, - s3_file_id uuid -); \ No newline at end of file diff --git a/services/migrations-schema/migrations/00084_add-file_id-to-s3_files.sql b/services/migrations-schema/migrations/00084_add-file_id-to-s3_files.sql deleted file mode 100644 index fa08332..0000000 --- a/services/migrations-schema/migrations/00084_add-file_id-to-s3_files.sql +++ /dev/null @@ -1,2 +0,0 @@ -ALTER TABLE api.s3_files - ADD COLUMN file_id TEXT; diff --git a/services/migrations-schema/migrations/00085_sync-s3_files.sql b/services/migrations-schema/migrations/00085_sync-s3_files.sql deleted file mode 100644 index 94493cf..0000000 --- a/services/migrations-schema/migrations/00085_sync-s3_files.sql +++ /dev/null @@ -1,9 +0,0 @@ --- undo last because it was redundant. we don't need file_id because the same data is in s3_id -ALTER TABLE api.s3_files - DROP COLUMN file_id; - --- add created_at and updated_at to match strapi data -ALTER TABLE api.s3_files - ADD COLUMN created_at timestamp(6) without time zone; -ALTER TABLE api.s3_files - ADD COLUMN updated_at timestamp(6) without time zone; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00086_add-cdn_url-to-s3_files.sql b/services/migrations-schema/migrations/00086_add-cdn_url-to-s3_files.sql deleted file mode 100644 index db2a82d..0000000 --- a/services/migrations-schema/migrations/00086_add-cdn_url-to-s3_files.sql +++ /dev/null @@ -1,2 +0,0 @@ -ALTER TABLE api.s3_files - ADD COLUMN cdn_url TEXT; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00087_remove-id_deprecated.sql b/services/migrations-schema/migrations/00087_remove-id_deprecated.sql deleted file mode 100644 index 1c73d22..0000000 --- a/services/migrations-schema/migrations/00087_remove-id_deprecated.sql +++ /dev/null @@ -1,2 +0,0 @@ -ALTER TABLE api.s3_files - DROP COLUMN id_deprecated; diff --git a/services/migrations-schema/migrations/00088_remove-id_deprecated-from-vtubers.sql b/services/migrations-schema/migrations/00088_remove-id_deprecated-from-vtubers.sql deleted file mode 100644 index 30cb5cb..0000000 --- a/services/migrations-schema/migrations/00088_remove-id_deprecated-from-vtubers.sql +++ /dev/null @@ -1,2 +0,0 @@ -ALTER TABLE api.vtubers - DROP COLUMN id_deprecated; diff --git a/services/migrations-schema/migrations/00089_create-vods_s3_files_join.sql b/services/migrations-schema/migrations/00089_create-vods_s3_files_join.sql deleted file mode 100644 index 1d7e59b..0000000 --- a/services/migrations-schema/migrations/00089_create-vods_s3_files_join.sql +++ /dev/null @@ -1,10 +0,0 @@ --- vods_s3_files_join table schema -CREATE TABLE api.vods_s3_files_join ( - id UUID, - vod_id UUID, - s3_file UUID -); - --- roles & permissions -GRANT all ON api.vods_s3_files_join TO automation; -GRANT SELECT ON api.vods_s3_files_join TO web_anon; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00090_rename-vods-s3-files-joins.sql b/services/migrations-schema/migrations/00090_rename-vods-s3-files-joins.sql deleted file mode 100644 index a154599..0000000 --- a/services/migrations-schema/migrations/00090_rename-vods-s3-files-joins.sql +++ /dev/null @@ -1,19 +0,0 @@ --- we are using joins instead of join because it matches all other table names being plural - --- drop table -DROP TABLE api.vods_s3_files_join CASCADE; - - --- vods_s3_files_joins table schema -CREATE TABLE api.vods_s3_files_joins ( - id UUID, - id_old INT, - vod_id UUID, - s3_file UUID -); - --- roles & permissions -GRANT all ON api.vods_s3_files_joins TO automation; -GRANT SELECT ON api.vods_s3_files_joins TO web_anon; - - diff --git a/services/migrations-schema/migrations/00091_drop-vods_s3_file_join.sql b/services/migrations-schema/migrations/00091_drop-vods_s3_file_join.sql deleted file mode 100644 index d56d214..0000000 --- a/services/migrations-schema/migrations/00091_drop-vods_s3_file_join.sql +++ /dev/null @@ -1,2 +0,0 @@ --- drop table -DROP TABLE api.vods_s3_file_join CASCADE; diff --git a/services/migrations-schema/migrations/00092_use-s3_file_id.sql b/services/migrations-schema/migrations/00092_use-s3_file_id.sql deleted file mode 100644 index d852ed4..0000000 --- a/services/migrations-schema/migrations/00092_use-s3_file_id.sql +++ /dev/null @@ -1,5 +0,0 @@ -ALTER TABLE api.vods_s3_files_joins - DROP COLUMN s3_file; - -ALTER TABLE api.vods_s3_files_joins - ADD COLUMN s3_file_id UUID; diff --git a/services/migrations-schema/migrations/00093_enable-pg_trgm.sql b/services/migrations-schema/migrations/00093_enable-pg_trgm.sql deleted file mode 100644 index c77035b..0000000 --- a/services/migrations-schema/migrations/00093_enable-pg_trgm.sql +++ /dev/null @@ -1,2 +0,0 @@ --- enabling pg_trgm for drupal -CREATE EXTENSION IF NOT EXISTS pg_trgm; diff --git a/services/migrations-schema/migrations/00094_add-patrons-table.sql b/services/migrations-schema/migrations/00094_add-patrons-table.sql deleted file mode 100644 index 44a4865..0000000 --- a/services/migrations-schema/migrations/00094_add-patrons-table.sql +++ /dev/null @@ -1,7 +0,0 @@ -CREATE TABLE api.patrons ( - id uuid PRIMARY KEY DEFAULT gen_random_uuid(), - full_name TEXT NOT NULL, - link TEXT -); -GRANT all ON api.patrons TO automation; -GRANT SELECT ON api.patrons TO web_anon; diff --git a/services/migrations-schema/migrations/00095_create-contributors.sql b/services/migrations-schema/migrations/00095_create-contributors.sql deleted file mode 100644 index 1a6b312..0000000 --- a/services/migrations-schema/migrations/00095_create-contributors.sql +++ /dev/null @@ -1,7 +0,0 @@ -CREATE TABLE api.contributors ( - id uuid PRIMARY KEY DEFAULT gen_random_uuid(), - full_name TEXT NOT NULL, - link TEXT -); -GRANT all ON api.contributors TO automation; -GRANT SELECT ON api.contributors TO web_anon; diff --git a/services/migrations-schema/migrations/00096_add-id_num-to-s3_files.sql b/services/migrations-schema/migrations/00096_add-id_num-to-s3_files.sql deleted file mode 100644 index 17dec9c..0000000 --- a/services/migrations-schema/migrations/00096_add-id_num-to-s3_files.sql +++ /dev/null @@ -1,4 +0,0 @@ --- for strapi to postgrest migration. this col is meant to be deleted later - -ALTER TABLE IF EXISTS api.s3_files - ADD COLUMN IF NOT EXISTS id_num INT; diff --git a/services/migrations-schema/migrations/00097_add-id_num-to-mux-assets.sql b/services/migrations-schema/migrations/00097_add-id_num-to-mux-assets.sql deleted file mode 100644 index 0a41bc8..0000000 --- a/services/migrations-schema/migrations/00097_add-id_num-to-mux-assets.sql +++ /dev/null @@ -1,4 +0,0 @@ --- for strapi to postgrest migration. this col is meant to be deleted later - -ALTER TABLE IF EXISTS api.mux_assets - ADD COLUMN IF NOT EXISTS id_num INT; diff --git a/services/migrations-schema/migrations/00098_add-id_num-to-streams.sql b/services/migrations-schema/migrations/00098_add-id_num-to-streams.sql deleted file mode 100644 index 226ae11..0000000 --- a/services/migrations-schema/migrations/00098_add-id_num-to-streams.sql +++ /dev/null @@ -1,6 +0,0 @@ --- for strapi to postgrest migration. this col is meant to be deleted later -ALTER TABLE IF EXISTS api.streams - ADD COLUMN IF NOT EXISTS id_num INT; - -ALTER TABLE IF EXISTS api.vtubers - ADD COLUMN IF NOT EXISTS id_num INT; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00099_add-vtuber_num.sql b/services/migrations-schema/migrations/00099_add-vtuber_num.sql deleted file mode 100644 index d2e3ee4..0000000 --- a/services/migrations-schema/migrations/00099_add-vtuber_num.sql +++ /dev/null @@ -1,6 +0,0 @@ --- adding an INT col to make it compatible with the strapi migration --- this is meant to be deleted after migration to UUID - - -ALTER TABLE IF EXISTS api.streams - ADD COLUMN IF NOT EXISTS vtuber_num INT; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00100_add-id_num-to-tags.sql b/services/migrations-schema/migrations/00100_add-id_num-to-tags.sql deleted file mode 100644 index 226a049..0000000 --- a/services/migrations-schema/migrations/00100_add-id_num-to-tags.sql +++ /dev/null @@ -1,2 +0,0 @@ -ALTER TABLE IF EXISTS api.tags - ADD COLUMN IF NOT EXISTS id_num INT; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00101_add-missing-cols-to-tags.sql b/services/migrations-schema/migrations/00101_add-missing-cols-to-tags.sql deleted file mode 100644 index 78d8103..0000000 --- a/services/migrations-schema/migrations/00101_add-missing-cols-to-tags.sql +++ /dev/null @@ -1,11 +0,0 @@ -ALTER TABLE IF EXISTS api.tags - ADD COLUMN IF NOT EXISTS created_at TIMESTAMP WITHOUT TIME ZONE; - -ALTER TABLE IF EXISTS api.tags - ADD COLUMN IF NOT EXISTS updated_at TIMESTAMP WITHOUT TIME ZONE; - -ALTER TABLE IF EXISTS api.tags - ADD COLUMN IF NOT EXISTS created_by_id INT; - -ALTER TABLE IF EXISTS api.tags - ADD COLUMN IF NOT EXISTS updated_by_id INT; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00102_add-toy_id_num-to-tags.sql b/services/migrations-schema/migrations/00102_add-toy_id_num-to-tags.sql deleted file mode 100644 index 6314af2..0000000 --- a/services/migrations-schema/migrations/00102_add-toy_id_num-to-tags.sql +++ /dev/null @@ -1,3 +0,0 @@ - -ALTER TABLE IF EXISTS api.tags - ADD COLUMN IF NOT EXISTS toy_id_num INT; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00103_add-toy_num.sql b/services/migrations-schema/migrations/00103_add-toy_num.sql deleted file mode 100644 index e0b7524..0000000 --- a/services/migrations-schema/migrations/00103_add-toy_num.sql +++ /dev/null @@ -1,21 +0,0 @@ - -ALTER TABLE IF EXISTS api.toys - ADD COLUMN IF NOT EXISTS toy_id_num INT; - -ALTER TABLE IF EXISTS api.toys - ADD COLUMN IF NOT EXISTS created_at TIMESTAMP WITHOUT TIME ZONE; - -ALTER TABLE IF EXISTS api.toys - ADD COLUMN IF NOT EXISTS updated_at TIMESTAMP WITHOUT TIME ZONE; - -ALTER TABLE IF EXISTS api.toys - ADD COLUMN IF NOT EXISTS created_by_id_num INT; - -ALTER TABLE IF EXISTS api.toys - ADD COLUMN IF NOT EXISTS updated_by_id_num INT; - -ALTER TABLE IF EXISTS api.toys - ADD COLUMN IF NOT EXISTS created_by_id UUID; - -ALTER TABLE IF EXISTS api.toys - ADD COLUMN IF NOT EXISTS updated_by_id UUID; diff --git a/services/migrations-schema/migrations/00104_add-toy_num-fix.sql b/services/migrations-schema/migrations/00104_add-toy_num-fix.sql deleted file mode 100644 index cb461e0..0000000 --- a/services/migrations-schema/migrations/00104_add-toy_num-fix.sql +++ /dev/null @@ -1,7 +0,0 @@ - - -ALTER TABLE IF EXISTS api.toys - DROP COLUMN IF EXISTS toy_id_num; - -ALTER TABLE IF EXISTS api.toys - ADD COLUMN IF NOT EXISTS id_num INT; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00105_add-id_num-to-vods_links.sql b/services/migrations-schema/migrations/00105_add-id_num-to-vods_links.sql deleted file mode 100644 index 4797c5a..0000000 --- a/services/migrations-schema/migrations/00105_add-id_num-to-vods_links.sql +++ /dev/null @@ -1,18 +0,0 @@ - -ALTER TABLE IF EXISTS api.vods_mux_asset_links - ADD COLUMN IF NOT EXISTS id_num INT; - -ALTER TABLE IF EXISTS api.vods_stream_links - ADD COLUMN IF NOT EXISTS id_num INT; - -ALTER TABLE IF EXISTS api.vods_thumbnail_links - ADD COLUMN IF NOT EXISTS id_num INT; - -ALTER TABLE IF EXISTS api.vods_uploader_links - ADD COLUMN IF NOT EXISTS id_num INT; - -ALTER TABLE IF EXISTS api.vods_video_src_b_2_links - ADD COLUMN IF NOT EXISTS id_num INT; - -ALTER TABLE IF EXISTS api.vods_vtuber_links - ADD COLUMN IF NOT EXISTS id_num INT; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00106_add-id_num-to-vods.sql b/services/migrations-schema/migrations/00106_add-id_num-to-vods.sql deleted file mode 100644 index 61bacf8..0000000 --- a/services/migrations-schema/migrations/00106_add-id_num-to-vods.sql +++ /dev/null @@ -1,2 +0,0 @@ -ALTER TABLE IF EXISTS api.vods - ADD COLUMN IF NOT EXISTS id_num INT; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00107_remove-uuid.sql b/services/migrations-schema/migrations/00107_remove-uuid.sql deleted file mode 100644 index 002fee2..0000000 --- a/services/migrations-schema/migrations/00107_remove-uuid.sql +++ /dev/null @@ -1,125 +0,0 @@ --- I was planning on migrating strapi database to postgrest and also migrating from INT to UUID. --- Doing both at once proved far too complex, so I'm reverting the schmea to use INT ids in order to migrate. --- Once that is done we can switch to uuids, but the two steps should be separate - -DO $$ -BEGIN - -- api.builds - ALTER TABLE api.builds DROP COLUMN id CASCADE; - ALTER TABLE api.builds ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - ALTER TABLE api.builds DROP COLUMN vod_id CASCADE; - ALTER TABLE api.builds ADD COLUMN vod_id INT; - - -- api.contributors - ALTER TABLE api.contributors DROP COLUMN id CASCADE; - ALTER TABLE api.contributors ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - - -- api.discord_interactions - ALTER TABLE api.discord_interactions DROP COLUMN id CASCADE; - ALTER TABLE api.discord_interactions ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - - -- api.mux_assets - ALTER TABLE api.mux_assets DROP COLUMN id CASCADE; - ALTER TABLE api.mux_assets ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - - -- api.patrons - ALTER TABLE api.patrons DROP COLUMN id CASCADE; - ALTER TABLE api.patrons ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - - -- api.recordings - ALTER TABLE api.recordings DROP COLUMN id CASCADE; - ALTER TABLE api.recordings ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - ALTER TABLE api.recordings DROP COLUMN discord_interaction_id CASCADE; - ALTER TABLE api.recordings ADD COLUMN discord_interaction_id INT; - - -- api.s3_files - ALTER TABLE api.s3_files DROP COLUMN id CASCADE; - ALTER TABLE api.s3_files ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - - -- api.segments - ALTER TABLE api.segments DROP COLUMN id CASCADE; - ALTER TABLE api.segments ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - ALTER TABLE api.segments DROP COLUMN vod_id CASCADE; - ALTER TABLE api.segments ADD COLUMN vod_id INT; - - -- api.segments_vod_links - ALTER TABLE api.segments_vod_links DROP COLUMN id CASCADE; - ALTER TABLE api.segments_vod_links ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - ALTER TABLE api.segments_vod_links DROP COLUMN segment_id CASCADE; - ALTER TABLE api.segments_vod_links ADD COLUMN segment_id INT; - ALTER TABLE api.segments_vod_links DROP COLUMN vod_id CASCADE; - ALTER TABLE api.segments_vod_links ADD COLUMN vod_id INT; - - -- api.streams - ALTER TABLE api.streams DROP COLUMN id CASCADE; - ALTER TABLE api.streams ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - ALTER TABLE api.streams DROP COLUMN vtuber CASCADE; - ALTER TABLE api.streams ADD COLUMN vtuber INT; - - -- api.tags - ALTER TABLE api.tags DROP COLUMN id CASCADE; - ALTER TABLE api.tags ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - ALTER TABLE api.tags DROP COLUMN toy_id CASCADE; - ALTER TABLE api.tags ADD COLUMN toy_id INT; - - -- api.tags_vods - ALTER TABLE api.tags_vods DROP COLUMN id CASCADE; - ALTER TABLE api.tags_vods ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - ALTER TABLE api.tags_vods DROP COLUMN tag_id CASCADE; - ALTER TABLE api.tags_vods ADD COLUMN tag_id INT; - ALTER TABLE api.tags_vods DROP COLUMN vod_id CASCADE; - ALTER TABLE api.tags_vods ADD COLUMN vod_id INT; - - -- api.toys - ALTER TABLE api.toys DROP COLUMN id CASCADE; - ALTER TABLE api.toys ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - ALTER TABLE api.toys DROP COLUMN created_by_id CASCADE; - ALTER TABLE api.toys ADD COLUMN created_by_id INT; - ALTER TABLE api.toys DROP COLUMN updated_by_id CASCADE; - ALTER TABLE api.toys ADD COLUMN updated_by_id INT; - - -- api.toys_tags - ALTER TABLE api.toys_tags DROP COLUMN id CASCADE; - ALTER TABLE api.toys_tags ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - ALTER TABLE api.toys_tags DROP COLUMN tag_id CASCADE; - ALTER TABLE api.toys_tags ADD COLUMN tag_id INT; - ALTER TABLE api.toys_tags DROP COLUMN toy_id CASCADE; - ALTER TABLE api.toys_tags ADD COLUMN toy_id INT; - - -- api.toys_vtubers - ALTER TABLE api.toys_vtubers DROP COLUMN id CASCADE; - ALTER TABLE api.toys_vtubers ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - ALTER TABLE api.toys_vtubers DROP COLUMN toy_id CASCADE; - ALTER TABLE api.toys_vtubers ADD COLUMN toy_id INT; - ALTER TABLE api.toys_vtubers DROP COLUMN vtuber_id CASCADE; - ALTER TABLE api.toys_vtubers ADD COLUMN vtuber_id INT; - - -- api.vods - ALTER TABLE api.vods DROP COLUMN id CASCADE; - ALTER TABLE api.vods ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - ALTER TABLE api.vods DROP COLUMN mux_asset CASCADE; - ALTER TABLE api.vods ADD COLUMN mux_asset INT; - ALTER TABLE api.vods DROP COLUMN recording_id CASCADE; - ALTER TABLE api.vods ADD COLUMN recording_id INT; - ALTER TABLE api.vods DROP COLUMN s3_file CASCADE; - ALTER TABLE api.vods ADD COLUMN s3_file INT; - ALTER TABLE api.vods DROP COLUMN stream_id CASCADE; - ALTER TABLE api.vods ADD COLUMN stream_id INT; - ALTER TABLE api.vods DROP COLUMN thumbnail CASCADE; - ALTER TABLE api.vods ADD COLUMN thumbnail INT; - ALTER TABLE api.vods DROP COLUMN vtuber CASCADE; - ALTER TABLE api.vods ADD COLUMN vtuber INT; - - -- api.vods_s3_files_joins - ALTER TABLE api.vods_s3_files_joins DROP COLUMN id CASCADE; - ALTER TABLE api.vods_s3_files_joins ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - ALTER TABLE api.vods_s3_files_joins DROP COLUMN s3_file_id CASCADE; - ALTER TABLE api.vods_s3_files_joins ADD COLUMN s3_file_id INT; - ALTER TABLE api.vods_s3_files_joins DROP COLUMN vod_id CASCADE; - ALTER TABLE api.vods_s3_files_joins ADD COLUMN vod_id INT; - - -- api.vtubers - ALTER TABLE api.vtubers DROP COLUMN id CASCADE; - ALTER TABLE api.vtubers ADD COLUMN id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY; - -END $$; diff --git a/services/migrations-schema/migrations/00108_add-strapi-tables.sql b/services/migrations-schema/migrations/00108_add-strapi-tables.sql deleted file mode 100644 index 33c24b5..0000000 --- a/services/migrations-schema/migrations/00108_add-strapi-tables.sql +++ /dev/null @@ -1,47 +0,0 @@ - -CREATE TABLE IF NOT EXISTS api.streams_vtuber_links -( - id integer NOT NULL, - stream_id integer, - vtuber_id integer, - stream_order double precision, - CONSTRAINT streams_vtuber_links_pkey PRIMARY KEY (id), - CONSTRAINT streams_vtuber_links_unique UNIQUE (stream_id, vtuber_id), - CONSTRAINT streams_vtuber_links_fk FOREIGN KEY (stream_id) - REFERENCES api.streams (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT streams_vtuber_links_inv_fk FOREIGN KEY (vtuber_id) - REFERENCES api.vtubers (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.streams_vtuber_links - OWNER to postgres; --- Index: streams_vtuber_links_fk - --- DROP INDEX IF EXISTS api.streams_vtuber_links_fk; - -CREATE INDEX IF NOT EXISTS streams_vtuber_links_fk - ON api.streams_vtuber_links USING btree - (stream_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: streams_vtuber_links_inv_fk - --- DROP INDEX IF EXISTS api.streams_vtuber_links_inv_fk; - -CREATE INDEX IF NOT EXISTS streams_vtuber_links_inv_fk - ON api.streams_vtuber_links USING btree - (vtuber_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: streams_vtuber_links_order_inv_fk - --- DROP INDEX IF EXISTS api.streams_vtuber_links_order_inv_fk; - -CREATE INDEX IF NOT EXISTS streams_vtuber_links_order_inv_fk - ON api.streams_vtuber_links USING btree - (stream_order ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00109_use-consistent-description_1.sql b/services/migrations-schema/migrations/00109_use-consistent-description_1.sql deleted file mode 100644 index 3f7109d..0000000 --- a/services/migrations-schema/migrations/00109_use-consistent-description_1.sql +++ /dev/null @@ -1,5 +0,0 @@ -ALTER TABLE api.vtubers DROP COLUMN description1; -ALTER TABLE api.vtubers DROP COLUMN description2; - -ALTER TABLE api.vtubers ADD COLUMN description_1 TEXT; -ALTER TABLE api.vtubers ADD COLUMN description_2 TEXT; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00110_unseed-vtubers.sql b/services/migrations-schema/migrations/00110_unseed-vtubers.sql deleted file mode 100644 index 6af82b6..0000000 --- a/services/migrations-schema/migrations/00110_unseed-vtubers.sql +++ /dev/null @@ -1,5 +0,0 @@ --- we are removing vtuber rows because vtubers are data. --- data is not supposed to be part of schema migrations --- @see https://github.com/zakpatterson/postgres-schema-migrations?tab=readme-ov-file#schema-migrations-vs-data-migrations - -DELETE FROM api.vtubers; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00111_create-strapi-tables-tvr.sql b/services/migrations-schema/migrations/00111_create-strapi-tables-tvr.sql deleted file mode 100644 index 8bca59b..0000000 --- a/services/migrations-schema/migrations/00111_create-strapi-tables-tvr.sql +++ /dev/null @@ -1,33 +0,0 @@ - -CREATE TABLE IF NOT EXISTS api.tag_vod_relations -( - id integer NOT NULL, - votes integer, - creator_id integer, - created_at timestamp(6) without time zone, - updated_at timestamp(6) without time zone, - created_by_id integer, - updated_by_id integer, - CONSTRAINT tag_vod_relations_pkey PRIMARY KEY (id) -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.tag_vod_relations - OWNER to postgres; --- Index: tag_vod_relations_created_by_id_fk - --- DROP INDEX IF EXISTS api.tag_vod_relations_created_by_id_fk; - -CREATE INDEX IF NOT EXISTS tag_vod_relations_created_by_id_fk - ON api.tag_vod_relations USING btree - (created_by_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: tag_vod_relations_updated_by_id_fk - --- DROP INDEX IF EXISTS api.tag_vod_relations_updated_by_id_fk; - -CREATE INDEX IF NOT EXISTS tag_vod_relations_updated_by_id_fk - ON api.tag_vod_relations USING btree - (updated_by_id ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00112_create-tvr_tag_links.sql b/services/migrations-schema/migrations/00112_create-tvr_tag_links.sql deleted file mode 100644 index 48fdcb8..0000000 --- a/services/migrations-schema/migrations/00112_create-tvr_tag_links.sql +++ /dev/null @@ -1,38 +0,0 @@ - -CREATE TABLE IF NOT EXISTS api.tag_vod_relations_tag_links -( - id integer NOT NULL, - tag_vod_relation_id integer, - tag_id integer, - CONSTRAINT tag_vod_relations_tag_links_pkey PRIMARY KEY (id), - CONSTRAINT tag_vod_relations_tag_links_unique UNIQUE (tag_vod_relation_id, tag_id), - CONSTRAINT tag_vod_relations_tag_links_fk FOREIGN KEY (tag_vod_relation_id) - REFERENCES api.tag_vod_relations (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT tag_vod_relations_tag_links_inv_fk FOREIGN KEY (tag_id) - REFERENCES api.tags (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.tag_vod_relations_tag_links - OWNER to postgres; --- Index: tag_vod_relations_tag_links_fk - --- DROP INDEX IF EXISTS api.tag_vod_relations_tag_links_fk; - -CREATE INDEX IF NOT EXISTS tag_vod_relations_tag_links_fk - ON api.tag_vod_relations_tag_links USING btree - (tag_vod_relation_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: tag_vod_relations_tag_links_inv_fk - --- DROP INDEX IF EXISTS api.tag_vod_relations_tag_links_inv_fk; - -CREATE INDEX IF NOT EXISTS tag_vod_relations_tag_links_inv_fk - ON api.tag_vod_relations_tag_links USING btree - (tag_id ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00113_create-tvr_vod_links.sql b/services/migrations-schema/migrations/00113_create-tvr_vod_links.sql deleted file mode 100644 index 76abe0e..0000000 --- a/services/migrations-schema/migrations/00113_create-tvr_vod_links.sql +++ /dev/null @@ -1,47 +0,0 @@ - -CREATE TABLE IF NOT EXISTS api.tag_vod_relations_vod_links -( - id integer NOT NULL, - tag_vod_relation_id integer, - vod_id integer, - tag_vod_relation_order double precision, - CONSTRAINT tag_vod_relations_vod_links_pkey PRIMARY KEY (id), - CONSTRAINT tag_vod_relations_vod_links_unique UNIQUE (tag_vod_relation_id, vod_id), - CONSTRAINT tag_vod_relations_vod_links_fk FOREIGN KEY (tag_vod_relation_id) - REFERENCES api.tag_vod_relations (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT tag_vod_relations_vod_links_inv_fk FOREIGN KEY (vod_id) - REFERENCES api.vods (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.tag_vod_relations_vod_links - OWNER to postgres; --- Index: tag_vod_relations_vod_links_fk - --- DROP INDEX IF EXISTS api.tag_vod_relations_vod_links_fk; - -CREATE INDEX IF NOT EXISTS tag_vod_relations_vod_links_fk - ON api.tag_vod_relations_vod_links USING btree - (tag_vod_relation_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: tag_vod_relations_vod_links_inv_fk - --- DROP INDEX IF EXISTS api.tag_vod_relations_vod_links_inv_fk; - -CREATE INDEX IF NOT EXISTS tag_vod_relations_vod_links_inv_fk - ON api.tag_vod_relations_vod_links USING btree - (vod_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: tag_vod_relations_vod_links_order_inv_fk - --- DROP INDEX IF EXISTS api.tag_vod_relations_vod_links_order_inv_fk; - -CREATE INDEX IF NOT EXISTS tag_vod_relations_vod_links_order_inv_fk - ON api.tag_vod_relations_vod_links USING btree - (tag_vod_relation_order ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00114_create-tags_toy_links.sql b/services/migrations-schema/migrations/00114_create-tags_toy_links.sql deleted file mode 100644 index 70cbe39..0000000 --- a/services/migrations-schema/migrations/00114_create-tags_toy_links.sql +++ /dev/null @@ -1,47 +0,0 @@ - -CREATE TABLE IF NOT EXISTS api.tags_toy_links -( - id integer NOT NULL, - tag_id integer, - toy_id integer, - tag_order double precision, - CONSTRAINT tags_toy_links_pkey PRIMARY KEY (id), - CONSTRAINT tags_toy_links_unique UNIQUE (tag_id, toy_id), - CONSTRAINT tags_toy_links_fk FOREIGN KEY (tag_id) - REFERENCES api.tags (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT tags_toy_links_inv_fk FOREIGN KEY (toy_id) - REFERENCES api.toys (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.tags_toy_links - OWNER to postgres; --- Index: tags_toy_links_fk - --- DROP INDEX IF EXISTS api.tags_toy_links_fk; - -CREATE INDEX IF NOT EXISTS tags_toy_links_fk - ON api.tags_toy_links USING btree - (tag_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: tags_toy_links_inv_fk - --- DROP INDEX IF EXISTS api.tags_toy_links_inv_fk; - -CREATE INDEX IF NOT EXISTS tags_toy_links_inv_fk - ON api.tags_toy_links USING btree - (toy_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: tags_toy_links_order_inv_fk - --- DROP INDEX IF EXISTS api.tags_toy_links_order_inv_fk; - -CREATE INDEX IF NOT EXISTS tags_toy_links_order_inv_fk - ON api.tags_toy_links USING btree - (tag_order ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00115_create-tags_vods_links.sql b/services/migrations-schema/migrations/00115_create-tags_vods_links.sql deleted file mode 100644 index 7d7f78e..0000000 --- a/services/migrations-schema/migrations/00115_create-tags_vods_links.sql +++ /dev/null @@ -1,55 +0,0 @@ -CREATE TABLE IF NOT EXISTS api.tags_vods_links -( - id integer NOT NULL, - tag_id integer, - vod_id integer, - vod_order double precision, - tag_order double precision, - CONSTRAINT tags_vods_links_pkey PRIMARY KEY (id), - CONSTRAINT tags_vods_links_unique UNIQUE (tag_id, vod_id), - CONSTRAINT tags_vods_links_fk FOREIGN KEY (tag_id) - REFERENCES api.tags (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT tags_vods_links_inv_fk FOREIGN KEY (vod_id) - REFERENCES api.vods (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.tags_vods_links - OWNER to postgres; --- Index: tags_vods_links_fk - --- DROP INDEX IF EXISTS api.tags_vods_links_fk; - -CREATE INDEX IF NOT EXISTS tags_vods_links_fk - ON api.tags_vods_links USING btree - (tag_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: tags_vods_links_inv_fk - --- DROP INDEX IF EXISTS api.tags_vods_links_inv_fk; - -CREATE INDEX IF NOT EXISTS tags_vods_links_inv_fk - ON api.tags_vods_links USING btree - (vod_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: tags_vods_links_order_fk - --- DROP INDEX IF EXISTS api.tags_vods_links_order_fk; - -CREATE INDEX IF NOT EXISTS tags_vods_links_order_fk - ON api.tags_vods_links USING btree - (vod_order ASC NULLS LAST) - TABLESPACE pg_default; --- Index: tags_vods_links_order_inv_fk - --- DROP INDEX IF EXISTS api.tags_vods_links_order_inv_fk; - -CREATE INDEX IF NOT EXISTS tags_vods_links_order_inv_fk - ON api.tags_vods_links USING btree - (tag_order ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00116_create-timestamps.sql b/services/migrations-schema/migrations/00116_create-timestamps.sql deleted file mode 100644 index f8d1168..0000000 --- a/services/migrations-schema/migrations/00116_create-timestamps.sql +++ /dev/null @@ -1,210 +0,0 @@ --- Table: api.timestamps - --- DROP TABLE IF EXISTS api.timestamps; - -CREATE TABLE IF NOT EXISTS api.timestamps -( - id integer NOT NULL, - "time" integer, - creator_id integer, - created_at timestamp(6) without time zone, - updated_at timestamp(6) without time zone, - created_by_id integer, - updated_by_id integer, - CONSTRAINT timestamps_pkey PRIMARY KEY (id) -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.timestamps - OWNER to postgres; --- Index: timestamps_created_by_id_fk - --- DROP INDEX IF EXISTS api.timestamps_created_by_id_fk; - -CREATE INDEX IF NOT EXISTS timestamps_created_by_id_fk - ON api.timestamps USING btree - (created_by_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: timestamps_updated_by_id_fk - --- DROP INDEX IF EXISTS api.timestamps_updated_by_id_fk; - -CREATE INDEX IF NOT EXISTS timestamps_updated_by_id_fk - ON api.timestamps USING btree - (updated_by_id ASC NULLS LAST) - TABLESPACE pg_default; - - --- Table: api.timestamps - --- DROP TABLE IF EXISTS api.timestamps; - -CREATE TABLE IF NOT EXISTS api.timestamps -( - id integer NOT NULL, - "time" integer, - creator_id integer, - created_at timestamp(6) without time zone, - updated_at timestamp(6) without time zone, - created_by_id integer, - updated_by_id integer, - CONSTRAINT timestamps_pkey PRIMARY KEY (id) -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.timestamps - OWNER to postgres; --- Index: timestamps_created_by_id_fk - --- DROP INDEX IF EXISTS api.timestamps_created_by_id_fk; - -CREATE INDEX IF NOT EXISTS timestamps_created_by_id_fk - ON api.timestamps USING btree - (created_by_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: timestamps_updated_by_id_fk - --- DROP INDEX IF EXISTS api.timestamps_updated_by_id_fk; - -CREATE INDEX IF NOT EXISTS timestamps_updated_by_id_fk - ON api.timestamps USING btree - (updated_by_id ASC NULLS LAST) - TABLESPACE pg_default; - - --- Table: api.timestamps - --- DROP TABLE IF EXISTS api.timestamps; - -CREATE TABLE IF NOT EXISTS api.timestamps -( - id integer NOT NULL, - "time" integer, - creator_id integer, - created_at timestamp(6) without time zone, - updated_at timestamp(6) without time zone, - created_by_id integer, - updated_by_id integer, - CONSTRAINT timestamps_pkey PRIMARY KEY (id) -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.timestamps - OWNER to postgres; --- Index: timestamps_created_by_id_fk - --- DROP INDEX IF EXISTS api.timestamps_created_by_id_fk; - -CREATE INDEX IF NOT EXISTS timestamps_created_by_id_fk - ON api.timestamps USING btree - (created_by_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: timestamps_updated_by_id_fk - --- DROP INDEX IF EXISTS api.timestamps_updated_by_id_fk; - -CREATE INDEX IF NOT EXISTS timestamps_updated_by_id_fk - ON api.timestamps USING btree - (updated_by_id ASC NULLS LAST) - TABLESPACE pg_default; - - - --- Table: api.timestamps_upvoters_links - --- DROP TABLE IF EXISTS api.timestamps_upvoters_links; - -CREATE TABLE IF NOT EXISTS api.timestamps_upvoters_links -( - id integer NOT NULL, - timestamp_id integer, - user_id integer, - user_order double precision, - CONSTRAINT timestamps_upvoters_links_pkey PRIMARY KEY (id), - CONSTRAINT timestamps_upvoters_links_unique UNIQUE (timestamp_id, user_id), - CONSTRAINT timestamps_upvoters_links_fk FOREIGN KEY (timestamp_id) - REFERENCES api.timestamps (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.timestamps_upvoters_links - OWNER to postgres; --- Index: timestamps_upvoters_links_fk - --- DROP INDEX IF EXISTS api.timestamps_upvoters_links_fk; - -CREATE INDEX IF NOT EXISTS timestamps_upvoters_links_fk - ON api.timestamps_upvoters_links USING btree - (timestamp_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: timestamps_upvoters_links_inv_fk - --- DROP INDEX IF EXISTS api.timestamps_upvoters_links_inv_fk; - -CREATE INDEX IF NOT EXISTS timestamps_upvoters_links_inv_fk - ON api.timestamps_upvoters_links USING btree - (user_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: timestamps_upvoters_links_order_fk - --- DROP INDEX IF EXISTS api.timestamps_upvoters_links_order_fk; - -CREATE INDEX IF NOT EXISTS timestamps_upvoters_links_order_fk - ON api.timestamps_upvoters_links USING btree - (user_order ASC NULLS LAST) - TABLESPACE pg_default; - - - --- Table: api.timestamps_upvoters_links - --- DROP TABLE IF EXISTS api.timestamps_upvoters_links; - -CREATE TABLE IF NOT EXISTS api.timestamps_upvoters_links -( - id integer NOT NULL, - timestamp_id integer, - user_id integer, - user_order double precision, - CONSTRAINT timestamps_upvoters_links_pkey PRIMARY KEY (id), - CONSTRAINT timestamps_upvoters_links_unique UNIQUE (timestamp_id, user_id), - CONSTRAINT timestamps_upvoters_links_fk FOREIGN KEY (timestamp_id) - REFERENCES api.timestamps (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.timestamps_upvoters_links - OWNER to postgres; --- Index: timestamps_upvoters_links_fk - --- DROP INDEX IF EXISTS api.timestamps_upvoters_links_fk; - -CREATE INDEX IF NOT EXISTS timestamps_upvoters_links_fk - ON api.timestamps_upvoters_links USING btree - (timestamp_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: timestamps_upvoters_links_inv_fk - --- DROP INDEX IF EXISTS api.timestamps_upvoters_links_inv_fk; - -CREATE INDEX IF NOT EXISTS timestamps_upvoters_links_inv_fk - ON api.timestamps_upvoters_links USING btree - (user_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: timestamps_upvoters_links_order_fk - --- DROP INDEX IF EXISTS api.timestamps_upvoters_links_order_fk; - -CREATE INDEX IF NOT EXISTS timestamps_upvoters_links_order_fk - ON api.timestamps_upvoters_links USING btree - (user_order ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00117_create-timestamps_tag_links.sql b/services/migrations-schema/migrations/00117_create-timestamps_tag_links.sql deleted file mode 100644 index 5c16aae..0000000 --- a/services/migrations-schema/migrations/00117_create-timestamps_tag_links.sql +++ /dev/null @@ -1,41 +0,0 @@ --- Table: api.timestamps_tag_links - --- DROP TABLE IF EXISTS api.timestamps_tag_links; - -CREATE TABLE IF NOT EXISTS api.timestamps_tag_links -( - id integer NOT NULL, - timestamp_id integer, - tag_id integer, - CONSTRAINT timestamps_tag_links_pkey PRIMARY KEY (id), - CONSTRAINT timestamps_tag_links_unique UNIQUE (timestamp_id, tag_id), - CONSTRAINT timestamps_tag_links_fk FOREIGN KEY (timestamp_id) - REFERENCES api.timestamps (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT timestamps_tag_links_inv_fk FOREIGN KEY (tag_id) - REFERENCES api.tags (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.timestamps_tag_links - OWNER to postgres; --- Index: timestamps_tag_links_fk - --- DROP INDEX IF EXISTS api.timestamps_tag_links_fk; - -CREATE INDEX IF NOT EXISTS timestamps_tag_links_fk - ON api.timestamps_tag_links USING btree - (timestamp_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: timestamps_tag_links_inv_fk - --- DROP INDEX IF EXISTS api.timestamps_tag_links_inv_fk; - -CREATE INDEX IF NOT EXISTS timestamps_tag_links_inv_fk - ON api.timestamps_tag_links USING btree - (tag_id ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00118_create-timestamps_vod_links.sql b/services/migrations-schema/migrations/00118_create-timestamps_vod_links.sql deleted file mode 100644 index 9f4d198..0000000 --- a/services/migrations-schema/migrations/00118_create-timestamps_vod_links.sql +++ /dev/null @@ -1,50 +0,0 @@ --- Table: api.timestamps_vod_links - --- DROP TABLE IF EXISTS api.timestamps_vod_links; - -CREATE TABLE IF NOT EXISTS api.timestamps_vod_links -( - id integer NOT NULL, - timestamp_id integer, - vod_id integer, - timestamp_order double precision, - CONSTRAINT timestamps_vod_links_pkey PRIMARY KEY (id), - CONSTRAINT timestamps_vod_links_unique UNIQUE (timestamp_id, vod_id), - CONSTRAINT timestamps_vod_links_fk FOREIGN KEY (timestamp_id) - REFERENCES api.timestamps (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT timestamps_vod_links_inv_fk FOREIGN KEY (vod_id) - REFERENCES api.vods (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.timestamps_vod_links - OWNER to postgres; --- Index: timestamps_vod_links_fk - --- DROP INDEX IF EXISTS api.timestamps_vod_links_fk; - -CREATE INDEX IF NOT EXISTS timestamps_vod_links_fk - ON api.timestamps_vod_links USING btree - (timestamp_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: timestamps_vod_links_inv_fk - --- DROP INDEX IF EXISTS api.timestamps_vod_links_inv_fk; - -CREATE INDEX IF NOT EXISTS timestamps_vod_links_inv_fk - ON api.timestamps_vod_links USING btree - (vod_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: timestamps_vod_links_order_inv_fk - --- DROP INDEX IF EXISTS api.timestamps_vod_links_order_inv_fk; - -CREATE INDEX IF NOT EXISTS timestamps_vod_links_order_inv_fk - ON api.timestamps_vod_links USING btree - (timestamp_order ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00119_create-toys_link_tag_links.sql b/services/migrations-schema/migrations/00119_create-toys_link_tag_links.sql deleted file mode 100644 index 3b06a90..0000000 --- a/services/migrations-schema/migrations/00119_create-toys_link_tag_links.sql +++ /dev/null @@ -1,41 +0,0 @@ --- Table: api.toys_link_tag_links - --- DROP TABLE IF EXISTS api.toys_link_tag_links; - -CREATE TABLE IF NOT EXISTS api.toys_link_tag_links -( - id integer NOT NULL, - toy_id integer, - tag_id integer, - CONSTRAINT toys_link_tag_links_pkey PRIMARY KEY (id), - CONSTRAINT toys_link_tag_links_unique UNIQUE (toy_id, tag_id), - CONSTRAINT toys_link_tag_links_fk FOREIGN KEY (toy_id) - REFERENCES api.toys (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT toys_link_tag_links_inv_fk FOREIGN KEY (tag_id) - REFERENCES api.tags (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.toys_link_tag_links - OWNER to postgres; --- Index: toys_link_tag_links_fk - --- DROP INDEX IF EXISTS api.toys_link_tag_links_fk; - -CREATE INDEX IF NOT EXISTS toys_link_tag_links_fk - ON api.toys_link_tag_links USING btree - (toy_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: toys_link_tag_links_inv_fk - --- DROP INDEX IF EXISTS api.toys_link_tag_links_inv_fk; - -CREATE INDEX IF NOT EXISTS toys_link_tag_links_inv_fk - ON api.toys_link_tag_links USING btree - (tag_id ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00120_create-vods_mux_asset_links.sql b/services/migrations-schema/migrations/00120_create-vods_mux_asset_links.sql deleted file mode 100644 index 2b29cd5..0000000 --- a/services/migrations-schema/migrations/00120_create-vods_mux_asset_links.sql +++ /dev/null @@ -1,41 +0,0 @@ --- Table: api.vods_mux_asset_links - --- DROP TABLE IF EXISTS api.vods_mux_asset_links; - -CREATE TABLE IF NOT EXISTS api.vods_mux_asset_links -( - id integer NOT NULL, - vod_id integer, - mux_asset_id integer, - CONSTRAINT vods_mux_asset_links_pkey PRIMARY KEY (id), - CONSTRAINT vods_mux_asset_links_unique UNIQUE (vod_id, mux_asset_id), - CONSTRAINT vods_mux_asset_links_fk FOREIGN KEY (vod_id) - REFERENCES api.vods (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT vods_mux_asset_links_inv_fk FOREIGN KEY (mux_asset_id) - REFERENCES api.mux_assets (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.vods_mux_asset_links - OWNER to postgres; --- Index: vods_mux_asset_links_fk - --- DROP INDEX IF EXISTS api.vods_mux_asset_links_fk; - -CREATE INDEX IF NOT EXISTS vods_mux_asset_links_fk - ON api.vods_mux_asset_links USING btree - (vod_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: vods_mux_asset_links_inv_fk - --- DROP INDEX IF EXISTS api.vods_mux_asset_links_inv_fk; - -CREATE INDEX IF NOT EXISTS vods_mux_asset_links_inv_fk - ON api.vods_mux_asset_links USING btree - (mux_asset_id ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00121_create-vods_stream_links.sql b/services/migrations-schema/migrations/00121_create-vods_stream_links.sql deleted file mode 100644 index 57b37b9..0000000 --- a/services/migrations-schema/migrations/00121_create-vods_stream_links.sql +++ /dev/null @@ -1,50 +0,0 @@ --- Table: api.vods_stream_links - --- DROP TABLE IF EXISTS api.vods_stream_links; - -CREATE TABLE IF NOT EXISTS api.vods_stream_links -( - id integer NOT NULL, - vod_id integer, - stream_id integer, - vod_order double precision, - CONSTRAINT vods_stream_links_pkey PRIMARY KEY (id), - CONSTRAINT vods_stream_links_unique UNIQUE (vod_id, stream_id), - CONSTRAINT vods_stream_links_fk FOREIGN KEY (vod_id) - REFERENCES api.vods (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT vods_stream_links_inv_fk FOREIGN KEY (stream_id) - REFERENCES api.streams (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.vods_stream_links - OWNER to postgres; --- Index: vods_stream_links_fk - --- DROP INDEX IF EXISTS api.vods_stream_links_fk; - -CREATE INDEX IF NOT EXISTS vods_stream_links_fk - ON api.vods_stream_links USING btree - (vod_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: vods_stream_links_inv_fk - --- DROP INDEX IF EXISTS api.vods_stream_links_inv_fk; - -CREATE INDEX IF NOT EXISTS vods_stream_links_inv_fk - ON api.vods_stream_links USING btree - (stream_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: vods_stream_links_order_inv_fk - --- DROP INDEX IF EXISTS api.vods_stream_links_order_inv_fk; - -CREATE INDEX IF NOT EXISTS vods_stream_links_order_inv_fk - ON api.vods_stream_links USING btree - (vod_order ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00122_create-vods_thumbnail_links.sql b/services/migrations-schema/migrations/00122_create-vods_thumbnail_links.sql deleted file mode 100644 index 269838c..0000000 --- a/services/migrations-schema/migrations/00122_create-vods_thumbnail_links.sql +++ /dev/null @@ -1,87 +0,0 @@ - --- Table: api.b2_files - --- DROP TABLE IF EXISTS api.b2_files; - -CREATE TABLE IF NOT EXISTS api.b2_files -( - id integer NOT NULL, - url character varying(255) COLLATE pg_catalog."default", - key character varying(255) COLLATE pg_catalog."default", - upload_id character varying(255) COLLATE pg_catalog."default", - created_at timestamp(6) without time zone, - updated_at timestamp(6) without time zone, - created_by_id integer, - updated_by_id integer, - cdn_url character varying(255) COLLATE pg_catalog."default", - CONSTRAINT b2_files_pkey PRIMARY KEY (id) -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.b2_files - OWNER to postgres; --- Index: b2_files_created_by_id_fk - --- DROP INDEX IF EXISTS api.b2_files_created_by_id_fk; - -CREATE INDEX IF NOT EXISTS b2_files_created_by_id_fk - ON api.b2_files USING btree - (created_by_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: b2_files_updated_by_id_fk - --- DROP INDEX IF EXISTS api.b2_files_updated_by_id_fk; - -CREATE INDEX IF NOT EXISTS b2_files_updated_by_id_fk - ON api.b2_files USING btree - (updated_by_id ASC NULLS LAST) - TABLESPACE pg_default; - - - - --- Table: api.vods_thumbnail_links - --- DROP TABLE IF EXISTS api.vods_thumbnail_links; - -CREATE TABLE IF NOT EXISTS api.vods_thumbnail_links -( - id integer NOT NULL, - vod_id integer, - b_2_file_id integer, - CONSTRAINT vods_thumbnail_links_pkey PRIMARY KEY (id), - CONSTRAINT vods_thumbnail_links_unique UNIQUE (vod_id, b_2_file_id), - CONSTRAINT vods_thumbnail_links_fk FOREIGN KEY (vod_id) - REFERENCES api.vods (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT vods_thumbnail_links_inv_fk FOREIGN KEY (b_2_file_id) - REFERENCES api.b2_files (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.vods_thumbnail_links - OWNER to postgres; --- Index: vods_thumbnail_links_fk - --- DROP INDEX IF EXISTS api.vods_thumbnail_links_fk; - -CREATE INDEX IF NOT EXISTS vods_thumbnail_links_fk - ON api.vods_thumbnail_links USING btree - (vod_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: vods_thumbnail_links_inv_fk - --- DROP INDEX IF EXISTS api.vods_thumbnail_links_inv_fk; - -CREATE INDEX IF NOT EXISTS vods_thumbnail_links_inv_fk - ON api.vods_thumbnail_links USING btree - (b_2_file_id ASC NULLS LAST) - TABLESPACE pg_default; - - - diff --git a/services/migrations-schema/migrations/00123_create-vods_uploader_links.sql b/services/migrations-schema/migrations/00123_create-vods_uploader_links.sql deleted file mode 100644 index 43f4e5f..0000000 --- a/services/migrations-schema/migrations/00123_create-vods_uploader_links.sql +++ /dev/null @@ -1,37 +0,0 @@ --- Table: api.vods_uploader_links - --- DROP TABLE IF EXISTS api.vods_uploader_links; - -CREATE TABLE IF NOT EXISTS api.vods_uploader_links -( - id integer NOT NULL, - vod_id integer, - user_id integer, - CONSTRAINT vods_uploader_links_pkey PRIMARY KEY (id), - CONSTRAINT vods_uploader_links_unique UNIQUE (vod_id, user_id), - CONSTRAINT vods_uploader_links_fk FOREIGN KEY (vod_id) - REFERENCES api.vods (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.vods_uploader_links - OWNER to postgres; --- Index: vods_uploader_links_fk - --- DROP INDEX IF EXISTS api.vods_uploader_links_fk; - -CREATE INDEX IF NOT EXISTS vods_uploader_links_fk - ON api.vods_uploader_links USING btree - (vod_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: vods_uploader_links_inv_fk - --- DROP INDEX IF EXISTS api.vods_uploader_links_inv_fk; - -CREATE INDEX IF NOT EXISTS vods_uploader_links_inv_fk - ON api.vods_uploader_links USING btree - (user_id ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00124_create-vods_video_src_b_2_links.sql b/services/migrations-schema/migrations/00124_create-vods_video_src_b_2_links.sql deleted file mode 100644 index 1f37371..0000000 --- a/services/migrations-schema/migrations/00124_create-vods_video_src_b_2_links.sql +++ /dev/null @@ -1,41 +0,0 @@ --- Table: api.vods_video_src_b_2_links - --- DROP TABLE IF EXISTS api.vods_video_src_b_2_links; - -CREATE TABLE IF NOT EXISTS api.vods_video_src_b_2_links -( - id integer NOT NULL, - vod_id integer, - b_2_file_id integer, - CONSTRAINT vods_video_src_b_2_links_pkey PRIMARY KEY (id), - CONSTRAINT vods_video_src_b_2_links_unique UNIQUE (vod_id, b_2_file_id), - CONSTRAINT vods_video_src_b_2_links_fk FOREIGN KEY (vod_id) - REFERENCES api.vods (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT vods_video_src_b_2_links_inv_fk FOREIGN KEY (b_2_file_id) - REFERENCES api.b2_files (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.vods_video_src_b_2_links - OWNER to postgres; --- Index: vods_video_src_b_2_links_fk - --- DROP INDEX IF EXISTS api.vods_video_src_b_2_links_fk; - -CREATE INDEX IF NOT EXISTS vods_video_src_b_2_links_fk - ON api.vods_video_src_b_2_links USING btree - (vod_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: vods_video_src_b_2_links_inv_fk - --- DROP INDEX IF EXISTS api.vods_video_src_b_2_links_inv_fk; - -CREATE INDEX IF NOT EXISTS vods_video_src_b_2_links_inv_fk - ON api.vods_video_src_b_2_links USING btree - (b_2_file_id ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00125_create-vods_vtuber_links.sql b/services/migrations-schema/migrations/00125_create-vods_vtuber_links.sql deleted file mode 100644 index a0f280f..0000000 --- a/services/migrations-schema/migrations/00125_create-vods_vtuber_links.sql +++ /dev/null @@ -1,50 +0,0 @@ --- Table: api.vods_vtuber_links - --- DROP TABLE IF EXISTS api.vods_vtuber_links; - -CREATE TABLE IF NOT EXISTS api.vods_vtuber_links -( - id integer NOT NULL, - vod_id integer, - vtuber_id integer, - vod_order double precision, - CONSTRAINT vods_vtuber_links_pkey PRIMARY KEY (id), - CONSTRAINT vods_vtuber_links_unique UNIQUE (vod_id, vtuber_id), - CONSTRAINT vods_vtuber_links_fk FOREIGN KEY (vod_id) - REFERENCES api.vods (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT vods_vtuber_links_inv_fk FOREIGN KEY (vtuber_id) - REFERENCES api.vtubers (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.vods_vtuber_links - OWNER to postgres; --- Index: vods_vtuber_links_fk - --- DROP INDEX IF EXISTS api.vods_vtuber_links_fk; - -CREATE INDEX IF NOT EXISTS vods_vtuber_links_fk - ON api.vods_vtuber_links USING btree - (vod_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: vods_vtuber_links_inv_fk - --- DROP INDEX IF EXISTS api.vods_vtuber_links_inv_fk; - -CREATE INDEX IF NOT EXISTS vods_vtuber_links_inv_fk - ON api.vods_vtuber_links USING btree - (vtuber_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: vods_vtuber_links_order_inv_fk - --- DROP INDEX IF EXISTS api.vods_vtuber_links_order_inv_fk; - -CREATE INDEX IF NOT EXISTS vods_vtuber_links_order_inv_fk - ON api.vods_vtuber_links USING btree - (vod_order ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00126_create-vtubers_toy_links.sql b/services/migrations-schema/migrations/00126_create-vtubers_toy_links.sql deleted file mode 100644 index 57c816f..0000000 --- a/services/migrations-schema/migrations/00126_create-vtubers_toy_links.sql +++ /dev/null @@ -1,105 +0,0 @@ --- Table: api.vtubers_toy_links - --- DROP TABLE IF EXISTS api.vtubers_toy_links; - -CREATE TABLE IF NOT EXISTS api.vtubers_toy_links -( - id integer NOT NULL, - vtuber_id integer, - toy_id integer, - vtuber_order double precision, - CONSTRAINT vtubers_toy_links_pkey PRIMARY KEY (id), - CONSTRAINT vtubers_toy_links_unique UNIQUE (vtuber_id, toy_id), - CONSTRAINT vtubers_toy_links_fk FOREIGN KEY (vtuber_id) - REFERENCES api.vtubers (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT vtubers_toy_links_inv_fk FOREIGN KEY (toy_id) - REFERENCES api.toys (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.vtubers_toy_links - OWNER to postgres; --- Index: vtubers_toy_links_fk - --- DROP INDEX IF EXISTS api.vtubers_toy_links_fk; - -CREATE INDEX IF NOT EXISTS vtubers_toy_links_fk - ON api.vtubers_toy_links USING btree - (vtuber_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: vtubers_toy_links_inv_fk - --- DROP INDEX IF EXISTS api.vtubers_toy_links_inv_fk; - -CREATE INDEX IF NOT EXISTS vtubers_toy_links_inv_fk - ON api.vtubers_toy_links USING btree - (toy_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: vtubers_toy_links_order_inv_fk - --- DROP INDEX IF EXISTS api.vtubers_toy_links_order_inv_fk; - -CREATE INDEX IF NOT EXISTS vtubers_toy_links_order_inv_fk - ON api.vtubers_toy_links USING btree - (vtuber_order ASC NULLS LAST) - TABLESPACE pg_default; - - - --- also adding vtubers_toys_links - --- Table: api.vtubers_toys_links - --- DROP TABLE IF EXISTS api.vtubers_toys_links; - -CREATE TABLE IF NOT EXISTS api.vtubers_toys_links -( - id integer NOT NULL, - vtuber_id integer, - toy_id integer, - toy_order double precision, - CONSTRAINT vtubers_toys_links_pkey PRIMARY KEY (id), - CONSTRAINT vtubers_toys_links_unique UNIQUE (vtuber_id, toy_id), - CONSTRAINT vtubers_toys_links_fk FOREIGN KEY (vtuber_id) - REFERENCES api.vtubers (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE, - CONSTRAINT vtubers_toys_links_inv_fk FOREIGN KEY (toy_id) - REFERENCES api.toys (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE -) - -TABLESPACE pg_default; - -ALTER TABLE IF EXISTS api.vtubers_toys_links - OWNER to postgres; --- Index: vtubers_toys_links_fk - --- DROP INDEX IF EXISTS api.vtubers_toys_links_fk; - -CREATE INDEX IF NOT EXISTS vtubers_toys_links_fk - ON api.vtubers_toys_links USING btree - (vtuber_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: vtubers_toys_links_inv_fk - --- DROP INDEX IF EXISTS api.vtubers_toys_links_inv_fk; - -CREATE INDEX IF NOT EXISTS vtubers_toys_links_inv_fk - ON api.vtubers_toys_links USING btree - (toy_id ASC NULLS LAST) - TABLESPACE pg_default; --- Index: vtubers_toys_links_order_fk - --- DROP INDEX IF EXISTS api.vtubers_toys_links_order_fk; - -CREATE INDEX IF NOT EXISTS vtubers_toys_links_order_fk - ON api.vtubers_toys_links USING btree - (toy_order ASC NULLS LAST) - TABLESPACE pg_default; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00127_grant-permissions-for-vods_vtuber_links.sql b/services/migrations-schema/migrations/00127_grant-permissions-for-vods_vtuber_links.sql deleted file mode 100644 index 1f7228b..0000000 --- a/services/migrations-schema/migrations/00127_grant-permissions-for-vods_vtuber_links.sql +++ /dev/null @@ -1,2 +0,0 @@ -GRANT all ON api.vods_vtuber_links TO automation; -GRANT SELECT ON api.vods_vtuber_links TO web_anon; diff --git a/services/migrations-schema/migrations/00128_add-vtuber-fk-to-vods.sql b/services/migrations-schema/migrations/00128_add-vtuber-fk-to-vods.sql deleted file mode 100644 index c6e14cb..0000000 --- a/services/migrations-schema/migrations/00128_add-vtuber-fk-to-vods.sql +++ /dev/null @@ -1,6 +0,0 @@ - -ALTER TABLE IF EXISTS api.vods - ADD CONSTRAINT vods_vtuber_fk FOREIGN KEY (vtuber) - REFERENCES api.vtubers (id) MATCH SIMPLE - ON UPDATE NO ACTION - ON DELETE CASCADE \ No newline at end of file diff --git a/services/migrations-schema/migrations/00129_add-vtuber_id-to-vods.sql b/services/migrations-schema/migrations/00129_add-vtuber_id-to-vods.sql deleted file mode 100644 index 096f800..0000000 --- a/services/migrations-schema/migrations/00129_add-vtuber_id-to-vods.sql +++ /dev/null @@ -1,9 +0,0 @@ -ALTER TABLE api.vods ADD COLUMN vtuber_id INTEGER; - -ALTER TABLE api.vods - DROP CONSTRAINT vods_vtuber_fk; - -ALTER TABLE api.vods - ADD CONSTRAINT vods_vtuber_fk FOREIGN KEY (vtuber_id) - REFERENCES api.vtubers (id) - ON DELETE CASCADE; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00130_add-s3_files-fk-to-vods.sql b/services/migrations-schema/migrations/00130_add-s3_files-fk-to-vods.sql deleted file mode 100644 index 4c6fcdc..0000000 --- a/services/migrations-schema/migrations/00130_add-s3_files-fk-to-vods.sql +++ /dev/null @@ -1,6 +0,0 @@ -ALTER TABLE api.vods ADD COLUMN s3_file_id INTEGER; - -ALTER TABLE api.vods - ADD CONSTRAINT vods_s3_file_fk FOREIGN KEY (s3_file_id) - REFERENCES api.s3_files (id) - ON DELETE CASCADE; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00131_grant-perms-on-s3_files.sql b/services/migrations-schema/migrations/00131_grant-perms-on-s3_files.sql deleted file mode 100644 index 6593f67..0000000 --- a/services/migrations-schema/migrations/00131_grant-perms-on-s3_files.sql +++ /dev/null @@ -1,7 +0,0 @@ -GRANT all ON api.s3_files TO automation; --- GRANT SELECT ON api.s3_files TO web_anon; - --- we only want to grant access to this resource to patrons. how do we do that? @TODO - - - diff --git a/services/migrations-schema/migrations/00132_rename-vtuber_num-to-vtuber_id.sql b/services/migrations-schema/migrations/00132_rename-vtuber_num-to-vtuber_id.sql deleted file mode 100644 index 4c05ef0..0000000 --- a/services/migrations-schema/migrations/00132_rename-vtuber_num-to-vtuber_id.sql +++ /dev/null @@ -1,7 +0,0 @@ -ALTER TABLE IF EXISTS api.streams - ADD COLUMN IF NOT EXISTS vtuber_id INT; - -ALTER TABLE api.streams - ADD CONSTRAINT streams_vtuber_fk FOREIGN KEY (vtuber_id) - REFERENCES api.vtubers (id) - ON DELETE CASCADE; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00133_add-mux_assets-fk-to-vods.sql b/services/migrations-schema/migrations/00133_add-mux_assets-fk-to-vods.sql deleted file mode 100644 index 04e7d0a..0000000 --- a/services/migrations-schema/migrations/00133_add-mux_assets-fk-to-vods.sql +++ /dev/null @@ -1,6 +0,0 @@ -ALTER TABLE api.vods ADD COLUMN mux_asset_id INTEGER; - -ALTER TABLE api.vods - ADD CONSTRAINT vods_mux_asset_fk FOREIGN KEY (mux_asset_id) - REFERENCES api.mux_assets (id) - ON DELETE CASCADE; diff --git a/services/migrations-schema/migrations/00134_add-thumbnail-fk-to-vods.sql b/services/migrations-schema/migrations/00134_add-thumbnail-fk-to-vods.sql deleted file mode 100644 index 6dbafc0..0000000 --- a/services/migrations-schema/migrations/00134_add-thumbnail-fk-to-vods.sql +++ /dev/null @@ -1,6 +0,0 @@ -ALTER TABLE api.vods ADD COLUMN thumbnail_id INTEGER; - -ALTER TABLE api.vods - ADD CONSTRAINT vods_thumbnail_fk FOREIGN KEY (thumbnail_id) - REFERENCES api.b2_files (id) - ON DELETE CASCADE; diff --git a/services/migrations-schema/migrations/00135_add-b2_files-permissions.sql b/services/migrations-schema/migrations/00135_add-b2_files-permissions.sql deleted file mode 100644 index b30808a..0000000 --- a/services/migrations-schema/migrations/00135_add-b2_files-permissions.sql +++ /dev/null @@ -1,2 +0,0 @@ -GRANT all ON api.b2_files TO automation; -GRANT SELECT ON api.b2_files TO web_anon; diff --git a/services/migrations-schema/migrations/00136_add-uuid-to-streams.sql b/services/migrations-schema/migrations/00136_add-uuid-to-streams.sql deleted file mode 100644 index 69151dc..0000000 --- a/services/migrations-schema/migrations/00136_add-uuid-to-streams.sql +++ /dev/null @@ -1,8 +0,0 @@ -ALTER TABLE IF EXISTS api.streams - ADD COLUMN IF NOT EXISTS uuid UUID - DEFAULT gen_random_uuid(); - -ALTER TABLE IF EXISTS api.vods - ADD COLUMN IF NOT EXISTS uuid UUID - DEFAULT gen_random_uuid(); - diff --git a/services/migrations-schema/migrations/00137_add-vods-fk-to-streams.sql b/services/migrations-schema/migrations/00137_add-vods-fk-to-streams.sql deleted file mode 100644 index c38ceaa..0000000 --- a/services/migrations-schema/migrations/00137_add-vods-fk-to-streams.sql +++ /dev/null @@ -1,4 +0,0 @@ -ALTER TABLE api.vods - ADD CONSTRAINT vods_stream_fk FOREIGN KEY (stream_id) - REFERENCES api.streams (id) - ON DELETE CASCADE; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00138_add-date_2-to-streams-and-vods.sql b/services/migrations-schema/migrations/00138_add-date_2-to-streams-and-vods.sql deleted file mode 100644 index e234f37..0000000 --- a/services/migrations-schema/migrations/00138_add-date_2-to-streams-and-vods.sql +++ /dev/null @@ -1,7 +0,0 @@ - - -ALTER TABLE IF EXISTS api.streams - ADD COLUMN IF NOT EXISTS date_2 TIMESTAMP(6) WITH TIME ZONE; - -ALTER TABLE IF EXISTS api.vods - ADD COLUMN IF NOT EXISTS date_2 TIMESTAMP(6) WITH TIME ZONE; \ No newline at end of file diff --git a/services/migrations-schema/migrations/00139_add-timestamps-permissions.sql b/services/migrations-schema/migrations/00139_add-timestamps-permissions.sql deleted file mode 100644 index 1106229..0000000 --- a/services/migrations-schema/migrations/00139_add-timestamps-permissions.sql +++ /dev/null @@ -1,2 +0,0 @@ -GRANT all ON api.timestamps TO automation; -GRANT SELECT ON api.timestamps TO web_anon; diff --git a/services/migrations-schema/migrations/00140_add-permissions-for-links.sql b/services/migrations-schema/migrations/00140_add-permissions-for-links.sql deleted file mode 100644 index 6ebbce7..0000000 --- a/services/migrations-schema/migrations/00140_add-permissions-for-links.sql +++ /dev/null @@ -1,15 +0,0 @@ - -GRANT all ON api.timestamps_vod_links TO automation; -GRANT SELECT ON api.timestamps_vod_links TO web_anon; - - -GRANT all ON api.timestamps_tag_links TO automation; -GRANT SELECT ON api.timestamps_tag_links TO web_anon; - - -GRANT all ON api.toys_link_tag_links TO automation; -GRANT SELECT ON api.toys_link_tag_links TO web_anon; - -GRANT all ON api.vods_mux_asset_links TO automation; -GRANT SELECT ON api.vods_mux_asset_links TO web_anon; - diff --git a/services/migrations-schema/migrations/00141_add-permissions-timestamps_vod_links.sql b/services/migrations-schema/migrations/00141_add-permissions-timestamps_vod_links.sql deleted file mode 100644 index d479f9f..0000000 --- a/services/migrations-schema/migrations/00141_add-permissions-timestamps_vod_links.sql +++ /dev/null @@ -1,2 +0,0 @@ -GRANT all ON api.timestamps_vod_links TO automation; -GRANT SELECT ON api.timestamps_vod_links TO web_anon; diff --git a/services/migrations-schema/migrations/00142_add-permissions-for-tag_vod_relations.sql b/services/migrations-schema/migrations/00142_add-permissions-for-tag_vod_relations.sql deleted file mode 100644 index a354df7..0000000 --- a/services/migrations-schema/migrations/00142_add-permissions-for-tag_vod_relations.sql +++ /dev/null @@ -1,4 +0,0 @@ - - -GRANT all ON api.tag_vod_relations TO automation; -GRANT SELECT ON api.tag_vod_relations TO web_anon; diff --git a/services/migrations-schema/migrations/00143_add-permissions-for-tag_vod_relations_vod_links.sql b/services/migrations-schema/migrations/00143_add-permissions-for-tag_vod_relations_vod_links.sql deleted file mode 100644 index 9a35c3b..0000000 --- a/services/migrations-schema/migrations/00143_add-permissions-for-tag_vod_relations_vod_links.sql +++ /dev/null @@ -1,6 +0,0 @@ - - - - -GRANT all ON api.tag_vod_relations_vod_links TO automation; -GRANT SELECT ON api.tag_vod_relations_vod_links TO web_anon; diff --git a/services/migrations-schema/migrations/00144_add-permissions-for-tags_vods_links.sql b/services/migrations-schema/migrations/00144_add-permissions-for-tags_vods_links.sql deleted file mode 100644 index 6b2f805..0000000 --- a/services/migrations-schema/migrations/00144_add-permissions-for-tags_vods_links.sql +++ /dev/null @@ -1,2 +0,0 @@ -GRANT all ON api.tags_vods_links TO automation; -GRANT SELECT ON api.tags_vods_links TO web_anon; \ No newline at end of file diff --git a/services/migrations-schema/package.json b/services/migrations-schema/package.json deleted file mode 100644 index 2019d04..0000000 --- a/services/migrations-schema/package.json +++ /dev/null @@ -1,19 +0,0 @@ -{ - "name": "@futureporn/migrations-schema", - "type": "module", - "version": "0.6.0", - "description": "", - "main": "index.js", - "scripts": { - "test": "echo \"Error: no test specified\" && exit 0", - "start": "node index.js" - }, - "packageManager": "pnpm@9.6.0", - "keywords": [], - "author": "@CJ_Clippy", - "license": "Unlicense", - "dependencies": { - "dotenv": "^16.4.5", - "postgres-schema-migrations": "^6.1.0" - } -} diff --git a/services/migrations-schema/pnpm-lock.yaml b/services/migrations-schema/pnpm-lock.yaml deleted file mode 100644 index 535b76b..0000000 --- a/services/migrations-schema/pnpm-lock.yaml +++ /dev/null @@ -1,183 +0,0 @@ -lockfileVersion: '9.0' - -settings: - autoInstallPeers: true - excludeLinksFromLockfile: false - -importers: - - .: - dependencies: - dotenv: - specifier: ^16.4.5 - version: 16.4.5 - postgres-schema-migrations: - specifier: ^6.1.0 - version: 6.1.0 - - ../..: {} - - ../../packages/fetchers: {} - - ../../packages/infra: {} - - ../../packages/storage: {} - - ../../packages/types: {} - - ../../packages/utils: {} - - ../bot: {} - - ../capture: {} - - ../factory: {} - - ../htmx: {} - - ../mailbox: {} - - ../migrations-data: {} - - ../next: {} - - ../scout: {} - - ../strapi: {} - - ../uppy: {} - -packages: - - dotenv@16.4.5: - resolution: {integrity: sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==} - engines: {node: '>=12'} - - pg-cloudflare@1.1.1: - resolution: {integrity: sha512-xWPagP/4B6BgFO+EKz3JONXv3YDgvkbVrGw2mTo3D6tVDQRh1e7cqVGvyR3BE+eQgAvx1XhW/iEASj4/jCWl3Q==} - - pg-connection-string@2.6.4: - resolution: {integrity: sha512-v+Z7W/0EO707aNMaAEfiGnGL9sxxumwLl2fJvCQtMn9Fxsg+lPpPkdcyBSv/KFgpGdYkMfn+EI1Or2EHjpgLCA==} - - pg-int8@1.0.1: - resolution: {integrity: sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw==} - engines: {node: '>=4.0.0'} - - pg-pool@3.6.2: - resolution: {integrity: sha512-Htjbg8BlwXqSBQ9V8Vjtc+vzf/6fVUuak/3/XXKA9oxZprwW3IMDQTGHP+KDmVL7rtd+R1QjbnCFPuTHm3G4hg==} - peerDependencies: - pg: '>=8.0' - - pg-protocol@1.6.1: - resolution: {integrity: sha512-jPIlvgoD63hrEuihvIg+tJhoGjUsLPn6poJY9N5CnlPd91c2T18T/9zBtLxZSb1EhYxBRoZJtzScCaWlYLtktg==} - - pg-types@2.2.0: - resolution: {integrity: sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==} - engines: {node: '>=4'} - - pg@8.12.0: - resolution: {integrity: sha512-A+LHUSnwnxrnL/tZ+OLfqR1SxLN3c/pgDztZ47Rpbsd4jUytsTtwQo/TLPRzPJMp/1pbhYVhH9cuSZLAajNfjQ==} - engines: {node: '>= 8.0.0'} - peerDependencies: - pg-native: '>=3.0.1' - peerDependenciesMeta: - pg-native: - optional: true - - pgpass@1.0.5: - resolution: {integrity: sha512-FdW9r/jQZhSeohs1Z3sI1yxFQNFvMcnmfuj4WBMUTxOrAyLMaTcE1aAMBiTlbMNaXvBCQuVi0R7hd8udDSP7ug==} - - postgres-array@2.0.0: - resolution: {integrity: sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA==} - engines: {node: '>=4'} - - postgres-bytea@1.0.0: - resolution: {integrity: sha512-xy3pmLuQqRBZBXDULy7KbaitYqLcmxigw14Q5sj8QBVLqEwXfeybIKVWiqAXTlcvdvb0+xkOtDbfQMOf4lST1w==} - engines: {node: '>=0.10.0'} - - postgres-date@1.0.7: - resolution: {integrity: sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q==} - engines: {node: '>=0.10.0'} - - postgres-interval@1.2.0: - resolution: {integrity: sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==} - engines: {node: '>=0.10.0'} - - postgres-schema-migrations@6.1.0: - resolution: {integrity: sha512-d1LJ+A9Lg4kAwuh91S8ozF8q3adFNJlStbpUF/sbjMTzSIzJClpmg4D6qyd9nvKt2el0rnZJjXZQ2r01Y5OpzA==} - engines: {node: '>10.17.0'} - hasBin: true - - split2@4.2.0: - resolution: {integrity: sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==} - engines: {node: '>= 10.x'} - - sql-template-strings@2.2.2: - resolution: {integrity: sha512-UXhXR2869FQaD+GMly8jAMCRZ94nU5KcrFetZfWEMd+LVVG6y0ExgHAhatEcKZ/wk8YcKPdi+hiD2wm75lq3/Q==} - engines: {node: '>=4.0.0'} - - xtend@4.0.2: - resolution: {integrity: sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==} - engines: {node: '>=0.4'} - -snapshots: - - dotenv@16.4.5: {} - - pg-cloudflare@1.1.1: - optional: true - - pg-connection-string@2.6.4: {} - - pg-int8@1.0.1: {} - - pg-pool@3.6.2(pg@8.12.0): - dependencies: - pg: 8.12.0 - - pg-protocol@1.6.1: {} - - pg-types@2.2.0: - dependencies: - pg-int8: 1.0.1 - postgres-array: 2.0.0 - postgres-bytea: 1.0.0 - postgres-date: 1.0.7 - postgres-interval: 1.2.0 - - pg@8.12.0: - dependencies: - pg-connection-string: 2.6.4 - pg-pool: 3.6.2(pg@8.12.0) - pg-protocol: 1.6.1 - pg-types: 2.2.0 - pgpass: 1.0.5 - optionalDependencies: - pg-cloudflare: 1.1.1 - - pgpass@1.0.5: - dependencies: - split2: 4.2.0 - - postgres-array@2.0.0: {} - - postgres-bytea@1.0.0: {} - - postgres-date@1.0.7: {} - - postgres-interval@1.2.0: - dependencies: - xtend: 4.0.2 - - postgres-schema-migrations@6.1.0: - dependencies: - pg: 8.12.0 - sql-template-strings: 2.2.2 - transitivePeerDependencies: - - pg-native - - split2@4.2.0: {} - - sql-template-strings@2.2.2: {} - - xtend@4.0.2: {} diff --git a/services/next/.dockerignore b/services/next/.dockerignore deleted file mode 100644 index 83f3821..0000000 --- a/services/next/.dockerignore +++ /dev/null @@ -1,3 +0,0 @@ -## This is the WRONG place to add dockerignore spec! -## We are using monorepo which menas .dockerignore must be in the root of context (../../) -## Add dockerignore specs to ../../.dockerignore instead. diff --git a/services/next/.eslintrc.json b/services/next/.eslintrc.json deleted file mode 100644 index bffb357..0000000 --- a/services/next/.eslintrc.json +++ /dev/null @@ -1,3 +0,0 @@ -{ - "extends": "next/core-web-vitals" -} diff --git a/services/next/.gitignore b/services/next/.gitignore deleted file mode 100644 index c43ed62..0000000 --- a/services/next/.gitignore +++ /dev/null @@ -1,48 +0,0 @@ -# Created by https://www.toptal.com/developers/gitignore/api/nextjs -# Edit at https://www.toptal.com/developers/gitignore?templates=nextjs - -fuck.mjs - -.vscode/ - -.env -.env.* -dist/ - -### NextJS ### -# dependencies -/node_modules -/.pnp -.pnp.js - -# testing -/coverage - -# next.js -/.next/ -/out/ - -# production -/build - -# misc -.DS_Store -*.pem - -# debug -npm-debug.log* -yarn-debug.log* -yarn-error.log* -.pnpm-debug.log* - -# local env files -.env*.local - -# vercel -.vercel - -# typescript -*.tsbuildinfo -next-env.d.ts - -# End of https://www.toptal.com/developers/gitignore/api/nextjs diff --git a/services/next/.nvmrc b/services/next/.nvmrc deleted file mode 100644 index 9de2256..0000000 --- a/services/next/.nvmrc +++ /dev/null @@ -1 +0,0 @@ -lts/iron diff --git a/services/next/CHECKS b/services/next/CHECKS deleted file mode 100644 index 6731352..0000000 --- a/services/next/CHECKS +++ /dev/null @@ -1 +0,0 @@ -/ futureporn.net \ No newline at end of file diff --git a/services/next/LICENSE b/services/next/LICENSE deleted file mode 100644 index 7c53cea..0000000 --- a/services/next/LICENSE +++ /dev/null @@ -1,21 +0,0 @@ -MIT License - -Copyright (c) 2020 Rogier van den Berg - -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/services/next/README.md b/services/next/README.md deleted file mode 100644 index 9ae2830..0000000 --- a/services/next/README.md +++ /dev/null @@ -1,3 +0,0 @@ -# next - -HTML5 frontend for https://futureporn.net \ No newline at end of file diff --git a/services/next/app.json b/services/next/app.json deleted file mode 100644 index 31825ab..0000000 --- a/services/next/app.json +++ /dev/null @@ -1,14 +0,0 @@ -{ - "healthchecks": { - "web": [ - { - "type": "startup", - "name": "web check", - "description": "Checking for expecting string at /api", - "path": "/api", - "content": "Application Programmable Interface", - "attempts": 3 - } - ] - } -} \ No newline at end of file diff --git a/services/next/app/404.tsx b/services/next/app/404.tsx deleted file mode 100644 index bf9ef9c..0000000 --- a/services/next/app/404.tsx +++ /dev/null @@ -1,3 +0,0 @@ -export default function Custom404() { - return

404~~~~ - Page Not Found

-} \ No newline at end of file diff --git a/services/next/app/about/page.tsx b/services/next/app/about/page.tsx deleted file mode 100644 index a936463..0000000 --- a/services/next/app/about/page.tsx +++ /dev/null @@ -1,37 +0,0 @@ -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faExternalLinkAlt } from "@fortawesome/free-solid-svg-icons"; -import Link from 'next/link'; -// import { getProgress } from '../lib/vods' - -export default async function Page() { - // const { complete, total } = await getProgress('projektmelody') - - return ( - <> -
- -
-
-

About

-

It's the worst feeling when a VOD disappears from the internet. It means you missed out, it's gone, and you may never experience what your peers got to take part in.

-

Futureporn is created by fans, for fans. Missed a stream? We got you, bro.

-

Together we can end 404s and create an everlasting archive of lewdtuber livestreams.

- -
- -
- - -
- -
-
-

Futureporn needs financial support to continue improving. If you enjoy this website, please consider becoming a patron.

-
-
-
- -
- - ) -} diff --git a/services/next/app/api/auth/[...nextauth]/route.ts b/services/next/app/api/auth/[...nextauth]/route.ts deleted file mode 100644 index 98f6330..0000000 --- a/services/next/app/api/auth/[...nextauth]/route.ts +++ /dev/null @@ -1,17 +0,0 @@ -/** - * api/auth/[...nextauth]/route.ts - */ - -import { authOptions } from "@/app/lib/auth" -import NextAuth from "next-auth" - - -// v4 -const handler = NextAuth(authOptions) -export { handler as GET, handler as POST } - - - - -// v5 -// export { GET, POST } from "@/app/lib/auth" diff --git a/services/next/app/api/blogs/route.ts b/services/next/app/api/blogs/route.ts deleted file mode 100644 index dca5232..0000000 --- a/services/next/app/api/blogs/route.ts +++ /dev/null @@ -1,10 +0,0 @@ -import { NextResponse } from 'next/server' - -export async function GET() { - const res = await fetch('https://dummyjson.com/posts', { - next: { revalidate: 60 }, - }); - const data = await res.json(); - - return NextResponse.json(data); -} diff --git a/services/next/app/api/page.tsx b/services/next/app/api/page.tsx deleted file mode 100644 index ace3230..0000000 --- a/services/next/app/api/page.tsx +++ /dev/null @@ -1,58 +0,0 @@ -'use client'; - -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faExternalLinkAlt } from "@fortawesome/free-solid-svg-icons"; -import { postgrestUrl } from '@/app/lib/constants' -import Link from 'next/link' -import { useSession } from "next-auth/react"; -import { signIn } from "next-auth/react" - - -export default function Page() { - const { data: session } = useSession(); - return ( -
-
-

Futureporn API

-

Futureporn Application Programmable Interface (API) for developers and power users

-
-
- - -
-
-

Data API

-

The Data API contains all the data served by this website in JSON format. This API is unstable, meaning it's subject to changes that may break your integrations.

-
Futureporn API Version 1
-
Futureporn API Version 2
-
-
- -
-
-

Client Session (test)

-
{JSON.stringify(session)}
- - - - -
-
- -
-
- ) -} \ No newline at end of file diff --git a/services/next/app/api/patreon/currently-entitled-tiers/route.ts b/services/next/app/api/patreon/currently-entitled-tiers/route.ts deleted file mode 100644 index f989cb4..0000000 --- a/services/next/app/api/patreon/currently-entitled-tiers/route.ts +++ /dev/null @@ -1,65 +0,0 @@ -'use server'; - -import { NextResponse } from "next/server"; -import { getServerSession } from "next-auth"; -import { getPatreonMemberships } from "@/app/lib/patreon"; -import { authOptions } from "@/app/lib/auth"; - -export async function GET(req: Request, res: Response) { - - // return NextResponse.json(['test']) - - console.log('lets run getServerSession()') - - - // @TODO @TODO @TODO @todo @todo @todo THE PROBLEM IS WITH getServerSession()!!!! - // or rather, the JWE we generated in middleware.ts is invalid and the error is manifesting within the getServerSession() invocation. - const session = await getServerSession(authOptions); - // console.log('session as follows') - // console.log(session) - - if (session) { - - // console.log('currently-entitled-tiers lets get user id') - // console.log(session) - - - let patreonTiersList - - if (!session.token?.access_token) { - console.error('session.token.access_token was missing') - return NextResponse.json({ error: `Failed to get access token from Session`}, { status: 500 }) - } - - if (!session.token.sub) { - return NextResponse.json({ error: `failed to get profile.sub from Session` }, { status: 400 }) - } - - // try { - // idpToken = await get(session.token.access_token) - // } catch (e) { - // console.error(e) - // return NextResponse.json({ error: `Failed to get Patreon token (Keycloak IDP). e=${e}`}, { status: 401 }) - // } - - try { - patreonTiersList = await getPatreonMemberships(idpToken) - } catch (e) { - console.error(e) - return NextResponse.json({ error: `Failed to get patreon memberships. e=${e}`}, { status: 401 }) - } - - - // side effect which grants the appropriate keycloak roles to the user - try { - await syncronizeKeycloakRoles(session.token.sub, patreonTiersList) - } catch (e) { - console.error(e) - return NextResponse.json({ error: `Failed to syncronize roles` }, { status: 500 }) - } - - - return NextResponse.json(patreonTiersList); - } - return NextResponse.json({ error: "You must be logged in." }, { status: 401 }); -} diff --git a/services/next/app/api/patreon/session/route.ts b/services/next/app/api/patreon/session/route.ts deleted file mode 100644 index 6a2b155..0000000 --- a/services/next/app/api/patreon/session/route.ts +++ /dev/null @@ -1,18 +0,0 @@ -'use server'; - -import { NextResponse } from "next/server"; -import { getServerSession } from "next-auth"; -import { authOptions } from "@/app/lib/auth"; - - -export async function GET() { - - - const session = await getServerSession(authOptions); - - if (session) { - - return NextResponse.json(session) - } - return NextResponse.json({ error: "You must be logged in." }, { status: 401 }); -} diff --git a/services/next/app/api/profile/route.ts b/services/next/app/api/profile/route.ts deleted file mode 100644 index 5c58f29..0000000 --- a/services/next/app/api/profile/route.ts +++ /dev/null @@ -1,51 +0,0 @@ - - -// 'use server'; - -// import { NextRequest, NextResponse } from "next/server"; -// import { getServerSession } from "next-auth"; -// import { authOptions } from "@/app/lib/auth"; -// import { getUMAToken } from "@/app/lib/keycloak"; -// import { configs } from "@/app/config/configs"; - -// export async function POST(req: NextRequest) { -// const session = await getServerSession(authOptions); - -// console.log(`a user is attempting to POST their profile. session as follows`) -// console.log(session) - -// if (session) { -// if (!session.token?.access_token) { -// console.error('session.token.access_token was missing') -// return NextResponse.json({ error: `Failed to get access token from Session`}, { status: 500 }) -// } - -// const uma = await getUMAToken(); -// const userId = session.user.id; // Assuming session.user.id is the Keycloak user ID -// const updatedAttributes = await req.json(); // Assuming the updated attributes are sent in the request body - -// console.log('updatedAttributes as follows') -// console.log(updatedAttributes) - -// const response = await fetch(`${configs.keycloakLocalUrl}/admin/realms/futureporn/users/${userId}`, { -// method: 'PUT', -// headers: { -// 'Authorization': `Bearer ${uma.access_token}`, -// 'Content-Type': 'application/json' -// }, -// body: JSON.stringify({ -// attributes: updatedAttributes -// }) -// }); - -// // /admin/realms/futureporn/users/9dd538cd-56a9-4f43-81cd-fa86473214f5 - -// if (!response.ok) { -// console.error(`failed. res.status=${response.status} res.statusText=${response.statusText}`) -// return NextResponse.json({ error: 'Failed to update user attributes' }, { status: 500 }); -// } else { -// return NextResponse.json({ message: 'User attributes updated successfully' }); -// } -// } -// return NextResponse.json({ error: "You must be logged in." }, { status: 401 }); -// } \ No newline at end of file diff --git a/services/next/app/api/revalidate/route.ts b/services/next/app/api/revalidate/route.ts deleted file mode 100644 index bad85ce..0000000 --- a/services/next/app/api/revalidate/route.ts +++ /dev/null @@ -1,26 +0,0 @@ -import { NextRequest, NextResponse } from 'next/server'; -import { revalidateTag } from 'next/cache'; - -export const dynamic = 'force-dynamic'; - -export async function GET(request: NextRequest) { - const token = request.nextUrl.searchParams.get('token') - const tag = request.nextUrl.searchParams.get('tag') - - - - if (!token) { - return NextResponse.json({ message: 'Missing token param' }, { status: 400}) - } - - if (!tag) { - return NextResponse.json({ message: 'Missing tag param' }, { status: 400 }) - } - - if (token !== process.env.REVALIDATION_TOKEN) { - return NextResponse.json({ message: 'Invalid token' }, { status: 401 }) - } - - revalidateTag(tag) - return NextResponse.json({ revalidated: true, now: Date.now() }) -} \ No newline at end of file diff --git a/services/next/app/api/service.json/route.ts b/services/next/app/api/service.json/route.ts deleted file mode 100644 index 9aed096..0000000 --- a/services/next/app/api/service.json/route.ts +++ /dev/null @@ -1,139 +0,0 @@ -import { NextResponse } from 'next/server' - - -const serviceConfig = { - "cluster": { - "peername": "replace-this-with-a-super-cool-peer-name", - "secret": "3acade7f761c91f5fe3d34c4f4d15a17f817bc3463ab4395958f302b222a023b", - "leave_on_shutdown": false, - "listen_multiaddress": [ - "/ip4/0.0.0.0/tcp/9096" - ], - "connection_manager": { - "high_water": 400, - "low_water": 100, - "grace_period": "2m0s" - }, - "dial_peer_timeout": "3s", - "state_sync_interval": "10m", - "pin_recover_interval": "12m", - "ipfs_sync_interval": "130s", - "replication_factor_min": -1, - "replication_factor_max": -1, - "monitor_ping_interval": "30s", - "peer_watch_interval": "10s", - "mdns_interval": "10s", - "disable_repinning": true, - "follower_mode": true, - "peer_addresses": [ - "/dns4/cluster.sbtp.xyz/tcp/9096/p2p/12D3KooWJmCsFadow1UvqAqCGtuKpqrS3puyPUYujJj4dRRCTfXf" - ] - }, - "consensus": { - "crdt": { - "cluster_name": "futureporn.net", - "trusted_peers": [ - "12D3KooWJmCsFadow1UvqAqCGtuKpqrS3puyPUYujJj4dRRCTfXf" - ], - "rebroadcast_interval": "1m", - "peerset_metric": "ping", - "batching": { - "max_batch_size": 0, - "max_batch_age": "0s", - "max_queue_size": 50000 - } - } - }, - "ipfs_connector": { - "ipfshttp": { - "node_multiaddress": "/ip4/127.0.0.1/tcp/5001", - "connect_swarms_delay": "30s", - "ipfs_request_timeout": "5m", - "repogc_timeout": "24h", - "pin_timeout": "3m", - "unpin_timeout": "3h", - "unpin_disable": false - } - }, - "pin_tracker": { - "stateless": { - "max_pin_queue_size": 1000000, - "concurrent_pins": 8, - "priority_pin_max_age" : "24h", - "priority_pin_max_retries" : 5 - } - }, - "monitor": { - "pubsubmon": { - "check_interval": "15s", - "failure_threshold": 3 - } - }, - "informer": { - "disk": { - "metric_ttl": "5m", - "metric_type": "freespace" - }, - "tags": { - "metric_ttl": "30s", - "tags": {} - } - }, - "allocator": { - "balanced": { - "allocate_by": ["freespace"] - } - }, - "observations": { - "metrics": { - "enable_stats": false, - "prometheus_endpoint": "/ip4/0.0.0.0/tcp/8888", - "reporting_interval": "2s" - }, - "tracing": { - "enable_tracing": false, - "jaeger_agent_endpoint": "/ip4/0.0.0.0/udp/6831", - "sampling_prob": 0.3, - "service_name": "cluster-daemon" - } - }, - "datastore": { - "badger": { - "gc_discard_ratio": 0.2, - "gc_interval": "15m0s", - "gc_sleep": "10s", - "badger_options": { - "dir": "", - "value_dir": "", - "sync_writes": true, - "table_loading_mode": 0, - "value_log_loading_mode": 0, - "num_versions_to_keep": 1, - "max_table_size": 67108864, - "level_size_multiplier": 10, - "max_levels": 7, - "value_threshold": 32, - "num_memtables": 5, - "num_level_zero_tables": 5, - "num_level_zero_tables_stall": 10, - "level_one_size": 268435456, - "value_log_file_size": 1073741823, - "value_log_max_entries": 1000000, - "num_compactors": 2, - "compact_l_0_on_close": true, - "read_only": false, - "truncate": false - } - } - } - } - -export const dynamic = 'force-dynamic' -export async function GET() { - const options = { - headers: { - "Content-Type": "application/json", - } - }; - return new NextResponse(JSON.stringify(serviceConfig), options); -} \ No newline at end of file diff --git a/services/next/app/api/session/route.ts b/services/next/app/api/session/route.ts deleted file mode 100644 index 7cff2d7..0000000 --- a/services/next/app/api/session/route.ts +++ /dev/null @@ -1,12 +0,0 @@ -import { getServerSession } from "next-auth"; -import { authOptions } from "@/app/lib/auth"; -import { NextResponse } from "next/server"; - -export async function GET(request: Request) { - const session = await getServerSession(authOptions); - - return NextResponse.json({ - authenticated: !!session, - session, - }); -} diff --git a/services/next/app/api/user/route.ts b/services/next/app/api/user/route.ts deleted file mode 100644 index 48abd02..0000000 --- a/services/next/app/api/user/route.ts +++ /dev/null @@ -1,3 +0,0 @@ -import { getServerSession } from "next-auth"; -import { NextResponse, NextRequest } from "next/server"; - diff --git a/services/next/app/api/v1.json/route.ts b/services/next/app/api/v1.json/route.ts deleted file mode 100644 index 600a279..0000000 --- a/services/next/app/api/v1.json/route.ts +++ /dev/null @@ -1,91 +0,0 @@ - -import { getVodTitle } from '@/app/components/vod-page'; -import { getUrl, getAllVods } from "@/app/lib/vods" -import { IVod } from "@/app/lib/vods" - - -/* - * this is a legacy format - * - * for API version 1. - * - * @deprecated - */ -interface IVod1 { - title: string; - videoSrcHash: string; - video720Hash: string; - video480Hash: string; - video360Hash: string; - video240Hash: string; - thinHash: string; - thiccHash: string; - announceTitle: string; - announceUrl: string; - date: string; - note: string; - url: string; -} - -interface IAPI1 { - vods: IVod1[] -} - - -export async function GET(): Promise { - try { - const vodsRaw = await getAllVods(); - if (!vodsRaw) { - const options = { - headers: { - "Content-Type": "application/json", - }, - status: 500, - }; - return new Response('{}', options); - } - - const vods: IVod1[] = vodsRaw.map((v: IVod): IVod1 => ({ - title: getVodTitle(v), - videoSrcHash: v.videoSrcHash, - video720Hash: '', - video480Hash: '', - video360Hash: '', - video240Hash: v.video240Hash, - thinHash: '', - thiccHash: '', - announceTitle: v.announce_title, - announceUrl: v.announce_url, - date: v.date_2, - note: v.note || '', - url: getUrl(v, v.vtuber.slug, v.date_2), - })); - - const response = { - vods: vods, - }; - - const options = { - headers: { - "Content-Type": "application/json", - }, - }; - - return new Response(JSON.stringify(response), options); - } catch (error) { - console.error("Error fetching VODs:", error); - - const errorResponse = { - error: "An error occurred while fetching VODs", - }; - - const options = { - headers: { - "Content-Type": "application/json", - }, - status: 500, - }; - - return new Response(JSON.stringify(errorResponse), options); - } -} \ No newline at end of file diff --git a/services/next/app/archive/[uuid]/not-found.tsx b/services/next/app/archive/[uuid]/not-found.tsx deleted file mode 100644 index 785331b..0000000 --- a/services/next/app/archive/[uuid]/not-found.tsx +++ /dev/null @@ -1,12 +0,0 @@ -import Link from 'next/link' - -export default function NotFound() { - return ( -
-

404 Not Found

-

Could not find that stream archive.

- - Return to archive list -
- ) -} \ No newline at end of file diff --git a/services/next/app/archive/[uuid]/page.tsx b/services/next/app/archive/[uuid]/page.tsx deleted file mode 100644 index b9f03a8..0000000 --- a/services/next/app/archive/[uuid]/page.tsx +++ /dev/null @@ -1,21 +0,0 @@ - -import StreamPage from '@/app/components/stream-page'; -import { getStreamByUUID } from '@/app/lib/streams'; - - -interface IPageParams { - params: { - uuid: string; - } -} - - -export default async function Page ({ params: { uuid } }: IPageParams) { - const stream = await getStreamByUUID(uuid); - console.log(`getting stream by uuid. uuid=${uuid}`) - return ( - <> - - - ) -} \ No newline at end of file diff --git a/services/next/app/archive/page.tsx b/services/next/app/archive/page.tsx deleted file mode 100644 index c38ca75..0000000 --- a/services/next/app/archive/page.tsx +++ /dev/null @@ -1,31 +0,0 @@ -import Pager from "@/app/components/pager"; -import StreamsList from "@/app/components/streams-list"; -import StreamsTable from '@/app/components/streams-table'; -import { getAllStreams, getStreamsForVtuber } from "@/app/lib/streams"; -// import { getAllVtubers } from "@/app/lib/vtubers"; -import { notFound } from "next/navigation"; - - - -export default async function Page() { - - - return ( -
- {/*
-                

here are the streams object

- - {JSON.stringify(streams, null, 2)} - -
*/} - - - -

Stream Archive

- - - {/* - */} -
- ) -} \ No newline at end of file diff --git a/services/next/app/components/access-denied-screen.tsx b/services/next/app/components/access-denied-screen.tsx deleted file mode 100644 index e1f692d..0000000 --- a/services/next/app/components/access-denied-screen.tsx +++ /dev/null @@ -1,24 +0,0 @@ -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome" -import { faCircleXmark, faExternalLinkAlt } from "@fortawesome/free-solid-svg-icons" -import Link from "next/link" - -export default function AccessDeniedScreen(requiredUserRole: string, featureName: string = 'This feature') { - return <> -
-
-

Access denied

-
-
-

{featureName} is only available to {requiredUserRole}s

-

To unlock this feature, make a pledge at - patreon.com/CJ_Clippy - - -

-
-
- -} \ No newline at end of file diff --git a/services/next/app/components/archive-progress.tsx b/services/next/app/components/archive-progress.tsx deleted file mode 100644 index 1dda4b7..0000000 --- a/services/next/app/components/archive-progress.tsx +++ /dev/null @@ -1,36 +0,0 @@ -import { getAllStreamsForVtuber, getStreamCountForVtuber } from "@/app/lib/streams"; -import { getVodsForVtuber } from "@/app/lib/vods"; -import { IVtuber } from "@futureporn/types"; - -export interface IArchiveProgressProps { - vtuber: IVtuber; -} - -export default async function ArchiveProgress ({ vtuber }: IArchiveProgressProps) { - - // // Check if totalStreams is not zero before calculating completedPercentage - // const completedPercentage = (totalStreams !== 0) ? Math.round(eligibleStreams / totalStreams * 100) : 0; - const goodStreams = await getStreamCountForVtuber(vtuber.id, ['good']) - const issueStreams = await getStreamCountForVtuber(vtuber.id, ['issue']) - const totalStreams = await getStreamCountForVtuber(vtuber.id) - const eligibleStreams = goodStreams+issueStreams - const completedPercentage = Math.floor((eligibleStreams / totalStreams) * 100) - return ( -
- {/*

- {totalStreams} known streams
- {goodStreams} streams archived
- {issueStreams} streams with issues
-

*/} -

{eligibleStreams}/{totalStreams} Streams Archived ({completedPercentage}%)

- {completedPercentage}% -
- ) - // @todo - - // return ( - //
- //

{(vods) ? vods.data.length : 0} vods

- //
- // ) -} \ No newline at end of file diff --git a/services/next/app/components/auth-buttons.tsx b/services/next/app/components/auth-buttons.tsx deleted file mode 100644 index eb2efcb..0000000 --- a/services/next/app/components/auth-buttons.tsx +++ /dev/null @@ -1,41 +0,0 @@ -"use client"; - -import { signIn, signOut } from "next-auth/react"; -import Link from "next/link"; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faUser } from "@fortawesome/free-solid-svg-icons"; - -export const LoginButton = () => { - return ( - - ); -}; - -export const RegisterButton = () => { - return ( - - Register - - ); -}; - -export const LogoutButton = () => { - return ( - - ); -}; - -export const ProfileButton = () => { - return ( - - Profile - - ) -}; diff --git a/services/next/app/components/auth-provider.tsx b/services/next/app/components/auth-provider.tsx deleted file mode 100644 index 04e162a..0000000 --- a/services/next/app/components/auth-provider.tsx +++ /dev/null @@ -1,11 +0,0 @@ -"use client"; - -import { SessionProvider } from "next-auth/react"; - -type Props = { - children: React.ReactNode; -} - -export default function AuthProvider({ children }: Props) { - return {children}; -} \ No newline at end of file diff --git a/services/next/app/components/contributors.tsx b/services/next/app/components/contributors.tsx deleted file mode 100644 index 7f0d8d7..0000000 --- a/services/next/app/components/contributors.tsx +++ /dev/null @@ -1,40 +0,0 @@ -import Skeleton, { SkeletonTheme } from "react-loading-skeleton"; -import { getContributors } from "../lib/contributors"; -import Link from 'next/link'; -import { faExternalLinkAlt } from "@fortawesome/free-solid-svg-icons"; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; - -export default async function Contributors() { - try { - const contributors = await getContributors(); - if (!contributors || contributors.length < 1) return ( - - - - ) - const contributorList = contributors.map((contributor, index) => ( - - {contributor.attributes.url ? ( - - {contributor.attributes.name} - - - ) : ( - contributor.attributes.name - )} - {index !== contributors.length - 1 ? ", " : ""} - - )); - return ( - <>{contributorList} - ) - } catch (e) { - if (e instanceof Error) { - console.error(e) - } - return

Failed to fetch contributor list

- } -} \ No newline at end of file diff --git a/services/next/app/components/error-card.tsx b/services/next/app/components/error-card.tsx deleted file mode 100644 index d16beec..0000000 --- a/services/next/app/components/error-card.tsx +++ /dev/null @@ -1,9 +0,0 @@ -export function ErrorCard(props: { message: string }) { - const { message } = props - return ( -
- - {message} -
- ) -} \ No newline at end of file diff --git a/services/next/app/components/footer.tsx b/services/next/app/components/footer.tsx deleted file mode 100644 index 3f162f5..0000000 --- a/services/next/app/components/footer.tsx +++ /dev/null @@ -1,117 +0,0 @@ -import Link from "next/link"; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faExternalLinkAlt } from "@fortawesome/free-solid-svg-icons"; -import { faGit, faReddit, faDiscord, faPatreon } from "@fortawesome/free-brands-svg-icons"; -import Contributors from "./contributors"; -import PatronsList from "./patrons-list"; - -export default function Footer() { - return ( - <> -
-
- -
-
-

Sitemap

-
    -
  • ↑ Top of page
  • -
  • Vtubers
  • -
  • Archive
  • -
  • About
  • -
  • FAQ
  • -
  • Goals
  • -
  • Patrons
  • -
  • Tags
  • -
  • RSS Feed
  • -
  • API
  • -
  • Status
  • - {/*
  • Upload
  • */} -
  • Profile
  • -
-
-
-

- Futureporn.net is made with ❤️ by CJ_Clippy -

-

- Made possible by generous - - donations - - - - from - -

-

- VOD contributions by -

-

- - - Git Repo - - -

- -

- - - Reddit Thread - - -

- -

- - - Discord Server - - -

-
-
- -
- -
- - ) -} \ No newline at end of file diff --git a/services/next/app/components/funding-goal.tsx b/services/next/app/components/funding-goal.tsx deleted file mode 100644 index 2a96cbc..0000000 --- a/services/next/app/components/funding-goal.tsx +++ /dev/null @@ -1,88 +0,0 @@ - -import { getCampaign } from "@/app/lib/patreon"; -import { getGoals, IGoals } from '@/app/lib/pm' -import Image from "next/legacy/image"; -import React from 'react'; -import Link from 'next/link' - - - -export default async function FundingGoal(): Promise { - const campaignData = await getCampaign(); - const { pledgeSum, patronCount } = campaignData; - - const goals = await getGoals(pledgeSum); - if (!goals || !goals?.featuredFunded?.amountCents || !goals?.featuredUnfunded?.amountCents || !goals?.featuredFunded?.amountCents || !goals?.featuredUnfunded?.completedPercentage || !goals?.featuredFunded?.completedPercentage ) return <> - - return ( - <> - {/*

- pledgeSum:{JSON.stringify(pledgeSum, null, 2)} -

-

- patronCount:{JSON.stringify(patronCount, null, 2)} -

-

featuredFunded:{JSON.stringify(goals.featuredFunded)}

-

featuredUnfunded:{JSON.stringify(goals.featuredUnfunded)}

*/} - - {/*
-                
-                    {JSON.stringify(goals, null, 2)}
-                
-            
*/} -
- - -
-
- Funding Goal -
- - CJ_Clippy - -
-
-
-
- {/* the most recently funded goal */} -
- {/* const { featuredFunded, featuredUnfunded } = goals; - if (!featuredFunded?.amountCents || !featuredFunded?.completedPercentage) return <> - if (!featuredUnfunded?.amountCents || !featuredUnfunded?.completedPercentage) return <> */} - -

${(goals.featuredFunded.amountCents * (goals.featuredFunded.completedPercentage * 0.01) / 100)} of {goals.featuredFunded.amountCents / 100} ({goals.featuredFunded.completedPercentage}%) -

-
- FUNDED -
-

{goals.featuredFunded.description}

-
- - {/* the next unfunded goal */} -
-

${(goals.featuredUnfunded.amountCents * (goals.featuredUnfunded.completedPercentage * 0.01) / 100) | 0} of ${goals.featuredUnfunded.amountCents / 100} ({goals.featuredUnfunded.completedPercentage}%)

- - {goals.featuredUnfunded.completedPercentage}% - -

{goals.featuredUnfunded.description}

-
-
- -

- Thank you, Patrons! -

-
-
-
- - ); -}; - diff --git a/services/next/app/components/icons/carrd.tsx b/services/next/app/components/icons/carrd.tsx deleted file mode 100644 index d900cda..0000000 --- a/services/next/app/components/icons/carrd.tsx +++ /dev/null @@ -1,8 +0,0 @@ -import * as React from "react" -const SvgComponent = (props: any) => ( - - {"Carrd"} - - -) -export default SvgComponent diff --git a/services/next/app/components/icons/chaturbate.tsx b/services/next/app/components/icons/chaturbate.tsx deleted file mode 100644 index 31c641f..0000000 --- a/services/next/app/components/icons/chaturbate.tsx +++ /dev/null @@ -1,14 +0,0 @@ -import * as React from "react" -const SvgComponent = (props: any) => ( - - - -) -export default SvgComponent diff --git a/services/next/app/components/icons/fansly.tsx b/services/next/app/components/icons/fansly.tsx deleted file mode 100644 index 03a78dc..0000000 --- a/services/next/app/components/icons/fansly.tsx +++ /dev/null @@ -1,20 +0,0 @@ -import * as React from "react" -const SvgComponent = (props: any) => ( - - - - -) -export default SvgComponent diff --git a/services/next/app/components/icons/linktree.tsx b/services/next/app/components/icons/linktree.tsx deleted file mode 100644 index 3e17f8b..0000000 --- a/services/next/app/components/icons/linktree.tsx +++ /dev/null @@ -1,30 +0,0 @@ -import * as React from "react" -const SvgComponent = (props: any) => ( - - - - - - - - - - - -) -export default SvgComponent diff --git a/services/next/app/components/icons/onlyfans.tsx b/services/next/app/components/icons/onlyfans.tsx deleted file mode 100644 index 81a568a..0000000 --- a/services/next/app/components/icons/onlyfans.tsx +++ /dev/null @@ -1,55 +0,0 @@ -import * as React from "react" -const SvgComponent = (props: any) => ( - - - - - - -) -export default SvgComponent diff --git a/services/next/app/components/icons/pornhub.tsx b/services/next/app/components/icons/pornhub.tsx deleted file mode 100644 index 5f7a746..0000000 --- a/services/next/app/components/icons/pornhub.tsx +++ /dev/null @@ -1,23 +0,0 @@ -import * as React from "react" -const SvgComponent = (props: any) => ( - - - - - - - -) -export default SvgComponent diff --git a/services/next/app/components/icons/throne.tsx b/services/next/app/components/icons/throne.tsx deleted file mode 100644 index 897285c..0000000 --- a/services/next/app/components/icons/throne.tsx +++ /dev/null @@ -1,31 +0,0 @@ -import * as React from "react" -const SvgComponent = (props: any) => ( - - - - - - - - - - - - -) -export default SvgComponent diff --git a/services/next/app/components/ipfs-cid.tsx b/services/next/app/components/ipfs-cid.tsx deleted file mode 100644 index efbde66..0000000 --- a/services/next/app/components/ipfs-cid.tsx +++ /dev/null @@ -1,42 +0,0 @@ -'use client'; - -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faCopy, faCheck } from "@fortawesome/free-solid-svg-icons"; -import { useState } from "react"; -import styles from '@/assets/styles/cid.module.css' - -interface IIpfsCidProps { - label?: string; - cid: string; -} - - -export function IpfsCid({ label, cid }: IIpfsCidProps) { - - const [isCopied, setIsCopied] = useState(false); - - - - return ( -
- {label} -
{cid}
- {(isCopied) ? - - : - { - navigator.clipboard.writeText(cid) - setIsCopied(true) - setTimeout(() => setIsCopied(false), 3000) - }} - > - } -
- ) -} diff --git a/services/next/app/components/ipfs-logo.tsx b/services/next/app/components/ipfs-logo.tsx deleted file mode 100644 index 3875418..0000000 --- a/services/next/app/components/ipfs-logo.tsx +++ /dev/null @@ -1,18 +0,0 @@ -import React from 'react'; - -interface LogoProps { - size: number; - color: string; -} - -const IPFSLogo: React.FC = ({ size = 32, color = '#65C2CB' }) => { - - return ( - - IPFS - - - ); -}; - -export default IPFSLogo; \ No newline at end of file diff --git a/services/next/app/components/ipfs.tsx b/services/next/app/components/ipfs.tsx deleted file mode 100644 index cacfea8..0000000 --- a/services/next/app/components/ipfs.tsx +++ /dev/null @@ -1,39 +0,0 @@ -'use client'; - -// import { type Helia, createHelia } from 'helia'; -// import React, { useState, useEffect } from 'react'; - -// export default function Ipfs () { -// const [id, setId] = useState(null) -// const [helia, setHelia] = useState(null) -// const [isOnline, setIsOnline] = useState(false) - -// useEffect(() => { -// const init = async () => { -// if (helia) return - -// const heliaNode = await createHelia(); - -// const nodeId = heliaNode.libp2p.peerId.toString(); -// const nodeIsOnline = heliaNode.libp2p.isStarted(); - -// setHelia(heliaNode); -// setId(nodeId); -// setIsOnline(nodeIsOnline); -// } - -// init() -// }, [helia]) - -// if (!helia || !id) { -// return

Connecting to IPFS...

-// } - -// return ( -//
-//

ID: {id.toString()}

-//

Status: {isOnline ? 'Online' : 'Offline'}

-//
-// ) -// } - diff --git a/services/next/app/components/linkable-heading.tsx b/services/next/app/components/linkable-heading.tsx deleted file mode 100644 index c5e782a..0000000 --- a/services/next/app/components/linkable-heading.tsx +++ /dev/null @@ -1,28 +0,0 @@ -import Link from "next/link"; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { IconDefinition, faLink } from "@fortawesome/free-solid-svg-icons"; - -interface ILinkableHeadingProps { - icon?: IconDefinition; - text: string; - slug: string; -} - -export default function LinkableHeading({ icon, text, slug }: ILinkableHeadingProps) { - return ( -

- {icon && } - {text} - - - - -

- ) -} \ No newline at end of file diff --git a/services/next/app/components/localized-date.tsx b/services/next/app/components/localized-date.tsx deleted file mode 100644 index 44ad6ec..0000000 --- a/services/next/app/components/localized-date.tsx +++ /dev/null @@ -1,26 +0,0 @@ -import { formatISO } from "date-fns"; - -interface ILocalizedDateProps { - date: Date; -} - -/** - * - * this causes hydration issues - * - * Error: Text content does not match server-rendered HTML. - * - * Warning: Text content did not match. Server: "2024-01-01" Client: "2023-12-31" - * - * See more info here: https://nextjs.org/docs/messages/react-hydration-error - */ - -export function LocalizedDate ({ date }: ILocalizedDateProps) { - const isoDateTime = formatISO(date); - const isoDate = formatISO(date, { representation: 'date' }); - return ( - <> - - - ) -} \ No newline at end of file diff --git a/services/next/app/components/navbar.tsx b/services/next/app/components/navbar.tsx deleted file mode 100644 index 8d8dc1d..0000000 --- a/services/next/app/components/navbar.tsx +++ /dev/null @@ -1,93 +0,0 @@ -'use client' - -import { useState } from 'react' -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faExternalLinkAlt, faUpload } from "@fortawesome/free-solid-svg-icons"; -import Link from 'next/link' -import { LoginButton, ProfileButton } from './auth-buttons'; -import { Spinner } from './spinner'; -import ProtectedRoute from "../components/protected-route"; - -export default function Navbar() { - const [isExpanded, setExpanded] = useState(false); - - - - const handleBurgerClick = () => { - setExpanded(!isExpanded); - }; - - return ( - <> - - - ) -} \ No newline at end of file diff --git a/services/next/app/components/notification-center.tsx b/services/next/app/components/notification-center.tsx deleted file mode 100644 index 31218c6..0000000 --- a/services/next/app/components/notification-center.tsx +++ /dev/null @@ -1,13 +0,0 @@ -'use client'; - -import { ToastContainer } from 'react-toastify'; -import 'react-toastify/dist/ReactToastify.css'; - - -export default function NotificationCenter() { - return ( - - ) -} \ No newline at end of file diff --git a/services/next/app/components/notifications.tsx b/services/next/app/components/notifications.tsx deleted file mode 100644 index 1968ecd..0000000 --- a/services/next/app/components/notifications.tsx +++ /dev/null @@ -1,10 +0,0 @@ - -export function DangerNotification ({ errors }: { errors: String[] }): JSX.Element { - return ( -
- {errors && errors.map((error, index) => ( -

Error:{error}

- ))} -
- ); -} \ No newline at end of file diff --git a/services/next/app/components/pager.tsx b/services/next/app/components/pager.tsx deleted file mode 100644 index d1680e4..0000000 --- a/services/next/app/components/pager.tsx +++ /dev/null @@ -1,92 +0,0 @@ -import Link from 'next/link'; - -interface IPagerProps { - baseUrl: string; // Pass the base URL as a prop - page: number; - pageCount: number; -} - -export default function Pager({ baseUrl, page, pageCount }: IPagerProps): React.JSX.Element { - const pageNumbers = Array.from({ length: pageCount }, (_, i) => i + 1); - - const getPagePath = (page: any): string => { - const pageNumber = parseInt(page); - return `${baseUrl}/${pageNumber}`; - }; - - const getNextPagePath = (page: any): string => { - const pageNumber = parseInt(page); - return `${baseUrl}/${pageNumber+1}`; - } - - const getPreviousPagePath = (page: any): string => { - const pageNumber = parseInt(page); - return `${baseUrl}/${pageNumber-1}` - } - - // Define the number of page links to show around the current page - const maxPageLinksToShow = 3; - - // Calculate the range of page numbers to display - const startPage = Math.max(1, page - Math.floor(maxPageLinksToShow / 2)); - const endPage = Math.min(pageCount, startPage + maxPageLinksToShow - 1); - - return ( -
- -
- ); -} diff --git a/services/next/app/components/patron-perks.tsx b/services/next/app/components/patron-perks.tsx deleted file mode 100644 index b6f796e..0000000 --- a/services/next/app/components/patron-perks.tsx +++ /dev/null @@ -1,51 +0,0 @@ -import PatronRoute from "../components/protected-route"; -import { useMutateMetadata, useMetadata } from "../profile/hooks/useMetadata"; -import { faCheckCircle } from "@fortawesome/free-solid-svg-icons"; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import VibrateTest from "./vibrate-test"; -import { useState, useEffect } from "react"; -import { faSpinner, faXmarkCircle, faSave } from "@fortawesome/free-solid-svg-icons"; -import Skeleton, { SkeletonTheme } from "react-loading-skeleton" -import 'react-loading-skeleton/dist/skeleton.css'; - -export default function PatronPerks() { - - - - - const { status, data, error, isFetching, isPending } = useMetadata() - const mutateMetadata = useMutateMetadata() - - - - const username = `${data?.metadata?.first_name || ''} ${data?.metadata?.last_name || ''}`.trim(); - const { metadata } = data || {}; - - const [ isUsernamePublic, setIsUsernamePublic ] = useState(metadata?.isUsernamePublic || false) - useEffect(() => { - mutateMetadata.mutate({ isUsernamePublic }) - }, [ isUsernamePublic ]) - - - - return ( - <> -

Patron Perks

-

Website Shoutout

- -
- Display {(!!username) ? username : } publicly? - setIsUsernamePublic(evt.target.checked)}/> - {(mutateMetadata.status === 'success') && } - {(mutateMetadata.status === 'pending') && } - {(mutateMetadata.status === 'error') && } -
-
- - -

Vibrate (test)

- - - - ) -} \ No newline at end of file diff --git a/services/next/app/components/patrons-list.tsx b/services/next/app/components/patrons-list.tsx deleted file mode 100644 index 0990914..0000000 --- a/services/next/app/components/patrons-list.tsx +++ /dev/null @@ -1,52 +0,0 @@ -import Skeleton, { SkeletonTheme } from 'react-loading-skeleton'; -import 'react-loading-skeleton/dist/skeleton.css'; -import { getPatrons } from '../lib/patreon'; -import Link from 'next/link' - - -interface PatronsListProps { - displayStyle: string; -} - -export default async function PatronsList({ displayStyle }: PatronsListProps) { - const patrons = await getPatrons() - console.log('patrons list as follows') - console.log(patrons) - - if (!patrons || patrons.length === 0) return ( - - - - ); - - if (displayStyle === 'box') { - return ( -
- {patrons.map((patron) => ( - -
-
-
-
-
- {patron.full_name && ( - - {patron.full_name} - - )} -
-
-
-
-
- ))} -
- ); - } else if (displayStyle === 'list') { - const patronNames = patrons.map((patron) => patron.full_name.trim()).join(', '); - return {patronNames}; - } else { - return ; // Handle unsupported display styles or provide a default display style - } -} - diff --git a/services/next/app/components/permissions-table.tsx b/services/next/app/components/permissions-table.tsx deleted file mode 100644 index f162268..0000000 --- a/services/next/app/components/permissions-table.tsx +++ /dev/null @@ -1,8 +0,0 @@ - -// @see https://gitea.futureporn.net/futureporn/pm/issues/106 - -export default function PermissionsTable() { - return <> -

@todo permissions table

- -} \ No newline at end of file diff --git a/services/next/app/components/protected-route.tsx b/services/next/app/components/protected-route.tsx deleted file mode 100644 index 17cf6db..0000000 --- a/services/next/app/components/protected-route.tsx +++ /dev/null @@ -1,53 +0,0 @@ -'use client' - -import { useSession } from "next-auth/react" -import { Spinner } from "./spinner"; - -// type AuthContentProps = { -// loadingSlot?: React.ReactNode; -// authedSlot: React.ReactNode; -// unauthSlot: React.ReactNode; -// }; - -// export default function AuthContent({ -// loadingSlot, -// authedSlot, -// unauthSlot, -// }: AuthContentProps) { -// const { data: session, status } = useSession(); - -// if (status === 'loading') { -// return <>{!!loadingSlot ? loadingSlot : } -// } - -// if (status === 'authenticated') { -// return <>{authedSlot}; -// } - -// return <>{unauthSlot}; -// } - - -import React from "react"; -import AccessDeniedScreen from './access-denied-screen'; - -interface ProtectedRouteProps extends React.PropsWithChildren { - requiredUserRole: string; - featureName?: string; - loading?: React.ReactNode; - accessDenied?: React.ReactNode; -} - -const ProtectedRoute = (props: ProtectedRouteProps) => { - const { data: session, status } = useSession(); - - if (status === 'loading') return (props.loading) ? props.loading : ; - if (status !== 'authenticated' || !session.roles || !session.roles.includes(props.requiredUserRole)) { - return (!!props?.accessDenied) ? props.accessDenied : AccessDeniedScreen(props.requiredUserRole, props.featureName); - } else { - return props.children; - } - -} - -export default ProtectedRoute \ No newline at end of file diff --git a/services/next/app/components/query-provider.tsx b/services/next/app/components/query-provider.tsx deleted file mode 100644 index 99ea319..0000000 --- a/services/next/app/components/query-provider.tsx +++ /dev/null @@ -1,27 +0,0 @@ -// @see https://github.com/TanStack/query/issues/4933#issuecomment-1416892904 - -'use client'; - -import { QueryClient, QueryClientProvider } from '@tanstack/react-query' -import { useState } from 'react'; - - - -export default function QueryProvider({ - children, -}: { - children: React.ReactNode -}) { - const [queryClient] = useState(() => new QueryClient({ - defaultOptions: { - queries: { - staleTime: 60 * 1000 - } - } - })) - return ( - - {children} - - ); -} \ No newline at end of file diff --git a/services/next/app/components/sortable-tags.tsx b/services/next/app/components/sortable-tags.tsx deleted file mode 100644 index 82a01ee..0000000 --- a/services/next/app/components/sortable-tags.tsx +++ /dev/null @@ -1,70 +0,0 @@ -'use client' - -import React, { useState } from 'react'; -import { ITag } from '../lib/tags'; -import Link from 'next/link'; -import slugify from 'slugify'; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faFilter } from "@fortawesome/free-solid-svg-icons"; - -interface ISortableTagsProps { - tags: ITag[]; -} - -export default function SortableTags({ tags }: ISortableTagsProps) { - const [filterText, setFilterText] = useState(''); - const [sortOption, setSortOption] = useState('Sort'); - - const filteredTags = tags.filter((tag: ITag) => - tag.name.toLowerCase().includes(filterText.toLowerCase()) - ); - - const sortedTags = [...filteredTags].sort((a, b) => { - if (sortOption === 'Alphabetical') { - return a.name.localeCompare(b.name); - } else if (sortOption === 'Frequency') { - return b.count - a.count; - } - return 0; - }); - - return ( - <> -
-
- setFilterText(e.target.value)} - /> - - - -
-
-
- -
-
-
-
- {sortedTags.map((tag: ITag) => ( - - - {tag.name} ({tag.count}) - - - ))} -
- - ); -} diff --git a/services/next/app/components/spinner.tsx b/services/next/app/components/spinner.tsx deleted file mode 100644 index 23a2ba3..0000000 --- a/services/next/app/components/spinner.tsx +++ /dev/null @@ -1,6 +0,0 @@ -import { faSpinner } from "@fortawesome/free-solid-svg-icons"; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; - -export function Spinner() { - return -} \ No newline at end of file diff --git a/services/next/app/components/stream-button.tsx b/services/next/app/components/stream-button.tsx deleted file mode 100644 index 97b946f..0000000 --- a/services/next/app/components/stream-button.tsx +++ /dev/null @@ -1,17 +0,0 @@ -import { IStream } from "@futureporn/types"; -import Link from "next/link" -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faCalendar } from "@fortawesome/free-solid-svg-icons"; - -export function StreamButton({ stream }: { stream: IStream }) { - if (!stream) return <> - - return ( - - {new Date(stream.date).toLocaleDateString()} - - ) -} \ No newline at end of file diff --git a/services/next/app/components/stream-page.tsx b/services/next/app/components/stream-page.tsx deleted file mode 100644 index da4e9a8..0000000 --- a/services/next/app/components/stream-page.tsx +++ /dev/null @@ -1,203 +0,0 @@ -import { IStream } from "@futureporn/types"; -import { IVod } from "@/app/lib/vods"; -import Link from "next/link"; -import Image from "next/legacy/image"; -import { LocalizedDate } from "./localized-date"; -import { FontAwesomeIcon, FontAwesomeIconProps } from "@fortawesome/react-fontawesome"; -import { faTriangleExclamation, faCircleInfo, faThumbsUp, IconDefinition, faO, faX, faExternalLinkAlt } from "@fortawesome/free-solid-svg-icons"; -import { faXTwitter } from "@fortawesome/free-brands-svg-icons"; -import { notFound } from "next/navigation"; -import ProtectedRoute from "./protected-route"; - -export interface IStreamProps { - stream: IStream; -} -type Status = 'missing' | 'issue' | 'good'; -interface StyleDef { - heading: string; - icon: IconDefinition; - description: string; - prompt: string; -} - -function capitalizeFirstLetter(string: string): string { - return string.charAt(0).toUpperCase() + string.slice(1); -} - -function hasNote(vod: IVod) { - if (!!vod?.note) return true; - else return false; -} - -function determineStatus(stream: IStream): Status { - if (stream.vods.length < 1) { - return 'missing' - } else { - if (stream.vods.some((vod: IVod) => !hasNote(vod))) { - return 'good'; - } else { - return 'issue'; - } - } -} - -export default function StreamPage({ stream }: IStreamProps) { - console.log('StreamPage function has been invoked! stream as follows') - console.log(stream) - if (!stream) notFound() - const displayName = stream.vtuber.display_name; - const date = new Date(stream.date); - const selectedStatus = determineStatus(stream); - - const styleMap: Record = { - 'missing': { - heading: 'is-danger', - icon: faTriangleExclamation, - description: "We don't have a VOD for this stream.", - prompt: 'Know someone who does?' - }, - 'issue': { - heading: 'is-warning', - icon: faCircleInfo, - description: "We have a VOD for this stream, but it's not full quality.", - prompt: 'Have a better copy?' - }, - 'good': { - heading: 'is-success', - icon: faThumbsUp, - description: "We have a VOD for this stream, and we think it's the best quality possible.", - prompt: "Have one that's even better?" - } - }; - const { heading, icon, description, prompt } = styleMap[selectedStatus] || {}; - - - if (!stream) return

NotFound

- // - - // return

- //

-    //         
-    //             {JSON.stringify(stream, null, 2)}
-
-    //         
-    //     
- - //

- // const platformsList = [ - // stream.attributes.isChaturbateStream ? 'Chaturbate' : null, - // stream.attributes.isFanslyStream ? 'Fansly' : null - // ].filter(Boolean).join(', '); - // platformsList = platformsArray.length > 0 ? platformsArray.join(', ') : 'None'; - - // const platformsList = [ - // (stream.attributes.isChaturbateStream && 'CB'), - // (stream.attributes.isFanslyStream && 'Fansly') - // ].filter(Boolean).join(', ') - - const platformsList = [ - (stream.is_chaturbate_stream && 'CB'), - (stream.is_fansly_stream && 'Fansly') - ].filter(Boolean).join(', ') || '!!!'; - - - - return ( - <> - - -
-
-

{displayName} Stream Archive

-
- -
-
-
-
-
- - - - - - - - - - - - - - - - - - - - - -
DescriptionDetails
Platform{platformsList}
UTC Datetime
Local Datetime{date.toLocaleDateString()} {date.toLocaleTimeString()}
- -
-
-
-
- - -
-
-
- VOD {capitalizeFirstLetter(selectedStatus)} -
-
- -

{description}

- } > -

{prompt}
- Upload it here.

-
-
-
-
- -
- - - {stream.vods.length !== 0 && -
-

VODs

- - - - - - {/* - */} - - - - - - - {stream.vods.map((vod: IVod) => ( - - {/*

{JSON.stringify(vod, null, 2)}

*/} - - - {/* - */} - - - - - ))} - -
IDUpload DateThumbnailDurationTagsTimestampsNote
{vod.uuid}{vod.publishedAt}{(!!vod?.attributes?.thumbnail?.data?.attributes?.cdnUrl) ? : }{(!!vod?.attributes?.duration) ? vod.duration : }{vod.tagVodRelations.length}{vod.timestamps.length}{(!!vod.note) ? : }
-
} - -
- - - ) -} diff --git a/services/next/app/components/stream.tsx b/services/next/app/components/stream.tsx deleted file mode 100644 index 2e73f35..0000000 --- a/services/next/app/components/stream.tsx +++ /dev/null @@ -1,86 +0,0 @@ -import { IStream } from "@futureporn/types"; -import NotFound from "@/app/vt/[slug]/not-found"; -import { LocalizedDate } from "./localized-date"; -import Link from "next/link"; -import ChaturbateIcon from "@/app/components/icons/chaturbate"; -import FanslyIcon from "@/app/components/icons/fansly"; -import Image from "next/legacy/image"; - -export interface IStreamProps { - stream: IStream; -} - - -export function Stream({ stream }: IStreamProps) { - if (!stream) return - return ( -
-
-                
-                    {JSON.stringify(stream, null, 2)}
-                
-            
- {/*

Stream {stream.attributes.date}

*/} -
- ) -} - - - -export function StreamSummary ({ stream }: IStreamProps) { - if (!stream) return - - // return ( - //
-    //         
-    //             {JSON.stringify(stream, null, 2)}
-    //         
-    //     
- // ) - - const archiveStatus = stream.attributes.archiveStatus; - const archiveStatusClassName = (() => { - if (archiveStatus === 'missing') return 'is-danger'; - if (archiveStatus === 'good') return 'is-success'; - if (archiveStatus === 'issue') return 'is-warning'; - })(); - - return ( - -
- {/*
-                    
-                        {JSON.stringify(stream, null, 2)}
-                    
-                
*/} -
- - {stream.vtuber.display_name} - -
-
- {stream.vtuber.display_name} -
-
- -
-
- {(stream.is_chaturbate_stream) && } - {(stream.is_fansly_stream) && } -
-
-
{stream.archive_status}
-
-
-
-
-
- - ) -} \ No newline at end of file diff --git a/services/next/app/components/streams-list.tsx b/services/next/app/components/streams-list.tsx deleted file mode 100644 index 9b0edee..0000000 --- a/services/next/app/components/streams-list.tsx +++ /dev/null @@ -1,58 +0,0 @@ -import React from 'react' -import Link from 'next/link'; -import { IVtuber, IStream } from '@futureporn/types'; -import { notFound } from 'next/navigation'; -import { getAllStreams } from '@/app/lib/streams'; -import { StreamSummary } from '@/app/components/stream'; - -interface IStreamsListProps { - vtubers: IVtuber[]; - page: number; - pageSize: number; -} - - -interface IStreamsListHeadingProps { - slug: string; - displayName: string; -} - -export function StreamsListHeading({ slug, displayName }: IStreamsListHeadingProps): React.JSX.Element { - return ( -
-

- {displayName} Streams -

-
- ) -} - - -export default async function StreamsList({ vtubers, page = 1, pageSize = 24 }: IStreamsListProps): Promise { - if (!vtubers) return
vtubers is not defined. vtubers:{JSON.stringify(vtubers, null, 2)}
- - // const streams = await getStreamsForVtuber(vtubers[0].id); - const streams = await getAllStreams(['missing', 'issue', 'good']); - - if (!streams) return notFound(); - - - // @todo [ ] pagination - // @todo [ ] sortability - return ( - <> - -

Stream Archive

- - - ); -} diff --git a/services/next/app/components/streams-table.tsx b/services/next/app/components/streams-table.tsx deleted file mode 100644 index fd7a2ee..0000000 --- a/services/next/app/components/streams-table.tsx +++ /dev/null @@ -1,287 +0,0 @@ -'use client' -import React from 'react' -import ReactDOM from 'react-dom/client' -import Link from 'next/link' -import { defaultImageBlur } from '../lib/constants' -import { - keepPreviousData, - QueryClient, - useQuery, -} from '@tanstack/react-query' -import { format } from 'date-fns' -import Image from "next/legacy/image" -import { - PaginationState, - useReactTable, - getCoreRowModel, - ColumnDef, - flexRender, -} from '@tanstack/react-table' -import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' -import { faSpinner } from '@fortawesome/free-solid-svg-icons' - -import { fetchStreamData } from '@/app/lib/streams' -import { IStream } from '@futureporn/types' - -const queryClient = new QueryClient() - -function getStatusClass(value: string) { - switch (value) { - case 'issue': - return 'is-warning'; - case 'missing': - return 'is-danger'; - case 'good': - return 'is-success'; - default: - return ''; - } -} - -export default function StreamsTable() { - const rerender = React.useReducer(() => ({}), {})[1] - - // image & name - // title - // platform - // date & time - // archiveStatus - const columns = React.useMemo[]>( - () => [ - { - header: 'VTuber', - accessorFn: d => ({ - displayName: d.vtuber.display_name, - image: d.vtuber.image, - imageBlur: d.vtuber.image_blur - }), - cell: info => { - const { displayName, image, imageBlur } = info.getValue<{ displayName: string, image: string, imageBlur: string }>(); - return ( - <> -
-
-
- {displayName} -
-
-
- {displayName} -
-
- - ) - } - }, - { - header: 'Date', - accessorFn: d => format(new Date(d.date), 'yyyy-MM-dd HH:mm'), - sortingFn: 'datetime', - sortDescFirst: true, - cell: info => ({info.getValue() as string}) - }, - { - header: 'Platform', - accessorFn: d => [ - (d.is_chaturbate_stream && 'CB'), - (d.is_fansly_stream && 'Fansly') - ].filter(Boolean).join(', ') || '???' - }, - { - header: 'Status', - accessorFn: d => { - if (!d.archive_status) return 'missing'; - return d.archive_status - } - }, - // { - // header: 'Name', - // footer: props => props.column.id, - // columns: [ - // { - // accessorKey: 'firstName', - // cell: info => info.getValue(), - // footer: props => props.column.id, - // }, - // { - // accessorFn: row => row.lastName, - // id: 'lastName', - // cell: info => info.getValue(), - // header: () => Last Name, - // footer: props => props.column.id, - // }, - // ], - // }, - ], - [] - ) - - const [pagination, setPagination] = React.useState({ - pageIndex: 0, - pageSize: 50, - }) - - const { data, error, isPending } = useQuery({ - queryKey: ['streams', pagination.pageIndex, pagination.pageSize], - queryFn: () => fetchStreamData(pagination), - placeholderData: keepPreviousData, // don't have 0 rows flash while changing pages/loading next page, - staleTime: 1000, - }, queryClient) - - const defaultData = React.useMemo(() => [], []) - - const table = useReactTable({ - data: data?.rows ?? defaultData, - columns, - // pageCount: dataQuery.data?.pageCount ?? -1, //you can now pass in `rowCount` instead of pageCount and `pageCount` will be calculated internally (new in v8.13.0) - rowCount: data?.rowCount, // new in v8.13.0 - alternatively, just pass in `pageCount` directly - state: { - pagination, - }, - onPaginationChange: setPagination, - getCoreRowModel: getCoreRowModel(), - manualPagination: true, //we're doing manual "server-side" pagination - // getPaginationRowModel: getPaginationRowModel(), // If only doing manual pagination, you don't need this - debugTable: true, - }) - - return ( -
-
- - {isPending && } - {!isPending && <> - - - {table.getHeaderGroups().map(headerGroup => ( - - {headerGroup.headers.map(header => { - return ( - - ) - })} - - ))} - - - {table.getRowModel().rows.map(row => { - return ( - - {row.getVisibleCells().map(cell => { - return ( - - ) - })} - - ) - })} - -
- {header.isPlaceholder ? null : ( -
- {flexRender( - header.column.columnDef.header, - header.getContext() - )} -
- )} -
- {flexRender( - cell.column.columnDef.cell, - cell.getContext() - )} -
- - -
-
- - - - -
-
- Page - - {table.getState().pagination.pageIndex + 1} of{' '}{table.getPageCount().toLocaleString()} - -
-
- - {/* second row with page number input and pages-per-screen select */} -
-
- Go to page: -
-
- { - const page = e.target.value ? Number(e.target.value) - 1 : 0 - table.setPageIndex(page) - }} - className="input" - /> -
-
-
- -
-
-
- } - - - -
- - ) -} diff --git a/services/next/app/components/tag-button.tsx b/services/next/app/components/tag-button.tsx deleted file mode 100644 index dda1ea5..0000000 --- a/services/next/app/components/tag-button.tsx +++ /dev/null @@ -1,8 +0,0 @@ - -import { useState } from 'react'; - -export function TagButton ({ name, selectedTag, setSelectedTag }: { name: string, selectedTag: string | null, setSelectedTag: Function }) { - return ( - - ) -} \ No newline at end of file diff --git a/services/next/app/components/tag.tsx b/services/next/app/components/tag.tsx deleted file mode 100644 index 9d49264..0000000 --- a/services/next/app/components/tag.tsx +++ /dev/null @@ -1,56 +0,0 @@ -'use client'; - -import { ITagVodRelation } from "@/app/lib/tag-vod-relations" -import { isWithinInterval, subHours } from "date-fns"; -import { faTrash } from "@fortawesome/free-solid-svg-icons"; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { useContext, useEffect, useState } from "react"; -import { useRouter } from 'next/navigation'; -import { postgrestLocalUrl } from "@/app/lib/constants"; - -export interface ITagParams { - tvr: ITagVodRelation; -} - - -function isCreatedByMeRecently(userId: number | undefined, tvr: ITagVodRelation) { - if (!userId) return false; - if (userId !== tvr.attributes.creatorId) return false; - const last24H: Interval = { start: subHours(new Date(), 24), end: new Date() }; - if (!isWithinInterval(new Date(tvr.attributes.createdAt), last24H)) return false; - return true; -} - -async function handleDelete(authContext: IUseAuth | null, tvr: ITagVodRelation): Promise { - if (!authContext) return; - const { authData } = authContext; - const res = await fetch(`${postgrestLocalUrl}/api/tag-vod-relations/deleteMine/${tvr.id}`, { - method: 'DELETE', - headers: { - 'Authorization': `Bearer ${authData?.accessToken}`, - 'Content-Type': 'application/json' - } - }) - if (!res.ok) throw new Error(res.statusText) -} - -export function Tag({ tvr }: ITagParams) { - const authContext = useContext(AuthContext); - const router = useRouter() - const [shouldRenderDeleteButton, setShouldRenderDeleteButton] = useState(false); - - useEffect(() => { - setShouldRenderDeleteButton(isCreatedByMeRecently(authContext?.authData?.user?.id, tvr)); - }, [authContext?.authData?.user?.id, tvr]); - - return ( - - {tvr.attributes.tag.name} - {shouldRenderDeleteButton && { - handleDelete(authContext, tvr); router.refresh() - } - } className="tag">} - - ) -} \ No newline at end of file diff --git a/services/next/app/components/tagger.tsx b/services/next/app/components/tagger.tsx deleted file mode 100644 index a42ecea..0000000 --- a/services/next/app/components/tagger.tsx +++ /dev/null @@ -1,241 +0,0 @@ -'use client'; - -import { useState, useCallback, useEffect, useContext } from 'react'; -import { IVod } from '@/app/lib/vods'; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faPlus, faX, faTags } from "@fortawesome/free-solid-svg-icons"; -import { formatTimestamp } from '@/app/lib/dates'; -import { readOrCreateTagVodRelation } from '@/app/lib/tag-vod-relations'; -import { readOrCreateTag } from '@/app/lib/tags'; -import { debounce } from 'lodash'; -import { postgrestLocalUrl } from '@/app/lib/constants'; -import { VideoContext } from './video-context'; -import { useForm } from "react-hook-form"; -import { ITimestamp, createTimestamp } from '@/app/lib/timestamps'; -import { useRouter } from 'next/navigation'; -import styles from '@/assets/styles/fp.module.css' -import qs from 'qs'; -import { toast } from 'react-toastify'; -import slugify from 'slugify'; - -interface ITaggerProps { - vod: IVod; - setTimestamps: Function; -} - -export interface ITagSuggestion { - id: number; - name: string; - createdAt: string; -} - - -type FormData = { - tagName: string; - isTimestamp: boolean; -}; - - - - - -export function Tagger({ vod, setTimestamps }: ITaggerProps): React.JSX.Element { - return

@todo Tagger

- const { register, setValue, setError, setFocus, handleSubmit, watch, clearErrors, formState: { errors } } = useForm({ - defaultValues: { - tagName: '', - isTimestamp: true - } - }); - const [isEditor, setIsEditor] = useState(false); - const [isAuthed, setIsAuthed] = useState(false); - const [tagSuggestions, setTagSuggestions] = useState([]); - const { authData } = useAuth(); - const { timeStamp, tvrs, setTvrs } = useContext(VideoContext); - const router = useRouter(); - - const request = debounce((value: string) => { - search(value); - }, 300); - - const debounceRequest = useCallback((v: string) => request(v), [request]); - - - // Callback version of watch. It's your responsibility to unsubscribe when done. - useEffect(() => { - const subscription = watch((value, { name, type }) => { - const tagNameValue = value.tagName as string; - if (name === 'tagName' && type === 'change' && value.tagName !== '') debounceRequest(tagNameValue); - }); - return () => subscription.unsubscribe(); - }, [watch, debounceRequest]); - - - useEffect(() => { - if (isEditor) { - setFocus('tagName'); - getRandomSuggestions(); - } - }, [isEditor, setFocus]); - - useEffect(() => { - if (authData?.accessToken) { - setIsAuthed(true); - } - }, [isAuthed, authData]); - - - async function getRandomSuggestions() { - // @todo https://gitea.futureporn.net/futureporn/pm/issues/129 - setTagSuggestions([]) - // const res = await fetch(`${postgrestLocalUrl}/tag/random`); - // const tags = await res.json(); - // setTagSuggestions(tags) - } - - async function search(value: string) { - const query = qs.stringify( - { - filters: { - tags: { - publishedAt: { - $notNull: true - } - } - }, - query: value - } - ) - if (!value) return; - const res = await fetch(`${postgrestLocalUrl}/fuzzy-search/search?${query}`, { - headers: { - 'Authorization': `Bearer ${authData?.accessToken}` - } - }) - const json = await res.json() - if (!res.ok) { - toast('failed to get recomended tags', { type: 'error', theme: 'dark' }); - } else { - setTagSuggestions(json.tags) - } - } - - - async function onError(errors: any) { - console.error('submit handler encoutnered an error'); - console.error(errors); - toast('there was an error'); - } - - async function onSubmit(values: { tagName: string, isTimestamp: boolean }) { - if (!authData?.accessToken) { - toast('must be logged in', { type: 'error', theme: 'dark' }); - return - } - try { - - const tag = await readOrCreateTag(authData.accessToken, slugify(values.tagName)); - if (!tag) throw new Error(`readOrCreateTag failed`); - - - const tvr = await readOrCreateTagVodRelation(authData.accessToken, tag.id, vod.id); - console.log(`now we check to see if we have a TVR`); - console.log(tvr) - - if (values.isTimestamp) { - console.log(`user specified that we must create a timestamp`); - const timestamp = await createTimestamp(authData, tag.id, vod.id, timeStamp); - console.log(timestamp) - if (!timestamp) throw new Error(`failed to create timestamp`) - setTimestamps((prevTimestamps: ITimestamp[]) => [...prevTimestamps, timestamp]); - } - - setValue('tagName', ''); - router.refresh(); - } catch (e) { - toast(`${e}`, { type: 'error', theme: 'dark' }); - } - } - - // if (!isAuthed) { - // return <> - // } else { - if (isEditor) { - return ( -
- - -
-

Tagger

- -
-
-
-
- - -
-
- Suggestions - {tagSuggestions.length > 0 && tagSuggestions.map((tag: ITagSuggestion) => ())} -
-
- -
- -
- {(!!errors?.root?.serverError) &&
{errors.root.serverError.message}
} - - -
-
-
-
- ) - } else { - return ( - - ); - } - // } - - -} diff --git a/services/next/app/components/thumbnail.tsx b/services/next/app/components/thumbnail.tsx deleted file mode 100644 index a64aeb0..0000000 --- a/services/next/app/components/thumbnail.tsx +++ /dev/null @@ -1,24 +0,0 @@ -'use client' -import LinkableHeading from "./linkable-heading" -import Image from "next/legacy/image" -import { useState } from "react" -import styles from '@/assets/styles/fp.module.css' - - -export default function Thumbnail({url}: { url: string }) { - const [isBig, setIsBig] = useState(false) - - return ( -
- setIsBig(!isBig)} - src={url} - alt="A composite image showing several frames from the source video" - width={(isBig) ? 1920 : 192} - height={(isBig) ? 1080 : 108} - quality={(isBig) ? 100 : 75} - > -
- ) -} \ No newline at end of file diff --git a/services/next/app/components/timestamps-list.tsx b/services/next/app/components/timestamps-list.tsx deleted file mode 100644 index 6bf09c1..0000000 --- a/services/next/app/components/timestamps-list.tsx +++ /dev/null @@ -1,68 +0,0 @@ -import React, { useContext, useState, useEffect } from "react"; -import { IVod } from "@/app/lib/vods"; -import { - ITimestamp, - deleteTimestamp -} from "@/app/lib/timestamps"; -import { - formatTimestamp, - formatUrlTimestamp, -} from "@/app/lib/dates"; -import Link from 'next/link'; -import { faClock, faLink, faTrash } from "@fortawesome/free-solid-svg-icons"; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { isWithinInterval, subHours, Interval } from 'date-fns'; -import { useRouter } from 'next/navigation'; - -export interface ITimestampsProps { - vod: IVod; - timestamps: ITimestamp[]; - setTimestamps: Function; -} - -function isCreatedByMeRecently(authData: any, ts: ITimestamp) { - if (!authData?.user) return false; - if (authData.user.id !== ts.creator_id) return false; - const last24H: Interval = { start: subHours(new Date(), 24), end: new Date() }; - return isWithinInterval(new Date(ts.created_at), last24H); -} - - -export function TimestampsList({ vod, timestamps, setTimestamps }: ITimestampsProps): React.JSX.Element { - // const throttledTimestampFetch = throttle(getRawTimestampsForVod); - - - const hasTimestamps = timestamps.length > 0; - - return ( -
- - {hasTimestamps && ( - timestamps.map((ts: ITimestamp) => ( -

- {JSON.stringify(ts, null, 2)}


- - {formatTimestamp(ts.time)} - {' '} - {/* {ts.tag.name} */} - {isCreatedByMeRecently({}, ts) && ( - - )} -

- )) - )} - - {!hasTimestamps &&

This VOD has no timestamps

} -
- ); -} diff --git a/services/next/app/components/toys.tsx b/services/next/app/components/toys.tsx deleted file mode 100644 index 1739f78..0000000 --- a/services/next/app/components/toys.tsx +++ /dev/null @@ -1,80 +0,0 @@ -import React from 'react'; -import { IToy, IToysResponse } from '@/app/lib/toys'; -import { IVtuber } from '@futureporn/types'; -import Link from 'next/link'; -import Image from "next/legacy/image"; - -export interface IToyProps { - toy: IToy; -} - -export interface IToysListsProps { - vtuber: IVtuber; - toys: IToysResponse; - page: number; - pageSize: number; -} - -// interface VodsListProps { -// vtuber: IVtuber; -// vods: IVods; -// page: number; -// pageSize: number; -// } - - - -export function ToysListHeading({ slug, displayName }: { slug: string, displayName: string }): React.JSX.Element { - return ( -
-

- {displayName}'s Toys -

-
- ) -} - -// export interface IToy { -// id: number; -// tags: ITag[]; -// linkTag: ITag; -// make: string; -// model: string; -// aspectRatio: string; -// image2: string; -// } - -export function ToyItem({ toy }: IToyProps) { - const displayName = `${toy.attributes.make} ${toy.attributes.model}`; - // if (!toy?.linkTag) return
toy.linkTag is missing which is a problem
- return ( -
- - -
- {displayName} -
-

{toy.attributes.model}

- -
- ); -}; - -export function ToysList({ vtuber, toys, page = 1, pageSize = 24 }: IToysListsProps) { - return ( -
- {/*
{JSON.stringify(toys, null, 2)} toys:{toys.data.length} page:{page} pageSize:{pageSize}
*/} -
- {toys.data.map((toy: IToy) => ( - //

{JSON.stringify(toy, null, 2)}

- - ))} -
-
- ) -}; diff --git a/services/next/app/components/upload-form.tsx b/services/next/app/components/upload-form.tsx deleted file mode 100644 index 25fbc46..0000000 --- a/services/next/app/components/upload-form.tsx +++ /dev/null @@ -1,457 +0,0 @@ -'use client'; - - -import { IVtuber } from "@futureporn/types"; -import { useSearchParams } from 'next/navigation'; -import React from 'react'; -import AwsS3 from '@uppy/aws-s3'; -import RemoteSources from '@uppy/remote-sources'; -import { Dashboard } from '@uppy/react'; -import styles from '@/assets/styles/fp.module.css' -import { projektMelodyEpoch } from "@/app/lib/constants"; -import add from "date-fns/add"; -import sub from "date-fns/sub"; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faEraser, faInfoCircle, faPaperPlane, faSpinner, faUpload, faX, faXmark } from "@fortawesome/free-solid-svg-icons"; -import { useForm, ValidationMode } from 'react-hook-form'; -import { yupResolver } from '@hookform/resolvers/yup'; -import * as Yup from 'yup'; -import { toast } from "react-toastify"; -import { ErrorMessage } from "@hookform/error-message" -import Uppy from '@uppy/core'; -import { companionUrl } from '@/app/lib/constants'; - - - -interface IUploadFormProps { - vtubers: IVtuber[]; -} - -interface IValidationResults { - valid: boolean; - issues: string[] | null; -} - -interface IFormSchema extends Yup.InferType { }; - - -const validationSchema = Yup.object().shape({ - vtuber: Yup.number() - .required('VTuber is required'), - streamCuid: Yup.string().optional(), - date: Yup.date() - .typeError('Invalid date') // https://stackoverflow.com/a/72985532/1004931 - .min(sub(projektMelodyEpoch, { days: 1 }), 'Date must be after February 7 2020') - .max(add(new Date(), { days: 1 }), 'Date cannot be in the future') - .required('Date is required'), - notes: Yup.string().optional(), - attribution: Yup.boolean().optional(), - files: Yup.array() - .of( - Yup.object().shape({ - key: Yup.string().required('key is required'), - uploadId: Yup.string().required('uploadId is required') - }), - ) - .min(1, 'At least one file is required'), -}); - - - -export default function UploadForm({ vtubers }: IUploadFormProps) { - const searchParams = useSearchParams(); - const cuid = searchParams.get('cuid'); - - return ( -
-
-
-
- VOD Uploading -
-
- -

Coming soon, patrons will be able to upload VODs to Futureporn's archive. Together we can achieve 100% archival!

-
-
-
-
- ) - - // const uppy = new Uppy( - // { - // autoProceed: true, - // debug: true, - // logger: { - // debug: console.info, - // warn: console.log, - // error: console.error - // }, - // } - // ) - // .use(RemoteSources, { - // companionUrl, - // sources: [ - // 'GoogleDrive', - // 'Dropbox', - // 'Url' - // ] - // }) - // .use(AwsS3, { - // companionUrl, - // shouldUseMultipart: true, - // abortMultipartUpload: () => {}, // @see https://github.com/transloadit/uppy/issues/1197#issuecomment-491756118 - // companionHeaders: { - // 'authorization': `Bearer ${authData?.accessToken}` - // } - // }) - - - - const formOptions = { - resolver: yupResolver(validationSchema), - mode: 'onChange' as keyof ValidationMode, - }; - const { - register, - handleSubmit, - setError, - clearErrors, - formState: { - errors, - isValid, - isSubmitted, - isSubmitSuccessful, - isSubmitting - }, - setValue, - watch, - reset - } = useForm(formOptions); - - - // useEffect(() => { - // if (!cuid) return; - // (async () => { - // console.log('query') - // const query = qs.stringify({ - // filters: { - // cuid: { - // '$eq': cuid - // } - // } - // }); - // const res = await fetch(`${process.env.NEXT_PUBLIC_STRAPI_URL}/streams?${query}`); - // if (!res.ok) return; - // const matchingStream = (await res.json()).data as IStream; - // console.log(matchingStream); - // setValue('vtuber', matchingStream.attributes.vtuber.data.id); - // })(); - // }, [cuid]); - - // setValue('streamCuid', cuid||''); - const files = watch('files'); - - - - async function createUSC(data: IFormSchema) { - try { - const res = await fetch(`${process.env.NEXT_PUBLIC_STRAPI_URL}/user-submitted-contents/createFromUppy`, { - method: 'POST', - headers: { - 'authorization': `Bearer ${authData?.accessToken}`, - 'content-type': 'application/json', - 'accept': 'application/json' - }, - body: JSON.stringify({ - data: { - files: data.files, - attribution: data.attribution, - notes: data.notes, - vtuber: data.vtuber, - date: data.date, - streamCuid: cuid - } - }) - }); - - if (!res.ok) { - console.error('failed to fetch /user-submitted-contents/createFromUppy'); - const body = await res.json(); - const error = body.error; - toast.error(`${error.type} ${error.message}`, { theme: 'dark' }); - - setError('root.serverError', { - type: error.type, - message: error.message - }) - } - } catch (e) { - if (e instanceof Error) { - toast.error(`${e.message}`, { theme: 'dark' }); - setError('root.serverError', { - type: "remote", - message: e.message, - }); - } else { - toast.error(`Something went wrong. Please try again.`, { theme: 'dark' }); - setError('root.serverError', { - type: 'remote', - message: 'Something went wrong. Please try again.' - }) - } - } - } - - - uppy.on('complete', async (result: any) => { - for (const s of result.successful) { - if (!s?.s3Multipart) { - const m = 'file was missing s3Multipart' - toast.error(`${m}`, { theme: 'dark' }); - setError('root.serverError', { - type: 'remote', - message: m - }) - throw new Error(m) - } - } - console.log('uppy complete! ') - console.log(result) - toast.success(`upload complete`); - let files = result.successful.map((f: any) => ({ key: f.s3Multipart.key, uploadId: f.s3Multipart.uploadId })); - setValue('files', files); - }); - - - return ( - <> - -
-

Upload VOD

- -

Together we can archive all lewdtuber livestreams!

- - {(!authData?.accessToken) - ? - <> - - - - : ( - - - -
-
- - - {(!isSubmitSuccessful) &&
-
-
-

- Step 1 -

-

- Upload the file -

-
-
-
- - - {/* - Here is how we upload the files to the server. - From uppy, we get a list of files. - we add the files to a hidden input box. - the input box is part of the form which gets POSTed. - */} - - - - - - {errors.files &&

{errors.files.message?.toString()}

} - -
-
} - - {(!isSubmitSuccessful) &&
- {/* {(!cuid) && } */} - -
-
-

- Step 2 -

-

- Tell us about the VOD -

-
-
- -
- - - - {/* */} - - -
- -
- -
-

Choose the VTuber this VOD belongs to. (More VTubers will be added when storage/bandwidth funding is secured.)

- {errors.vtuber &&

vtuber error

} - -
- -
- - setDate(evt.target.value)} - > -

The date when the VOD was originally streamed.

- {errors.date &&

{errors.date.message?.toString()}

} - -
- -
- - -

If there are any issues with the VOD, put a note here. If there are no VOD issues, leave this field blank.

-
- -
- - -
- -
- -
} - - -
-
-
-

- Step 3 -

-

- Send the form -

-
-
-
- - - - - {errors.root?.serverError && ( -
- - -
- )} - - - - {!isSubmitSuccessful && ( - - )} - - {isSubmitting && ( -

- -

- )} - {isSubmitSuccessful && ( - <> - - - - )} - - - -
-
- -
-
- - - ) - } - -
- - - ) - -} diff --git a/services/next/app/components/user-controls.tsx b/services/next/app/components/user-controls.tsx deleted file mode 100644 index 6a76717..0000000 --- a/services/next/app/components/user-controls.tsx +++ /dev/null @@ -1,230 +0,0 @@ -'use client'; - -import React, { useState } from 'react'; -import { patreonQuantumSupporterId } from '../lib/constants'; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faSave, faTimes, faCheck } from "@fortawesome/free-solid-svg-icons"; -import Skeleton from 'react-loading-skeleton'; - -interface IArchiveSupporterProps { - isNamePublic: boolean; - setIsNamePublic: Function; -} - -interface ISaveButtonProps { - isDirty: boolean; - isLoading: boolean; - isSuccess: boolean; - isNamePublic: boolean; - isLinkPublic: boolean; - vanityLink: string; - setVanityLink: Function; - setIsLoading: Function; - setIsSuccess: Function; - setIsDirty: Function; - setAuthData: Function; - errors: String[]; - setErrors: Function; -} - -interface IQuantumSupporterProps { - isLinkPublic: boolean; - hasUrlBenefit: boolean; - setIsLinkPublic: Function; - vanityLink: string; - setVanityLink: Function; -} - - -export default function UserControls() { - const [isLoading, setIsLoading] = useState(false); - const [isSuccess, setIsSuccess] = useState(false); - const [isDirty, setIsDirty] = useState(false); - const [isNamePublic, setIsNamePublic] = useState(false); - const [isLinkPublic, setIsLinkPublic] = useState(false); - const [errors, setErrors] = useState([]) - const [vanityLink, setVanityLink] = useState('') - - return

@todo user-controls.tsx

- const { authData, setAuthData } = useAuth() - - - if (!authData) return

Loading...

- - - const hasUrlBenefit = (authData?.user?.patreonBenefits) ? authData.user.patreonBenefits.split(' ').includes(patreonQuantumSupporterId) : false; - - return ( -
-
-

Patron Perks

- - - - - - -
-
- ); -}; - - -export function SaveButton({ - isDirty, - setIsDirty, - isLoading, - setIsLoading, - setIsSuccess, - isSuccess, - isNamePublic, - isLinkPublic, - vanityLink, - setVanityLink, - setAuthData, - errors, - setErrors, -}: ISaveButtonProps) { - return

@todo user-controls.tsx

- const { authData } = useAuth(); - const handleClick = async () => { - if (!authData?.user) return; - try { - setIsLoading(true); - - const response = await fetch(`${postgrestLocalUrl}/profile/${authData.user.id}`, { - method: 'PUT', - headers: { - 'Content-Type': 'application/json', - 'Authorization': `Bearer ${authData.accessToken}` - }, - body: JSON.stringify({ - isNamePublic, - isLinkPublic, - vanityLink - }) - }); - - setIsLoading(false); - setIsDirty(true); - - if (!response.ok) { - setIsSuccess(false); - } else { - setIsSuccess(true); - - // Update authData if needed - const updatedAuthData = { ...authData }; - if (!updatedAuthData?.user) return; - updatedAuthData.user.vanityLink = vanityLink; - updatedAuthData.user.isNamePublic = isNamePublic; - updatedAuthData.user.isLinkPublic = isLinkPublic; - setAuthData(updatedAuthData); - } - } catch (error) { - if (error instanceof Error) { - setErrors(errors.concat([error.message])) - } - } - }; - - - return ( - - ) -} - -export function Thanks() { - return

Thank you so much for supporting Futureporn!

-} - -export function QuantumSupporterPerks({ isLinkPublic, setIsLinkPublic, setVanityLink, vanityLink, hasUrlBenefit }: IQuantumSupporterProps) { - const { authData } = useAuth() - - return ( -
- -
- -
-
- setVanityLink(e.target.value)} - /> -
- -
- ) -} - -export function AdvancedArchiveSupporterPerks() { - -} - -export function ArchiveSupporterPerks({ isNamePublic, setIsNamePublic }: IArchiveSupporterProps) { - const { authData } = useAuth() - - return ( -
- -
- -
-
- ) -} \ No newline at end of file diff --git a/services/next/app/components/user-metadata.tsx b/services/next/app/components/user-metadata.tsx deleted file mode 100644 index 60ead00..0000000 --- a/services/next/app/components/user-metadata.tsx +++ /dev/null @@ -1,37 +0,0 @@ -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome" -import { faCheckCircle, faSpinner, faXmarkCircle } from "@fortawesome/free-solid-svg-icons" -import { useMetadata, useMutateMetadata } from "../profile/hooks/useMetadata" -import { useState, useEffect } from 'react' - -export default function UserMetadata() { - - - const { status, data, error, isFetching, isPending } = useMetadata() - const mutateMetadata = useMutateMetadata() - - - const username = `${data?.metadata?.first_name || ''} ${data?.metadata?.last_name || ''}`.trim(); - const { metadata } = data || {}; - - const [ isUsernamePublic, setIsUsernamePublic ] = useState(metadata?.isUsernamePublic || false) - useEffect(() => { - mutateMetadata.mutate({ isUsernamePublic }) - }, [ isUsernamePublic ]) - - return ( - <> - - {isPending && } - -
- Display {username} publicly? - setIsUsernamePublic(evt.target.checked)}/> - {(mutateMetadata.status === 'success') && } - {(mutateMetadata.status === 'pending') && } - {(mutateMetadata.status === 'error') && } -
- - ) -} - - diff --git a/services/next/app/components/vibrate-test.tsx b/services/next/app/components/vibrate-test.tsx deleted file mode 100644 index 2eccfa8..0000000 --- a/services/next/app/components/vibrate-test.tsx +++ /dev/null @@ -1,29 +0,0 @@ -'use client'; - -import { useVibrate } from '@react-hookz/web'; -import { useState } from 'react'; -import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'; -import { faStop, faPlay } from '@fortawesome/free-solid-svg-icons'; - -export default function VibrateTest() { - const [doVibrate, setDoVibrate] = useState(false) - - useVibrate( - doVibrate, - [100, 30, 100, 30, 100, 30, 200, 30, 200, 30, 200, 30, 100, 30, 100, 30, 100], - true - ); - return ( - <> -

This experimental test is meant to activate your device's vibration.

- - - ) -} \ No newline at end of file diff --git a/services/next/app/components/video-context.tsx b/services/next/app/components/video-context.tsx deleted file mode 100644 index 97c5599..0000000 --- a/services/next/app/components/video-context.tsx +++ /dev/null @@ -1,55 +0,0 @@ - -import { createContext } from "react"; -import { ITagVodRelation } from "@/app/lib/tag-vod-relations"; - -export interface IVideoContextValue { - timeStamp: number; - setTimeStamp: Function; - tvrs: ITagVodRelation[]; - setTvrs: Function; -} - -// const defaultContextValue = { -// timeStamp: 3, -// setTimeStamp: () => null, -// ref: null, -// } - -export const VideoContext = createContext({} as IVideoContextValue); - - -// export function VideoContextProvider({ children }: IAuthContextProps): React.JSX.Element { -// const { value: authData, set: setAuthData } = useLocalStorageValue('authData', { -// defaultValue: null, -// }); - -// const { value: lastVisitedPath, set: setLastVisitedPath } = useLocalStorageValue('lastVisitedPath', { -// defaultValue: '/profile', -// initializeWithValue: false, -// }); -// const router = useRouter(); - -// const login = async () => { -// const currentPath = window.location.pathname; -// setLastVisitedPath(currentPath); -// router.push(`${postgrestLocalUrl}/connect/patreon`); -// }; - -// const logout = () => { -// setAuthData({ accessToken: null, user: null }); -// }; - -// return ( -// -// {children} -// -// ); -// } \ No newline at end of file diff --git a/services/next/app/components/video-interactive.tsx b/services/next/app/components/video-interactive.tsx deleted file mode 100644 index 5d21819..0000000 --- a/services/next/app/components/video-interactive.tsx +++ /dev/null @@ -1,149 +0,0 @@ -'use client'; - -import { IVod } from "@/app/lib/vods"; -import { useRef, useState, useEffect, useCallback } from "react"; -import { VideoPlayer } from "./video-player"; -import { Tagger } from './tagger'; -import { ITimestamp, getTimestampsVodLinksForVod } from "@/app/lib/timestamps"; -import { TimestampsList } from "./timestamps-list"; -import { ITagVodRelation } from "@/app/lib/tag-vod-relations"; -import { VideoContext } from "./video-context"; -import { getVodTitle } from "./vod-page"; -import { useSearchParams } from 'next/navigation'; -import type VideoApiElement from "@mux/mux-player"; -import { parseUrlTimestamp } from "@/app/lib/dates"; -import { faTags, faNoteSticky, faClock } from "@fortawesome/free-solid-svg-icons"; -import { Tag } from './tag'; -import VodNav from './vod-nav'; -import LinkableHeading from "./linkable-heading"; - - -export interface IVideoInteractiveProps { - vod: IVod; -} - - -function secondsToHumanReadable(timestampInSeconds: number): string { - const hours = Math.floor(timestampInSeconds / 3600); - const minutes = Math.floor((timestampInSeconds % 3600) / 60); - const seconds = timestampInSeconds % 60; - - return `${hours}h${minutes}m${seconds}s`; -} - - -function humanReadableTimestampToSeconds(timestamp: string): number | null { - const parts = timestamp.split(':'); - - if (parts.length !== 3) { - // Invalid format, return null or throw an error as appropriate - return null; - } - - const hours = parseInt(parts[0], 10); - const minutes = parseInt(parts[1], 10); - const seconds = parseInt(parts[2], 10); - - if (isNaN(hours) || isNaN(minutes) || isNaN(seconds)) { - // Invalid numeric values, return null or throw an error as appropriate - return null; - } - - const totalSeconds = hours * 3600 + minutes * 60 + seconds; - - return totalSeconds; -} - - - - -export function VideoInteractive({ vod }: IVideoInteractiveProps): React.JSX.Element { - - const [timeStamp, setTimeStamp] = useState(0); - const [tvrs, setTvrs] = useState([]); - const [isPlayerReady, setIsPlayerReady] = useState(false); - const [timestamps, setTimestamps] = useState([]); - const [currentTsPage, setCurrentTsPage] = useState(1); - - const getTimestampPage = useCallback(async (page: number) => { - const timestamps = await getTimestampsVodLinksForVod(vod.id, page); - setTimestamps(timestamps); - }, [vod.id, setTimestamps]); // IGNORE TS LINTER! DO NOT PUT timestamps HERE! IT CAUSES SELF-DDOS! - - const ref = useRef(null); - const searchParams = useSearchParams(); - const t = searchParams.get('t'); - - - - useEffect(() => { - getTimestampPage(currentTsPage); - }, [vod.id, getTimestampPage, currentTsPage]); - - useEffect(() => { - if (!t) return; - if (!ref?.current) return; - const videoRef = ref.current as VideoApiElement; - const seconds = parseUrlTimestamp(t) - if (seconds === null) return; - videoRef.currentTime = seconds; - }, [t, isPlayerReady, ref]) - - - return ( - - - - -

- {getVodTitle(vod)} -

- - -
- {vod.note && ( - <> - -
{vod.note}
- - )} - - -
-                    
-                        {JSON.stringify(vod, null, 2)}
-                    
-                
-
- { - (!vod?.tag_vod_relations || vod.tag_vod_relations.length === 0) - ?

This vod has no tags

- : - vod.tag_vod_relations.map((tvr: ITagVodRelation) => ( - - )) - } - {/* */} -
- -
-                        
-                            {JSON.stringify(timestamps, null, 2)}
-                        
-                    
- - -
- -
- ) -} \ No newline at end of file diff --git a/services/next/app/components/video-player.tsx b/services/next/app/components/video-player.tsx deleted file mode 100644 index e467371..0000000 --- a/services/next/app/components/video-player.tsx +++ /dev/null @@ -1,145 +0,0 @@ -'use client'; - -import { useEffect, useState, forwardRef, useContext, Ref } from 'react'; -import { IVod } from '@/app/lib/vods'; -import { getVodTitle } from './vod-page'; -import { VideoSourceSelector } from '@/app/components/video-source-selector' -import { buildIpfsUrl } from '@/app/lib/ipfs'; -import { postgrestLocalUrl } from '@/app/lib/constants'; -import MuxPlayer from '@mux/mux-player-react/lazy'; -import { VideoContext } from './video-context'; -import MuxPlayerElement from '@mux/mux-player'; -import type VideoApiElement from "@mux/mux-player"; - -interface IPlayerProps { - vod: IVod; - setIsPlayerReady: Function; -} - -interface ITokens { - playbackToken: string; - storyboardToken: string; - thumbnailToken: string; -} - -async function getMuxPlaybackTokens(playback_id: string, jwt: string): Promise { - const res = await fetch(`${postgrestLocalUrl}/api/mux-asset/secure?id=${playback_id}`, { - headers: { - 'Authorization': `Bearer ${jwt}` - } - }) - const json = await res.json() - - return { - playbackToken: json.playbackToken, - storyboardToken: json.storyboardToken, - thumbnailToken: json.thumbnailToken - } -} - -function hexToRgba(hex: string, alpha: number) { - const r = parseInt(hex.slice(1, 3), 16); - const g = parseInt(hex.slice(3, 5), 16); - const b = parseInt(hex.slice(5, 7), 16); - return `rgba(${r}, ${g}, ${b}, ${alpha})`; -} - - - -export const VideoPlayer = forwardRef(function VideoPlayer( props: IPlayerProps, ref: Ref ): React.JSX.Element { - const { vod, setIsPlayerReady } = props - const title: string = getVodTitle(vod); - - const [selectedVideoSource, setSelectedVideoSource] = useState(''); - const [isEntitledToCDN, setIsEntitledToCDN] = useState(false); - const [hlsSource, setHlsSource] = useState(''); - const [isClient, setIsClient] = useState(false); - const [playback_id, setplayback_id] = useState(''); - const [src, setSrc] = useState(''); - const [tokens, setTokens] = useState({}); - const { setTimeStamp } = useContext(VideoContext); - - - - useEffect(() => { - setIsClient(true); - const playback_id = vod?.mux_asset?.playback_id; - - - if (selectedVideoSource === 'Mux') { - if (!!token && !!playback_id) { - try { - getMuxPlaybackTokens(vod.mux_asset.playback_id, token) - .then((tokens) => { - setTokens({ - playback: tokens.playbackToken, - storyboard: tokens.storyboardToken, - thumbnail: tokens.thumbnailToken - }) - setHlsSource(vod.mux_asset.playback_id) - setplayback_id(vod.mux_asset.playback_id) - }); - } - - catch (e) { - console.error(e) - } - } - } else if (selectedVideoSource === 'B2') { - if (!vod.s3_file) return; // This shouldn't happen because videoSourceSelector won't choose B2 if there is no b2. This return is only for satisfying TS - setHlsSource(vod.s3_file.cdn_url); - setplayback_id(''); - setSrc(vod.s3_file.cdn_url); - } - }, [selectedVideoSource, vod, setHlsSource]); - - - if (!isClient) return <> - - - return ( - <> - { - setIsPlayerReady(true)} - } - ref={ref} - preload="auto" - crossOrigin="*" - loading="viewport" - playback_id={playback_id} - src={src} - tokens={tokens} - primaryColor="#FFFFFF" - metadata={{ - video_title: getVodTitle(vod) - }} - - streamType="on-demand" - onTimeUpdate={(evt) => { - const muxPlayer = evt.target as VideoApiElement - const { currentTime } = muxPlayer; - setTimeStamp(currentTime) - }} - muted - > - - {/* {vod?.attributes?.videoSrcB2?.data?.attributes?.cdn_url && (<> -

CDN2

- - )} */} - - - {/* */} - - ) -}) \ No newline at end of file diff --git a/services/next/app/components/video-source-selector.tsx b/services/next/app/components/video-source-selector.tsx deleted file mode 100644 index 408fe24..0000000 --- a/services/next/app/components/video-source-selector.tsx +++ /dev/null @@ -1,142 +0,0 @@ -'use client'; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome" -import { faPatreon } from "@fortawesome/free-brands-svg-icons"; -import { faGlobe, faTriangleExclamation } from "@fortawesome/free-solid-svg-icons"; -import { useState, useEffect } from 'react'; - -interface IVSSProps { - isMux: boolean; - isB2: boolean; - isIPFSSource: boolean; - isIPFS240: boolean; - isEntitledToCDN: boolean; - setSelectedVideoSource: (option: string) => void; - selectedVideoSource: string; -} - -export function VideoSourceSelector({ - isMux, - isB2, - isIPFSSource, - isIPFS240, - isEntitledToCDN, - selectedVideoSource, - setSelectedVideoSource, -}: IVSSProps): React.JSX.Element { - - // Check for user's entitlements and saved preference when component mounts - useEffect(() => { - // Function to determine the best video source based on entitlements and preferences - const determineBestVideoSource = () => { - if (isEntitledToCDN) { - if (selectedVideoSource === 'Mux' && isMux) { - return 'Mux'; - } - } - - // if the user has B2 as their preference or they have no preference, use B2 - if (selectedVideoSource === 'B2' || !selectedVideoSource) { - return 'B2' - } - - // use IPFS only if the user has opted to use it - if (selectedVideoSource === 'IPFSSource' && isIPFSSource) { - return 'IPFSSource'; - } else if (isIPFS240) { - return 'IPFS240'; - } - // If no sources are available, return an empty string - return ''; - }; - - // If selectedVideoSource is unset, find the value to use - if (selectedVideoSource === '') { - // Load the user's saved preference from storage (e.g., local storage) - const savedPreference = localStorage.getItem('videoSourcePreference'); - - // Check if the saved preference is valid based on entitlements and available sources - if (savedPreference === 'Mux' && isMux && isEntitledToCDN) { - setSelectedVideoSource('Mux'); - } else if (savedPreference === 'B2') { - setSelectedVideoSource('B2'); - } else if (savedPreference === 'IPFSSource') { - setSelectedVideoSource('IPFSSource'); - } else if (savedPreference === 'IPFS240') { - if (isIPFS240) { - setSelectedVideoSource('IPFS240'); - } else { - setSelectedVideoSource('IPFSSource'); - } - } else { - // Determine the best video source if the saved preference is invalid or not available - const bestSource = determineBestVideoSource(); - setSelectedVideoSource(bestSource); - } - } - - - }, [isMux, isB2, isIPFSSource, isIPFS240, isEntitledToCDN, selectedVideoSource, setSelectedVideoSource]); - - // Handle button click to change the selected video source - const handleSourceClick = (source: string) => { - if ( - (source === 'Mux' && isMux && isEntitledToCDN) || - (source === 'B2' && isB2) || - (source === 'IPFSSource') || - (source === 'IPFS240') - ) { - setSelectedVideoSource(source); - // Save the user's preference to storage (e.g., local storage) - localStorage.setItem('videoSourcePreference', source); - } - }; - - return ( - <> -
- -
- - ) -} \ No newline at end of file diff --git a/services/next/app/components/vod-card.tsx b/services/next/app/components/vod-card.tsx deleted file mode 100644 index ea653fe..0000000 --- a/services/next/app/components/vod-card.tsx +++ /dev/null @@ -1,70 +0,0 @@ -import Link from "next/link"; -import { getSafeDate } from '@/app/lib/dates'; -import { IVtuber } from '@futureporn/types'; -import Image from "next/legacy/image" -import { LocalizedDate } from '@/app/components/localized-date' -import Skeleton from "react-loading-skeleton"; - -interface IVodCardProps { - id: number; - title: string; - date: string; - muxAsset: string | undefined; - thumbnail: string | undefined; - vtuber: IVtuber; -} - - -export default function VodCard({ id, title, date, muxAsset, thumbnail = 'https://futureporn-b2.b-cdn.net/default-thumbnail.webp', vtuber }: IVodCardProps) { - - - // return (

@todo W VodCard

) - - if (!vtuber?.slug) return ( -
-
-
-
VOD {id} is missing VTuber.slug
-
-
-
- ) - - - return ( -
-
- -
- {thumbnail && -
- {title} -
- } - {!thumbnail &&

missing thumbnail

} -
-
-

{vtuber.display_name}

- -

{title}

-
- -
-
- - ) -} - - - - - diff --git a/services/next/app/components/vod-nav.tsx b/services/next/app/components/vod-nav.tsx deleted file mode 100644 index 7c21383..0000000 --- a/services/next/app/components/vod-nav.tsx +++ /dev/null @@ -1,76 +0,0 @@ -'use client'; - -import { faVideo, faExternalLinkAlt, faShareAlt } from "@fortawesome/free-solid-svg-icons"; -import { faXTwitter } from '@fortawesome/free-brands-svg-icons'; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import Image from "next/legacy/image"; -import Link from 'next/link'; -import { IVod } from '@/app/lib/vods'; -import { buildIpfsUrl } from '@/app/lib/ipfs'; -import { getSafeDate } from "@/app/lib/dates"; -import { StreamButton } from '@/app/components/stream-button'; -import VtuberButton from "./vtuber-button"; -import { LocalizedDate } from "./localized-date"; - -export function getDownloadLink(cid: string, safeDate: string, slug: string, quality: string) { - return buildIpfsUrl(`${cid}?filename=${slug}-${safeDate}-${quality}.mp4`) -} - - -export interface IVodNavProps { - vod: IVod; -} - -export default function VodNav ({ vod }: IVodNavProps) { - const safeDate = getSafeDate(vod.date_2); - return ( - - ) -} \ No newline at end of file diff --git a/services/next/app/components/vod-page-2.tsx b/services/next/app/components/vod-page-2.tsx deleted file mode 100644 index e2c1de8..0000000 --- a/services/next/app/components/vod-page-2.tsx +++ /dev/null @@ -1,61 +0,0 @@ -import { IVod } from "@futureporn/types"; -import Link from "next/link"; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faChevronLeft, faChevronRight } from "@fortawesome/free-solid-svg-icons"; -import { getNextVod, getPreviousVod, getUrl } from "../lib/vods"; -import { getLocalizedDate } from "../lib/vods"; -import Footer from "./footer"; - -export default async function VodPage2 ({ vod }: { vod: IVod }) { - - // use vod 337 for prototyping, because 337 has both timestamps and tags. - - const nextVod = await getNextVod(vod) - const previousVod = await getPreviousVod(vod) - const slug = vod.vtuber.slug - - return ( - <> -

VodPage2

-

@todo insert video player here

-

@todo insert title here

-

@todo insert notes

-

@todo insert tags here

-

@todo insert timestamps here

- - - ) -} - diff --git a/services/next/app/components/vod-page.tsx b/services/next/app/components/vod-page.tsx deleted file mode 100644 index 7e9a61e..0000000 --- a/services/next/app/components/vod-page.tsx +++ /dev/null @@ -1,106 +0,0 @@ -import { getUrl, getNextVod, getPreviousVod, getLocalizedDate } from '@/app/lib/vods'; -import { IVod } from '@/app/lib/vods'; -import Link from 'next/link'; -import { VideoInteractive } from './video-interactive'; -import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'; -import { faChevronLeft, faChevronRight, faGlobe, faImage, faLink } from "@fortawesome/free-solid-svg-icons"; -import { notFound } from 'next/navigation'; -import { IpfsCid } from './ipfs-cid'; -import LinkableHeading from './linkable-heading'; -import Image from "next/legacy/image"; -import Thumbnail from './thumbnail'; - -export function getVodTitle(vod: IVod): string { - // console.log('lets getVodTitle, ey?') - // console.log(JSON.stringify(vod, null, 2)) - return vod.title || vod.announce_title || `VOD ${vod.id}` - // return vod.title || vod.announceTitle || (vod?.date2 && vod?.vtuber?.display_name) ? `${vod.vtuber.display_name} ${vod.date_2}` : `VOD ${vod.id}`; -} - -export function buildMuxUrl(playbackId: string, token: string) { - return `https://stream.mux.com/${playbackId}.m3u8?token=${token}` -} - -export function buildMuxSignedPlaybackId(playbackId: string, token: string) { - return `${playbackId}?token=${token}` -} - -export function buildMuxThumbnailUrl(playbackId: string, token: string) { - return `https://image.mux.com/${playbackId}/storyboard.vtt?token=${token}` -} - - -export default async function VodPage({vod}: { vod: IVod }) { - - if (!vod) notFound(); - if (!vod.vtuber) { - throw new Error(`vod.vtuber was falsy.`) - } - const slug = vod.vtuber.slug; - const previousVod = await getPreviousVod(vod); - const nextVod = await getNextVod(vod); - - // return
{JSON.stringify(previousVod, null, 2)}
- // return

{slug} VOD @todo previousVod={previousVod.title} nextVod={nextVod?.title}

- return ( - -
-
- - {/*
{JSON.stringify(vod, null, 2)}
*/} - - {(vod.thumbnail) && (
- - -
)} - - {(vod.ipfs_cid) && ( - <> - - {vod.ipfs_cid && ( - - )} - - )} - - - - - -
-
- ); -} diff --git a/services/next/app/components/vods-list.tsx b/services/next/app/components/vods-list.tsx deleted file mode 100644 index b7dd8fd..0000000 --- a/services/next/app/components/vods-list.tsx +++ /dev/null @@ -1,66 +0,0 @@ -import React from 'react' -import Link from 'next/link'; -import VodCard from './vod-card'; -import { IVtuber } from '@futureporn/types'; -import { IVod } from '@/app/lib/vods'; -import { getVodTitle } from './vod-page'; -import { notFound } from 'next/navigation'; - -interface IVodsListProps { - vtuber?: IVtuber; - vods: IVod[]; - page: number; - pageSize: number; -} - - -interface IVodsListHeadingProps { - slug: string; - displayName: string; -} - -export function VodsListHeading({ slug, displayName }: IVodsListHeadingProps): React.JSX.Element { - return ( -
-

- {displayName} Vods -

-
- ) -} - - -export default function VodsList({ vods, page = 1, pageSize = 24 }: IVodsListProps): React.JSX.Element { - // if (!vtuber) return
vtuber is not defined. vtuber:{JSON.stringify(vtuber, null, 2)}
- // if (!vods) return
failed to load vods
; - if (!vods) return notFound() - - // @todo [x] pagination - // @todo [x] sortability - return ( - <> -

VodsList on page {page}, pageSize {pageSize}, with {vods.length} vods

- - {/*
-                
-                    {JSON.stringify(vods, null, 2)}
-                
-            
*/} - - -
- {vods.map((vod: IVod) => ( - - ))} -
- - ); -} diff --git a/services/next/app/components/vtuber-button.tsx b/services/next/app/components/vtuber-button.tsx deleted file mode 100644 index 22f5bc2..0000000 --- a/services/next/app/components/vtuber-button.tsx +++ /dev/null @@ -1,29 +0,0 @@ -import Image from "next/legacy/image" - -interface VtuberButtonProps { - image: string; - displayName: string; - size?: string; -} - -export default function VtuberButton ({ image, displayName, size }: VtuberButtonProps) { - const sizeClass = (() => { - if (size === 'large') return 'is-large'; - if (size === 'medium') return 'is-medium'; - if (size === 'small') return 'is-small' - })(); - return ( -
- - {displayName} - - {displayName} -
- ); -} \ No newline at end of file diff --git a/services/next/app/components/vtuber-card.tsx b/services/next/app/components/vtuber-card.tsx deleted file mode 100644 index 739f54e..0000000 --- a/services/next/app/components/vtuber-card.tsx +++ /dev/null @@ -1,44 +0,0 @@ -import Link from "next/link"; -import type { IVtuber } from '@futureporn/types'; -import { getVodsForVtuber } from "@/app/lib/vods"; -import Image from "next/legacy/image" -import NotFound from "@/app/vt/[slug]/not-found"; -import ArchiveProgress from "./archive-progress"; - -export default async function VTuberCard(vtuber: IVtuber) { - const { id, slug, display_name, image_blur, image } = vtuber; - if (!image_blur) return
This VTuberCard is missing image_blur
- const vods = await getVodsForVtuber(id) - if (!vods) return - return ( - -
-
-
-
-
- {display_name} -
-
-
-

{display_name}

- -
-
-
-
- - ) - } \ No newline at end of file diff --git a/services/next/app/config/clientConfigs.ts b/services/next/app/config/clientConfigs.ts deleted file mode 100644 index 2442989..0000000 --- a/services/next/app/config/clientConfigs.ts +++ /dev/null @@ -1,21 +0,0 @@ - -if (!process.env.NEXT_PUBLIC_API_DOMAIN) throw new Error('Missing NEXT_PUBLIC_API_DOMAIN env var'); -if (!process.env.NEXT_PUBLIC_WEBSITE_DOMAIN) throw new Error('Missing NEXT_PUBLIC_WEBSITE_DOMAIN env var'); -if (!process.env.NEXT_PUBLIC_UPPY_COMPANION_URL) throw new Error('Missing NEXT_PUBLIC_UPPY_COMPANION_URL env var'); -if (!process.env.NEXT_PUBLIC_SITE_URL) throw new Error('Missing NEXT_PUBLIC_SITE_URL env var'); -if (!process.env.NEXT_PUBLIC_STRAPI_URL) throw new Error('Missing NEXT_PUBLIC_STRAPI_URL env var'); -if (!process.env.NEXT_PUBLIC_POSTGREST_URL) throw new Error('Missing NEXT_PUBLIC_POSTGREST_URL env var'); - - -// const apiDomain = process.env.NEXT_PUBLIC_API_DOMAIN! -// const websiteDomain = process.env.NEXT_PUBLIC_WEBSITE_DOMAIN! - -// export interface ClientConfig { -// apiDomain: string; -// websiteDomain: string; -// } - -// export const configs: ClientConfig = { -// apiDomain, -// websiteDomain, -// } diff --git a/services/next/app/config/configs.ts b/services/next/app/config/configs.ts deleted file mode 100644 index c0134ea..0000000 --- a/services/next/app/config/configs.ts +++ /dev/null @@ -1,39 +0,0 @@ - -// When adding env vars here, make sure to also add the assertion in ./server.ts - -const patreonClientId = process.env.PATREON_CLIENT_ID! -const patreonClientSecret = process.env.PATREON_CLIENT_SECRET! -const keycloakClientId = process.env.KEYCLOAK_CLIENT_ID! -const keycloakClientSecret = process.env.KEYCLOAK_CLIENT_SECRET! -const keycloakIssuer = process.env.KEYCLOAK_ISSUER! -const keycloakLocalUrl = process.env.KEYCLOAK_LOCAL_URL! -const keycloakUrl = process.env.KEYCLOAK_URL! -const nextAuthSecret = process.env.NEXTAUTH_SECRET! -const nextAuthUrl = process.env.NEXTAUTH_URL! -const nextUrl = process.env.NEXT_PUBLIC_URL! - -export interface ServerConfig { - patreonClientId: string; - patreonClientSecret: string; - keycloakClientId: string; - keycloakClientSecret: string; - keycloakIssuer: string; - keycloakLocalUrl: string; - keycloakUrl: string; - nextAuthSecret: string; - nextAuthUrl: string; - nextUrl: string; -} - -export const configs: ServerConfig = { - patreonClientId, - patreonClientSecret, - keycloakClientId, - keycloakClientSecret, - keycloakIssuer, - keycloakLocalUrl, - keycloakUrl, - nextAuthSecret, - nextAuthUrl, - nextUrl, -} \ No newline at end of file diff --git a/services/next/app/config/server.ts b/services/next/app/config/server.ts deleted file mode 100644 index a70e23c..0000000 --- a/services/next/app/config/server.ts +++ /dev/null @@ -1,20 +0,0 @@ -/** - * this file exists to give us verbose errors when we forget to define a server-side env var. - * configs should go in ./configs - */ - -'use server'; - -export async function main () { - if (!process.env.PATREON_CLIENT_ID) throw new Error("PATREON_CLIENT_ID was missing from env"); - if (!process.env.PATREON_CLIENT_SECRET) throw new Error('Missing PATREON_CLIENT_SECRET env var'); - if (!process.env.KEYCLOAK_CLIENT_ID) throw new Error(`KEYCLOAK_CLIENT_ID missing in env!`); - if (!process.env.KEYCLOAK_CLIENT_SECRET) throw new Error(`KEYCLOAK_CLIENT_SECRET missing in env!`); - if (!process.env.KEYCLOAK_ISSUER) throw new Error(`KEYCLOAK_ISSUER missing in env!`); - if (!process.env.KEYCLOAK_LOCAL_URL) throw new Error(`KEYCLOAK_LOCAL_URL missing in env!`); - if (!process.env.NEXTAUTH_SECRET) throw new Error(`NEXTAUTH_SECRET missing in env!`); - if (!process.env.NEXTAUTH_URL) throw new Error(`NEXTAUTH_URL missing in env!`); - if (!process.env.NEXT_PUBLIC_URL) throw new Error(`NEXT_PUBLIC_URL missing in env!`); -} - -main() diff --git a/services/next/app/connect/patreon/redirect/page.tsx b/services/next/app/connect/patreon/redirect/page.tsx deleted file mode 100644 index 40b65a0..0000000 --- a/services/next/app/connect/patreon/redirect/page.tsx +++ /dev/null @@ -1,122 +0,0 @@ -'use client' - -import { useSearchParams, useRouter } from 'next/navigation' -import Link from 'next/link' -import { useEffect, useState } from 'react' -import { postgrestLocalUrl } from '@/app/lib/constants' -import { DangerNotification } from '@/app/components/notifications' - -export type AccessToken = string | null; - - -export default function Page() { - const searchParams = useSearchParams() - const router = useRouter() - const { authData, setAuthData, lastVisitedPath } = useAuth() - const [errors, setErrors] = useState([]) - - const initAuth = async () => { - try { - const accessToken: AccessToken = getAccessTokenFromURL(); - const json = await getJwt(accessToken); - if (!json) { - setErrors(errors.concat(['Unable to get access token from portal. Please try again later or check Futureporn Discord.'])) - } else { - storeJwtJson(json) - redirect(); - } - } catch (error) { - console.error(error); - } - }; - - const storeJwtJson = (json: IJWT) => { - - - // Store the JWT and other relevant data in your state management system - const data: IAuthData = { - accessToken: json.jwt, - user: json.user, - } - setAuthData(data); - } - - - const getAccessTokenFromURL = () => { - const accessToken: AccessToken = searchParams?.get('access_token'); - if (!accessToken) { - throw new Error('Failed to get access_token from auth portal.'); - } - return accessToken; - }; - - const getJwt = async (accessToken: AccessToken): Promise => { - - try { - const response = await fetch(`${postgrestLocalUrl}/api/auth/patreon/callback?access_token=${accessToken}`); - - if (!response.ok) { - // Handle non-2xx HTTP response status - throw new Error(`Failed to fetch. Status: ${response.status}`); - } - - const json = await response.json(); - - if (!json.jwt) { - throw new Error('Failed to get auth token. Please try again later.'); - } - - return json; - } catch (error) { - console.error(error); - return null; // Return null or handle the error in an appropriate way - } - }; - - - const redirect = () => { - if (!lastVisitedPath) return; // on first render, it's likely null - router.push(lastVisitedPath); - }; - - - useEffect(() => { - initAuth() - }) - - - - - - - {/* - After user auths, - they are redirected to this page. - - This page grabs the access_token from the query string, - exchanges it with strapi for a jwt - then persists the jwt - - After a jwt is stored, this page redirects the user - to whatever page they were previously on. - */} - - // @todo get query parameters - // @todo save account info to session - // @todo ??? - // @todo profit - // const searchParams = useSearchParams() - // const accessToken = searchParams?.get('access_token'); - // const refreshToken = searchParams?.get('refresh_token'); - // const lastVisitedPath = '@todo!' - - return ( -
- {errors && errors.length > 0 && ( - - )} -

Redirecting...

- Click here if you are not automatically redirected -
- ) -} \ No newline at end of file diff --git a/services/next/app/faq/page.tsx b/services/next/app/faq/page.tsx deleted file mode 100644 index fad05ac..0000000 --- a/services/next/app/faq/page.tsx +++ /dev/null @@ -1,65 +0,0 @@ -import Link from 'next/link'; -import { getVtuberBySlug } from '../lib/vtubers' -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faExternalLinkAlt } from "@fortawesome/free-solid-svg-icons"; -import { faLink } from '@fortawesome/free-solid-svg-icons'; -import { projektMelodyEpoch } from '@/app/lib/constants'; -import LinkableHeading from '@/app/components/linkable-heading'; - -export default async function Page() { - return ( -
-
-

Frequently Asked Questions (FAQ)

-
- - - -
- -

VTuber is a portmantou of the words Virtual and Youtuber. Originally started in Japan, VTubing uses cameras and/or motion capture technology to replicate human movement and facial expressions onto a virtual character in realtime.

-
- -
- -

Lewdtubers are sexually explicit vtubers. ProjektMelody was the first Vtuber to livestream on Chaturbate on {projektMelodyEpoch.toDateString()}. Many more followed after her.

-
- - -
-
-
- -
-
-

You may get an error when clicking on a video link. Errors such as DNS_PROBE_FINISHED_NXDOMAIN

- -

This is a DNS server error that occurs when a web browser isn't able to translate the domain name into an IP address.

- -

If this happens, using a different DNS server can fix it. There are many gratis services to choose from, including Cloudflare DNS or Google DNS.

- -

Often, using a DNS server other than the one provided to you by your ISP can improve your internet browsing experience for all websites.

-
-
-
- - -
-
- -
-

Yes. Futureporn aims to become the galaxy's best VTuber hentai site.

-
- -
-
- -

Bandwidth and rental fees are expensive, so Futureporn needs financial assistance to keep servers online and videos streaming.

-

Patrons gain access to perks like our video Content Delivery Network (CDN), and optional shoutouts on the patrons page.

-

Additionally, help is needed populating our archive with vods from past lewdtuber streams.

-
-
- -
- ) -} \ No newline at end of file diff --git a/services/next/app/favicon.ico b/services/next/app/favicon.ico deleted file mode 100644 index 2ed11c7480c90fcd2c32df0465864598baa02c05..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 318 zcmbV|I}U&_3_~4>12_v}3uBMTy&1qML -
-
-
- -

RSS Feed

- -

Keep up to date with new VODs using Real Simple Syndication (RSS).

- -

Don't have a RSS reader? Futureporn recommends Fraidycat

- -
-

ATOM

-

RSS

-

JSON

-
-
-
-
- - ) -} - diff --git a/services/next/app/feed/rss.xml/route.ts b/services/next/app/feed/rss.xml/route.ts deleted file mode 100644 index c32d021..0000000 --- a/services/next/app/feed/rss.xml/route.ts +++ /dev/null @@ -1,11 +0,0 @@ -import { generateFeeds } from "@/app/lib/rss" - -export async function GET() { - const { rss2 } = await generateFeeds() - const options = { - headers: { - "Content-Type": "application/rss+xml" - } - } - return new Response(rss2, options) -} \ No newline at end of file diff --git a/services/next/app/goals/page.tsx b/services/next/app/goals/page.tsx deleted file mode 100644 index e23ae22..0000000 --- a/services/next/app/goals/page.tsx +++ /dev/null @@ -1,82 +0,0 @@ -import { getGoals } from "@/app/lib/pm"; -import { getCampaign } from "@/app/lib/patreon"; - -interface IFundingStatusBadgeProps { - completedPercentage: number; -} - -function FundingStatusBadge({ completedPercentage }: IFundingStatusBadgeProps) { - if (completedPercentage === 100) return Funded; - return ( - - - {completedPercentage}% Funded - - - ); -} - - - -// export interface IGoals { -// complete: IIssue[]; -// inProgress: IIssue[]; -// planned: IIssue[]; -// featuredFunded: IIssue; -// featuredUnfunded: IIssue; -// } -export default async function Page() { - const { pledgeSum } = await getCampaign() - const goals = await getGoals(pledgeSum); - if (!goals) return

failed to get goals

- const { inProgress, planned, complete } = goals; - return ( - <> -
-
- -
-

Goals

-

Shooting for the stars

-
-
-

- In Progress -

-
    - {inProgress.map((goal) => ( -
  • - ☐ {goal.title} {(!!goal?.amountCents && !!goal.completedPercentage) && } -
  • - ))} -
-
-
-

- Planned -

-
    - {planned.map((goal) => ( -
  • - ☐ {goal.title} {(!!goal?.amountCents && !!goal.completedPercentage) && } -
  • - ))} -
-
-
-

- Completed -

-
    - {complete.map((goal) => ( -
  • - ✅ {goal.title} {(!!goal?.amountCents && !!goal.completedPercentage) && } -
  • - ))} -
-
-
-
- - ) -} \ No newline at end of file diff --git a/services/next/app/health/page.tsx b/services/next/app/health/page.tsx deleted file mode 100644 index 89cf9c0..0000000 --- a/services/next/app/health/page.tsx +++ /dev/null @@ -1,10 +0,0 @@ - -export default async function Page() { - return ( -
-
-

Healthy!

-
-
- ) -} \ No newline at end of file diff --git a/services/next/app/latest-vods/[page]/page.tsx b/services/next/app/latest-vods/[page]/page.tsx deleted file mode 100644 index 17ebc96..0000000 --- a/services/next/app/latest-vods/[page]/page.tsx +++ /dev/null @@ -1,29 +0,0 @@ -import VodsList from '@/app/components/vods-list'; -import { getVods } from '@/app/lib/vods'; -import Pager from '@/app/components/pager'; - -interface IPageParams { - params: { - page: number; - }; -} - -export default async function Page({ params: { page } }: IPageParams) { - - const pageSize = 24 - const { count, vods } = await getVods(page, pageSize, true); - - - return ( - <> -

Latest VODs

-

page {page}

- - - - ); -} diff --git a/services/next/app/latest-vods/page.tsx b/services/next/app/latest-vods/page.tsx deleted file mode 100644 index 3a750d2..0000000 --- a/services/next/app/latest-vods/page.tsx +++ /dev/null @@ -1,25 +0,0 @@ - -import VodsList from '@/app/components/vods-list'; -import { IVodsResponse } from '@/app/lib/vods'; -import Pager from '@/app/components/pager'; -import { getVods } from '@/app/lib/vods'; - - -interface IPageParams { - params: { - slug: string; - } -} - -export default async function Page({ params }: IPageParams) { - const pageSize = 24 - const { vods, count } = await getVods(1, pageSize); - return ( - <> -

Latest VODs

-

page 1

- - - - ) -} \ No newline at end of file diff --git a/services/next/app/layout.tsx b/services/next/app/layout.tsx deleted file mode 100644 index 3d448ca..0000000 --- a/services/next/app/layout.tsx +++ /dev/null @@ -1,58 +0,0 @@ -import { ReactNode } from 'react' -import Footer from "./components/footer" -import Navbar from "./components/navbar" -import "bulma" -import "../assets/styles/global.css" -import "@fortawesome/fontawesome-svg-core/styles.css" -import type { Metadata } from 'next' -import NotificationCenter from './components/notification-center' -import QueryProvider from './components/query-provider' -import AuthProvider from './components/auth-provider' - -export const metadata: Metadata = { - title: 'Futureporn.net', - description: "The Galaxy's Best VTuber Hentai Site", - other: { - RATING: 'RTA-5042-1996-1400-1577-RTA' - }, - metadataBase: new URL('https://futureporn.net'), - twitter: { - site: '@futureporn_net', - creator: '@cj_clippy' - }, - alternates: { - types: { - 'application/atom+xml': '/feed/feed.xml', - 'application/rss+xml': '/feed/rss.xml', - 'application/json': '/feed/feed.json' - } - } -} - -type Props = { - children: ReactNode; -} - - -export default function RootLayout({ - children, -}: Props) { - - - return ( - - - - - - -
- {children} - {/*
*/} -
-
-
- - - ) -} diff --git a/services/next/app/lib/auth.ts b/services/next/app/lib/auth.ts deleted file mode 100644 index 44e3579..0000000 --- a/services/next/app/lib/auth.ts +++ /dev/null @@ -1,173 +0,0 @@ -import KeycloakProvider, { type KeycloakProfile } from "next-auth/providers/keycloak"; -import { NextAuthOptions } from "next-auth"; -import { jwtDecode } from "jwt-decode"; -import { type JWT } from "next-auth/jwt"; -import NextAuth, { User, Profile } from "next-auth"; -import { configs } from "../config/configs"; -import { extractCurrentlyEntitledTiers, mapTierIdsToRoles } from "./patreon"; -import { getTierNameFromTierId } from "./keycloak"; -import type { OAuthConfig, OAuthUserConfig } from "next-auth/providers/oauth"; - - -declare module "next-auth" { - interface Session { - user: User; - token: JWT | undefined; - profile: Profile; - } - - interface Profile { - realm_access: any; - preferred_username: string; - username_visibility?: "public" | "private"; - } - - interface Session { - roles: any - } - -} - -declare module "next-auth/jwt" { - interface JWT { - expires_at: number; - access_token: string; - refresh_token: string; - profile: Profile; - } -} - - -export interface PatreonProfile extends Record { - sub: string - nickname: string - email: string - picture: string -} - - - -export default function Patreon

( - options: OAuthUserConfig

-): OAuthConfig

{ - return { - id: "patreon", - name: "Patreon", - type: "oauth", - version: "2.0", - authorization: { - url: "https://www.patreon.com/oauth2/authorize", - params: { scope: "identity" }, - }, - token: "https://www.patreon.com/api/oauth2/token", - userinfo: "https://www.patreon.com/api/oauth2/v2/identity?fields%5Buser%5D=about,created,email,first_name,full_name,image_url,last_name,thumb_url,url,vanity&include=memberships,memberships.currently_entitled_tiers,memberships.currently_entitled_tiers.benefits", - profile(profile) { - console.log(`profile callback!!!! userinfo as follows`) - // console.log(profile) - - const tiers = extractCurrentlyEntitledTiers(profile) - console.log(tiers) - - return { - id: profile.data.id, - name: profile.data.attributes.full_name, - email: profile.data.attributes.email, - image: profile.data.attributes.image_url, - currently_entitled_tiers: tiers - } - }, - style: { logo: "/patreon.svg", bg: "#e85b46", text: "#fff" }, - options, - } -} - - - - - -export const authOptions: NextAuthOptions = { - session: { - strategy: "jwt" - }, - providers: [ - Patreon({ - clientId: configs.patreonClientId, - clientSecret: configs.patreonClientSecret, - }) - ], - callbacks: { - // Using the `...rest` parameter to be able to narrow down the type based on `trigger` - jwt({ token, trigger, session, account, profile, user }) { - console.log(`next-auth jwt callback! trigger=${trigger}`) - if (account) { - console.log(`>>>> account was present!`) - console.log(account) - console.log(user) - token.currently_entitled_tiers = user.currently_entitled_tiers - } - if (trigger === 'signIn' && profile) { - console.log('trigger is signIn and profile as follows') - console.log(profile) - token.test = true - return token - } - if (trigger === "update" && session?.name) { - // Note, that `session` can be any arbitrary object, remember to validate it! - token.name = session.name - } - - token.test = 'yessir' - return token - }, - async session({ session, token, user }) { - console.log('Send properties to the client, like an access_token and user id from a provider.') - console.log(token) - console.log(session) - console.log(user) - session.token = token - if (token?.currently_entitled_tiers) { - session.roles = mapTierIdsToRoles(token.currently_entitled_tiers) - } - - - return session - } - } -} - - -export const { handler, signIn, signOut, auth } = NextAuth(authOptions) - - - -// async jwt({ token, account, profile }) { -// try { -// if (account) { -// const decodedToken = jwtDecode(account.access_token as any) -// if (token == null){ -// throw new Error("Unable to decode token") -// } -// console.log('decodedToken as follows') -// console.log(JSON.stringify(decodedToken, null, 2)) -// // Do something here to add more info, maybe just overwrite profile (thats the one that should have this info) -// profile = decodedToken -// token.account = account -// } -// if (profile) { -// console.log('profile as follows') -// console.log(profile) -// token.profile = profile -// // Then do here the assignation of roles elements to token so session has access -// // This can be modified so uses by client, realm or account BE AWARE OF THAT! -// // Modify the "resource_access['next-auth-AFB']" value to the one your resource/realm/accout -// // json scope roles you need - -// // While the info is already on profile, we could make a new key on the json response of session -// const clientRoles = profile.realm_access.roles -// token.client_roles = clientRoles -// } -// } catch (error) { -// console.log(error) -// } -// return token -// }, \ No newline at end of file diff --git a/services/next/app/lib/b2File.ts b/services/next/app/lib/b2File.ts deleted file mode 100644 index 389ab10..0000000 --- a/services/next/app/lib/b2File.ts +++ /dev/null @@ -1,2 +0,0 @@ -import { IMeta } from "@futureporn/types"; - diff --git a/services/next/app/lib/constants.ts b/services/next/app/lib/constants.ts deleted file mode 100644 index 30bde6b..0000000 --- a/services/next/app/lib/constants.ts +++ /dev/null @@ -1,27 +0,0 @@ -if (!process.env.NEXT_PUBLIC_SITE_URL) console.error('NEXT_PUBLIC_SITE_URL was missing in env'); -if (!process.env.NEXT_PUBLIC_STRAPI_URL) console.error('NEXT_PUBLIC_STRAPI_URL was missing in env'); -if (!process.env.NEXT_PUBLIC_UPPY_COMPANION_URL) console.error('NEXT_PUBLIC_UPPY_COMPANION_URL undefined in env'); - -export const companionUrl = ''+process.env.NEXT_PUBLIC_UPPY_COMPANION_URL -export const siteUrl = ''+process.env.NEXT_PUBLIC_SITE_URL -export const postgrestUrl = ''+process.env.NEXT_PUBLIC_POSTGREST_URL -export const postgrestLocalUrl = 'http://postgrest.futureporn.svc.cluster.local:9000' -export const patreonCampaignId: string = '8012692' -export const patreonSupporterBenefitId: string = '4760169' -export const patreonQuantumSupporterId: string = '10663202' -export const patreonVideoAccessBenefitId: string = '13462019' -export const patreonApiIdentityUrl = 'https://www.patreon.com/api/oauth2/v2/identity' -export const skeletonHeight = '32pt' -export const skeletonBaseColor = '#000' -export const skeletonHighlightColor = '#000' -export const skeletonBorderRadius = 0 -export const description = "The Galaxy's Best VTuber Hentai Site" -export const title = "Futureporn.net" -export const siteImage = 'https://futureporn.net/images/futureporn-icon.png' -export const favicon = 'https://futureporn.net/favicon.ico' -export const authorName = 'CJ_Clippy' -export const authorEmail = 'cj@futureporn.net' -export const authorLink = 'https://futureporn.net' -export const giteaUrl = 'https://gitea.futureporn.net' -export const projektMelodyEpoch = new Date('2020-02-07T23:21:48.000Z') -export const defaultImageBlur = '' \ No newline at end of file diff --git a/services/next/app/lib/contributors.ts b/services/next/app/lib/contributors.ts deleted file mode 100644 index 22a4798..0000000 --- a/services/next/app/lib/contributors.ts +++ /dev/null @@ -1,22 +0,0 @@ -import { postgrestLocalUrl } from "./constants"; -import fetchAPI from "./fetch-api"; - -export interface IContributor { - id: number; - name: string; - url?: string; - isFinancialDonor: boolean; - isVodProvider: boolean; -} - - -export async function getContributors(): Promise { - try { - const res = await fetchAPI(`/contributors`); - return res.data; - } catch (e) { - console.error(`error while fetching contributors`) - console.error(e); - return null; - } -} diff --git a/services/next/app/lib/dates.ts b/services/next/app/lib/dates.ts deleted file mode 100644 index 330dab2..0000000 --- a/services/next/app/lib/dates.ts +++ /dev/null @@ -1,59 +0,0 @@ -import { parse } from 'date-fns'; -import { format } from 'date-fns-tz' -import { toZonedTime, fromZonedTime } from 'date-fns-tz' - -const safeDateFormatString: string = "yyyyMMdd'T'HHmmss'Z'" -const localTimeZone = Intl.DateTimeFormat().resolvedOptions().timeZone; - - -export function getSafeDate(date: string | Date): string { - let dateString: string; - if (!date) throw new Error(`date passed to getSafeDate() was falsy. ${date}`); - if (typeof date === 'string') { - const dateObject = toZonedTime(date, 'UTC'); - dateString = format(dateObject, safeDateFormatString, { timeZone: 'UTC' }); - } else { - dateString = format(date, safeDateFormatString, { timeZone: 'UTC' }); - } - - return dateString; -} - - -export function getDateFromSafeDate(safeDate: string): Date { - console.log(`getDateFromSafeDate(). safeDate=${safeDate}`) - const date = parse(safeDate, safeDateFormatString, new Date()) - const utcDate = fromZonedTime(date, 'UTC') - return utcDate; -} - - -export function formatTimestamp(seconds: number = 0): string { - return new Date(seconds * 1000).toISOString().slice(11, 19); -} - -export function formatUrlTimestamp(timestampInSeconds: number): string { - const hours = Math.floor(timestampInSeconds / 3600); - const minutes = Math.floor((timestampInSeconds % 3600) / 60); - const seconds = timestampInSeconds % 60; - return `${hours}h${minutes}m${seconds}s`; -} - -export function parseUrlTimestamp(timestamp: string): number | null { - // Regular expression to match the "XhYmZs" format - const regex = /^(\d+)h(\d+)m(\d+)s$/; - const match = timestamp.match(regex); - - if (match) { - const hours = parseInt(match[1], 10); - const minutes = parseInt(match[2], 10); - const seconds = parseInt(match[3], 10); - - if (!isNaN(hours) && !isNaN(minutes) && !isNaN(seconds)) { - return hours * 3600 + minutes * 60 + seconds; - } - } - - // If the format doesn't match or parsing fails, return null - return null; -} \ No newline at end of file diff --git a/services/next/app/lib/fetch-api.ts b/services/next/app/lib/fetch-api.ts deleted file mode 100644 index 8db3165..0000000 --- a/services/next/app/lib/fetch-api.ts +++ /dev/null @@ -1,35 +0,0 @@ -// greets https://github.com/strapi/nextjs-corporate-starter/blob/main/frontend/src/app/%5Blang%5D/utils/fetch-api.tsx#L4 - -import qs from "qs"; -import { postgrestLocalUrl } from "./constants"; - -export default async function fetchAPI( - path: string, - urlParamsObject = {}, - options = {} -) { - try { - // Merge default and user options - const mergedOptions = { - next: { revalidate: 60 }, - headers: { - "Content-Type": "application/json", - }, - ...options, - }; - - // Build request URL - const queryString = qs.stringify(urlParamsObject); - const requestUrl = `${postgrestLocalUrl}${path}${queryString ? `?${queryString}` : ""}`; - console.log(`fetching--> ${requestUrl}`) - - // Trigger API call - const response = await fetch(requestUrl, mergedOptions); - const data = await response.json(); - return data; - - } catch (error) { - console.error(error); - throw new Error(`Error while fetching data from API. ${path}`); - } -} \ No newline at end of file diff --git a/services/next/app/lib/fetchers.ts b/services/next/app/lib/fetchers.ts deleted file mode 100644 index 98b6e5a..0000000 --- a/services/next/app/lib/fetchers.ts +++ /dev/null @@ -1,42 +0,0 @@ -import { postgrestLocalUrl } from "./constants"; - -/* - * https://postgrest.org/en/latest/references/api/pagination_count.html - * HTTP/1.1 206 Partial Content - * Content-Range: 0-24/3572000 - */ -export function getCountFromHeaders(res: Response) { - const count = parseInt(res.headers.get('Content-Range')?.split('/').at(-1) || '0') - return count -} - -export async function fetchPaginatedData(apiEndpoint: string, pageSize: number, queryParams: Record = {}): Promise { - let data: any[] = []; - let totalDataCount: number = 0; - let totalRequestsNeeded: number = 1; - - for (let requestCounter = 0; requestCounter < totalRequestsNeeded; requestCounter++) { - const humanReadableRequestCount = requestCounter + 1; - const params = new URLSearchParams({ - 'pagination[page]': humanReadableRequestCount.toString(), - 'pagination[pageSize]': pageSize.toString(), - ...queryParams, - }); - const url = `${postgrestLocalUrl}${apiEndpoint}?${params}`; - - const response = await fetch(url, { - method: 'GET' - }); - - const responseData = await response.json(); - - - if (requestCounter === 0) { - totalDataCount = responseData.meta.pagination.total; - totalRequestsNeeded = Math.ceil(totalDataCount / pageSize); - } - data = data.concat(responseData.data); - } - - return data; -} diff --git a/services/next/app/lib/ipfs.ts b/services/next/app/lib/ipfs.ts deleted file mode 100644 index ffb3f04..0000000 --- a/services/next/app/lib/ipfs.ts +++ /dev/null @@ -1,7 +0,0 @@ -export function buildIpfsUrl (urlFragment: string): string { - return `https://ipfs.io/ipfs/${urlFragment}` -} - -export function buildPatronIpfsUrl (cid: string, token: string): string { - return `https://gw.futureporn.net/ipfs/${cid}?token=${token}` -} diff --git a/services/next/app/lib/keycloak.ts b/services/next/app/lib/keycloak.ts deleted file mode 100644 index bbd6181..0000000 --- a/services/next/app/lib/keycloak.ts +++ /dev/null @@ -1,210 +0,0 @@ -import { type Tier, type TiersList, tiers } from './patreon' -import { configs } from '../config/configs'; - -export interface UMAToken { - upgraded: boolean; - access_token: string; - expires_in: number; - refresh_expires_in: number; - token_type: string; - "not-before-policy": number; -} - -export interface KeycloakGroup { - id: string; - name: string; - path: string; - subGroups: KeycloakGroup[]; -} - -// Maps patreon tiers to keycloak groups. -export const patreonToKeycloakMap: Record = { - everyone: 'ccaece3a-bb62-4fc9-be1d-6326ecdec65c', - free: 'ccaece3a-bb62-4fc9-be1d-6326ecdec65c', - archiveSupporter: '9e70fe60-5015-44df-ab77-d93b90e86738', - stealthSupporter: 'fd5358fe-d5e8-43be-85f2-27ddff711bd0', - tuneItUp: '4127e407-0c09-4c0a-b231-0bc058fb3dbd', - maxQ: '25b3d3af-2015-49ea-949f-b052fc82c6a3', - archiveCollector: '9e70fe60-5015-44df-ab77-d93b90e86738', - advancedArchiveSupporter: '7f875c63-4c74-4ae1-b33c-eefa7904d031', - quantumSupporter: '13076175-3a6f-4d80-84f2-6a8d5711e5f5', - sneakyQuantumSupporter: '13076175-3a6f-4d80-84f2-6a8d5711e5f5', - luberPlusPlus: '205d9c95-68ec-4b20-a6c1-28162af8a8f5' -}; - - -export function getTierNameFromTierId(tierId: string): string { - const tierName = Object.entries(tiers).find(([_, id]) => id === tierId)?.[0]; - if (!tierName) throw new Error(`getTierNameFromTierId failed to convert tierId=${tierId} into a tier name.`); - return tierName -} - -export function getKeycloakIdFromTierId(tierId: string): string | undefined { - // Find the tier name corresponding to the provided tier ID - const tierName = getTierNameFromTierId(tierId) - - if (!tierName) { - console.warn(`Tier ID ${tierId} not found in tiers map.`); - return undefined; // Return undefined if the tier ID is not found - } - - // Use the tier name to get the corresponding Keycloak group ID - const keycloakId = patreonToKeycloakMap[tierName]; - - if (!keycloakId) { - console.warn(`Keycloak ID for tier ${tierName} not found in patreonToKeycloakMap.`); - } - - return keycloakId; -} - - - -/** - * This gets us an access_token with our service-account-futureporn user privs. - * This service account has manage-users role so it can modify their group memebership. - */ -export async function getUMAToken(): Promise { - const res = await fetch(`${configs.keycloakLocalUrl}/realms/futureporn/protocol/openid-connect/token`, { - method: 'POST', - headers: { - 'Content-Type': 'application/x-www-form-urlencoded' - }, - body: new URLSearchParams({ - grant_type: 'urn:ietf:params:oauth:grant-type:uma-ticket', - client_id: configs.keycloakClientId, - client_secret: configs.keycloakClientSecret, - audience: 'futureporn' - }).toString() - }) - const data = await res.json() - if (!res.ok) { - throw new Error(`failed to getUMAToken. body=${JSON.stringify(data, null, 2)}, res.status=${res.status}, res.statusText=${res.statusText}`); - } - return data -} - - - - -export async function getUserGroups(uma: UMAToken, userId: string): Promise { - if (!uma) throw new Error('getUserGroups() requires a UMAToken as second param, but it was undefined.'); - if (!userId) throw new Error('getUserGroups() requires a userId as second param, but it was undefined.'); - const res = await fetch(`${configs.keycloakLocalUrl}/admin/realms/futureporn/users/${userId}/groups`, { - headers: { - 'Authorization': `Bearer ${uma.access_token}` - } - }) - const data = await res.json() as KeycloakGroup[] - if (!res.ok) { - throw new Error(`failed to getUserGroups. body=${JSON.stringify(data, null, 2)}, res.status=${res.status}, res.statusText=${res.statusText}`); - } - return data.map((g) => g.id) -} - - - -export async function addUserGroup(uma: UMAToken, userId: string, groupId: string): Promise { - const res = await fetch(`${configs.keycloakLocalUrl}/admin/realms/futureporn/users/${userId}/groups/${groupId}`, { - method: 'PUT', - headers: { - 'Authorization': `Bearer ${uma.access_token}` - } - }) - if (!res.ok) { - throw new Error(`failed to addUserGroup. res.status=${res.status}, res.statusText=${res.statusText}`); - } -} - - -export async function removeUserGroup(uma: UMAToken, userId: string, groupId: string): Promise { - const res = await fetch(`${configs.keycloakLocalUrl}/admin/realms/futureporn/users/${userId}/groups/${groupId}`, { - method: 'DELETE', - headers: { - 'Authorization': `Bearer ${uma.access_token}` - } - }) - if (!res.ok) { - throw new Error(`failed to removeUserGroup. res.status=${res.status}, res.statusText=${res.statusText}`); - } -} - - - - -/** - * Identifies the changes required to sync the actual state with the desired state. - * - * @param desired - Array of desired group IDs. - * @param actual - Array of actual group IDs. - * @returns An object with: - * - `toAdd`: Groups in `desired` but not in `actual` (need to be added). - * - `toRemove`: Groups in `actual` but not in `desired` (need to be removed). - */ -export function calculateSyncChanges(desired: string[], actual: string[]) { - const toAdd = Array.from(new Set(desired.filter(item => !actual.includes(item)))); - const toRemove = Array.from(new Set(actual.filter(item => !desired.includes(item)))); - return { toAdd, toRemove }; -} - -export async function syncronizeKeycloakRoles(keycloakUserId: string, patreonTiersList: TiersList): Promise { - - if (!keycloakUserId) throw new Error('syncronizeKeycloakRoles() requires userId as first param, but it was undefined.'); - if (!patreonTiersList) throw new Error('patreonTiersList() requires userId as second param, but it was undefined.'); - - // 1. [-] get Keycloak service account uma token - // POST https://keycloak.fp.sbtp.xyz/realms/futureporn/protocol/openid-connect/token - // 2. [-] use the map to determine the groups the user needs to be assigned to - // 3. [-] get a list of groups the user is currently a member of - // GET https://keycloak.fp.sbtp.xyz/admin/realms/futureporn/users/fbec8417-e5e3-4282-a98d-ed2fbc4a7e82/groups - // GET https://keycloak.fp.sbtp.xyz/admin/realms/futureporn/users/fbec8417-e5e3-4282-a98d-ed2fbc4a7e82/role-mappings/realm - // 4. [ ] add any groups the user should be a member of - // PUT https://keycloak.fp.sbtp.xyz/admin/realms/futureporn/users/fbec8417-e5e3-4282-a98d-ed2fbc4a7e82/groups/ccaece3a-bb62-4fc9-be1d-6326ecdec65c - // 5. [ ] remove any groups the user should not be a member of - // DELETE https://keycloak.fp.sbtp.xyz/admin/realms/futureporn/users/fbec8417-e5e3-4282-a98d-ed2fbc4a7e82/groups/ccaece3a-bb62-4fc9-be1d-6326ecdec65c - // - - - const uma = await getUMAToken() - // console.log('uma token as follows') - // console.log(uma) - // console.log('got uma token') - - - const desiredKeycloakGroupIds = patreonTiersList - .map((entitledTierId) => getKeycloakIdFromTierId(entitledTierId)) - .filter((groupId): groupId is string => groupId !== undefined); - // console.log(`got desiredKeycloakGroupIds=${desiredKeycloakGroupIds.join(', ')}`) - - - if (desiredKeycloakGroupIds.length === 0) { - throw new Error(`failed to get keycloak group id from the following patreon tiers: ${patreonTiersList.join(',')}`); - } - // console.log('desiredKeycloakGroupIds as follows') - // console.log(desiredKeycloakGroupIds) - - - const actualKeycloakGroupIds = await getUserGroups(uma, keycloakUserId); - // console.log(`actualKeyclaokGroupIds as follows`) - // console.log(actualKeycloakGroupIds) - - - const { toAdd, toRemove } = calculateSyncChanges(desiredKeycloakGroupIds, actualKeycloakGroupIds); - - - if (toAdd.length > 0) { - // console.log(`Groups to add: ${toAdd.join(', ')}`); - await Promise.all(toAdd.map(groupId => addUserGroup(uma, keycloakUserId, groupId))); - } - - if (toRemove.length > 0) { - // console.log(`Groups to remove: ${toRemove.join(', ')}`); - await Promise.all(toRemove.map(groupId => removeUserGroup(uma, keycloakUserId, groupId))); - } - - - console.log('finished updating keycloak groups.') - - -} - diff --git a/services/next/app/lib/patreon.ts b/services/next/app/lib/patreon.ts deleted file mode 100644 index 574cdcb..0000000 --- a/services/next/app/lib/patreon.ts +++ /dev/null @@ -1,277 +0,0 @@ -import { postgrestLocalUrl, patreonVideoAccessBenefitId, giteaUrl } from './constants' -import { type IPatron } from '@futureporn/types' -import { PublicPatron } from '@futureporn/utils/patron.ts'; -import { Session } from 'next-auth'; -import { JWT } from 'next-auth/jwt'; -import { type UMAToken, getKeycloakIdFromTierId, getUMAToken, syncronizeKeycloakRoles } from './keycloak'; -import { patreonApiIdentityUrl } from './constants'; - - -export type TiersList = string[] -interface KeycloakToken { - account: { - access_token: string - } -} -interface KeycloakIdpToken { - access_token: string; - expires_in: number; - refresh_expires_in: number; - refresh_token: string; - token_type: string; - "not-before-policy": number; - scope: string; - version: string; -} -export interface PatreonResponse { - data: PatreonUserData; - included: IncludedItem[]; - links: { - self: string; - }; -} - -export interface PatreonUserData { - attributes: UserAttributes; - id: string; - relationships: { - memberships: { - data: Relationship[]; - }; - }; - type: "user"; -} - -interface UserAttributes { - about: string | null; - created: string; // ISO 8601 date string - first_name: string; - full_name: string; - image_url: string; - last_name: string; - thumb_url: string; - url: string; - vanity: string | null; -} - -interface Relationship { - id: string; - type: string; -} - -type IncludedItem = Member | Tier | Benefit; - -interface Member { - attributes: Record; - id: string; - relationships: { - currently_entitled_tiers: { - data: Relationship[]; - }; - }; - type: "member"; -} - -export interface Tier { - attributes: Record; - id: string; - relationships: { - benefits: { - data: Relationship[]; - }; - }; - type: "tier"; -} - -interface Benefit { - attributes: Record; - id: string; - type: "benefit"; -} - -export interface ICampaign { - pledgeSum: number; - patronCount: number; -} - - -export interface IMarshalledCampaign { - data: { - attributes: { - pledge_sum: number, - patron_count: number - } - } -} - - -export const tiers = { - everyone: '-1', - free: '10620388', - archiveSupporter: '8154170', - stealthSupporter: '9561793', - tuneItUp: '9184994', - maxQ: '22529959', - archiveCollector: '8154171', - advancedArchiveSupporter: '8686045', - quantumSupporter: '8694826', - sneakyQuantumSupporter: '9560538', - luberPlusPlus: '8686022' -} - - -// Maps patreon tiers to keycloak roles -// @todo this is a dirty hack, this should be removed once we figure out how to sync keycloak roles at login. -export const tiersToRolesMap: Record = { - everyone: ['cdn_2', 'default-roles-futureporn', 'offline_access', 'uma_authorization'], - free: ['cdn_2', 'default-roles-futureporn', 'offline_access', 'uma_authorization'], - archiveSupporter: ['cdn_2', 'cdn_1', 'patron', 'website_shoutout', 'offline_access', 'default-roles-futureporn', 'uma_authorization'], - stealthSupporter: ['hacker', 'cdn_2', 'cdn_1', 'patron', 'website_shoutout', 'offline_access', 'default-roles-futureporn', 'uma_authorization'], - tuneItUp: ['cdn_2', 'cdn_1', 'patron', 'website_shoutout', 'offline_access', 'default-roles-futureporn', 'uma_authorization'], - maxQ: ['cdn_2', 'cdn_1', 'patron', 'website_shoutout', 'offline_access', 'default-roles-futureporn', 'uma_authorization', 'uploader'], - archiveCollector: ['cdn_2', 'cdn_1', 'patron', 'website_shoutout', 'offline_access', 'default-roles-futureporn', 'uma_authorization', 'uploader'], - advancedArchiveSupporter: ['cdn_2', 'cdn_1', 'patron', 'website_shoutout', 'offline_access', 'default-roles-futureporn', 'uma_authorization', 'uploader'], - quantumSupporter: ['cdn_2', 'cdn_1', 'patron', 'website_shoutout', 'offline_access', 'default-roles-futureporn', 'uma_authorization', 'uploader'], - sneakyQuantumSupporter: ['cdn_2', 'cdn_1', 'patron', 'website_shoutout', 'offline_access', 'default-roles-futureporn', 'uma_authorization', 'uploader'], - luberPlusPlus: ['cdn_2', 'cdn_1', 'patron', 'website_shoutout', 'offline_access', 'default-roles-futureporn', 'uma_authorization', 'uploader'] -}; - - -export const mapTierIdsToRoles = (tierIds: string[]): string[] => { - - console.log(`mapTierIdsToRoles tierIds=${JSON.stringify(tierIds, null, 2)}`) - // Reverse map of tiers for easier lookup by tier ID - const idToTier = Object.entries(tiers).reduce>( - (acc, [tierName, tierId]) => { - acc[tierId] = tierName; - return acc; - }, - {} - ); - - // Aggregate roles for all tier IDs - const roles = tierIds.flatMap((tierId) => { - const tierName = idToTier[tierId]; - return tierName ? tiersToRolesMap[tierName] : []; - }); - - // Remove duplicates and return - return [...new Set(roles)]; -}; - - -export async function updateKeycloakUserPatreonEntitlements(token: JWT): Promise { - - console.log(`updateKeycloakUserPatreonEntitlements() invoked.`) - - const userId = token?.sub - if (!userId) { - throw new Error(`failed to get userId from token.sub`); - } - - const keycloakidpToken = await getKeycloakIdpToken(token.access_token) - - if (!keycloakidpToken) { - throw new Error(`failed to get keycloakIdpToken; it was falsy.`) - } - - const patreonTiersList = await getPatreonMemberships(keycloakidpToken) - - - await syncronizeKeycloakRoles(userId, patreonTiersList) - - return patreonTiersList -} - -export async function getKeycloakIdpToken(access_token: string): Promise { - // console.log(`getKeycloakIdpToken() using access_token=${access_token}`) - - // @todo check the access_token. if it is expired, use the refresh_token to get a new access_token. - - const res = await fetch(`https://keycloak.fp.sbtp.xyz/realms/futureporn/broker/patreon/token`, { - headers: { - 'Authorization': `Bearer ${access_token}` - } - }) - - if (!res.ok) { - const bod = await res.text() - const msg = `req.status=${res.status} req.statusText=${res.statusText} body=${bod}` - console.log(msg) - throw new Error(`Failed to getKeycloakIdpToken. ${msg}`) - } - - const idpToken = await res.json() - return idpToken -} - -export function extractCurrentlyEntitledTiers(response: PatreonResponse): string[] { - return response.included - .filter((item): item is Member => item.type === "member") - .flatMap(member => member.relationships.currently_entitled_tiers.data) - .map((t) => t.id) // from the currently_entitled_tiers, we only want the id. - .filter((t) => Object.values(tiers).includes(t)) // we filter out any non-futureporn patreon tiers. -} - -export async function getPatreonMemberships(token: KeycloakIdpToken): Promise { - // console.log(`getPatreonMemberships with keycloakidpToken as follows`) - // console.log(token) - const query = 'fields%5Buser%5D=about,created,email,first_name,full_name,image_url,last_name,thumb_url,url,vanity&include=memberships,memberships.currently_entitled_tiers,memberships.currently_entitled_tiers.benefits'; - const res = await fetch(`${patreonApiIdentityUrl}?${query}`, { - headers: { - 'Authorization': `Bearer ${token.access_token}` - } - }) - const data = await res.json() as PatreonResponse - - // from the currently_entitled_tiers, we only want the id. - // we filter out any non-futureporn patreon tiers. - return extractCurrentlyEntitledTiers(data) - .map((rel) => rel.id) - .filter((t) => Object.values(tiers).includes(t)) -} - - - -export function isEntitledToPatronVideoAccess(session: Session): boolean { - if (!session.user?.patreonBenefits) return false; - const patreonBenefits = session.user.patreonBenefits - return (patreonBenefits.includes(patreonVideoAccessBenefitId)) -} - - -export async function getPatrons(): Promise { - let patrons: PublicPatron[] = [] - try { - const url = `${postgrestLocalUrl}/patrons` - console.log(`GET requesting ${url}`) - const res = await fetch(url); - const data = await res.json(); - if (!res.ok) throw new Error(`failed to get /patrons. res.status=${res.status}, res.statusText=${res.statusText}`); - if (!data) throw new Error(`no patron data was available. ${JSON.stringify(data)}`); - patrons = data - } catch (e) { - console.error('failed to get patrons~ list') - console.error(e) - return [] as PublicPatron[] - } - return patrons -} - - -export async function getCampaign(): Promise { - const res = await fetch('https://www.patreon.com/api/campaigns/8012692', { - headers: { - accept: 'application/json' - }, - next: { - revalidate: 43200 // 12 hour cache - } - }) - const campaignData = await res.json(); - const data = { - patronCount: campaignData.patron_count, - pledgeSum: campaignData.campaign_pledge_sum - } - return data -} diff --git a/services/next/app/lib/pm.ts b/services/next/app/lib/pm.ts deleted file mode 100644 index cc17e5a..0000000 --- a/services/next/app/lib/pm.ts +++ /dev/null @@ -1,139 +0,0 @@ -import matter from 'gray-matter'; - -const CACHE_TIME = 3600; -const GOAL_LABEL = 'Goal'; - -export interface IIssue { - id: number; - title: string; - comments: number; - updatedAt: string; - createdAt: string; - assignee: string | null; - name: string | null; - completedPercentage: number | null; - amountCents: number | null; - description: string | null; -} - -export interface IGoals { - complete: IIssue[]; - inProgress: IIssue[]; - planned: IIssue[]; - featuredFunded: IIssue; - featuredUnfunded: IIssue; -} - - -export interface IGiteaIssue { - id: number; - title: string; - body: string; - comments: number; - updated_at: string; - created_at: string; - assignee: string | null; -} - -const bigHairyAudaciousGoal: IIssue = { - id: 55234234, - title: 'BHAG', - comments: 0, - updatedAt: '2023-09-20T08:54:01.373Z', - createdAt: '2023-09-20T08:54:01.373Z', - assignee: null, - name: 'Big Hairy Audacious Goal', - description: 'World domination!!!!!1', - amountCents: 100000000, - completedPercentage: 0.04 -}; - -const defaultGoal: IIssue = { - id: 55234233, - title: 'e', - comments: 0, - updatedAt: '2023-09-20T08:54:01.373Z', - createdAt: '2023-09-20T08:54:01.373Z', - assignee: null, - name: 'Generic', - description: 'Getting started', - amountCents: 200, - completedPercentage: 1 -}; - -export function calcPercent(goalAmountCents: number, totalPledgeSumCents: number): number { - if (!goalAmountCents || totalPledgeSumCents <= 0) { - return 0; - } - const output = Math.min(100, Math.floor((totalPledgeSumCents / goalAmountCents) * 100)); - return output; -} - -export async function getGoals(pledgeSum: number): Promise { - try { - const openData = await fetchAndParseData('open', pledgeSum); - const closedData = await fetchAndParseData('closed', pledgeSum); - - - const inProgress = filterByAssignee(openData); - const planned = filterByAssignee(openData, true); - const funded = filterAndSortGoals(openData.concat(closedData), true); - const unfunded = filterAndSortGoals(openData.concat(closedData), false); - - // console.log('the following are unfunded goals') - // console.log(unfunded) - - return { - complete: closedData, - inProgress, - planned, - featuredFunded: funded[funded.length - 1] || defaultGoal, - featuredUnfunded: unfunded[0] || bigHairyAudaciousGoal - }; - } catch (error) { - console.error('Error fetching goals:', error); - return null; - } -} - -function filterByAssignee(issues: IIssue[], isPlanned: boolean = false): IIssue[] { - return issues.filter((issue) => (isPlanned ? issue.assignee === null : issue.assignee !== null)) -} - -async function fetchAndParseData(state: 'open' | 'closed', pledgeSum: number): Promise { - const response = await fetch(`https://gitea.futureporn.net/api/v1/repos/futureporn/pm/issues?state=${state}&labels=${GOAL_LABEL}`, { - next: { - revalidate: CACHE_TIME, - tags: ['goals'] - }, - }); - - if (!response.ok) return []; - - return response.json().then(issues => issues.map((g: IGiteaIssue) => parseGiteaGoal(g, pledgeSum))); -} - - - -function filterAndSortGoals(issues: IIssue[], isFunded: boolean): IIssue[] { - return issues - .filter((issue) => issue.amountCents) - .filter((issue) => (issue.completedPercentage === 100) === isFunded) - .sort((b, a) => b.amountCents! - a.amountCents!); -} - -function parseGiteaGoal(giteaIssue: IGiteaIssue, pledgeSum: number): IIssue { - const headMatter: any = matter(giteaIssue.body); - return { - id: giteaIssue.id, - title: giteaIssue.title, - comments: giteaIssue.comments, - updatedAt: giteaIssue.updated_at, - createdAt: giteaIssue.created_at, - assignee: giteaIssue.assignee, - name: headMatter.data.name || '', - description: headMatter.data.description || '', - amountCents: headMatter.data.amountCents || 0, - completedPercentage: calcPercent(headMatter.data.amountCents, pledgeSum) - }; -} diff --git a/services/next/app/lib/retry.ts b/services/next/app/lib/retry.ts deleted file mode 100644 index 68a4d7d..0000000 --- a/services/next/app/lib/retry.ts +++ /dev/null @@ -1,13 +0,0 @@ -export async function retry(fn: Function, maxRetries: number) { - let retries = 0; - while (retries < maxRetries) { - try { - return await fn(); - } catch (error) { - console.error(`Error during fetch attempt ${retries + 1}:`, error); - retries++; - } - } - console.error(`Max retries (${maxRetries}) reached. Giving up.`); - return null; -} \ No newline at end of file diff --git a/services/next/app/lib/rss.ts b/services/next/app/lib/rss.ts deleted file mode 100644 index 74ba374..0000000 --- a/services/next/app/lib/rss.ts +++ /dev/null @@ -1,51 +0,0 @@ -import { authorName, authorEmail, siteUrl, title, description, siteImage, favicon, authorLink } from './constants' -import { Feed } from "feed"; -import { getVods, getUrl, IVod } from '@/app/lib/vods' -import { ITagVodRelation } from '@/app/lib/tag-vod-relations'; - -export async function generateFeeds() { - const feedOptions = { - id: siteUrl, - title: title, - description: description, - link: siteUrl, - language: 'en', - image: siteImage, - favicon: favicon, - copyright: '', - generator: ' ', - feedLinks: { - json: `${siteUrl}/feed/feed.json`, - atom: `${siteUrl}/feed/feed.xml` - }, - author: { - name: authorName, - email: authorEmail, - link: authorLink - } - }; - - const feed = new Feed(feedOptions); - - const vods = await getVods() - - vods.data.map((vod: IVod) => { - feed.addItem({ - title: vod.title || vod.announceTitle, - description: vod.title, // @todo vod.spoiler or vod.note could go here - content: vod.tagVodRelations.data.map((tvr: ITagVodRelation) => tvr.attributes.tag.name).join(' '), - link: getUrl(vod, vod.vtuber.slug, vod.date2), - date: new Date(vod.date2), - image: vod.vtuber.image - }) - }) - - - return { - atom1: feed.atom1(), - rss2: feed.rss2(), - json1: feed.json1() - } -} - - diff --git a/services/next/app/lib/shareRef.ts b/services/next/app/lib/shareRef.ts deleted file mode 100644 index 5a6dddc..0000000 --- a/services/next/app/lib/shareRef.ts +++ /dev/null @@ -1,16 +0,0 @@ -import type { MutableRefObject, RefCallback } from 'react'; - -type RefType = MutableRefObject | RefCallback | null; - -export const shareRef = (refA: RefType, refB: RefType): RefCallback => instance => { - if (typeof refA === 'function') { - refA(instance); - } else if (refA && 'current' in refA) { - (refA as MutableRefObject).current = instance as T; // Use type assertion to tell TypeScript the type - } - if (typeof refB === 'function') { - refB(instance); - } else if (refB && 'current' in refB) { - (refB as MutableRefObject).current = instance as T; // Use type assertion to tell TypeScript the type - } -}; diff --git a/services/next/app/lib/streams.ts b/services/next/app/lib/streams.ts deleted file mode 100644 index 4d0a993..0000000 --- a/services/next/app/lib/streams.ts +++ /dev/null @@ -1,415 +0,0 @@ - -import { postgrestLocalUrl, postgrestUrl, siteUrl } from './constants'; -import { getSafeDate } from './dates'; -import qs from 'qs'; -import { IStream } from '@futureporn/types'; - - - -const fetchStreamsOptions = { - next: { - tags: ['streams'], - revalidation: 1 - } -} - - -export async function getStreamByUUID(uuid: string): Promise { - // const query = qs.stringify({ - // filters: { - // cuid: { - // $eq: cuid - // } - // }, - // pagination: { - // limit: 1 - // }, - // populate: { - // vtuber: { - // fields: ['slug', 'displayName'] - // }, - // tweet: { - // fields: ['isChaturbateInvite', 'isFanslyInvite', 'url'] - // }, - // vods: { - // fields: ['note', 'cuid', 'publishedAt'], - // populate: { - // tagVodRelations: { - // fields: ['id'] - // }, - // timestamps: '*' - // } - // } - // } - // }); - const query = `select=*,vods(*),vtuber:vtubers(slug,display_name,id,image,image_blur)&uuid=eq.${uuid}&limit=1` - const res = await fetch(`${postgrestUrl}/streams?${query}`); - const json = await res.json(); - console.log(json) - return json[0]; -} - -export function getUrl(stream: IStream, slug: string, date: string): string { - console.log(`getUrl() invoked with idk idk`) - return `${siteUrl}/vt/${slug}/stream/${getSafeDate(date)}` -} - - -export function getPaginatedUrl(): (slug: string, pageNumber: number) => string { - return (slug: string, pageNumber: number) => { - return `${siteUrl}/vt/${slug}/streams/${pageNumber}` - } -} - - - -export function getLocalizedDate(stream: IStream): string { - return new Date(stream.date).toLocaleDateString() -} - - - - -export async function getStreamsForYear(year: number): Promise { - const startOfYear = new Date(year, 0, 0); - const endOfYear = new Date(year, 11, 31); - - const pageSize = 100; // Number of records per page - let currentPage = 0; - let allStreams: IStream[] = []; - - while (true) { - const query = qs.stringify({ - filters: { - date: { - $gte: startOfYear, - $lte: endOfYear, - }, - }, - populate: { - vtuber: { - fields: ['displayName'] - } - }, - pagination: { - page: currentPage, - pageSize: pageSize, - } - }); - - const res = await fetch(`${postgrestUrl}/streams?${query}`, fetchStreamsOptions); - - if (!res.ok) { - // Handle error if needed - console.error('here is the res.body') - - console.error((await res.text())); - throw new Error(`Error fetching streams: ${res.status}`); - } - - const json = await res.json(); - const streams = json as IStream[]; - - if (streams.length === 0) { - // No more records, break the loop - break; - } - - allStreams = [...allStreams, ...streams]; - currentPage += pageSize; - } - - return allStreams; - } - -export async function getStream(id: number): Promise { - const query = qs.stringify({ - filters: { - id: { - $eq: id - } - } - }); - const res = await fetch(`${postgrestUrl}/vods?${query}`, fetchStreamsOptions); - const json = await res.json(); - return json.data; -} - - - - -export async function getAllStreams(archiveStatuses = ['missing', 'issue', 'good']): Promise { - throw new Error('getAllStreams function is not performant. please use something more efficient.') - const pageSize = 100; // Adjust this value as needed - const sortDesc = true; // Adjust the sorting direction as needed - - const allStreams: IStream[] = []; - let currentPage = 1; - - while (true) { - const query = qs.stringify({ - populate: { - vtuber: { - fields: ['slug', 'displayName', 'image', 'imageBlur', 'themeColor'], - }, - muxAsset: { - fields: ['playbackId', 'assetId'], - }, - thumbnail: { - fields: ['cdnUrl', 'url'], - }, - tagstreamRelations: { - fields: ['tag'], - populate: ['tag'], - }, - videoSrcB2: { - fields: ['url', 'key', 'uploadId', 'cdnUrl'], - }, - tweet: { - fields: ['isChaturbateInvite', 'isFanslyInvite'] - } - }, - filters: { - archiveStatus: { - '$in': archiveStatuses - } - }, - sort: { - date: sortDesc ? 'desc' : 'asc', - }, - pagination: { - pageSize, - page: currentPage, - }, - }); - const response = await fetch(`${postgrestUrl}/streams?${query}`, fetchStreamsOptions); - const responseData = await response.json(); - - if (!responseData.data || responseData.data.length === 0) { - // No more data to fetch - break; - } - - allStreams.push(...responseData.data); - currentPage++; - } - - return allStreams; -} - -export async function getStreamForVtuber(vtuberId: number, safeDate: string): Promise { - const query = qs.stringify({ - populate: { - vods: { - fields: [ - 'id', - 'date' - ] - }, - tweet: { - fields: [ - 'id' - ] - } - } - }); - - const response = await fetch(`${postgrestUrl}/streams?${query}`, fetchStreamsOptions); - - if (response.status !== 200) throw new Error('network fetch error while attempting to getStreamForVtuber'); - - const responseData = await response.json(); - return responseData; -} - -export async function getAllStreamsForVtuber(vtuberId: number, archiveStatuses = ['missing', 'issue', 'good']): Promise { - const maxRetries = 3; - - let retries = 0; - let allStreams: IStream[] = []; - let currentPage = 1; - - while (retries < maxRetries) { - try { - const query = qs.stringify({ - populate: '*', - filters: { - archiveStatus: { - '$in': archiveStatuses - }, - vtuber: { - id: { - $eq: vtuberId - } - } - }, - sort: { - date: 'desc', - }, - pagination: { - pageSize: 100, - page: currentPage, - }, - }); - - // console.log(`postgrestLocalUrl=${postgrestLocalUrl}`) - const response = await fetch(`${postgrestUrl}/streams?${query}`, fetchStreamsOptions) - - if (response.status !== 200) { - // If the response status is not 200 (OK), consider it a network failure - const bod = await response.text(); - console.log(response.status); - console.log(bod); - retries++; - continue; - } - - const responseData = await response.json(); - - if (!responseData.data || responseData.data.length === 0) { - // No more data to fetch - break; - } - - allStreams.push(...responseData.data); - currentPage++; - } catch (error) { - // Network failure or other error occurred - retries++; - } - } - - if (retries === maxRetries) { - throw new Error(`Failed to fetch streams after ${maxRetries} retries.`); - } - - return allStreams; -} - -/** - * Used as table data on /archive page. - * .pageIndex, pagination.pageSize - */ - -export async function fetchStreamData({ pageIndex, pageSize }: { pageIndex: number, pageSize: number }) { - // console.log(`fetchStreamData() invoked`) - const offset = pageIndex * pageSize; - // const query = qs.stringify({ - // populate: { - // vtuber: { - // fields: ['slug', 'displayName', 'publishedAt', 'image', 'imageBlur'] - // } - // }, - // filters: { - // vtuber: { - // publishedAt: { - // $notNull: true - // } - // } - // }, - // pagination: { - // start: offset, - // limit: pageSize, - // withCount: true - // }, - // sort: ['date:desc'] - // }) - const query = 'select=*,vtuber:vtubers(id,slug,image,image_blur,theme_color,display_name)' - const response = await fetch( - `${postgrestUrl}/streams?${query}` - ); - const data = await response.json(); - // console.log(data) - const filtered = data.filter((datum: any) => !!datum?.vtuber) - const d = { - rows: filtered, - pageCount: Math.ceil(filtered.length / pageSize), - rowCount: filtered, - } - // console.log(`fetchStreamData with pageIndex=${pageIndex}, pageSize=${pageSize}\n\n${JSON.stringify(d, null, 2)}`) - return d; -} - -export async function getStreamCountForVtuber(vtuberId: number, archiveStatuses = ['missing', 'issue', 'good']): Promise { - if (!vtuberId) throw new Error(`getStreamCountForVtuber requires a vtuberId, but it was undefined.`); - const res = await fetch( - `${postgrestLocalUrl}/streams?vtuber_id=eq.${vtuberId}&archive_status=in.(${archiveStatuses.join(',')})&limit=1`, - Object.assign(fetchStreamsOptions, { - headers: { - 'Prefer': 'count=exact' - } - }) - ) - const total = res.headers.get('Content-Range')?.split('/').at(-1) - const data = await res.json() - // console.log(`getStreamCountForVtuber with vtuberId=${vtuberId}, archiveStatuses=${archiveStatuses}, Content-Range=${res.headers.get('Content-Range')}`) - // console.log(JSON.stringify(data, null, 2)) - if (!total) { - console.error(`Failed to getStreamCountForVtuber-- total was falsy.`) - return 1; - } - else return parseInt(total)+1; -} - -export async function getStreamsForVtuber(vtuberId: number, page: number = 1, pageSize: number = 25, sortDesc = true): Promise { - // console.log(`getStreamsForVtuber() with postgrestLocalUrl=${postgrestLocalUrl}`) - const query = qs.stringify( - { - populate: { - vtuber: { - fields: [ - 'id', - ] - } - }, - filters: { - vtuber: { - id: { - $eq: vtuberId - } - } - }, - pagination: { - page: page, - pageSize: pageSize - }, - sort: { - date: (sortDesc) ? 'desc' : 'asc' - } - } - ) - const res = await fetch(`${postgrestUrl}/streams?${query}`, fetchStreamsOptions) - const data = await res.json() - // console.log(data) - return data -} - - -// /** -// * This returns stale data, because futureporn-historian is broken. -// * @todo get live data from historian -// * @see https://gitea.futureporn.net/futureporn/futureporn-historian/issues/1 -// */ -// export async function getProgress(vtuberSlug: string): Promise<{ complete: number; total: number }> { -// const query = qs.stringify({ -// filters: { -// vtuber: { -// slug: { -// $eq: vtuberSlug -// } -// } -// } -// }) -// const data = await fetch(`${postgrestUrl}/streams?${query}`, fetchStreamsOptions) -// .then((res) => res.json()) -// .then((g) => { -// return g -// }) - -// const total = data.meta.pagination.total - -// return { -// complete: total, -// total: total -// } -// } \ No newline at end of file diff --git a/services/next/app/lib/tag-vod-relations.ts b/services/next/app/lib/tag-vod-relations.ts deleted file mode 100644 index dad7e93..0000000 --- a/services/next/app/lib/tag-vod-relations.ts +++ /dev/null @@ -1,174 +0,0 @@ -/** - * Tag Vod Relations are an old name for what I'm now calling, "VodTag" - * - * VodTags are Tags related to Vods - * - */ - - -import qs from 'qs'; -import { postgrestLocalUrl } from './constants' -import { ITag, IToyTag } from './tags'; -import { IVod } from './vods'; -import { IMeta } from '@futureporn/types'; - - - - -export async function deleteTvr(authData: IAuthData, tagId: number) { - return fetch(`${postgrestLocalUrl}/tag-vod-relations/deleteMine/${tagId}`, { - method: 'DELETE', - headers: { - 'Authorization': `Bearer ${authData.accessToken}`, - 'Content-Type': 'application/json' - } - }) - .then((res) => { - if (!res.ok) throw new Error(res.statusText); - else return res.json(); - }) - .catch((e) => { - console.error(e); - // setError('root.serverError', { message: e.message }) - }) -} - -export async function readTagVodRelation(accessToken: string, tagId: number, vodId: number): Promise { - if (!tagId) throw new Error('readTagVodRelation requires tagId as second param'); - if (!vodId) throw new Error('readTagVodRelation requires vodId as second param'); - const findQuery = qs.stringify({ - filters: { - $and: [ - { - tag: tagId - }, { - vod: vodId - } - ] - } - }); - const res = await fetch(`${postgrestLocalUrl}/tag-vod-relations?${findQuery}`); - const json = await res.json(); - return json.data[0]; -} - -export async function createTagVodRelation(accessToken: string, tagId: number, vodId: number): Promise { - if (!accessToken) throw new Error('Must be logged in'); - if (!tagId) throw new Error('tagId is required.'); - if (!vodId) throw new Error('vodId is required.'); - const payload = { - tag: tagId, - vod: vodId - } - const res = await fetch(`${postgrestLocalUrl}/tag-vod-relations`, { - method: 'POST', - body: JSON.stringify({ data: payload }), - headers: { - authorization: `Bearer ${accessToken}`, - 'content-type': 'application/json' - } - }) - const json = await res.json(); - // console.log(json) - return json.data; -} - -export async function readOrCreateTagVodRelation (accessToken: string, tagId: number, vodId: number): Promise { - console.log(`Checking if the tagVodRelation with tagId=${tagId}, vodId=${vodId} already exists`); - const existingTagVodRelation = await readTagVodRelation(accessToken, tagId, vodId); - if (!!existingTagVodRelation) { - console.log(`there is an existing TVR so we return it`); - console.log(existingTagVodRelation); - return existingTagVodRelation - } - const newTagVodRelation = await createTagVodRelation(accessToken, tagId, vodId); - return newTagVodRelation; -} - -// export async function createTagAndTvr(setError: Function, authData: IAuthData, tagName: string, vodId: number) { -// if (!authData) throw new Error('Must be logged in'); -// if (!tagName || tagName === '') throw new Error('tagName cannot be empty'); -// const data = { -// tagName: tagName, -// vodId: vodId -// }; -// try { -// const res = await fetch(`${postgrestLocalUrl}/tag-vod-relations/tag`, { -// method: 'POST', -// body: JSON.stringify({ data }), -// headers: { -// 'Content-Type': 'application/json', -// 'Authorization': `Bearer ${authData.accessToken}` -// }, -// }); -// const json = await res.json(); -// return json.data; -// } catch (e) { -// setError('global', { type: 'idk', message: e }) -// } -// } - - -export async function getTagVodRelationsForVtuber(vtuberId: number, page: number = 1, pageSize: number = 25): Promise { - // get the tag-vod-relations where the vtuber is the vtuber we are interested in. - const query = qs.stringify( - { - populate: { - tag: { - fields: ['id', 'name'], - populate: { - toy: { - fields: ['linkTag', 'make', 'model', 'image2'], - populate: { - linkTag: { - fields: ['name'] - } - } - } - } - }, - vod: { - populate: { - vtuber: { - fields: ['slug'] - } - } - } - }, - filters: { - vod: { - vtuber: { - id: { - $eq: vtuberId - } - } - }, - tag: { - toy: { - linkTag: { - name: { - $notNull: true - } - } - } - } - }, - pagination: { - page: page, - pageSize: pageSize - }, - sort: { - id: 'desc' - } - } - ) - // we need to return an IToys object - // to get an IToys object, we have to get a list of toys from tvrs. - - - const res = await fetch(`${postgrestLocalUrl}/tag-vod-relations?${query}`); - if (!res.ok) return null; - const tvrs = await res.json() - return tvrs; -} - diff --git a/services/next/app/lib/tags.ts b/services/next/app/lib/tags.ts deleted file mode 100644 index 0850f5c..0000000 --- a/services/next/app/lib/tags.ts +++ /dev/null @@ -1,116 +0,0 @@ -import { postgrestLocalUrl } from './constants' -import { fetchPaginatedData } from './fetchers'; -import { IVod } from './vods'; -import slugify from 'slugify'; -import { IToy } from './toys'; -import qs from 'qs'; -import { IMeta } from '@futureporn/types'; - - -export interface ITag { - id: number; - name: string; - count: number; -} - - - -export function getTagHref(name: string): string { - return `/tags/${slugify(name)}` -} - - -export async function createTag(accessToken: string, tagName: string): Promise { - const payload = { - name: slugify(tagName) - }; - const res = await fetch(`${postgrestLocalUrl}/tags`, { - method: 'POST', - headers: { - 'authorization': `Bearer ${accessToken}`, - 'accept': 'application/json', - 'content-type': 'application/json' - }, - body: JSON.stringify({ data: payload }) - }); - const json = await res.json(); - // console.log(json); - if (!!json?.error) throw new Error(json.error.message); - if (!json?.data) throw new Error('created tag was missing data'); - return json.data as ITag; -} - -export async function readTag(accessToken: string, tagName: string): Promise { - - const findQuery = qs.stringify({ - filters: { - name: { - $eq: tagName - } - } - }); - const findResponse = await fetch(`${postgrestLocalUrl}/tags?${findQuery}`, { - method: 'GET', - headers: { - 'Authorization': `Bearer ${accessToken}`, - 'Accept': 'application/json', - 'Content-Type': 'application/json' - } - }); - - const json = await findResponse.json(); - return json.data[0]; -} - -export async function readOrCreateTag(accessToken: string, tagName: string): Promise { - console.log(`Checking if the tagName=${tagName} already exists`); - - const existingTag = await readTag(accessToken, tagName); - if (!!existingTag) { - console.log('there is an existing tag so we return it'); - console.log(existingTag); - return existingTag; - } - - const newTag = await createTag(accessToken, tagName); - return newTag; - - -} - -export async function getTags(): Promise { - const tagVodRelations = await fetchPaginatedData('/tag-vod-relations', 100, { 'populate[0]': 'tag', 'populate[1]': 'vod' }); - - // Create a Map to store tag data, including counts and IDs - const tagDataMap = new Map(); - - // Populate the tag data map with counts and IDs - tagVodRelations.forEach(tvr => { - const tagName = tvr.attributes.tag.name; - const tagId = tvr.attributes.tag.data.id; - - if (!tagDataMap.has(tagName)) { - tagDataMap.set(tagName, { id: tagId, count: 1 }); - } else { - const existingData = tagDataMap.get(tagName); - if (existingData) { - tagDataMap.set(tagName, { id: existingData.id, count: existingData.count + 1 }); - } - } - }); - - // Create an array of Tag objects with id, name, and count - const tags = Array.from(tagDataMap.keys()).map(tagName => { - const tagData = tagDataMap.get(tagName); - return { - id: tagData ? tagData.id : -1, - attributes: { - name: tagName, - count: tagData ? tagData.count : 0, - } - }; - }); - - - return tags; -} diff --git a/services/next/app/lib/timestamps.ts b/services/next/app/lib/timestamps.ts deleted file mode 100644 index 1f70423..0000000 --- a/services/next/app/lib/timestamps.ts +++ /dev/null @@ -1,125 +0,0 @@ - - -import qs from 'qs'; -import { postgrestLocalUrl, postgrestUrl } from './constants' -import { ITagsResponse, ITag, ITagResponse } from './tags'; -import { IMeta } from '@futureporn/types'; -import { getCountFromHeaders } from './fetchers'; - -export interface ITimestamp { - id: number; - time: number; - tag_name: string; - tn_short: string; - tag_id: number; - vod_id: number; - tag: ITagResponse; - created_at: string; - creator_id: number; -} - - - -export interface ITimestampResponse { - data: ITimestamp; - meta: IMeta; -} - -export interface ITimestampsResponse { - data: ITimestamp[]; - meta: IMeta; -} - -function truncateString(str: string, maxLength: number) { - if (str.length <= maxLength) { - return str; - } - return str.substring(0, maxLength - 1) + '…'; -} - -export function deleteTimestamp(authData: IAuthData, tsId: number) { - return fetch(`${postgrestLocalUrl}/timestamps/deleteMine/${tsId}`, { - method: 'DELETE', - headers: { - 'Authorization': `Bearer ${authData.accessToken}`, - 'Content-Type': 'application/json' - } - }) - .then((res) => { - if (!res.ok) throw new Error(res.statusText); - else return res.json(); - }) - .catch((e) => { - console.error(e); - // setError('root.serverError', { message: e.message }) - }) -} - -export async function createTimestamp( - authData: IAuthData, - tagId: number, - vodId: number, - time: number -): Promise { - if (!authData?.user?.id || !authData?.accessToken) throw new Error('User must be logged in to create timestamps'); - const query = qs.stringify({ - populate: '*' - }); - const response = await fetch(`${postgrestLocalUrl}/timestamps?${query}`, { - method: 'POST', - headers: { - 'Authorization': `Bearer ${authData.accessToken}`, - 'Content-Type': 'application/json' - }, - body: JSON.stringify({ - data: { - time: Math.floor(time), - tag: tagId, - vod: vodId, - creatorId: authData.user.id - } - }) - }); - - const json = await response.json(); - - if (!response.ok) { - throw new Error(json?.error?.message || response.statusText); - } - - return json.data; -} - - - -export async function getTimestampsVodLinksForVod(vodId: number, page: number = 1, pageSize: number = 25): Promise { - // const query = qs.stringify({ - // filters: { - // vod: { - // id: { - // $eq: vodId, - // }, - // }, - // }, - // populate: '*', - // sort: 'time:asc', - // pagination: { - // page: page, - // pageSize: pageSize, - // }, - // }); - - const query = `select=*,tags_vods(tag_id,tags(*))&id=eq.${vodId}`; - const response = await fetch(`${postgrestUrl}/vods?${query}`, { - headers: { - "Prefer": "count=exact" - } - }); - - // /vods?id=eq.326&select=timestamps_vod_links(timestamps(timestamps_tag_links(tags(id,name)))) - - - const data = await response.json() as ITimestamp[]; - - return data; -} \ No newline at end of file diff --git a/services/next/app/lib/toys.ts b/services/next/app/lib/toys.ts deleted file mode 100644 index 53cbea8..0000000 --- a/services/next/app/lib/toys.ts +++ /dev/null @@ -1,19 +0,0 @@ - -import { ITag, ITagResponse, ITagsResponse } from '@/app/lib/tags' -import { IMeta } from '@futureporn/types'; - - - - -/** This endpoint doesn't exist at the moment, but definitely could in the future */ -// export function getUrl(toy: IToy): string { -// return `${siteUrl}/toy/${toy.name}` -// } - -// export function getToysForVtuber(vtuberId: number, page: number = 1, pageSize: number = 25): Promise { -// const tvrs = await getTagVodRelationsForVtuber(vtuberId, page, pageNumber); -// return { -// data: tvrs.data. -// pagination: tvrs.pagination -// } -// } diff --git a/services/next/app/lib/tweets.ts b/services/next/app/lib/tweets.ts deleted file mode 100644 index 427c42f..0000000 --- a/services/next/app/lib/tweets.ts +++ /dev/null @@ -1,28 +0,0 @@ -import { IVtuberResponse, IMeta } from "@futureporn/types"; - - -export interface ITweet { - id: number; - attributes: { - date: string; - date2: string; - isChaturbateInvite: boolean; - isFanslyInvite: boolean; - cuid: string; - json: string; - id_str: string; - url: string; - vtuber: IVtuberResponse; - } -} - -export interface ITweetResponse { - data: ITweet; - meta: IMeta; -} - -export interface ITweetsResponse { - data: ITweet[]; - meta: IMeta; -} - diff --git a/services/next/app/lib/useForwardRef.ts b/services/next/app/lib/useForwardRef.ts deleted file mode 100644 index 13a97fd..0000000 --- a/services/next/app/lib/useForwardRef.ts +++ /dev/null @@ -1,27 +0,0 @@ -/** - * @see https://github.com/facebook/react/issues/24722#issue-1270749463 - */ - -import React, { useEffect, useRef, type ForwardedRef } from 'react'; - -const useForwardRef = ( - ref: ForwardedRef, - initialValue: any = null -) => { - const targetRef = useRef(initialValue); - - useEffect(() => { - if (!ref) return; - - if (typeof ref === 'function') { - ref(targetRef.current); - } else { - ref.current = targetRef.current; - } - }, [ref]); - - return targetRef; -}; - - -export default useForwardRef \ No newline at end of file diff --git a/services/next/app/lib/users.ts b/services/next/app/lib/users.ts deleted file mode 100644 index 188ae90..0000000 --- a/services/next/app/lib/users.ts +++ /dev/null @@ -1 +0,0 @@ -import { IMeta } from "@futureporn/types"; diff --git a/services/next/app/lib/vods.ts b/services/next/app/lib/vods.ts deleted file mode 100644 index ae3bc6f..0000000 --- a/services/next/app/lib/vods.ts +++ /dev/null @@ -1,414 +0,0 @@ - -import { postgrestLocalUrl, siteUrl } from './constants'; -import { getDateFromSafeDate, getSafeDate } from './dates'; -import { IVtuber, IStream, ITimestamp, IVod } from '@futureporn/types'; -import qs from 'qs'; -import { ITagVodRelation } from './tag-vod-relations'; -import { IMeta, IMuxAsset, IMuxAssetResponse } from '@futureporn/types'; -import { IS3File, IS3FileResponse } from '@/app/lib/b2File'; -import fetchAPI from './fetch-api'; -import { IUserResponse } from './users'; -import { getCountFromHeaders } from './fetchers'; - -/** - * Dec 2024 UUIDs were introduced. - * Going forward, use UUIDs where possible. - * safeDates are retained for backwards compatibility. - * - * @see https://www.w3.org/Provider/Style/URI - */ -export interface IVodPageProps { - params: { - safeDateOrUUID: string; - slug: string; - }; -} - - - - -const fetchVodsOptions = { - next: { - tags: ['vods'] - } -} - - -export async function getVodFromSafeDateOrUUID(safeDateOrUUID: string): Promise { - let vod: IVod | null; - let date: Date; - if (!safeDateOrUUID) { - console.log(`safeDateOrUUID was missing`); - return null; - } else if (/^[0-9a-f]{8}-[0-9a-f]{4}-[0-5][0-9a-f]{3}-[089ab][0-9a-f]{3}-[0-9a-f]{12}$/i.test(safeDateOrUUID)) { - console.log('this is a UUID!'); - vod = await getVodByUUID(safeDateOrUUID); - if (!vod) return null; - } else { - console.log(`this is a safe date. ${safeDateOrUUID}`) - date = getDateFromSafeDate(safeDateOrUUID); - console.log(`date=${date.toISOString()}`) - if (!date) { - console.log('there is no date') - return null; - } - vod = await getVodForDate(date); - } - return vod; -} - - - -export function getUrl(vod: IVod, slug: string, date: string): string { - return `/vt/${slug}/vod/${getSafeDate(date)}` -} - - -export function getPaginatedUrl(): (slug: string, pageNumber: number) => string { - return (slug: string, pageNumber: number) => { - return `${siteUrl}/vt/${slug}/vods/${pageNumber}` - } -} - - -/** @deprecated old format for futureporn.net/api/v1.json, which is deprecated. Please use getUrl() instead */ -export function getDeprecatedUrl(vod: IVod): string { - return `${siteUrl}/vods/${getSafeDate(vod.date_2)}` -} - - -export async function getNextVod(vod: IVod): Promise { - const query = new URLSearchParams({ - select: 'date_2,title,announce_title,vtuber:vtubers(slug)', - [`date_2`]: `gt.${vod.date_2}`, - [`vtuber.slug`]: `eq.${vod.vtuber.slug}`, - ['published_at']: 'not.is.null', - limit: '1', - order: 'date_2.asc', - }).toString(); - const res = await fetch(`${postgrestLocalUrl}/vods?${query}`, fetchVodsOptions) - const data = await res.json() - if (!res.ok) { - throw new Error(`res.status=${res.status} res.statusText=${res.statusText} \ndata=${JSON.stringify(data, null, 2)}`); - } - return data[0]; -} - - -export function getLocalizedDate(vod: IVod): string { - return new Date(vod.date_2).toLocaleDateString() -} - -export async function getPreviousVod(vod: IVod): Promise { - const query = new URLSearchParams({ - select: 'date_2,title,announce_title,vtuber:vtubers(slug)', - [`date_2`]: `lt.${vod.date_2}`, - [`vtuber.slug`]: `eq.${vod.vtuber.slug}`, - limit: '1', - order: 'date_2.desc', - }).toString(); - const res = await fetch(`${postgrestLocalUrl}/vods?${query}`, fetchVodsOptions) - const data = await res.json() - if (!res.ok) { - throw new Error(`res.status=${res.status} res.statusText=${res.statusText} \ndata=${JSON.stringify(data, null, 2)}`); - } - return data[0]; -} - -export async function getVodByUUID(uuid: string): Promise { - const query = new URLSearchParams({ - uuid: `eq.${uuid}`, - select: ` - vtuber(slug,displayName,image,imageBlur,themeColor), - muxAsset(playbackId,assetId), - thumbnail(cdnUrl,url), - tagVodRelations(tag,createdAt,creatorId,tag(*)), - videoSrcB2(url,key,uploadId,cdnUrl), - stream(archiveStatus,date,tweet,uuid), - timestamps_vod_links(timestamps(timestamps_tag_links(tags(id,name)))) - `.replace(/\s+/g, ''), // Remove whitespace for URL encoding - }); - - - try { - const res = await fetch(`${postgrestLocalUrl}/vods?${query}`, { cache: 'no-store', next: { tags: ['vods'] } }) - if (!res.ok) { - throw new Error('failed to fetch getVodByUUID') - } - const json = await res.json() - const vod = json.data[0] - if (!vod) return null; - return vod; - } catch (e) { - if (e instanceof Error) { - console.error(e) - } - return null; - } -} -export async function getVodForDate(date: Date): Promise { - const iso8601DateString = date.toISOString(); - console.log(`getVodForDate ison8601DateString=${iso8601DateString}`) - - const selectFields = [ - 'id', - 'date_2', - 'title', - 'announce_title', - 'vtuber:vtubers(slug,display_name,image,image_blur,theme_color)', - 'timestamps_vod_links(timestamps(timestamps_tag_links(tags(id,name))))', - ].join(','); - - const queryParams = { - select: selectFields, - date_2: `eq.${iso8601DateString}`, - published_at: 'not.is.null', - limit: '1', - }; - - const query = new URLSearchParams(queryParams).toString(); - - console.log(query) - - const res = await fetch(`${postgrestLocalUrl}/vods?${query}`, { - cache: 'no-store', - next: { tags: ['vods'] } - }); - - const json = await res.json(); - if (!res.ok) { - console.log(`res.status=${res.status} res.statusText=${res.statusText} body=${JSON.stringify(json, null, 2)}`) - throw new Error('Failed to fetch vodForDate'); - } - - const vod = json[0]; // PostgREST returns an array of results - if (!vod) return null; - - return vod; - -} - - -export async function getVod(id: number): Promise { - const query = qs.stringify( - { - filters: { - id: { - $eq: id - } - } - } - ) - const res = await fetch(`${postgrestLocalUrl}/vods?${query}`, fetchVodsOptions); - if (!res.ok) return null; - const data = await res.json(); - return data; -} - -export async function getVods(page: number = 1, pageSize: number = 25, sortDesc = true): Promise<{ vods: IVod[], count: number }> { - // const query = qs.stringify( - // { - // populate: { - // vtuber: { - // fields: ['slug', 'displayName', 'image', 'imageBlur'] - // }, - // muxAsset: { - // fields: ['playbackId', 'assetId'] - // }, - // thumbnail: { - // fields: ['cdnUrl', 'url'] - // }, - // tagVodRelations: { - // fields: ['tag'], - // populate: ['tag'] - // }, - // videoSrcB2: { - // fields: ['url', 'key', 'uploadId', 'cdnUrl'] - // } - // }, - // sort: { - // date: (sortDesc) ? 'desc' : 'asc' - // }, - // pagination: { - // pageSize: pageSize, - // page: page - // } - // } - // ) - - // console.log(`postgrestLocalUrl=${postgrestLocalUrl} query=${query}`) - // const url = `${configs.postgrestUrl}/vods?select=*,segments(*),recording:recordings(is_aborted)&id=eq.${vodId}` - const selects = [ - '*', - 'vtuber:vtubers(id,slug,image,display_name,image_blur)', - // 'mux_asset:mux_assets(playback_id,asset_id)', - 'thumbnail:b2_files(cdn_url)' - ] - const queryObject = { - select: selects.join(','), - limit: pageSize, - offset: page*pageSize - }; - const query = qs.stringify(queryObject, { encode: false }); - - const res = await fetch( - `${postgrestLocalUrl}/vods?${query}`, - Object.assign({}, fetchVodsOptions, { headers: { - 'Prefer': 'count=exact' - }}) - ); - const data = await res.json() - if (!res.ok) { - throw new Error(`Failed to fetch vods status=${res.status}, statusText=${res.statusText}, data=${data}`); - } - - console.log(`${data.length} vods. sample as follows.`) - console.log(data[0]) - - const count = getCountFromHeaders(res) - - return { - vods: data, - count: count - } -} - - - -export async function getAllVods(): Promise { - const pageSize = 100; // Adjust this value as needed - const sortDesc = true; // Adjust the sorting direction as needed - - const allVods: IVod[] = []; - let currentPage = 1; - - while (true) { - const query = qs.stringify({ - populate: { - vtuber: { - fields: ['slug', 'displayName', 'image', 'imageBlur'], - }, - muxAsset: { - fields: ['playbackId', 'assetId'], - }, - thumbnail: { - fields: ['cdnUrl', 'url'], - }, - tagVodRelations: { - fields: ['tag'], - populate: ['tag'], - }, - videoSrcB2: { - fields: ['url', 'key', 'uploadId', 'cdnUrl'], - }, - }, - sort: { - date: sortDesc ? 'desc' : 'asc', - }, - pagination: { - pageSize, - page: currentPage, - }, - }); - - try { - const response = await fetch(`${postgrestLocalUrl}/vods?${query}`, fetchVodsOptions); - - if (!response.ok) { - // Handle non-successful response (e.g., HTTP error) - throw new Error(`HTTP error! Status: ${response.status}`); - } - - const responseData = await response.json(); - - if (!responseData.data || responseData.data.length === 0) { - // No more data to fetch - break; - } - - allVods.push(...responseData.data); - currentPage++; - } catch (error) { - // Handle fetch error - if (error instanceof Error) { - console.error('Error fetching data:', error.message); - } - return null; - } - } - - return allVods; -} - - - -export async function getVodsForVtuber(vtuberId: number, page: number = 1, pageSize: number = 25, sortDesc = true): Promise<{ vods: IVod[] | null, count: number }> { - const res = await fetch(`${postgrestLocalUrl}/vods?limit=${pageSize}&offset=${page*pageSize}&select=*,thumbnail:b2_files(cdn_url),vtuber:vtubers(id,slug,image,display_name,image_blur)&vtuber.id=eq.${vtuberId}`, fetchVodsOptions) - if (!res.ok) { - const body = await res.text() - console.error(`getVodsForVtuber() failed. ok=${res.ok} status=${res.status} statusText=${res.statusText} body=${body}`); - return { vods: null, count: 0 }; - } - const vods = await res.json() as IVod[]; - const count = getCountFromHeaders(res) - return { vods, count }; -} - - -export async function getVodsForTag(tag: string): Promise { - // const query = qs.stringify( - // { - // populate: { - // vtuber: { - // fields: ['slug', 'displayName', 'image', 'imageBlur'] - // }, - // videoSrcB2: { - // fields: ['url', 'key', 'uploadId', 'cdnUrl'] - // } - // }, - // filters: { - // tagVodRelations: { - // tag: { - // name: { - // $eq: tag - // } - // } - // } - // } - // } - // ) - const query = 'select(*),vtuber:vtubers(id,slug,image,display_name,image_blur),thumbnail:b2_files(cdn_url)' - const res = await fetch(`${postgrestLocalUrl}/vods?${query}`, fetchVodsOptions) - if (!res.ok) return null; - const vods = await res.json() - return vods; -} - -/** - * This returns stale data, because futureporn-historian is broken. - * @todo get live data from historian - * @see https://git.futureporn.net/futureporn/futureporn-historian/issues/1 - */ -export async function getProgress(vtuberSlug: string): Promise<{ complete: number; total: number }> { - const query = qs.stringify({ - filters: { - vtuber: { - slug: { - $eq: vtuberSlug - } - } - } - }) - const data = await fetch(`${postgrestLocalUrl}/vods?${query}`, fetchVodsOptions) - .then((res) => res.json()) - .then((g) => { - return g - }) - - const total = data.meta.pagination.total - - return { - complete: total, - total: total - } -} \ No newline at end of file diff --git a/services/next/app/lib/vtubers.ts b/services/next/app/lib/vtubers.ts deleted file mode 100644 index 4289984..0000000 --- a/services/next/app/lib/vtubers.ts +++ /dev/null @@ -1,95 +0,0 @@ - - -import { IVod } from './vods' -import { postgrestLocalUrl, siteUrl } from './constants'; -import qs from 'qs'; -import { IMeta, IVtuber } from '@futureporn/types'; - - -const fetchVtubersOptions = { - next: { - tags: ['vtubers'] - } -} - - - - -export function getUrl(slug: string): string { - return `${siteUrl}/vt/${slug}` -} - - - - -export async function getVtuberBySlug(slug: string): Promise { - const res = await fetch(`${postgrestLocalUrl}/vtubers?slug=eq.${slug}`); - if (!res.ok) { - console.error(`error inside getVtuberBySlug-- ${res.statusText}`); - return null; - } - const vtubers = await res.json(); - return vtubers[0]; -} - -export async function getVtuberById(id: number): Promise { - const res = await fetch(`${postgrestLocalUrl}/vtubers?id=eq.${id}`); - if (!res.ok) { - console.error(`error inside getVtuberById-- ${res.statusText}`); - return null; - } - const vtuber = await res.json(); - return vtuber -} - -export async function getVtubers(): Promise { - const res = await fetch(`${postgrestLocalUrl}/vtubers`); - if (!res.ok) { - console.error(`error inside getVtubers-- ${res.statusText}`); - return null; - } - const vtubers = await res.json(); - return vtubers; - -} - -export async function getAllVtubers(): Promise { - const pageSize = 100; - - const allVtubers: IVtuber[] = []; - let currentPage = 1; - - while (true) { - const query = new URLSearchParams({ - offset: ''+currentPage, - limit: ''+pageSize - }).toString() - try { - console.log(`Getting ${postgrestLocalUrl}/vtubers page=${currentPage}`); - const response = await fetch(`${postgrestLocalUrl}/vtubers?${query}`, fetchVtubersOptions); - - if (!response.ok) { - // Handle non-successful response (e.g., HTTP error) - throw new Error(`HTTP error! Status: ${response.status}`); - } - - const responseData = await response.json(); - - if (!responseData.data || responseData.data.length === 0) { - // No more data to fetch - break; - } - - allVtubers.push(...responseData.data); - currentPage++; - } catch (error) { - // Handle fetch error - if (error instanceof Error) { - console.error('Error fetching data:', error.message); - } - return null; - } - } - - return allVtubers; -} \ No newline at end of file diff --git a/services/next/app/page.tsx b/services/next/app/page.tsx deleted file mode 100644 index 2985faa..0000000 --- a/services/next/app/page.tsx +++ /dev/null @@ -1,81 +0,0 @@ - -import FundingGoal from "@/app/components/funding-goal"; -import VodCard from "@/app/components/vod-card"; -import { getVodTitle } from "@/app/components/vod-page"; -import { getVods } from '@/app/lib/vods'; -import { IVod } from "@/app/lib/vods"; -import { IVtuber } from '@futureporn/types'; -import { getVtubers } from "./lib/vtubers"; -import VTuberCard from "./components/vtuber-card"; -import Link from 'next/link'; -import { notFound } from "next/navigation"; -import { ErrorCard } from "./components/error-card"; - - -export default async function Page() { - - const { vods, count } = await getVods(1, 9, true); - - const vtubers = await getVtubers(); - - return ( - <> -

-
-
-

- The Galaxy's Best VTuber Hentai Site -

-

For adults only (NSFW)

-
-
- - - - - -
- -

Latest VODs

-
- - - {(count === 0) &&

Error: Failed to fetch VODs from the database

} - - {/*
-                
-                  {JSON.stringify(vods, null, 2)}
-                
-              
*/} - {vods && vods.map((vod: IVod) => ( - - ))} - -
- - See all Latest Vods -
-
- -

VTubers

- -
-
- - ); -} diff --git a/services/next/app/patrons/page.tsx b/services/next/app/patrons/page.tsx deleted file mode 100644 index b44e91c..0000000 --- a/services/next/app/patrons/page.tsx +++ /dev/null @@ -1,46 +0,0 @@ - -import PatronsList from '../components/patrons-list'; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faExternalLinkAlt } from "@fortawesome/free-solid-svg-icons"; -import Link from 'next/link' -import { getCampaign } from '../lib/patreon'; - -export default async function Page() { - - const patreonCampaign = await getCampaign() - - return ( - <> -
-
-
-

Patron List

-

- Futureporn.net continues to improve thanks to - {patreonCampaign.patronCount} generous supporters. -

-
-
- - - -
-

Want to get your name on this list, and get perks like FAST video streaming?

- - Become a patron today! - - -

- Patron names are private by default--{' '} - Opt-in to have your name displayed. -

-
-
-
-
- - ); -} diff --git a/services/next/app/profile/hooks/patreonCurrentlyEntitledTiers.ts b/services/next/app/profile/hooks/patreonCurrentlyEntitledTiers.ts deleted file mode 100644 index 1440cda..0000000 --- a/services/next/app/profile/hooks/patreonCurrentlyEntitledTiers.ts +++ /dev/null @@ -1,16 +0,0 @@ -import { TiersList } from "@/app/lib/patreon" - -export async function fetchPatreonCurrentlyEntitledTiers (): Promise { - const response = await fetch(`/api/patreon/currently-entitled-tiers`, { - credentials: 'include' - }) - const data = await response.json() - - if (!response.ok) { - throw new Error(`failed to fetchPatreonCurrentlyEntitledTiers() res.status=${response.status}`) - } - - - console.log(data) - return data -} diff --git a/services/next/app/profile/hooks/updateProfile.ts b/services/next/app/profile/hooks/updateProfile.ts deleted file mode 100644 index 3c6bac4..0000000 --- a/services/next/app/profile/hooks/updateProfile.ts +++ /dev/null @@ -1,21 +0,0 @@ -import { configs } from "@/app/config/configs"; - - -export type UpdateIsUsernamePublicInput = { - isUsernamePublic: boolean; - access_token: string; -} - - -export async function updateIsUsernamePublic (data: UpdateIsUsernamePublicInput): Promise { - console.log(`upadateIsUernamePublic configs.nextUrl=${configs.nextUrl}`) - const res = await fetch(`${configs.nextUrl}/api/profile`, { - method: 'POST', - body: JSON.stringify({ - username_visibility: data.isUsernamePublic - }) - }) - const d = await res.json() - if (!res.ok) throw new Error(`failed to updateProfile. ${res.status} ${res.statusText}`); - return d -} \ No newline at end of file diff --git a/services/next/app/profile/hooks/useMetadata.ts b/services/next/app/profile/hooks/useMetadata.ts deleted file mode 100644 index 62149e9..0000000 --- a/services/next/app/profile/hooks/useMetadata.ts +++ /dev/null @@ -1,52 +0,0 @@ -// import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query' - -// export type MetadataPayload = { -// metadata: { -// first_name: string -// last_name: string -// patreon: { -// entitledTiers: string[], -// lifetimeSupportCents: number, -// currentlyEntitledSupportCents: number -// }, -// isUsernamePublic: boolean -// } -// } -// export type UpdateMetadataInput = { isUsernamePublic: boolean } - - -// const fetchMetadata = async (): Promise => { -// const response = await fetch(`/api/user/attributes`) -// const data = await response.json() -// return data -// } - -// const fetchMutateMetadata = async (data: UpdateMetadataInput): Promise => { -// const res = await fetch('/api/user/attributes', { -// method: 'POST', -// body: JSON.stringify({ -// isUsernamePublic: data.isUsernamePublic -// }) -// }) -// if (!res.ok) throw new Error(`failed to fetchMutateMetadata. ${res.status} ${res.statusText}`); -// } - -// const useMetadata = () => { -// return useQuery({ -// queryKey: ['user/attributes'], -// queryFn: () => fetchMetadata() -// }) -// } - -// const useMutateMetadata = () => { -// const queryClient = useQueryClient() - -// return useMutation({ -// mutationFn: (data: UpdateMetadataInput) => fetchMutateMetadata(data), -// onSuccess: () => { -// queryClient.invalidateQueries({ queryKey: ['user/attributes'] }) -// } -// }) -// } - -// export { useMetadata, fetchMetadata, useMutateMetadata } diff --git a/services/next/app/profile/page.tsx b/services/next/app/profile/page.tsx deleted file mode 100644 index dcf51ba..0000000 --- a/services/next/app/profile/page.tsx +++ /dev/null @@ -1,242 +0,0 @@ -'use client'; - -import 'react-loading-skeleton/dist/skeleton.css'; -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faSpinner, faCheckCircle, faXmarkCircle } from "@fortawesome/free-solid-svg-icons"; -import { useMutation, useQueryClient } from "@tanstack/react-query"; -// import { updateIsUsernamePublic } from "./hooks/updateProfile"; -import { useQuery } from '@tanstack/react-query'; -import ProtectedRoute from "../components/protected-route"; -import VibrateTest from "../components/vibrate-test"; -import { useSession } from "next-auth/react" -import { LoginButton, LogoutButton } from "../components/auth-buttons"; -import Skeleton, { SkeletonTheme } from 'react-loading-skeleton'; -import Link from 'next/link'; -import { useEffect } from 'react'; -import { useState } from 'react'; -import { fetchPatreonCurrentlyEntitledTiers } from './hooks/patreonCurrentlyEntitledTiers'; -import { Spinner } from '../components/spinner'; -import { signIn, signOut } from "next-auth/react"; - -export default function Page() { - - const queryClient = useQueryClient() - // const sessionContext = Session.useSessionContext() - // if (sessionContext.loading) { - // return - // } - const { data, status, update } = useSession() - - - async function renewJwt() { - await signOut({ redirect: false }); - await signIn("keycloak", { redirect: false }); - } - - // const { data: patreonTier, isPending, refetch } = useQuery({ - // queryKey: ['user/tier'], - // queryFn: () => fetchPatreonCurrentlyEntitledTiers(), - // }) - - // const keycloakGroupQuery = useQuery({ - // queryKey: ['user/group'], - // queryFn: () => fetchKeycloakGroup() - // }) - - // const attributesMutation = useMutation({ - // mutationFn: updateIsUsernamePublic, - // mutationKey: ['user/attributes'], - // onSuccess: () => queryClient.invalidateQueries({ queryKey: ['user/attributes'] }) - // }) - - // const metadataQuery = useQuery({ - // queryKey: ['user/metadata'], - // queryFn: () => fetchMetadata() - // }) - - // const [ isUsernamePublic, setIsUsernamePublic ] = useState() - // const [ username, setUsername ] = useState('') - // useEffect(() => { - // if (metadataQuery.data?.metadata) { - // setUsername(formatPatronUsername(metadataQuery.data.metadata.first_name, metadataQuery.data.metadata.last_name)) - // setIsUsernamePublic(metadataQuery.data.metadata.isUsernamePublic) - // } - // }, [metadataQuery.data]) - - const isUsernamePublic = (data?.profile?.username_visibility === 'public') - const [isGo, setIsGo] = useState() - - // useEffect(() => { - // const fetchData = async () => { - // const res = await fetch(`${process.env.KEYCLOAK_URL}/protocol/openid-connect/token`, { - // method: 'POST', - // headers: { 'Content-Type': 'application/x-www-form-urlencoded' }, - // body: new URLSearchParams({ - // client_id: process.env.KEYCLOAK_CLIENT_ID, - // client_secret: process.env.KEYCLOAK_CLIENT_SECRET, - // grant_type: 'refresh_token', - // refresh_token: session.refreshToken, - // }), - // }); - - // const data = await res.json(); - - // if (res.ok) { - // // Update session with new token - // session.accessToken = data.access_token; - // session.refreshToken = data.refresh_token; - // res.status(200).json(session); - // } else { - // res.status(400).json({ error: 'Token refresh failed' }); - // } - // } - - // fetchData(); - // }, [isGo]); - - - return ( -
- {/* {(metadataQuery.isPending) && } */} -
-

Debug section

- {/* {(isPending) ? Syncronizing Patron entitlements... : Patreon entitlements synced.} */} - {/*

metadataQuery as follows

*/} - {/*
{JSON.stringify(metadataQuery.data, null, 2)}
*/} -

session as follows

-
{JSON.stringify(data, null, 2)}
- {(status === 'unauthenticated') && } - - {/* */} - - -
- - - {(data?.user) && -
- -

Profile

- - - -
-
-
- {/*

{username}

*/} -

{data?.user?.name}

-
-
-
- - {/*

Patronage

-
-

You belong to the {patreonTier} tier. Thank you!

-
*/} - -

Patron Perks

-
-

Test Perk (not real)

- -

Hey look at you, you're granted access to this perk!

-
-
- -
-

Website Shoutout

- - {isUsernamePublic && {data.user.name} will be displayed on the /patrons page} - {!isUsernamePublic && Your username will be hidden from the /patrons page due to your privacy setting.} - {/* Users can update privacy pref at https://keycloak.fp.sbtp.xyz/realms/futureporn/account/ */} - -
- -
-

Premium Content Delivery Network (CDN1)

- -

You have access to Futureporn's Premium CDN for reliable video playback

-
-
- -
-

Economy Content Delivery Network (CDN2)

- -

You have access to Futureporn's Economy CDN for basic video playback

-
-
- - -
-

Vibrate (test)

- -
- -
- } -
- ) - - // if (metadataQuery.isPending) return ; - // return
{JSON.stringify(metadataQuery.data, null, 2)}
- // return

@todo

- // if (!isPending) return ; - // if (!data) return

there is no data

; - // if (!data?.metadata) return

there is no data.metadata

; - // const { first_name, last_name } = data.metadata - // const username = formatPatronUsername(first_name, last_name) - // const { metadata } = data || {} - // const { isUsernamePublic } = metadata - - - // return ( - // <> - - //
- //
- //

Profile

- - //
- //
Username
- //

- // {isPending && - // - // - // - // } - // {!isPending &&

{username}

} - //

- //
- - //
- //
User ID
- //
- //

{sessionContext.userId}

- //
- //
- - //

Patron Perks

- //

Website Shoutout

- // - //
- // Display {(!!username) ? username : } publicly? - // attributesMutation.mutate({ isUsernamePublic: evt.target.checked })}/> - // {(attributesMutation.status === 'success') && } - // {(attributesMutation.status === 'pending') && } - // {(attributesMutation.status === 'error') && } - //
- //
- - //

Content Delivery Network (CDN)

- // - //

You have access to Futureporn's CDN for reliable video playback

- //
- - - //

Vibrate (test)

- // - //
- //
- - // - // ) -} \ No newline at end of file diff --git a/services/next/app/sign-in.tsx b/services/next/app/sign-in.tsx deleted file mode 100644 index ae47881..0000000 --- a/services/next/app/sign-in.tsx +++ /dev/null @@ -1,19 +0,0 @@ -'use client'; - -type Props = { - onSignIn: () => Promise; -}; - -const SignIn = ({ onSignIn }: Props) => { - return ( -
{ - onSignIn(); - }} - > - Sign In -
- ); -}; - -export default SignIn; \ No newline at end of file diff --git a/services/next/app/sign-out.tsx b/services/next/app/sign-out.tsx deleted file mode 100644 index 504b139..0000000 --- a/services/next/app/sign-out.tsx +++ /dev/null @@ -1,19 +0,0 @@ -'use client'; - -type Props = { - onSignOut: () => Promise; -}; - -const SignOut = ({ onSignOut }: Props) => { - return ( -
{ - onSignOut(); - }} - > - Sign Out -
- ); -}; - -export default SignOut; \ No newline at end of file diff --git a/services/next/app/tags/[slug]/page.tsx b/services/next/app/tags/[slug]/page.tsx deleted file mode 100644 index b06e93c..0000000 --- a/services/next/app/tags/[slug]/page.tsx +++ /dev/null @@ -1,35 +0,0 @@ -import { getVodsForTag, IVod } from '@/app/lib/vods' -import VodCard from '@/app/components/vod-card' -import Link from 'next/link' -import { getVodTitle } from '@/app/components/vod-page' -import { notFound } from 'next/navigation' - -export default async function Page({ params }: { params: { slug: string }}) { - const vods = await getVodsForTag(params.slug) - if (!vods) return notFound() - return ( -
-
-

Tagged “{params.slug}”

-
- -
- {vods.map((vod: IVod) => ( - - ))} -
- -
-

See all tags.

-
-
- ) -} \ No newline at end of file diff --git a/services/next/app/tags/page.tsx b/services/next/app/tags/page.tsx deleted file mode 100644 index e025a49..0000000 --- a/services/next/app/tags/page.tsx +++ /dev/null @@ -1,15 +0,0 @@ -import { getTags } from '../lib/tags' -import SortableTags from '../components/sortable-tags' - -export default async function Page() { - const tags = await getTags(); - - return ( -
-
-

Tags

- -
-
- ) -} \ No newline at end of file diff --git a/services/next/app/upload/page.tsx b/services/next/app/upload/page.tsx deleted file mode 100644 index 8f633c9..0000000 --- a/services/next/app/upload/page.tsx +++ /dev/null @@ -1,30 +0,0 @@ - -import { getAllVtubers } from '@/app/lib/vtubers'; - - -import UploadForm from '@/app/components/upload-form'; -import { Suspense } from 'react'; - -import '@uppy/core/dist/style.min.css'; -import '@uppy/dashboard/dist/style.min.css'; -import { getStreamByUUID } from '@/app/lib/streams'; - - -export default async function Page() { - const vtubers = await getAllVtubers(); - - if (!vtubers) return ( - - ) - - return ( - <> - - - - - - - - ) -} \ No newline at end of file diff --git a/services/next/app/uppy.tsx b/services/next/app/uppy.tsx deleted file mode 100644 index 1fdda93..0000000 --- a/services/next/app/uppy.tsx +++ /dev/null @@ -1,57 +0,0 @@ -'use client'; - -import React, { useState, createContext, useContext, useEffect } from 'react'; -import Uppy from '@uppy/core'; -import AwsS3 from '@uppy/aws-s3'; -import RemoteSources from '@uppy/remote-sources'; -// import { useAuth } from './components/auth'; -import { companionUrl } from '@/app/lib/constants'; - - -// Uppy is a challenging react integration. Following are some references -// @see https://github.com/transloadit/uppy/issues/4727#issuecomment-1761118428 - - -export const UppyContext = createContext(new Uppy()); - -// export default function UppyProvider({ -// children -// }: { -// children: React.ReactNode -// }) { -// // const { authData } = useAuth(); -// const uppy = new Uppy( -// // const [uppy] = useState(() => new Uppy( -// { -// autoProceed: true, -// debug: true, -// logger: { -// debug: console.info, -// warn: console.log, -// error: console.error -// }, - -// } -// ) -// .use(RemoteSources, { -// companionUrl, -// title: 'testing 123', -// }) -// .use(AwsS3, { -// companionUrl, -// shouldUseMultipart: true, -// abortMultipartUpload: () => {}, // @see https://github.com/transloadit/uppy/issues/1197#issuecomment-491756118 -// companionHeaders: { -// 'authorization': `Bearer ${authData?.accessToken}` -// } -// }) -// // ); - - - -// return ( -// -// {children} -// -// ) -// } diff --git a/services/next/app/vods/[safeDateOrUUID]/page.tsx b/services/next/app/vods/[safeDateOrUUID]/page.tsx deleted file mode 100644 index b7914b3..0000000 --- a/services/next/app/vods/[safeDateOrUUID]/page.tsx +++ /dev/null @@ -1,16 +0,0 @@ - -import VodPage from '@/app/components/vod-page'; -import { IVodPageProps, getVodFromSafeDateOrUUID } from '@/app/lib/vods'; -import { notFound } from 'next/navigation'; - - -/** - * This route exists as backwards compatibility for Futureporn 0.0.0 which was on neocities - * @see https://www.w3.org/Provider/Style/URI - */ -export default async function Page({ params: { safeDateOrUUID, slug } }: IVodPageProps) { - const vod = await getVodFromSafeDateOrUUID(safeDateOrUUID); - if (!vod) notFound(); - return -} - diff --git a/services/next/app/vods/page.tsx b/services/next/app/vods/page.tsx deleted file mode 100644 index 175ff08..0000000 --- a/services/next/app/vods/page.tsx +++ /dev/null @@ -1,6 +0,0 @@ - -import { redirect } from 'next/navigation'; - -export default async function Page() { - redirect('/latest-vods/1') -} \ No newline at end of file diff --git a/services/next/app/vt/[slug]/not-found.tsx b/services/next/app/vt/[slug]/not-found.tsx deleted file mode 100644 index 87c153f..0000000 --- a/services/next/app/vt/[slug]/not-found.tsx +++ /dev/null @@ -1,11 +0,0 @@ -import Link from 'next/link' - -export default function NotFound() { - return ( -
-

404 Not Found

- - Return to vtuber list -
- ) -} \ No newline at end of file diff --git a/services/next/app/vt/[slug]/page.tsx b/services/next/app/vt/[slug]/page.tsx deleted file mode 100644 index fd9299b..0000000 --- a/services/next/app/vt/[slug]/page.tsx +++ /dev/null @@ -1,221 +0,0 @@ -import VodsList from '@/app/components/vods-list'; -import Link from 'next/link'; -import { getVtuberBySlug } from '@/app/lib/vtubers' -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { faExternalLinkAlt, faBagShopping } from "@fortawesome/free-solid-svg-icons"; -import { faFacebook, faInstagram, faPatreon, faYoutube, faTwitch, faTiktok, faXTwitter, faReddit, faDiscord } from "@fortawesome/free-brands-svg-icons"; -import Image from "next/legacy/image"; -import OnlyfansIcon from "@/app/components/icons/onlyfans"; -import PornhubIcon from '@/app/components/icons/pornhub'; -import ThroneIcon from '@/app/components/icons/throne'; -import LinktreeIcon from '@/app/components/icons/linktree'; -import FanslyIcon from '@/app/components/icons/fansly'; -import ChaturbateIcon from '@/app/components/icons/chaturbate'; -import CarrdIcon from '@/app/components/icons/carrd'; -import styles from '@/assets/styles/icon.module.css'; - -import { getVodsForVtuber } from '@/app/lib/vods'; -import { notFound } from 'next/navigation'; -import ArchiveProgress from '@/app/components/archive-progress'; -import { getAllStreamsForVtuber, getStreamsForVtuber } from '@/app/lib/streams'; -import LinkableHeading from '@/app/components/linkable-heading'; - - - -export default async function Page({ params }: { params: { slug: string } }) { - const vtuber = await getVtuberBySlug(params.slug); - if (!vtuber) notFound(); - - const pageSize = 9 - const page = 1 - const { vods, count } = await getVodsForVtuber(vtuber.id, page, pageSize); - if (!vods) notFound(); - - - return ( - <> - {vtuber && ( - <> -
- -
-
-

{vtuber.display_name}

-
-
-
- {vtuber.display_name} -
-
-
-

{vtuber.description_1}

-

{vtuber.description_2}

-
-
- -

- Socials -

- -
-
- {vtuber.patreon && ( -
- - Patreon - -
- )} - {vtuber.twitter && ( -
- - Twitter - -
- )} - {vtuber.youtube && ( -
- - YouTube - -
- )} - {vtuber.twitch && ( -
- - Twitch - -
- )} - {vtuber.tiktok && ( -
- - TikTok - -
- )} - {vtuber.fansly && ( -
- - Fansly - -
- )} - {vtuber.onlyfans && ( -
- - - - OnlyFans - -
- )} - {vtuber.pornhub && ( -
- - Pornhub - -
- )} - {vtuber.reddit && ( -
- - Reddit - -
- )} - {vtuber.discord && ( -
- - Discord - -
- )} - {vtuber.instagram && ( -
- - Instagram - -
- )} - {vtuber.facebook && ( -
- - Facebook - -
- )} - {vtuber.merch && ( -
- - Merch - -
- )} - {vtuber.chaturbate && ( -
- - Chaturbate - -
- )} - {vtuber.throne && ( -
- - Throne - -
- )} - {vtuber.linktree && ( -
- - Linktree - -
- )} - {vtuber.carrd && ( -
- - Carrd - -
- )} -
-
- - - {/*

- Toys -

- - <> - - {(toys.pagination.total > toySampleCount) && See all of {vtuber.display_name}'s toys} - */} - -

- Vods -

- - - { - (vods) ? ( - See all {vtuber.display_name} vods - ) : (

No VODs have been added, yet.

) - } - - -
- - )} - - ); -} diff --git a/services/next/app/vt/[slug]/stream/[safeDate]/page.tsx b/services/next/app/vt/[slug]/stream/[safeDate]/page.tsx deleted file mode 100644 index f2af6c1..0000000 --- a/services/next/app/vt/[slug]/stream/[safeDate]/page.tsx +++ /dev/null @@ -1,31 +0,0 @@ - -import { Stream } from '@/app/components/stream'; -import { getStreamForVtuber } from '@/app/lib/streams'; -import { getVtuberBySlug } from '@/app/lib/vtubers'; -import NotFound from '../../not-found'; - -interface IPageProps { - params: { - safeDate: string; - slug: string; - }; -} - -export default async function Page({ params: { safeDate, slug } }: IPageProps) { - const vtuber = await getVtuberBySlug(slug); - if (!vtuber) return - const stream = await getStreamForVtuber(vtuber.id, safeDate); - if (!stream) return - - return ( -
-
-

Stream Page!

-

slug={slug} safeDate={safeDate}

- - -
-
- ) -} - diff --git a/services/next/app/vt/[slug]/streams/page.tsx b/services/next/app/vt/[slug]/streams/page.tsx deleted file mode 100644 index 065836b..0000000 --- a/services/next/app/vt/[slug]/streams/page.tsx +++ /dev/null @@ -1,23 +0,0 @@ - -import { getVtuberBySlug } from '@/app/lib/vtubers'; -import { getStreamsForVtuber } from '@/app/lib/streams'; -import Pager from '@/app/components/pager'; -import { notFound } from 'next/navigation'; - -interface IPageParams { - params: { - slug: string; - } -} - -export default async function Page({ params }: IPageParams) { - const vtuber = await getVtuberBySlug(params.slug); - if (!vtuber) return

vtuber {params.slug} not found

- const streams = await getStreamsForVtuber(vtuber.id, 1, 24); - if (!streams) return

streams not found

; - return ( - <> - - - ) -} \ No newline at end of file diff --git a/services/next/app/vt/[slug]/toys/[page]/page.tsx b/services/next/app/vt/[slug]/toys/[page]/page.tsx deleted file mode 100644 index fb01d2a..0000000 --- a/services/next/app/vt/[slug]/toys/[page]/page.tsx +++ /dev/null @@ -1,33 +0,0 @@ - -// import VodsList, { VodsListHeading } from '@/app/components/vods-list' -// import { getVtuberBySlug } from '@/app/lib/vtubers' -// // import { IToys, getToysForVtuber } from '@/app/lib/toys' -// import { ToysList, ToysListHeading } from '@/app/components/toys' -// import Pager from '@/app/components/pager' - -// interface IPageParams { -// params: { -// name: string; -// page: number; -// } -// } - -export default async function Page() { - // const vtuber = await getVtuberBySlug(params.slug) - return

Toys pages coming soon

- // const toys: IToys = await getToysForVtuber(vtuber.id, params.page, 24) - // return ( - //
- //
- // - // - // - //
- //
- // ) -} \ No newline at end of file diff --git a/services/next/app/vt/[slug]/toys/page.tsx b/services/next/app/vt/[slug]/toys/page.tsx deleted file mode 100644 index 0062aae..0000000 --- a/services/next/app/vt/[slug]/toys/page.tsx +++ /dev/null @@ -1,33 +0,0 @@ - -// import VodsList, { VodsListHeading } from '@/app/components/vods-list' -// import { getVtuberBySlug } from '@/app/lib/vtubers' -// // import { IToys, getToysForVtuber } from '@/app/lib/toys' -// import { ToysList } from '@/app/components/toys' -// import Pager from '@/app/components/pager' - -interface IPageParams { - params: { - name: string; - } -} - -export default async function Page({ params }: IPageParams) { - // const vtuber = await getVtuberBySlug(params.slug) - return

toys pages coming soon

- // const toys: IToys = await getToysForVtuber(vtuber.id, 1, 24) - // return ( - //
- //
- // {/* */} - // {/* */} - // - // - //
- //
- // ) -} \ No newline at end of file diff --git a/services/next/app/vt/[slug]/vod/[safeDateOrUUID]/page.tsx b/services/next/app/vt/[slug]/vod/[safeDateOrUUID]/page.tsx deleted file mode 100644 index 49184b6..0000000 --- a/services/next/app/vt/[slug]/vod/[safeDateOrUUID]/page.tsx +++ /dev/null @@ -1,19 +0,0 @@ - -import VodPage2 from '@/app/components/vod-page-2' -import { IVodPageProps, getVodFromSafeDateOrUUID } from '@/app/lib/vods' -import { notFound } from 'next/navigation'; - - -export default async function Page({ params: { safeDateOrUUID } }: IVodPageProps) { - const vod = await getVodFromSafeDateOrUUID(safeDateOrUUID); - if (!vod) return notFound(); - // return ( - //
-    //         
-    //             {JSON.stringify(vod, null, 2)}
-    //         
-    //     
- // ) - return -} - diff --git a/services/next/app/vt/[slug]/vod/page.tsx b/services/next/app/vt/[slug]/vod/page.tsx deleted file mode 100644 index b62229f..0000000 --- a/services/next/app/vt/[slug]/vod/page.tsx +++ /dev/null @@ -1,15 +0,0 @@ -import React from 'react'; -import Link from 'next/link'; -import { redirect } from 'next/navigation'; - -interface IPageParams { - params: { - slug: string; - } -} - -export default function Page({ params: { slug } }: IPageParams) { - redirect(`/vt/${slug}/vods`) - return See {`/vt/${slug}/vods`} -} - diff --git a/services/next/app/vt/[slug]/vods/[page]/page.tsx b/services/next/app/vt/[slug]/vods/[page]/page.tsx deleted file mode 100644 index db67580..0000000 --- a/services/next/app/vt/[slug]/vods/[page]/page.tsx +++ /dev/null @@ -1,46 +0,0 @@ -import VodsList, { VodsListHeading } from '@/app/components/vods-list'; -import { getVtuberBySlug, getUrl } from '@/app/lib/vtubers'; -import { IVodsResponse, getVodsForVtuber } from '@/app/lib/vods'; -import Pager from '@/app/components/pager'; -import { notFound } from 'next/navigation'; - - -interface IPageParams { - params: { - slug: string; - page: string; - }; -} - -export default async function Page({ params }: IPageParams) { - const pageNumber = parseInt(params.page); - const pageSize = 24 - - console.log(`lets get us some vtuber and vods`); - const vtuber = await getVtuberBySlug(params.slug); - if (!vtuber) { - console.error('getVtuberBySlug failed to get us a vtuber object.') - return

failed to get vtuber

- // return notFound(); - } - - - const {vods, count} = await getVodsForVtuber(vtuber.id, pageNumber, 24, true); - if (!vods) { - return

failed to get vods

- console.error('getVodsForVtuber failed to get us a vods object.') - // return notFound(); - } - - return ( - <> - - - - - ); -} diff --git a/services/next/app/vt/[slug]/vods/page.tsx b/services/next/app/vt/[slug]/vods/page.tsx deleted file mode 100644 index 1849ddb..0000000 --- a/services/next/app/vt/[slug]/vods/page.tsx +++ /dev/null @@ -1,32 +0,0 @@ - -import VodsList, { VodsListHeading } from '@/app/components/vods-list' -import { getVtuberBySlug, getUrl } from '@/app/lib/vtubers' -import { IVodsResponse, getVodsForVtuber, getPaginatedUrl } from '@/app/lib/vods' -import Pager from '@/app/components/pager' -import { notFound } from 'next/navigation' - -interface IPageParams { - params: { - slug: string; - } -} - -export default async function Page({ params }: IPageParams) { - const vtuber = await getVtuberBySlug(params.slug) - const pageCount = 24 - if (!vtuber) notFound(); - const vods = await getVodsForVtuber(vtuber.id, 1, pageCount) - if (!vods) notFound(); - return ( - <> -
-                
-                    {JSON.stringify(vods, null, 2)}
-                
-            
- {/* - - */} - - ) -} \ No newline at end of file diff --git a/services/next/app/vt/page.tsx b/services/next/app/vt/page.tsx deleted file mode 100644 index 93a0cd9..0000000 --- a/services/next/app/vt/page.tsx +++ /dev/null @@ -1,36 +0,0 @@ -import { notFound } from 'next/navigation' -import VTuberCard from '../components/vtuber-card' -import { getVtubers } from '../lib/vtubers' -import { IVtuber } from '@futureporn/types' - - -export default async function Page() { - const vtubers = await getVtubers() - if (!vtubers) notFound() - // return ( - //
-    //         
-    //             {JSON.stringify(vtubers, null, 2)}
-    //         
-    //     
- // ) - return ( - <> - {/*
-                
-                    {JSON.stringify(vtubers, null, 2)}
-                
-            
*/} -
-
-

VTubers

- -
-
- - ) -} \ No newline at end of file diff --git a/services/next/assets/styles/cid.module.css b/services/next/assets/styles/cid.module.css deleted file mode 100644 index e195bb4..0000000 --- a/services/next/assets/styles/cid.module.css +++ /dev/null @@ -1,19 +0,0 @@ -.container { - display: flex; - align-items: center; -} - -.cid { - overflow: hidden; - text-overflow: ellipsis; - text-align: center; - flex: 1; -} - -.label { - width: 6em; -} - -.green { - color: rgb(52, 168, 115); -} \ No newline at end of file diff --git a/services/next/assets/styles/fp.module.css b/services/next/assets/styles/fp.module.css deleted file mode 100644 index 106e6b5..0000000 --- a/services/next/assets/styles/fp.module.css +++ /dev/null @@ -1,28 +0,0 @@ - -.noselect { - user-select: none; -} - -.tagButton { - height: 2em; - margin-bottom: 0.5rem; - border-radius: 4px; -} - -.isTiny { - height: 1.5em; -} - -.grade { - font-family: Arial, Helvetica, sans-serif; - font-size: 8rem; - font-weight: bolder; -} - -.zoomin { - cursor: zoom-in; -} - -.zoomout { - cursor: zoom-out; -} \ No newline at end of file diff --git a/services/next/assets/styles/global.css b/services/next/assets/styles/global.css deleted file mode 100644 index 4c04dc6..0000000 --- a/services/next/assets/styles/global.css +++ /dev/null @@ -1,58 +0,0 @@ -/* nunito-regular - latin */ -@font-face { - font-display: swap; /* Check https://developer.mozilla.org/en-US/docs/Web/CSS/@font-face/font-display for other options. */ - font-family: 'Nunito'; - font-style: normal; - font-weight: 400; - src: url('/fonts/nunito-v26-latin-regular.woff2') format('woff2'); /* Chrome 36+, Opera 23+, Firefox 39+, Safari 12+, iOS 10+ */ -} - -/* - palleton color scheme - https://paletton.com/#uid=54W0u0kv5npmjustvtxAGj4KwdK -*/ - -:root { - - /* mux player styling */ - --media-accent-color: #fb00ff; - - /* this brightens message headers (funding goal) which by default were too close to the bg color */ - --bulma-dark-l: 22%; - - /* Colors and Lightness values */ - --bulma-scheme-h: 221; - --bulma-scheme-s: 14%; - --bulma-light-l: 90%; - --bulma-light-invert-l: 20%; - --bulma-dark-invert-l: 90%; - --bulma-soft-l: 90%; - --bulma-bold-l: 20%; - --bulma-soft-invert-l: 20%; - --bulma-bold-invert-l: 90%; - /* etc. */ - - /* Color Palettes */ - --bulma-primary: hsla(var(--bulma-primary-h), var(--bulma-primary-s), var(--bulma-primary-l), 1); - --bulma-primary-base: hsla(var(--bulma-primary-h), var(--bulma-primary-s), var(--bulma-primary-l), 1); - --bulma-primary-rgb: 0, 209, 178; - --bulma-primary-h: 299deg; - --bulma-primary-s: 50%; - --bulma-primary-l: 50%; - --bulma-primary-00-l: 1%; - --bulma-primary-05-l: 6%; - --bulma-primary-10-l: 11%; - --bulma-primary-15-l: 16%; - --bulma-primary-20-l: 21%; - --bulma-info-l: 50%; - - /* Typography */ - --bulma-family-primary: Nunito, Inter, SF Pro, Segoe UI, Roboto, Oxygen, Ubuntu, Helvetica Neue, Helvetica, Arial, sans-serif; - --bulma-family-secondary: Inter, SF Pro, Segoe UI, Roboto, Oxygen, Ubuntu, Helvetica Neue, Helvetica, Arial, sans-serif; - --bulma-family-code: Inconsolata, Hack, SF Mono, Roboto Mono, Source Code Pro, Ubuntu Mono, monospace; - --bulma-size-small: 0.75rem; - --bulma-size-normal: 1rem; - --bulma-size-medium: 1.25rem; - --bulma-size-large: 1.5rem; - /* etc. */ -} \ No newline at end of file diff --git a/services/next/assets/styles/icon.module.css b/services/next/assets/styles/icon.module.css deleted file mode 100644 index 5d3efba..0000000 --- a/services/next/assets/styles/icon.module.css +++ /dev/null @@ -1,20 +0,0 @@ - -svg.icon { - width: 1em; - height: 1em; - vertical-align: -0.125em; - fill: rgb(208, 209, 205); -} - -svg.icon path { - fill: rgb(208, 209, 205); -} - -svg.icon g path { - fill: rgb(208, 209, 205) !important; -} - -svg.bigIcon { - width: 10em; - height: 10em; -} \ No newline at end of file diff --git a/services/next/assets/styles/player.module.css b/services/next/assets/styles/player.module.css deleted file mode 100644 index b3f531b..0000000 --- a/services/next/assets/styles/player.module.css +++ /dev/null @@ -1,4 +0,0 @@ - -.fpMediaPlayer { - --media-aspect-ratio: 1.7778; -} \ No newline at end of file diff --git a/services/next/assets/svg/README.md b/services/next/assets/svg/README.md deleted file mode 100644 index bbf6f41..0000000 --- a/services/next/assets/svg/README.md +++ /dev/null @@ -1,5 +0,0 @@ -# SVG in next/react - -see https://blog.logrocket.com/import-svgs-next-js-apps/ - -TL;DR: use https://react-svgr.com/playground/ to convert svg to jsx \ No newline at end of file diff --git a/services/next/assets/svg/carrd.svg b/services/next/assets/svg/carrd.svg deleted file mode 100644 index 9b1ead5..0000000 --- a/services/next/assets/svg/carrd.svg +++ /dev/null @@ -1 +0,0 @@ -Carrd diff --git a/services/next/assets/svg/chaturbate.svg b/services/next/assets/svg/chaturbate.svg deleted file mode 100644 index 0ef00ed..0000000 --- a/services/next/assets/svg/chaturbate.svg +++ /dev/null @@ -1,15 +0,0 @@ -import * as React from "react" -const SvgComponent = (props) => ( - - - -) -export default SvgComponent diff --git a/services/next/assets/svg/checkmark.svg b/services/next/assets/svg/checkmark.svg deleted file mode 100644 index 23cce6f..0000000 --- a/services/next/assets/svg/checkmark.svg +++ /dev/null @@ -1,5 +0,0 @@ - - - - - \ No newline at end of file diff --git a/services/next/assets/svg/fansly.tsx b/services/next/assets/svg/fansly.tsx deleted file mode 100644 index 03a78dc..0000000 --- a/services/next/assets/svg/fansly.tsx +++ /dev/null @@ -1,20 +0,0 @@ -import * as React from "react" -const SvgComponent = (props: any) => ( - - - - -) -export default SvgComponent diff --git a/services/next/assets/svg/ipfs.svg b/services/next/assets/svg/ipfs.svg deleted file mode 100644 index ea32d6e..0000000 --- a/services/next/assets/svg/ipfs.svg +++ /dev/null @@ -1 +0,0 @@ -IPFS \ No newline at end of file diff --git a/services/next/assets/svg/linktree.svg b/services/next/assets/svg/linktree.svg deleted file mode 100644 index 0f8d400..0000000 --- a/services/next/assets/svg/linktree.svg +++ /dev/null @@ -1,64 +0,0 @@ - - - - - - - - - - - - - - diff --git a/services/next/assets/svg/noun-adult-content-1731184.svg b/services/next/assets/svg/noun-adult-content-1731184.svg deleted file mode 100644 index 8509fd0..0000000 --- a/services/next/assets/svg/noun-adult-content-1731184.svg +++ /dev/null @@ -1 +0,0 @@ -Created by Anatolii Babiifrom the Noun Project \ No newline at end of file diff --git a/services/next/assets/svg/noun-anime-3890912.svg b/services/next/assets/svg/noun-anime-3890912.svg deleted file mode 100644 index c5c376f..0000000 --- a/services/next/assets/svg/noun-anime-3890912.svg +++ /dev/null @@ -1 +0,0 @@ -Created by Kevinfrom the Noun Project \ No newline at end of file diff --git a/services/next/assets/svg/noun-avatar-3546974.svg b/services/next/assets/svg/noun-avatar-3546974.svg deleted file mode 100644 index ec43ede..0000000 --- a/services/next/assets/svg/noun-avatar-3546974.svg +++ /dev/null @@ -1 +0,0 @@ -love charger copy 2Created by KEN111from the Noun Project \ No newline at end of file diff --git a/services/next/assets/svg/noun-girl-842331.svg b/services/next/assets/svg/noun-girl-842331.svg deleted file mode 100644 index f609fb4..0000000 --- a/services/next/assets/svg/noun-girl-842331.svg +++ /dev/null @@ -1,4 +0,0 @@ -Created by Zackary Cloefrom the Noun Project \ No newline at end of file diff --git a/services/next/assets/svg/noun-network-1603820.svg b/services/next/assets/svg/noun-network-1603820.svg deleted file mode 100644 index 8a3d5c0..0000000 --- a/services/next/assets/svg/noun-network-1603820.svg +++ /dev/null @@ -1 +0,0 @@ -Created by Three Six Fivefrom the Noun Project \ No newline at end of file diff --git a/services/next/assets/svg/onlyfans.svg b/services/next/assets/svg/onlyfans.svg deleted file mode 100644 index bba261f..0000000 --- a/services/next/assets/svg/onlyfans.svg +++ /dev/null @@ -1,9 +0,0 @@ - - - - - - - - - \ No newline at end of file diff --git a/services/next/assets/svg/pornhub.svg b/services/next/assets/svg/pornhub.svg deleted file mode 100644 index 4b1b6bd..0000000 --- a/services/next/assets/svg/pornhub.svg +++ /dev/null @@ -1,5 +0,0 @@ - - - - - diff --git a/services/next/assets/svg/throne.svg b/services/next/assets/svg/throne.svg deleted file mode 100644 index a396573..0000000 --- a/services/next/assets/svg/throne.svg +++ /dev/null @@ -1,12 +0,0 @@ - - - - - - - - - - - - diff --git a/services/next/next.config.js b/services/next/next.config.js deleted file mode 100644 index ccc33e5..0000000 --- a/services/next/next.config.js +++ /dev/null @@ -1,28 +0,0 @@ -/** @type {import('next').NextConfig} */ -const path = require("path"); -const nextConfig = { - output: 'standalone', - reactStrictMode: false, - sassOptions: { - includePaths: [path.join(__dirname, "assets", "styles")], - }, - images: { - remotePatterns: [ - { - protocol: 'https', - hostname: 'futureporn-b2.b-cdn.net', - port: '', - pathname: '/**', - }, - { - protocol: 'https', - hostname: 'fp-dev.b-cdn.net', - port: '', - pathname: '/**', - }, - ], - } -}; - - -module.exports = nextConfig; diff --git a/services/next/package.json b/services/next/package.json deleted file mode 100644 index d0d8130..0000000 --- a/services/next/package.json +++ /dev/null @@ -1,86 +0,0 @@ -{ - "name": "@futureporn/next", - "version": "2.0.0", - "private": true, - "scripts": { - "dev": "NODE_EXTRA_CA_CERTS=/app/certs/letsencrypt-stg-root-x1.pem next dev", - "build": "next build", - "start": "next start", - "lint": "next lint", - "preinstall": "npx only-allow pnpm", - "clean": "rm -rf dist", - "superclean": "rm -rf node_modules && rm -rf pnpm-lock.yaml && rm -rf dist" - }, - "dependencies": { - "@dicebear/collection": "^9.2.2", - "@dicebear/core": "^9.2.2", - "@fortawesome/fontawesome-free": "^6.6.0", - "@fortawesome/fontawesome-svg-core": "^6.6.0", - "@fortawesome/free-brands-svg-icons": "^6.6.0", - "@fortawesome/free-solid-svg-icons": "^6.6.0", - "@fortawesome/react-fontawesome": "^0.2.2", - "@futureporn/types": "workspace:*", - "@futureporn/utils": "workspace:^", - "@hookform/error-message": "^2.0.1", - "@hookform/resolvers": "^3.9.0", - "@logto/next": "^3.7.1", - "@mux/blurhash": "^0.1.2", - "@mux/mux-player": "^2.9.1", - "@mux/mux-player-react": "^2.9.1", - "@paralleldrive/cuid2": "^2.2.2", - "@react-hookz/web": "^24.0.4", - "@tanstack/react-query": "^5.53.3", - "@tanstack/react-table": "^8.20.5", - "@types/lodash": "^4.17.7", - "@types/qs": "^6.9.15", - "@types/react": "^18.3.5", - "@types/react-dom": "^18.3.0", - "@uppy/aws-s3": "^4.1.0", - "@uppy/aws-s3-multipart": "^4.0.0", - "@uppy/core": "^4.2.0", - "@uppy/dashboard": "^4.1.0", - "@uppy/drag-drop": "^4.0.2", - "@uppy/file-input": "^4.0.1", - "@uppy/progress-bar": "^4.0.0", - "@uppy/react": "^4.0.2", - "@uppy/remote-sources": "^2.2.0", - "bulma": "^1.0.2", - "date-fns": "^3.6.0", - "date-fns-tz": "^3.1.3", - "dayjs": "^1.11.13", - "dotenv": "^16.4.5", - "feed": "^4.2.2", - "gray-matter": "^4.0.3", - "hls.js": "^1.5.15", - "jose": "^5.9.6", - "jwt-decode": "^4.0.0", - "lodash": "^4.17.21", - "lunarphase-js": "^2.0.3", - "multiformats": "^13.2.2", - "next": "14.2.7", - "next-auth": "4.24.10", - "next-goatcounter": "^1.0.5", - "nextjs-cors": "^2.2.0", - "nextjs-toploader": "^3.6.15", - "plyr": "^3.7.8", - "qs": "^6.13.0", - "react": "^18.3.1", - "react-data-table-component": "^7.6.2", - "react-dom": "^18.3.1", - "react-hook-form": "^7.53.0", - "react-loading-skeleton": "^3.4.0", - "react-toastify": "^10.0.5", - "sass": "^1.77.8", - "sharp": "^0.33.5", - "slugify": "^1.6.6", - "styled-components": "6.1.13", - "yup": "^1.4.0" - }, - "devDependencies": { - "@types/node": "^22.5.2", - "eslint": "^9.9.1", - "eslint-config-next": "14.2.7", - "typescript": "5.5.4" - }, - "packageManager": "pnpm@9.14.4" -} diff --git a/services/next/pnpm-lock.yaml b/services/next/pnpm-lock.yaml deleted file mode 100644 index a7b4b90..0000000 --- a/services/next/pnpm-lock.yaml +++ /dev/null @@ -1,5293 +0,0 @@ -lockfileVersion: '9.0' - -settings: - autoInstallPeers: true - excludeLinksFromLockfile: false - -importers: - - .: - dependencies: - '@dicebear/collection': - specifier: ^9.2.2 - version: 9.2.2(@dicebear/core@9.2.2) - '@dicebear/core': - specifier: ^9.2.2 - version: 9.2.2 - '@fortawesome/fontawesome-free': - specifier: ^6.6.0 - version: 6.6.0 - '@fortawesome/fontawesome-svg-core': - specifier: ^6.6.0 - version: 6.6.0 - '@fortawesome/free-brands-svg-icons': - specifier: ^6.6.0 - version: 6.6.0 - '@fortawesome/free-solid-svg-icons': - specifier: ^6.6.0 - version: 6.6.0 - '@fortawesome/react-fontawesome': - specifier: ^0.2.2 - version: 0.2.2(@fortawesome/fontawesome-svg-core@6.6.0)(react@18.3.1) - '@futureporn/types': - specifier: workspace:* - version: link:../../packages/types - '@futureporn/utils': - specifier: workspace:^ - version: link:../../packages/utils - '@hookform/error-message': - specifier: ^2.0.1 - version: 2.0.1(react-dom@18.3.1(react@18.3.1))(react-hook-form@7.53.0(react@18.3.1))(react@18.3.1) - '@hookform/resolvers': - specifier: ^3.9.0 - version: 3.9.0(react-hook-form@7.53.0(react@18.3.1)) - '@logto/next': - specifier: ^3.7.1 - version: 3.7.1(next@14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8)) - '@mux/blurhash': - specifier: ^0.1.2 - version: 0.1.2 - '@mux/mux-player': - specifier: ^2.9.1 - version: 2.9.1 - '@mux/mux-player-react': - specifier: ^2.9.1 - version: 2.9.1(@types/react-dom@18.3.0)(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@paralleldrive/cuid2': - specifier: ^2.2.2 - version: 2.2.2 - '@react-hookz/web': - specifier: ^24.0.4 - version: 24.0.4(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@tanstack/react-query': - specifier: ^5.53.3 - version: 5.53.3(react@18.3.1) - '@tanstack/react-table': - specifier: ^8.20.5 - version: 8.20.5(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@types/lodash': - specifier: ^4.17.7 - version: 4.17.7 - '@types/qs': - specifier: ^6.9.15 - version: 6.9.15 - '@types/react': - specifier: ^18.3.5 - version: 18.3.5 - '@types/react-dom': - specifier: ^18.3.0 - version: 18.3.0 - '@uppy/aws-s3': - specifier: ^4.1.0 - version: 4.1.0(@uppy/core@4.2.0) - '@uppy/aws-s3-multipart': - specifier: ^4.0.0 - version: 4.0.0(@uppy/core@4.2.0) - '@uppy/core': - specifier: ^4.2.0 - version: 4.2.0 - '@uppy/dashboard': - specifier: ^4.1.0 - version: 4.1.0(@uppy/core@4.2.0) - '@uppy/drag-drop': - specifier: ^4.0.2 - version: 4.0.2(@uppy/core@4.2.0) - '@uppy/file-input': - specifier: ^4.0.1 - version: 4.0.1(@uppy/core@4.2.0) - '@uppy/progress-bar': - specifier: ^4.0.0 - version: 4.0.0(@uppy/core@4.2.0) - '@uppy/react': - specifier: ^4.0.2 - version: 4.0.2(@uppy/core@4.2.0)(@uppy/dashboard@4.1.0(@uppy/core@4.2.0))(@uppy/drag-drop@4.0.2(@uppy/core@4.2.0))(@uppy/file-input@4.0.1(@uppy/core@4.2.0))(@uppy/progress-bar@4.0.0(@uppy/core@4.2.0))(@uppy/status-bar@4.0.3(@uppy/core@4.2.0))(react@18.3.1) - '@uppy/remote-sources': - specifier: ^2.2.0 - version: 2.2.0(@uppy/core@4.2.0) - bulma: - specifier: ^1.0.2 - version: 1.0.2 - date-fns: - specifier: ^3.6.0 - version: 3.6.0 - date-fns-tz: - specifier: ^3.1.3 - version: 3.1.3(date-fns@3.6.0) - dayjs: - specifier: ^1.11.13 - version: 1.11.13 - dotenv: - specifier: ^16.4.5 - version: 16.4.5 - feed: - specifier: ^4.2.2 - version: 4.2.2 - gray-matter: - specifier: ^4.0.3 - version: 4.0.3 - hls.js: - specifier: ^1.5.15 - version: 1.5.15 - jose: - specifier: ^5.9.6 - version: 5.9.6 - jwt-decode: - specifier: ^4.0.0 - version: 4.0.0 - lodash: - specifier: ^4.17.21 - version: 4.17.21 - lunarphase-js: - specifier: ^2.0.3 - version: 2.0.3 - multiformats: - specifier: ^13.2.2 - version: 13.2.2 - next: - specifier: 14.2.7 - version: 14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8) - next-auth: - specifier: 4.24.10 - version: 4.24.10(next@14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8))(nodemailer@6.9.16)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - next-goatcounter: - specifier: ^1.0.5 - version: 1.0.5(next@14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8))(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - nextjs-cors: - specifier: ^2.2.0 - version: 2.2.0(next@14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8)) - nextjs-toploader: - specifier: ^3.6.15 - version: 3.6.15(next@14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8))(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - plyr: - specifier: ^3.7.8 - version: 3.7.8 - qs: - specifier: ^6.13.0 - version: 6.13.0 - react: - specifier: ^18.3.1 - version: 18.3.1 - react-data-table-component: - specifier: ^7.6.2 - version: 7.6.2(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)) - react-dom: - specifier: ^18.3.1 - version: 18.3.1(react@18.3.1) - react-hook-form: - specifier: ^7.53.0 - version: 7.53.0(react@18.3.1) - react-loading-skeleton: - specifier: ^3.4.0 - version: 3.4.0(react@18.3.1) - react-toastify: - specifier: ^10.0.5 - version: 10.0.5(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - sass: - specifier: ^1.77.8 - version: 1.77.8 - sharp: - specifier: ^0.33.5 - version: 0.33.5 - slugify: - specifier: ^1.6.6 - version: 1.6.6 - styled-components: - specifier: 6.1.13 - version: 6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - yup: - specifier: ^1.4.0 - version: 1.4.0 - devDependencies: - '@types/node': - specifier: ^22.5.2 - version: 22.5.2 - eslint: - specifier: ^9.9.1 - version: 9.9.1 - eslint-config-next: - specifier: 14.2.7 - version: 14.2.7(eslint@9.9.1)(typescript@5.5.4) - typescript: - specifier: 5.5.4 - version: 5.5.4 - -packages: - - '@babel/runtime@7.26.0': - resolution: {integrity: sha512-FDSOghenHTiToteC/QRlv2q3DhPZ/oOXTBoirfWNx1Cx3TMVcGWQtMMmQcSvb/JjpNeGzx8Pq/b4fKEJuWm1sw==} - engines: {node: '>=6.9.0'} - - '@dicebear/adventurer-neutral@9.2.2': - resolution: {integrity: sha512-XVAjhUWjav6luTZ7txz8zVJU/H0DiUy4uU1Z7IO5MDO6kWvum+If1+0OUgEWYZwM+RDI7rt2CgVP910DyZGd1w==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/adventurer@9.2.2': - resolution: {integrity: sha512-WjBXCP9EXbUul2zC3BS2/R3/4diw1uh/lU4jTEnujK1mhqwIwanFboIMzQsasNNL/xf+m3OHN7MUNJfHZ1fLZA==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/avataaars-neutral@9.2.2': - resolution: {integrity: sha512-pRj16P27dFDBI3LtdiHUDwIXIGndHAbZf5AxaMkn6/+0X93mVQ/btVJDXyW0G96WCsyC88wKAWr6/KJotPxU6Q==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/avataaars@9.2.2': - resolution: {integrity: sha512-WqJPQEt0OhBybTpI0TqU1uD1pSk9M2+VPIwvBye/dXo46b+0jHGpftmxjQwk6tX8z0+mRko8pwV5n+cWht1/+w==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/big-ears-neutral@9.2.2': - resolution: {integrity: sha512-IPHt8fi3dv9cyfBJBZ4s8T+PhFCrQvOCf91iRHBT3iOLNPdyZpI5GNLmGiV0XMAvIDP5NvA5+f6wdoBLhYhbDA==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/big-ears@9.2.2': - resolution: {integrity: sha512-hz4UXdPq4qqZpu0YVvlqM4RDFhk5i0WgPcuwj/MOLlgTjuj63uHUhCQSk6ZiW1DQOs12qpwUBMGWVHxBRBas9g==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/big-smile@9.2.2': - resolution: {integrity: sha512-D4td0GL8or1nTNnXvZqkEXlzyqzGPWs3znOnm1HIohtFTeIwXm72Ob2lNDsaQJSJvXmVlwaQQ0CCTvyCl8Stjw==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/bottts-neutral@9.2.2': - resolution: {integrity: sha512-lSgpqmSJtlnyxVuUgNdBwyzuA0O9xa5zRJtz7x2KyWbicXir5iYdX0MVMCkp1EDvlcxm9rGJsclktugOyakTlw==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/bottts@9.2.2': - resolution: {integrity: sha512-wugFkzw8JNWV1nftq/Wp/vmQsLAXDxrMtRK3AoMODuUpSVoP3EHRUfKS043xggOsQFvoj0HZ7kadmhn0AMLf5A==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/collection@9.2.2': - resolution: {integrity: sha512-vZAmXhPWCK3sf8Fj9/QflFC6XOLroJOT5K1HdnzHaPboEvffUQideGCrrEamnJtlH0iF0ZDXh8gqmwy2fu+yHA==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/core@9.2.2': - resolution: {integrity: sha512-ROhgHG249dPtcXgBHcqPEsDeAPRPRD/9d+tZCjLYyueO+cXDlIA8dUlxpwIVcOuZFvCyW6RJtqo8BhNAi16pIQ==} - engines: {node: '>=18.0.0'} - - '@dicebear/croodles-neutral@9.2.2': - resolution: {integrity: sha512-/4mNirxoQ+z1kHXnpDRbJ1JV1ZgXogeTeNp0MaFYxocCgHfJ7ckNM23EE1I7akoo9pqPxrKlaeNzGAjKHdS9vA==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/croodles@9.2.2': - resolution: {integrity: sha512-OzvAXQWsOgMwL3Sl+lBxCubqSOWoBJpC78c4TKnNTS21rR63TtXUyVdLLzgKVN4YHRnvMgtPf8F/W9YAgIDK4w==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/dylan@9.2.2': - resolution: {integrity: sha512-s7e3XliC1YXP+Wykj+j5kwdOWFRXFzYHYk/PB4oZ1F3sJandXiG0HS4chaNu4EoP0yZgKyFMUVTGZx+o6tMaYg==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/fun-emoji@9.2.2': - resolution: {integrity: sha512-M+rYTpB3lfwz18f+/i+ggNwNWUoEj58SJqXJ1wr7Jh/4E5uL+NmJg9JGwYNaVtGbCFrKAjSaILNUWGQSFgMfog==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/glass@9.2.2': - resolution: {integrity: sha512-imCMxcg+XScHYtQq2MUv1lCzhQSCUglMlPSezKEpXhTxgbgUpmGlSGVkOfmX5EEc7SQowKkF1W/1gNk6CXvBaQ==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/icons@9.2.2': - resolution: {integrity: sha512-Tqq2OVCdS7J02DNw58xwlgLGl40sWEckbqXT3qRvIF63FfVq+wQZBGuhuiyAURcSgvsc3h2oQeYFi9iXh7HTOA==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/identicon@9.2.2': - resolution: {integrity: sha512-POVKFulIrcuZf3rdAgxYaSm2XUg/TJg3tg9zq9150reEGPpzWR7ijyJ03dzAADPzS3DExfdYVT9+z3JKwwJnTQ==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/initials@9.2.2': - resolution: {integrity: sha512-/xNnsEmsstWjmF77htAOuwOMhFlP6eBVXgcgFlTl/CCH/Oc6H7t0vwX1he8KLQBBzjGpvJcvIAn4Wh9rE4D5/A==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/lorelei-neutral@9.2.2': - resolution: {integrity: sha512-Eys9Os6nt2Xll7Mvu66CfRR2YggTopWcmFcRZ9pPdohS96kT0MsLI2iTcfZXQ51K8hvT3IbwoGc86W8n0cDxAQ==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/lorelei@9.2.2': - resolution: {integrity: sha512-koXqVr/vcWUPo00VP5H6Czsit+uF1tmwd2NK7Q/e34/9Sd1f4QLLxHjjBNm/iNjCI1+UNTOvZ2Qqu0N5eo7Flw==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/micah@9.2.2': - resolution: {integrity: sha512-NCajcJV5yw8uMKiACp694w1T/UyYme2CUEzyTzWHgWnQ+drAuCcH8gpAoLWd67viNdQB/MTpNlaelUgTjmI4AQ==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/miniavs@9.2.2': - resolution: {integrity: sha512-vvkWXttdw+KHF3j+9qcUFzK+P0nbNnImGjvN48wwkPIh2h08WWFq0MnoOls4IHwUJC4GXBjWtiyVoCxz6hhtOA==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/notionists-neutral@9.2.2': - resolution: {integrity: sha512-AhOzk+lz6kB4uxGun8AJhV+W1nttnMlxmxd+5KbQ/txCIziYIaeD3il44wsAGegEpGFvAZyMYtR/jjfHcem3TA==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/notionists@9.2.2': - resolution: {integrity: sha512-Z9orRaHoj7Y9Ap4wEu8XOrFACsG1KbbBQUPV1R50uh6AHwsyNrm4cS84ICoGLvxgLNHHOae3YCjd8aMu2z19zg==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/open-peeps@9.2.2': - resolution: {integrity: sha512-6PeQDHYyjvKrGSl/gP+RE5dSYAQGKpcGnM65HorgyTIugZK7STo0W4hvEycedupZ3MCCEH8x/XyiChKM2sHXog==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/personas@9.2.2': - resolution: {integrity: sha512-705+ObNLC0w1fcgE/Utav+8bqO+Esu53TXegpX5j7trGEoIMf2bThqJGHuhknZ3+T2az3Wr89cGyOGlI0KLzLA==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/pixel-art-neutral@9.2.2': - resolution: {integrity: sha512-CdUY77H6Aj7dKLW3hdkv7tu0XQJArUjaWoXihQxlhl3oVYplWaoyu9omYy5pl8HTqs8YgVTGljjMXYoFuK0JUw==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/pixel-art@9.2.2': - resolution: {integrity: sha512-BvbFdrpzQl04+Y9UsWP63YGug+ENGC7GMG88qbEFWxb/IqRavGa4H3D0T4Zl2PSLiw7f2Ctv98bsCQZ1PtCznQ==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/rings@9.2.2': - resolution: {integrity: sha512-eD1J1k364Arny+UlvGrk12HP/XGG6WxPSm4BarFqdJGSV45XOZlwqoi7FlcMr9r9yvE/nGL8OizbwMYusEEdjw==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/shapes@9.2.2': - resolution: {integrity: sha512-e741NNWBa7fg0BjomxXa0fFPME2XCIR0FA+VHdq9AD2taTGHEPsg5x1QJhCRdK6ww85yeu3V3ucpZXdSrHVw5Q==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@dicebear/thumbs@9.2.2': - resolution: {integrity: sha512-FkPLDNu7n5kThLSk7lR/0cz/NkUqgGdZGfLZv6fLkGNGtv6W+e2vZaO7HCXVwIgJ+II+kImN41zVIZ6Jlll7pQ==} - engines: {node: '>=18.0.0'} - peerDependencies: - '@dicebear/core': ^9.0.0 - - '@edge-runtime/cookies@4.1.1': - resolution: {integrity: sha512-ATZLTOpnCUD9ZLNBIXhxOmP/UVx6BfhCjDy9P1YACpD8vrHb5Uw7YlG9RYUl1AMF7Y10TIIN3jhFbUSMiH2J7g==} - engines: {node: '>=16'} - - '@emnapi/runtime@1.2.0': - resolution: {integrity: sha512-bV21/9LQmcQeCPEg3BDFtvwL6cwiTMksYNWQQ4KOxCZikEGalWtenoZ0wCiukJINlGCIi2KXx01g4FoH/LxpzQ==} - - '@emotion/is-prop-valid@1.2.2': - resolution: {integrity: sha512-uNsoYd37AFmaCdXlg6EYD1KaPOaRWRByMCYzbKUX4+hhMfrxdVSelShywL4JVaAeM/eHUOSprYBQls+/neX3pw==} - - '@emotion/memoize@0.8.1': - resolution: {integrity: sha512-W2P2c/VRW1/1tLox0mVUalvnWXxavmv/Oum2aPsRcoDJuob75FC3Y8FbpfLwUegRcxINtGUMPq0tFCvYNTBXNA==} - - '@emotion/unitless@0.8.1': - resolution: {integrity: sha512-KOEGMu6dmJZtpadb476IsZBclKvILjopjUii3V+7MnXIQCYh8W3NgNcgwo21n9LXZX6EDIKvqfjYxXebDwxKmQ==} - - '@eslint-community/eslint-utils@4.4.0': - resolution: {integrity: sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==} - engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} - peerDependencies: - eslint: ^6.0.0 || ^7.0.0 || >=8.0.0 - - '@eslint-community/regexpp@4.11.0': - resolution: {integrity: sha512-G/M/tIiMrTAxEWRfLfQJMmGNX28IxBg4PBz8XqQhqUHLFI6TL2htpIB1iQCj144V5ee/JaKyT9/WZ0MGZWfA7A==} - engines: {node: ^12.0.0 || ^14.0.0 || >=16.0.0} - - '@eslint/config-array@0.18.0': - resolution: {integrity: sha512-fTxvnS1sRMu3+JjXwJG0j/i4RT9u4qJ+lqS/yCGap4lH4zZGzQ7tu+xZqQmcMZq5OBZDL4QRxQzRjkWcGt8IVw==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - - '@eslint/eslintrc@3.1.0': - resolution: {integrity: sha512-4Bfj15dVJdoy3RfZmmo86RK1Fwzn6SstsvK9JS+BaVKqC6QQQQyXekNaC+g+LKNgkQ+2VhGAzm6hO40AhMR3zQ==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - - '@eslint/js@9.9.1': - resolution: {integrity: sha512-xIDQRsfg5hNBqHz04H1R3scSVwmI+KUbqjsQKHKQ1DAUSaUjYPReZZmS/5PNiKu1fUvzDd6H7DEDKACSEhu+TQ==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - - '@eslint/object-schema@2.1.4': - resolution: {integrity: sha512-BsWiH1yFGjXXS2yvrf5LyuoSIIbPrGUWob917o+BTKuZ7qJdxX8aJLRxs1fS9n6r7vESrq1OUqb68dANcFXuQQ==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - - '@fortawesome/fontawesome-common-types@6.6.0': - resolution: {integrity: sha512-xyX0X9mc0kyz9plIyryrRbl7ngsA9jz77mCZJsUkLl+ZKs0KWObgaEBoSgQiYWAsSmjz/yjl0F++Got0Mdp4Rw==} - engines: {node: '>=6'} - - '@fortawesome/fontawesome-free@6.6.0': - resolution: {integrity: sha512-60G28ke/sXdtS9KZCpZSHHkCbdsOGEhIUGlwq6yhY74UpTiToIh8np7A8yphhM4BWsvNFtIvLpi4co+h9Mr9Ow==} - engines: {node: '>=6'} - - '@fortawesome/fontawesome-svg-core@6.6.0': - resolution: {integrity: sha512-KHwPkCk6oRT4HADE7smhfsKudt9N/9lm6EJ5BVg0tD1yPA5hht837fB87F8pn15D8JfTqQOjhKTktwmLMiD7Kg==} - engines: {node: '>=6'} - - '@fortawesome/free-brands-svg-icons@6.6.0': - resolution: {integrity: sha512-1MPD8lMNW/earme4OQi1IFHtmHUwAKgghXlNwWi9GO7QkTfD+IIaYpIai4m2YJEzqfEji3jFHX1DZI5pbY/biQ==} - engines: {node: '>=6'} - - '@fortawesome/free-solid-svg-icons@6.6.0': - resolution: {integrity: sha512-IYv/2skhEDFc2WGUcqvFJkeK39Q+HyPf5GHUrT/l2pKbtgEIv1al1TKd6qStR5OIwQdN1GZP54ci3y4mroJWjA==} - engines: {node: '>=6'} - - '@fortawesome/react-fontawesome@0.2.2': - resolution: {integrity: sha512-EnkrprPNqI6SXJl//m29hpaNzOp1bruISWaOiRtkMi/xSvHJlzc2j2JAYS7egxt/EbjSNV/k6Xy0AQI6vB2+1g==} - peerDependencies: - '@fortawesome/fontawesome-svg-core': ~1 || ~6 - react: '>=16.3' - - '@hookform/error-message@2.0.1': - resolution: {integrity: sha512-U410sAr92xgxT1idlu9WWOVjndxLdgPUHEB8Schr27C9eh7/xUnITWpCMF93s+lGiG++D4JnbSnrb5A21AdSNg==} - peerDependencies: - react: '>=16.8.0' - react-dom: '>=16.8.0' - react-hook-form: ^7.0.0 - - '@hookform/resolvers@3.9.0': - resolution: {integrity: sha512-bU0Gr4EepJ/EQsH/IwEzYLsT/PEj5C0ynLQ4m+GSHS+xKH4TfSelhluTgOaoc4kA5s7eCsQbM4wvZLzELmWzUg==} - peerDependencies: - react-hook-form: ^7.0.0 - - '@humanwhocodes/module-importer@1.0.1': - resolution: {integrity: sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==} - engines: {node: '>=12.22'} - - '@humanwhocodes/retry@0.3.0': - resolution: {integrity: sha512-d2CGZR2o7fS6sWB7DG/3a95bGKQyHMACZ5aW8qGkkqQpUoZV6C0X7Pc7l4ZNMZkfNBf4VWNe9E1jRsf0G146Ew==} - engines: {node: '>=18.18'} - - '@img/sharp-darwin-arm64@0.33.5': - resolution: {integrity: sha512-UT4p+iz/2H4twwAoLCqfA9UH5pI6DggwKEGuaPy7nCVQ8ZsiY5PIcrRvD1DzuY3qYL07NtIQcWnBSY/heikIFQ==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm64] - os: [darwin] - - '@img/sharp-darwin-x64@0.33.5': - resolution: {integrity: sha512-fyHac4jIc1ANYGRDxtiqelIbdWkIuQaI84Mv45KvGRRxSAa7o7d1ZKAOBaYbnepLC1WqxfpimdeWfvqqSGwR2Q==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [x64] - os: [darwin] - - '@img/sharp-libvips-darwin-arm64@1.0.4': - resolution: {integrity: sha512-XblONe153h0O2zuFfTAbQYAX2JhYmDHeWikp1LM9Hul9gVPjFY427k6dFEcOL72O01QxQsWi761svJ/ev9xEDg==} - cpu: [arm64] - os: [darwin] - - '@img/sharp-libvips-darwin-x64@1.0.4': - resolution: {integrity: sha512-xnGR8YuZYfJGmWPvmlunFaWJsb9T/AO2ykoP3Fz/0X5XV2aoYBPkX6xqCQvUTKKiLddarLaxpzNe+b1hjeWHAQ==} - cpu: [x64] - os: [darwin] - - '@img/sharp-libvips-linux-arm64@1.0.4': - resolution: {integrity: sha512-9B+taZ8DlyyqzZQnoeIvDVR/2F4EbMepXMc/NdVbkzsJbzkUjhXv/70GQJ7tdLA4YJgNP25zukcxpX2/SueNrA==} - cpu: [arm64] - os: [linux] - - '@img/sharp-libvips-linux-arm@1.0.5': - resolution: {integrity: sha512-gvcC4ACAOPRNATg/ov8/MnbxFDJqf/pDePbBnuBDcjsI8PssmjoKMAz4LtLaVi+OnSb5FK/yIOamqDwGmXW32g==} - cpu: [arm] - os: [linux] - - '@img/sharp-libvips-linux-s390x@1.0.4': - resolution: {integrity: sha512-u7Wz6ntiSSgGSGcjZ55im6uvTrOxSIS8/dgoVMoiGE9I6JAfU50yH5BoDlYA1tcuGS7g/QNtetJnxA6QEsCVTA==} - cpu: [s390x] - os: [linux] - - '@img/sharp-libvips-linux-x64@1.0.4': - resolution: {integrity: sha512-MmWmQ3iPFZr0Iev+BAgVMb3ZyC4KeFc3jFxnNbEPas60e1cIfevbtuyf9nDGIzOaW9PdnDciJm+wFFaTlj5xYw==} - cpu: [x64] - os: [linux] - - '@img/sharp-libvips-linuxmusl-arm64@1.0.4': - resolution: {integrity: sha512-9Ti+BbTYDcsbp4wfYib8Ctm1ilkugkA/uscUn6UXK1ldpC1JjiXbLfFZtRlBhjPZ5o1NCLiDbg8fhUPKStHoTA==} - cpu: [arm64] - os: [linux] - - '@img/sharp-libvips-linuxmusl-x64@1.0.4': - resolution: {integrity: sha512-viYN1KX9m+/hGkJtvYYp+CCLgnJXwiQB39damAO7WMdKWlIhmYTfHjwSbQeUK/20vY154mwezd9HflVFM1wVSw==} - cpu: [x64] - os: [linux] - - '@img/sharp-linux-arm64@0.33.5': - resolution: {integrity: sha512-JMVv+AMRyGOHtO1RFBiJy/MBsgz0x4AWrT6QoEVVTyh1E39TrCUpTRI7mx9VksGX4awWASxqCYLCV4wBZHAYxA==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm64] - os: [linux] - - '@img/sharp-linux-arm@0.33.5': - resolution: {integrity: sha512-JTS1eldqZbJxjvKaAkxhZmBqPRGmxgu+qFKSInv8moZ2AmT5Yib3EQ1c6gp493HvrvV8QgdOXdyaIBrhvFhBMQ==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm] - os: [linux] - - '@img/sharp-linux-s390x@0.33.5': - resolution: {integrity: sha512-y/5PCd+mP4CA/sPDKl2961b+C9d+vPAveS33s6Z3zfASk2j5upL6fXVPZi7ztePZ5CuH+1kW8JtvxgbuXHRa4Q==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [s390x] - os: [linux] - - '@img/sharp-linux-x64@0.33.5': - resolution: {integrity: sha512-opC+Ok5pRNAzuvq1AG0ar+1owsu842/Ab+4qvU879ippJBHvyY5n2mxF1izXqkPYlGuP/M556uh53jRLJmzTWA==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [x64] - os: [linux] - - '@img/sharp-linuxmusl-arm64@0.33.5': - resolution: {integrity: sha512-XrHMZwGQGvJg2V/oRSUfSAfjfPxO+4DkiRh6p2AFjLQztWUuY/o8Mq0eMQVIY7HJ1CDQUJlxGGZRw1a5bqmd1g==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm64] - os: [linux] - - '@img/sharp-linuxmusl-x64@0.33.5': - resolution: {integrity: sha512-WT+d/cgqKkkKySYmqoZ8y3pxx7lx9vVejxW/W4DOFMYVSkErR+w7mf2u8m/y4+xHe7yY9DAXQMWQhpnMuFfScw==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [x64] - os: [linux] - - '@img/sharp-wasm32@0.33.5': - resolution: {integrity: sha512-ykUW4LVGaMcU9lu9thv85CbRMAwfeadCJHRsg2GmeRa/cJxsVY9Rbd57JcMxBkKHag5U/x7TSBpScF4U8ElVzg==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [wasm32] - - '@img/sharp-win32-ia32@0.33.5': - resolution: {integrity: sha512-T36PblLaTwuVJ/zw/LaH0PdZkRz5rd3SmMHX8GSmR7vtNSP5Z6bQkExdSK7xGWyxLw4sUknBuugTelgw2faBbQ==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [ia32] - os: [win32] - - '@img/sharp-win32-x64@0.33.5': - resolution: {integrity: sha512-MpY/o8/8kj+EcnxwvrP4aTJSWw/aZ7JIGR4aBeZkZw5B7/Jn+tY9/VNwtcoGmdT7GfggGIU4kygOMSbYnOrAbg==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [x64] - os: [win32] - - '@isaacs/cliui@8.0.2': - resolution: {integrity: sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==} - engines: {node: '>=12'} - - '@logto/client@2.8.1': - resolution: {integrity: sha512-tUQC36l9U3knrTicXFjd+FiBqwG1/KSGn1o3wx9DFn+5iSTQa66B+Y88GaXcxAYsgjzmSHrDY3qxuQg729mleQ==} - - '@logto/js@4.2.0': - resolution: {integrity: sha512-kse29kNKuM/tw30YcOf2eOQuFBlGPZTr4rKI/LpdTKhk0fLGM5ZJKPQWNPiPFZdBfTldERXVNtd17/bZLWc/OQ==} - - '@logto/next@3.7.1': - resolution: {integrity: sha512-ZMk+yszQV9N7y21oT+V0c1bE88ffDdJJKJtXxpFHAEwjwXFgIfxlpWnmXppb5v3Zkf/SISFEyFTGYqIB9l7rDg==} - peerDependencies: - next: '>=12' - - '@logto/node@2.5.8': - resolution: {integrity: sha512-QNVWIZf6sLAbF6rDT/wAG64S4IH5OQYk38fJvJYuIuMRHhUDf2VD+ENZ5BBwX3sOHFf6OM4rJVt2vWVJ8kTVow==} - - '@mux/blurhash@0.1.2': - resolution: {integrity: sha512-fWLOXHS2l3CGyPHF4NSRLqZx9DDAz1WYC4YXD3du24xxibIKUyBYzg7PDtx54z5QaQ12ln5oPFvhH5LhaLzeZg==} - - '@mux/mux-player-react@2.9.1': - resolution: {integrity: sha512-1BpMs1J7P+d+/QCf9/mkTk/NPYR6sOskR4Ih0uFZjDAqNUN7/C9Q0FEJ6hF3sFXwAXo50RhnfCzsC5uYx3QHbA==} - peerDependencies: - '@types/react': ^17.0.0 || ^18 || ^19 - '@types/react-dom': '*' - react: ^17.0.2 || ^18 || ^19 - react-dom: ^17.0.2 || ^18 || ^19 - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@mux/mux-player@2.9.1': - resolution: {integrity: sha512-TAyoUSPTV0IXWGMOknL6O+IeGSEJ8aS154DzyzqZgdd3zDJHM8JpkyNHgtowatMHT2lB6h+qMtWfp4u3ijpo2A==} - - '@mux/mux-video@0.20.2': - resolution: {integrity: sha512-CqkK9EZZWdQE4U62JKlmWDskirT+D9C4suh2tSqKb2CA/0S4ybbbrVWcCKF7xfadUacfKO1yPsOKbe46F6phVQ==} - - '@mux/playback-core@0.25.2': - resolution: {integrity: sha512-vrBbCgLHwmPpVxF0QGj+sXHUVXSxgDJJhVm8pxPXEkbw0vjPNHTXgAd/Ty6JA0vZ0ZjoQuAa17AxJ+c02JYeWQ==} - - '@next/env@14.2.7': - resolution: {integrity: sha512-OTx9y6I3xE/eih+qtthppwLytmpJVPM5PPoJxChFsbjIEFXIayG0h/xLzefHGJviAa3Q5+Fd+9uYojKkHDKxoQ==} - - '@next/eslint-plugin-next@14.2.7': - resolution: {integrity: sha512-+7xh142AdhZGjY9/L0iFo7mqRBMJHe+q+uOL+hto1Lfo9DeWCGcR6no4StlFbVSVcA6fQLKEX6y6qhMsSKbgNQ==} - - '@next/swc-darwin-arm64@14.2.7': - resolution: {integrity: sha512-UhZGcOyI9LE/tZL3h9rs/2wMZaaJKwnpAyegUVDGZqwsla6hMfeSj9ssBWQS9yA4UXun3pPhrFLVnw5KXZs3vw==} - engines: {node: '>= 10'} - cpu: [arm64] - os: [darwin] - - '@next/swc-darwin-x64@14.2.7': - resolution: {integrity: sha512-ys2cUgZYRc+CbyDeLAaAdZgS7N1Kpyy+wo0b/gAj+SeOeaj0Lw/q+G1hp+DuDiDAVyxLBCJXEY/AkhDmtihUTA==} - engines: {node: '>= 10'} - cpu: [x64] - os: [darwin] - - '@next/swc-linux-arm64-gnu@14.2.7': - resolution: {integrity: sha512-2xoWtE13sUJ3qrC1lwE/HjbDPm+kBQYFkkiVECJWctRASAHQ+NwjMzgrfqqMYHfMxFb5Wws3w9PqzZJqKFdWcQ==} - engines: {node: '>= 10'} - cpu: [arm64] - os: [linux] - - '@next/swc-linux-arm64-musl@14.2.7': - resolution: {integrity: sha512-+zJ1gJdl35BSAGpkCbfyiY6iRTaPrt3KTl4SF/B1NyELkqqnrNX6cp4IjjjxKpd64/7enI0kf6b9O1Uf3cL0pw==} - engines: {node: '>= 10'} - cpu: [arm64] - os: [linux] - - '@next/swc-linux-x64-gnu@14.2.7': - resolution: {integrity: sha512-m6EBqrskeMUzykBrv0fDX/28lWIBGhMzOYaStp0ihkjzIYJiKUOzVYD1gULHc8XDf5EMSqoH/0/TRAgXqpQwmw==} - engines: {node: '>= 10'} - cpu: [x64] - os: [linux] - - '@next/swc-linux-x64-musl@14.2.7': - resolution: {integrity: sha512-gUu0viOMvMlzFRz1r1eQ7Ql4OE+hPOmA7smfZAhn8vC4+0swMZaZxa9CSIozTYavi+bJNDZ3tgiSdMjmMzRJlQ==} - engines: {node: '>= 10'} - cpu: [x64] - os: [linux] - - '@next/swc-win32-arm64-msvc@14.2.7': - resolution: {integrity: sha512-PGbONHIVIuzWlYmLvuFKcj+8jXnLbx4WrlESYlVnEzDsa3+Q2hI1YHoXaSmbq0k4ZwZ7J6sWNV4UZfx1OeOlbQ==} - engines: {node: '>= 10'} - cpu: [arm64] - os: [win32] - - '@next/swc-win32-ia32-msvc@14.2.7': - resolution: {integrity: sha512-BiSY5umlx9ed5RQDoHcdbuKTUkuFORDqzYKPHlLeS+STUWQKWziVOn3Ic41LuTBvqE0TRJPKpio9GSIblNR+0w==} - engines: {node: '>= 10'} - cpu: [ia32] - os: [win32] - - '@next/swc-win32-x64-msvc@14.2.7': - resolution: {integrity: sha512-pxsI23gKWRt/SPHFkDEsP+w+Nd7gK37Hpv0ngc5HpWy2e7cKx9zR/+Q2ptAUqICNTecAaGWvmhway7pj/JLEWA==} - engines: {node: '>= 10'} - cpu: [x64] - os: [win32] - - '@noble/hashes@1.5.0': - resolution: {integrity: sha512-1j6kQFb7QRru7eKN3ZDvRcP13rugwdxZqCjbiAVZfIJwgj2A65UmT4TgARXGlXgnRkORLTDTrO19ZErt7+QXgA==} - engines: {node: ^14.21.3 || >=16} - - '@nodelib/fs.scandir@2.1.5': - resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==} - engines: {node: '>= 8'} - - '@nodelib/fs.stat@2.0.5': - resolution: {integrity: sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==} - engines: {node: '>= 8'} - - '@nodelib/fs.walk@1.2.8': - resolution: {integrity: sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==} - engines: {node: '>= 8'} - - '@nolyfill/is-core-module@1.0.39': - resolution: {integrity: sha512-nn5ozdjYQpUCZlWGuxcJY/KpxkWQs4DcbMCmKojjyrYDEAGy4Ce19NN4v5MduafTwJlbKc99UA8YhSVqq9yPZA==} - engines: {node: '>=12.4.0'} - - '@panva/hkdf@1.2.1': - resolution: {integrity: sha512-6oclG6Y3PiDFcoyk8srjLfVKyMfVCKJ27JwNPViuXziFpmdz+MZnZN/aKY0JGXgYuO/VghU0jcOAZgWXZ1Dmrw==} - - '@paralleldrive/cuid2@2.2.2': - resolution: {integrity: sha512-ZOBkgDwEdoYVlSeRbYYXs0S9MejQofiVYoTbKzy/6GQa39/q5tQU2IX46+shYnUkpEl3wc+J6wRlar7r2EK2xA==} - - '@pkgjs/parseargs@0.11.0': - resolution: {integrity: sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==} - engines: {node: '>=14'} - - '@react-hookz/deep-equal@1.0.4': - resolution: {integrity: sha512-N56fTrAPUDz/R423pag+n6TXWbvlBZDtTehaGFjK0InmN+V2OFWLE/WmORhmn6Ce7dlwH5+tQN1LJFw3ngTJVg==} - - '@react-hookz/web@24.0.4': - resolution: {integrity: sha512-DcIM6JiZklDyHF6CRD1FTXzuggAkQ+3Ncq2Wln7Kdih8GV6ZIeN9JfS6ZaQxpQUxan8/4n0J2V/R7nMeiSrb2Q==} - engines: {node: '>=18.0.0'} - peerDependencies: - js-cookie: ^3.0.5 - react: ^16.8 || ^17 || ^18 - react-dom: ^16.8 || ^17 || ^18 - peerDependenciesMeta: - js-cookie: - optional: true - - '@rtsao/scc@1.1.0': - resolution: {integrity: sha512-zt6OdqaDoOnJ1ZYsCYGt9YmWzDXl4vQdKTyJev62gFhRGKdx7mcT54V9KIjg+d2wi9EXsPvAPKe7i7WjfVWB8g==} - - '@rushstack/eslint-patch@1.10.4': - resolution: {integrity: sha512-WJgX9nzTqknM393q1QJDJmoW28kUfEnybeTfVNcNAPnIx210RXm2DiXiHzfNPJNIUUb1tJnz/l4QGtJ30PgWmA==} - - '@silverhand/essentials@2.9.2': - resolution: {integrity: sha512-bD+82D9Dfa1F5xX1kfdR5ODIoJS41NOxTuHx4shVS5A4/ayEG+ZplpDDjB19fsa7kZXgSgD75R4sUCXjm88x6w==} - engines: {node: ^18.12.0 || ^20.9.0 || ^22.0.0, pnpm: ^9.0.0} - - '@swc/counter@0.1.3': - resolution: {integrity: sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ==} - - '@swc/helpers@0.5.5': - resolution: {integrity: sha512-KGYxvIOXcceOAbEk4bi/dVLEK9z8sZ0uBB3Il5b1rhfClSpcX0yfRO0KmTkqR2cnQDymwLB+25ZyMzICg/cm/A==} - - '@tanstack/query-core@5.53.3': - resolution: {integrity: sha512-ZfjAgd7NpqDx0e4aYBt7EmS2enbulPrJwowTy+mayRE93WUUH+sIYHun1TdRjpGwDPMNNZ5D6goh7n3CwoO+HA==} - - '@tanstack/react-query@5.53.3': - resolution: {integrity: sha512-286mN/91CeM7vC6CZFLKYDHSw+WyMX6ekIvzoTbpM4xyPb99VSyCKPLyPgaOatKqYm6ooMBquSq9NGRdKgsJfg==} - peerDependencies: - react: ^18 || ^19 - - '@tanstack/react-table@8.20.5': - resolution: {integrity: sha512-WEHopKw3znbUZ61s9i0+i9g8drmDo6asTWbrQh8Us63DAk/M0FkmIqERew6P71HI75ksZ2Pxyuf4vvKh9rAkiA==} - engines: {node: '>=12'} - peerDependencies: - react: '>=16.8' - react-dom: '>=16.8' - - '@tanstack/table-core@8.20.5': - resolution: {integrity: sha512-P9dF7XbibHph2PFRz8gfBKEXEY/HJPOhym8CHmjF8y3q5mWpKx9xtZapXQUWCgkqvsK0R46Azuz+VaxD4Xl+Tg==} - engines: {node: '>=12'} - - '@transloadit/prettier-bytes@0.3.4': - resolution: {integrity: sha512-8/SnIF9Q2k52mbjRVAYLranwkaDTLb+O9r4Z/uo8uNw//SjygKvvbF4BHSOuReufaAyum1q13602VcNud25Dfg==} - - '@types/json-schema@7.0.15': - resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==} - - '@types/json5@0.0.29': - resolution: {integrity: sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==} - - '@types/lodash@4.17.7': - resolution: {integrity: sha512-8wTvZawATi/lsmNu10/j2hk1KEP0IvjubqPE3cu1Xz7xfXXt5oCq3SNUz4fMIP4XGF9Ky+Ue2tBA3hcS7LSBlA==} - - '@types/node@22.5.2': - resolution: {integrity: sha512-acJsPTEqYqulZS/Yp/S3GgeE6GZ0qYODUR8aVr/DkhHQ8l9nd4j5x1/ZJy9/gHrRlFMqkO6i0I3E27Alu4jjPg==} - - '@types/prop-types@15.7.12': - resolution: {integrity: sha512-5zvhXYtRNRluoE/jAp4GVsSduVUzNWKkOZrCDBWYtE7biZywwdC2AcEzg+cSMLFRfVgeAFqpfNabiPjxFddV1Q==} - - '@types/qs@6.9.15': - resolution: {integrity: sha512-uXHQKES6DQKKCLh441Xv/dwxOq1TVS3JPUMlEqoEglvlhR6Mxnlew/Xq/LRVHpLyk7iK3zODe1qYHIMltO7XGg==} - - '@types/react-dom@18.3.0': - resolution: {integrity: sha512-EhwApuTmMBmXuFOikhQLIBUn6uFg81SwLMOAUgodJF14SOBOCMdU04gDoYi0WOJJHD144TL32z4yDqCW3dnkQg==} - - '@types/react@18.3.5': - resolution: {integrity: sha512-WeqMfGJLGuLCqHGYRGHxnKrXcTitc6L/nBUWfWPcTarG3t9PsquqUMuVeXZeca+mglY4Vo5GZjCi0A3Or2lnxA==} - - '@types/retry@0.12.2': - resolution: {integrity: sha512-XISRgDJ2Tc5q4TRqvgJtzsRkFYNJzZrhTdtMoGVBttwzzQJkPnS3WWTFc7kuDRoPtPakl+T+OfdEUjYJj7Jbow==} - - '@types/stylis@4.2.5': - resolution: {integrity: sha512-1Xve+NMN7FWjY14vLoY5tL3BVEQ/n42YLwaqJIPYhotZ9uBHt87VceMwWQpzmdEt2TNXIorIFG+YeCUUW7RInw==} - - '@typescript-eslint/parser@7.2.0': - resolution: {integrity: sha512-5FKsVcHTk6TafQKQbuIVkXq58Fnbkd2wDL4LB7AURN7RUOu1utVP+G8+6u3ZhEroW3DF6hyo3ZEXxgKgp4KeCg==} - engines: {node: ^16.0.0 || >=18.0.0} - peerDependencies: - eslint: ^8.56.0 - typescript: '*' - peerDependenciesMeta: - typescript: - optional: true - - '@typescript-eslint/scope-manager@7.2.0': - resolution: {integrity: sha512-Qh976RbQM/fYtjx9hs4XkayYujB/aPwglw2choHmf3zBjB4qOywWSdt9+KLRdHubGcoSwBnXUH2sR3hkyaERRg==} - engines: {node: ^16.0.0 || >=18.0.0} - - '@typescript-eslint/types@7.2.0': - resolution: {integrity: sha512-XFtUHPI/abFhm4cbCDc5Ykc8npOKBSJePY3a3s+lwumt7XWJuzP5cZcfZ610MIPHjQjNsOLlYK8ASPaNG8UiyA==} - engines: {node: ^16.0.0 || >=18.0.0} - - '@typescript-eslint/typescript-estree@7.2.0': - resolution: {integrity: sha512-cyxS5WQQCoBwSakpMrvMXuMDEbhOo9bNHHrNcEWis6XHx6KF518tkF1wBvKIn/tpq5ZpUYK7Bdklu8qY0MsFIA==} - engines: {node: ^16.0.0 || >=18.0.0} - peerDependencies: - typescript: '*' - peerDependenciesMeta: - typescript: - optional: true - - '@typescript-eslint/visitor-keys@7.2.0': - resolution: {integrity: sha512-c6EIQRHhcpl6+tO8EMR+kjkkV+ugUNXOmeASA1rlzkd8EPIriavpWoiEz1HR/VLhbVIdhqnV6E7JZm00cBDx2A==} - engines: {node: ^16.0.0 || >=18.0.0} - - '@uppy/aws-s3-multipart@4.0.0': - resolution: {integrity: sha512-tW7W1cBh/99PhYaYDQ4Bo5h0aCIjnCgx/n6d6h7OJMu4NRD703DV/D8AHKIRl5JgecDekiTbApy2iu3q4uKkyw==} - deprecated: Use @uppy/aws-s3 instead. - - '@uppy/aws-s3@4.1.0': - resolution: {integrity: sha512-xRip1Lo3He+3J3fP/SooEFQJKWMCVADTl8J375PzvpaeNnDFKa6W2XLEEl/fGy/K7vI4sH8Znz4+omdtSFCPSQ==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/box@3.1.0': - resolution: {integrity: sha512-d7PUhEdTNEr+7dTNc2lIv/AknRdcxOEXZML5Jp43S//VAGPu8pwdGgmHqi00s8RF4I6daSt1p+tl6VsMnzY3sQ==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/companion-client@4.1.0': - resolution: {integrity: sha512-nQ8CQfZcYVBNtFQ6ePj7FDIq38DXlH0YpzP/91LR9gnDVISJKKUuvWfr6tPktj1lRw9FZV8jLmlMKT2ituVKiw==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/core@4.2.0': - resolution: {integrity: sha512-/oQ2m/xubGfANR0UfMqYFR2mT94OpuXTp9N2cQLIQmWYZtpvfX2gyNBFtQJA3Njqpmox1RfIhOAsVFFuhYVa+Q==} - - '@uppy/dashboard@4.1.0': - resolution: {integrity: sha512-TSwgW1x0YDnZP4yJwxVrxO6vsW3GrWoqXwOdpcTGDwdIOvZZJBAajqr3QrVdkbz1L4YfQAgTtHZlyUhm5qMuzg==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/drag-drop@4.0.2': - resolution: {integrity: sha512-0/b8hBAX8tDBikkr2tORtKT3gEcCxQlygSBCJrbLTQTDh4poTpmHWyquqvsCcBtW7AqULhQn5h/xSSacBnEf/Q==} - peerDependencies: - '@uppy/core': ^4.1.1 - - '@uppy/dropbox@4.1.0': - resolution: {integrity: sha512-Ha+WhJwQyQ+VRMrfPKdr+tuEBKRUMMeCfqfDpjasiwJSgDxQUFgGhv81ZoHXYtxs/aOwucsMl3G4tt9FM3EK/w==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/facebook@4.1.0': - resolution: {integrity: sha512-0ZO8bx4yvlu97Zkh7QW25hqirSOpjumz4SVE3MHukCaRcYxMChYBqC0ABFTIWCsD9gmUfdQm4CN5qoBm6dqomA==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/file-input@4.0.1': - resolution: {integrity: sha512-UKZXl3089OVserkSA52TkPVBjJUsqEWcT9+UxuPUz/m+lS51uXonK0yl9ZrSCqkY+4yhLcdEdH0xW1XrJXNHYg==} - peerDependencies: - '@uppy/core': ^4.1.1 - - '@uppy/google-drive@4.1.0': - resolution: {integrity: sha512-djmPmElbkmNePzWPT38q+HkkhYPlfoN+7hUqp3YIIMXEvPZFZqUz/vPaly1qRvxGZiENlzD45jhwJQ3PhZ9N5A==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/google-photos@0.3.0': - resolution: {integrity: sha512-FY5W8fOuBVzFl6Y0VW772iKCcnCOMBMFwhuRL1922RI4roApX77b0mXG6vCPh3n7XpDRa4FFKA7lN4/9DQM4ag==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/informer@4.1.0': - resolution: {integrity: sha512-Dzq7bEnUUePd7Syy6bDgzwSc16Re1tDYqP/sivtvPDrqINz8gUIST6IxN0GxRoSH732EjGiMlSf3OjwV/N18xQ==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/instagram@4.1.0': - resolution: {integrity: sha512-+19SEIPWykha/pmOpgPrur8vqxImgwn4rLR3F3papbJdJuyUsVejWnrD4HEZaX4ay8HtMvBjz6AaY4KygONz+g==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/onedrive@4.1.0': - resolution: {integrity: sha512-PmYMD6IEbDiBqjLDNYmJ+lETrIudR05DM8rvNHAKzZu3m7oIw1iN7Tg9DLmL0DdnQ7F5rKSYL/SZBVB40yKLUQ==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/progress-bar@4.0.0': - resolution: {integrity: sha512-hCUjlfGWHlvBPQDO5YBH/8HEr+3+ZEobTblBg0Wbn3ecJSiKkSRi0GkDVp3OMnwfqgK2wm8Ve+tR/5Gds7vE0A==} - peerDependencies: - '@uppy/core': ^4.0.0 - - '@uppy/provider-views@4.0.1': - resolution: {integrity: sha512-oAOIVdCSPIpDZJXwU83o+13+qWYrIfRzJaXom7ZsJpj+WDbtFjML5iF3evDmqt22V3HwOC0N187lZvcO/9RwFA==} - peerDependencies: - '@uppy/core': ^4.1.1 - - '@uppy/react@4.0.2': - resolution: {integrity: sha512-q0IKlldfAmAWhpDyx19U3nCDO/aRXkjP/E3e/k1uWNMRTCBEtYdXPN/UIaYMsn3NcUwae/g/jOOV79OiB5WW0w==} - peerDependencies: - '@uppy/core': ^4.1.2 - '@uppy/dashboard': ^4.0.3 - '@uppy/drag-drop': ^4.0.2 - '@uppy/file-input': ^4.0.1 - '@uppy/progress-bar': ^4.0.0 - '@uppy/status-bar': ^4.0.3 - react: ^18.0.0 - peerDependenciesMeta: - '@uppy/dashboard': - optional: true - '@uppy/drag-drop': - optional: true - '@uppy/file-input': - optional: true - '@uppy/progress-bar': - optional: true - '@uppy/status-bar': - optional: true - - '@uppy/remote-sources@2.2.0': - resolution: {integrity: sha512-Spo+oc2vKW0oKRD4Rhf5kZXYv+5Sim/5TcMmxxfHxLPS46yh+WNLU/499ojKr31ZBHqhkXkzUdliSYunqlDwlw==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/status-bar@4.0.3': - resolution: {integrity: sha512-ckujiEQwHgpJGa5Q6OZF+hJ+3JSMgs/7vyl4aeBvV0zSWoPSg/W10TpyGeNvMaaAsbAs4UB+0LuUjVu/vSmFcw==} - peerDependencies: - '@uppy/core': ^4.1.2 - - '@uppy/store-default@4.1.0': - resolution: {integrity: sha512-z5VSc4PNXpAtrrUPg5hdKJO5Ul7u4ZYLyK+tYzvEgzgR4nLVZmpGzj/d4N90jXpUqEibWKXvevODEB5VlTLHzg==} - - '@uppy/thumbnail-generator@4.0.0': - resolution: {integrity: sha512-nwgRO/LHLzUqzyB1TDl6g8LNmqtkswXpvRNcPij0gOrPTTWjGY6ULv+ywXYiF5baWF2aGS+K62jJSUGBWonx0w==} - peerDependencies: - '@uppy/core': ^4.0.0 - - '@uppy/unsplash@4.1.0': - resolution: {integrity: sha512-r7RIO96igbDo9ENP3z7X0fqdOiA3zM1ihnG1bfBVgJax2t0VLfb/peB6XLILT9WSzOWi2izajElVCK9Q384Okw==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/url@4.1.0': - resolution: {integrity: sha512-7rxImH1h6vagjeqvAX/Go/6CrhE4SwBMZtJ3kgi5km1k9G4J/0rVfnylgdP4A5W2u7TR2RAShfAE+Nu/M2cMpw==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/utils@6.0.2': - resolution: {integrity: sha512-ZoNeAa1YTKSlcvXe1SP3POjzjRZ9jSojorbst03vwd1Ks9vHPGf6pne61DowTXHZ3HMj1vpcIaQ1VIEWeeADlA==} - - '@uppy/zoom@3.1.0': - resolution: {integrity: sha512-ubx5GjmYGUYjBTCaRFJAJaO78B5d4O4m2CkfG+R6VAcEGlrv7djujYJVUXfjzTevcRtq3qhDfK/WgNMjJv0ptQ==} - peerDependencies: - '@uppy/core': ^4.2.0 - - acorn-jsx@5.3.2: - resolution: {integrity: sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==} - peerDependencies: - acorn: ^6.0.0 || ^7.0.0 || ^8.0.0 - - acorn@8.12.1: - resolution: {integrity: sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==} - engines: {node: '>=0.4.0'} - hasBin: true - - ajv@6.12.6: - resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==} - - ansi-regex@5.0.1: - resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==} - engines: {node: '>=8'} - - ansi-regex@6.0.1: - resolution: {integrity: sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==} - engines: {node: '>=12'} - - ansi-styles@4.3.0: - resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} - engines: {node: '>=8'} - - ansi-styles@6.2.1: - resolution: {integrity: sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==} - engines: {node: '>=12'} - - anymatch@3.1.3: - resolution: {integrity: sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==} - engines: {node: '>= 8'} - - argparse@1.0.10: - resolution: {integrity: sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==} - - argparse@2.0.1: - resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==} - - aria-query@5.1.3: - resolution: {integrity: sha512-R5iJ5lkuHybztUfuOAznmboyjWq8O6sqNqtK7CLOqdydi54VNbORp49mb14KbWgG1QD3JFO9hJdZ+y4KutfdOQ==} - - array-buffer-byte-length@1.0.1: - resolution: {integrity: sha512-ahC5W1xgou+KTXix4sAO8Ki12Q+jf4i0+tmk3sC+zgcynshkHxzpXdImBehiUYKKKDwvfFiJl1tZt6ewscS1Mg==} - engines: {node: '>= 0.4'} - - array-includes@3.1.8: - resolution: {integrity: sha512-itaWrbYbqpGXkGhZPGUulwnhVf5Hpy1xiCFsGqyIGglbBxmG5vSjxQen3/WGOjPpNEv1RtBLKxbmVXm8HpJStQ==} - engines: {node: '>= 0.4'} - - array-union@2.1.0: - resolution: {integrity: sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==} - engines: {node: '>=8'} - - array.prototype.findlast@1.2.5: - resolution: {integrity: sha512-CVvd6FHg1Z3POpBLxO6E6zr+rSKEQ9L6rZHAaY7lLfhKsWYUBBOuMs0e9o24oopj6H+geRCX0YJ+TJLBK2eHyQ==} - engines: {node: '>= 0.4'} - - array.prototype.findlastindex@1.2.5: - resolution: {integrity: sha512-zfETvRFA8o7EiNn++N5f/kaCw221hrpGsDmcpndVupkPzEc1Wuf3VgC0qby1BbHs7f5DVYjgtEU2LLh5bqeGfQ==} - engines: {node: '>= 0.4'} - - array.prototype.flat@1.3.2: - resolution: {integrity: sha512-djYB+Zx2vLewY8RWlNCUdHjDXs2XOgm602S9E7P/UpHgfeHL00cRiIF+IN/G/aUJ7kGPb6yO/ErDI5V2s8iycA==} - engines: {node: '>= 0.4'} - - array.prototype.flatmap@1.3.2: - resolution: {integrity: sha512-Ewyx0c9PmpcsByhSW4r+9zDU7sGjFc86qf/kKtuSCRdhfbk0SNLLkaT5qvcHnRGgc5NP/ly/y+qkXkqONX54CQ==} - engines: {node: '>= 0.4'} - - array.prototype.tosorted@1.1.4: - resolution: {integrity: sha512-p6Fx8B7b7ZhL/gmUsAy0D15WhvDccw3mnGNbZpi3pmeJdxtWsj2jEaI4Y6oo3XiHfzuSgPwKc04MYt6KgvC/wA==} - engines: {node: '>= 0.4'} - - arraybuffer.prototype.slice@1.0.3: - resolution: {integrity: sha512-bMxMKAjg13EBSVscxTaYA4mRc5t1UAXa2kXiGTNfZ079HIWXEkKmkgFrh/nJqamaLSrXO5H4WFFkPEaLJWbs3A==} - engines: {node: '>= 0.4'} - - ast-types-flow@0.0.8: - resolution: {integrity: sha512-OH/2E5Fg20h2aPrbe+QL8JZQFko0YZaF+j4mnQ7BGhfavO7OpSLa8a0y9sBwomHdSbkhTS8TQNayBfnW5DwbvQ==} - - available-typed-arrays@1.0.7: - resolution: {integrity: sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==} - engines: {node: '>= 0.4'} - - axe-core@4.10.0: - resolution: {integrity: sha512-Mr2ZakwQ7XUAjp7pAwQWRhhK8mQQ6JAaNWSjmjxil0R8BPioMtQsTLOolGYkji1rcL++3dCqZA3zWqpT+9Ew6g==} - engines: {node: '>=4'} - - axobject-query@3.1.1: - resolution: {integrity: sha512-goKlv8DZrK9hUh975fnHzhNIO4jUnFCfv/dszV5VwUGDFjI6vQ2VwoyjYjYNEbBE8AH87TduWP5uyDR1D+Iteg==} - - balanced-match@1.0.2: - resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} - - base64-js@1.5.1: - resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==} - - binary-extensions@2.3.0: - resolution: {integrity: sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==} - engines: {node: '>=8'} - - bl@4.1.0: - resolution: {integrity: sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==} - - blurhash@1.1.5: - resolution: {integrity: sha512-a+LO3A2DfxTaTztsmkbLYmUzUeApi0LZuKalwbNmqAHR6HhJGMt1qSV/R3wc+w4DL28holjqO3Bg74aUGavGjg==} - - brace-expansion@1.1.11: - resolution: {integrity: sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==} - - brace-expansion@2.0.1: - resolution: {integrity: sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==} - - braces@3.0.3: - resolution: {integrity: sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==} - engines: {node: '>=8'} - - buffer@5.7.1: - resolution: {integrity: sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==} - - bulma@1.0.2: - resolution: {integrity: sha512-D7GnDuF6seb6HkcnRMM9E739QpEY9chDzzeFrHMyEns/EXyDJuQ0XA0KxbBl/B2NTsKSoDomW61jFGFaAxhK5A==} - - busboy@1.6.0: - resolution: {integrity: sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA==} - engines: {node: '>=10.16.0'} - - call-bind@1.0.7: - resolution: {integrity: sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==} - engines: {node: '>= 0.4'} - - callsites@3.1.0: - resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==} - engines: {node: '>=6'} - - camelcase-keys@7.0.2: - resolution: {integrity: sha512-Rjs1H+A9R+Ig+4E/9oyB66UC5Mj9Xq3N//vcLf2WzgdTi/3gUu3Z9KoqmlrEG4VuuLK8wJHofxzdQXz/knhiYg==} - engines: {node: '>=12'} - - camelcase@6.3.0: - resolution: {integrity: sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==} - engines: {node: '>=10'} - - camelize@1.0.1: - resolution: {integrity: sha512-dU+Tx2fsypxTgtLoE36npi3UqcjSSMNYfkqgmoEhtZrraP5VWq0K7FkWVTYa8eMPtnU/G2txVsfdCJTn9uzpuQ==} - - caniuse-lite@1.0.30001655: - resolution: {integrity: sha512-jRGVy3iSGO5Uutn2owlb5gR6qsGngTw9ZTb4ali9f3glshcNmJ2noam4Mo9zia5P9Dk3jNNydy7vQjuE5dQmfg==} - - castable-video@1.0.10: - resolution: {integrity: sha512-tJgUv+8/zE191y8EKojvB0eKIyKA9obIttd6Wpdm6x2qBmuwZ7wDgzVCSmf5cN2v9jBiuu0s7O5poz8a8cFX/w==} - - chalk@4.1.2: - resolution: {integrity: sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==} - engines: {node: '>=10'} - - chokidar@3.6.0: - resolution: {integrity: sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==} - engines: {node: '>= 8.10.0'} - - chownr@1.1.4: - resolution: {integrity: sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==} - - classnames@2.5.1: - resolution: {integrity: sha512-saHYOzhIQs6wy2sVxTM6bUDsQO4F50V9RQ22qBpEdCW+I+/Wmke2HOl6lS6dTpdxVhb88/I6+Hs+438c3lfUow==} - - client-only@0.0.1: - resolution: {integrity: sha512-IV3Ou0jSMzZrd3pZ48nLkT9DA7Ag1pnPzaiQhpW7c3RbcqqzvzzVu+L8gfqMp/8IM2MQtSiqaCxrrcfu8I8rMA==} - - clsx@2.1.1: - resolution: {integrity: sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==} - engines: {node: '>=6'} - - color-convert@2.0.1: - resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} - engines: {node: '>=7.0.0'} - - color-name@1.1.4: - resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} - - color-string@1.9.1: - resolution: {integrity: sha512-shrVawQFojnZv6xM40anx4CkoDP+fZsw/ZerEMsW/pyzsRbElpsL/DBVW7q3ExxwusdNXI3lXpuhEZkzs8p5Eg==} - - color@4.2.3: - resolution: {integrity: sha512-1rXeuUUiGGrykh+CeBdu5Ie7OJwinCgQY0bc7GCRxy5xVHy+moaqkpL/jqQq0MtQOeYcrqEz4abc5f0KtU7W4A==} - engines: {node: '>=12.5.0'} - - concat-map@0.0.1: - resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} - - cookie@0.6.0: - resolution: {integrity: sha512-U71cyTamuh1CRNCfpGY6to28lxvNwPG4Guz/EVjgf3Jmzv0vlDp1atT9eS5dDjMYHucpHbWns6Lwf3BKz6svdw==} - engines: {node: '>= 0.6'} - - cookie@0.7.1: - resolution: {integrity: sha512-6DnInpx7SJ2AK3+CTUE/ZM0vWTUboZCegxhC2xiIydHR9jNuTAASBrfEpHhiGOZw/nX51bHt6YQl8jsGo4y/0w==} - engines: {node: '>= 0.6'} - - core-js@3.38.1: - resolution: {integrity: sha512-OP35aUorbU3Zvlx7pjsFdu1rGNnD4pgw/CWoYzRY3t2EzoVT7shKHY1dlAy3f41cGIO7ZDPQimhGFTlEYkG/Hw==} - - cors@2.8.5: - resolution: {integrity: sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==} - engines: {node: '>= 0.10'} - - cross-spawn@7.0.3: - resolution: {integrity: sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==} - engines: {node: '>= 8'} - - css-color-keywords@1.0.0: - resolution: {integrity: sha512-FyyrDHZKEjXDpNJYvVsV960FiqQyXc/LlYmsxl2BcdMb2WPx0OGRVgTg55rPSyLSNMqP52R9r8geSp7apN3Ofg==} - engines: {node: '>=4'} - - css-to-react-native@3.2.0: - resolution: {integrity: sha512-e8RKaLXMOFii+02mOlqwjbD00KSEKqblnpO9e++1aXS1fPQOpS1YoqdVHBqPjHNoxeF2mimzVqawm2KCbEdtHQ==} - - csstype@3.1.3: - resolution: {integrity: sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==} - - custom-event-polyfill@1.0.7: - resolution: {integrity: sha512-TDDkd5DkaZxZFM8p+1I3yAlvM3rSr1wbrOliG4yJiwinMZN8z/iGL7BTlDkrJcYTmgUSb4ywVCc3ZaUtOtC76w==} - - custom-media-element@1.3.2: - resolution: {integrity: sha512-nDyMobZgoAVqz7mA8rsn7i1/6bjH6N9ab2Ge7LyyNxrvxAq7zQJPg8i3u2VH7wEB+Y1T1+C3/h1G774/D+ZLag==} - - damerau-levenshtein@1.0.8: - resolution: {integrity: sha512-sdQSFB7+llfUcQHUQO3+B8ERRj0Oa4w9POWMI/puGtuf7gFywGmkaLCElnudfTiKZV+NvHqL0ifzdrI8Ro7ESA==} - - data-view-buffer@1.0.1: - resolution: {integrity: sha512-0lht7OugA5x3iJLOWFhWK/5ehONdprk0ISXqVFn/NFrDu+cuc8iADFrGQz5BnRK7LLU3JmkbXSxaqX+/mXYtUA==} - engines: {node: '>= 0.4'} - - data-view-byte-length@1.0.1: - resolution: {integrity: sha512-4J7wRJD3ABAzr8wP+OcIcqq2dlUKp4DVflx++hs5h5ZKydWMI6/D/fAot+yh6g2tHh8fLFTvNOaVN357NvSrOQ==} - engines: {node: '>= 0.4'} - - data-view-byte-offset@1.0.0: - resolution: {integrity: sha512-t/Ygsytq+R995EJ5PZlD4Cu56sWa8InXySaViRzw9apusqsOO2bQP+SbYzAhR0pFKoB+43lYy8rWban9JSuXnA==} - engines: {node: '>= 0.4'} - - date-fns-tz@3.1.3: - resolution: {integrity: sha512-ZfbMu+nbzW0mEzC8VZrLiSWvUIaI3aRHeq33mTe7Y38UctKukgqPR4nTDwcwS4d64Gf8GghnVsroBuMY3eiTeA==} - peerDependencies: - date-fns: ^3.0.0 - - date-fns@3.6.0: - resolution: {integrity: sha512-fRHTG8g/Gif+kSh50gaGEdToemgfj74aRX3swtiouboip5JDLAyDE9F11nHMIcvOaXeOC6D7SpNhi7uFyB7Uww==} - - dayjs@1.11.13: - resolution: {integrity: sha512-oaMBel6gjolK862uaPQOVTA7q3TZhuSvuMQAAglQDOWYO9A91IrAOUJEyKVlqJlHE0vq5p5UXxzdPfMH/x6xNg==} - - debug@3.2.7: - resolution: {integrity: sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==} - peerDependencies: - supports-color: '*' - peerDependenciesMeta: - supports-color: - optional: true - - debug@4.3.6: - resolution: {integrity: sha512-O/09Bd4Z1fBrU4VzkhFqVgpPzaGbw6Sm9FEkBT1A/YBXQFGuuSxa1dN2nxgxS34JmKXqYx8CZAwEVoJFImUXIg==} - engines: {node: '>=6.0'} - peerDependencies: - supports-color: '*' - peerDependenciesMeta: - supports-color: - optional: true - - decompress-response@6.0.0: - resolution: {integrity: sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==} - engines: {node: '>=10'} - - deep-equal@2.2.3: - resolution: {integrity: sha512-ZIwpnevOurS8bpT4192sqAowWM76JDKSHYzMLty3BZGSswgq6pBaH3DhCSW5xVAZICZyKdOBPjwww5wfgT/6PA==} - engines: {node: '>= 0.4'} - - deep-extend@0.6.0: - resolution: {integrity: sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==} - engines: {node: '>=4.0.0'} - - deep-is@0.1.4: - resolution: {integrity: sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==} - - deepmerge@4.3.1: - resolution: {integrity: sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==} - engines: {node: '>=0.10.0'} - - define-data-property@1.1.4: - resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==} - engines: {node: '>= 0.4'} - - define-properties@1.2.1: - resolution: {integrity: sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==} - engines: {node: '>= 0.4'} - - detect-libc@2.0.3: - resolution: {integrity: sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw==} - engines: {node: '>=8'} - - dir-glob@3.0.1: - resolution: {integrity: sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==} - engines: {node: '>=8'} - - doctrine@2.1.0: - resolution: {integrity: sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==} - engines: {node: '>=0.10.0'} - - dotenv@16.4.5: - resolution: {integrity: sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==} - engines: {node: '>=12'} - - eastasianwidth@0.2.0: - resolution: {integrity: sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==} - - emoji-regex@8.0.0: - resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==} - - emoji-regex@9.2.2: - resolution: {integrity: sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==} - - end-of-stream@1.4.4: - resolution: {integrity: sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==} - - enhanced-resolve@5.17.1: - resolution: {integrity: sha512-LMHl3dXhTcfv8gM4kEzIUeTQ+7fpdA0l2tUf34BddXPkz2A5xJ5L/Pchd5BL6rdccM9QGvu0sWZzK1Z1t4wwyg==} - engines: {node: '>=10.13.0'} - - es-abstract@1.23.3: - resolution: {integrity: sha512-e+HfNH61Bj1X9/jLc5v1owaLYuHdeHHSQlkhCBiTK8rBvKaULl/beGMxwrMXjpYrv4pz22BlY570vVePA2ho4A==} - engines: {node: '>= 0.4'} - - es-define-property@1.0.0: - resolution: {integrity: sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==} - engines: {node: '>= 0.4'} - - es-errors@1.3.0: - resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==} - engines: {node: '>= 0.4'} - - es-get-iterator@1.1.3: - resolution: {integrity: sha512-sPZmqHBe6JIiTfN5q2pEi//TwxmAFHwj/XEuYjTuse78i8KxaqMTTzxPoFKuzRpDpTJ+0NAbpfenkmH2rePtuw==} - - es-iterator-helpers@1.0.19: - resolution: {integrity: sha512-zoMwbCcH5hwUkKJkT8kDIBZSz9I6mVG//+lDCinLCGov4+r7NIy0ld8o03M0cJxl2spVf6ESYVS6/gpIfq1FFw==} - engines: {node: '>= 0.4'} - - es-object-atoms@1.0.0: - resolution: {integrity: sha512-MZ4iQ6JwHOBQjahnjwaC1ZtIBH+2ohjamzAO3oaHcXYup7qxjF2fixyH+Q71voWHeOkI2q/TnJao/KfXYIZWbw==} - engines: {node: '>= 0.4'} - - es-set-tostringtag@2.0.3: - resolution: {integrity: sha512-3T8uNMC3OQTHkFUsFq8r/BwAXLHvU/9O9mE0fBc/MY5iq/8H7ncvO947LmYA6ldWw9Uh8Yhf25zu6n7nML5QWQ==} - engines: {node: '>= 0.4'} - - es-shim-unscopables@1.0.2: - resolution: {integrity: sha512-J3yBRXCzDu4ULnQwxyToo/OjdMx6akgVC7K6few0a7F/0wLtmKKN7I73AH5T2836UuXRqN7Qg+IIUw/+YJksRw==} - - es-to-primitive@1.2.1: - resolution: {integrity: sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==} - engines: {node: '>= 0.4'} - - escape-string-regexp@4.0.0: - resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} - engines: {node: '>=10'} - - eslint-config-next@14.2.7: - resolution: {integrity: sha512-ppmy+QdQ7qkuCHGDlPjWaoSbJvjGpWSBD4zEW8f1eWlxYXYpZK7QzBOer1EcHKT3uKhlY1JjUus9g7Kvv712rw==} - peerDependencies: - eslint: ^7.23.0 || ^8.0.0 - typescript: '>=3.3.1' - peerDependenciesMeta: - typescript: - optional: true - - eslint-import-resolver-node@0.3.9: - resolution: {integrity: sha512-WFj2isz22JahUv+B788TlO3N6zL3nNJGU8CcZbPZvVEkBPaJdCV4vy5wyghty5ROFbCRnm132v8BScu5/1BQ8g==} - - eslint-import-resolver-typescript@3.6.3: - resolution: {integrity: sha512-ud9aw4szY9cCT1EWWdGv1L1XR6hh2PaRWif0j2QjQ0pgTY/69iw+W0Z4qZv5wHahOl8isEr+k/JnyAqNQkLkIA==} - engines: {node: ^14.18.0 || >=16.0.0} - peerDependencies: - eslint: '*' - eslint-plugin-import: '*' - eslint-plugin-import-x: '*' - peerDependenciesMeta: - eslint-plugin-import: - optional: true - eslint-plugin-import-x: - optional: true - - eslint-module-utils@2.9.0: - resolution: {integrity: sha512-McVbYmwA3NEKwRQY5g4aWMdcZE5xZxV8i8l7CqJSrameuGSQJtSWaL/LxTEzSKKaCcOhlpDR8XEfYXWPrdo/ZQ==} - engines: {node: '>=4'} - peerDependencies: - '@typescript-eslint/parser': '*' - eslint: '*' - eslint-import-resolver-node: '*' - eslint-import-resolver-typescript: '*' - eslint-import-resolver-webpack: '*' - peerDependenciesMeta: - '@typescript-eslint/parser': - optional: true - eslint: - optional: true - eslint-import-resolver-node: - optional: true - eslint-import-resolver-typescript: - optional: true - eslint-import-resolver-webpack: - optional: true - - eslint-plugin-import@2.30.0: - resolution: {integrity: sha512-/mHNE9jINJfiD2EKkg1BKyPyUk4zdnT54YgbOgfjSakWT5oyX/qQLVNTkehyfpcMxZXMy1zyonZ2v7hZTX43Yw==} - engines: {node: '>=4'} - peerDependencies: - '@typescript-eslint/parser': '*' - eslint: ^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8 - peerDependenciesMeta: - '@typescript-eslint/parser': - optional: true - - eslint-plugin-jsx-a11y@6.9.0: - resolution: {integrity: sha512-nOFOCaJG2pYqORjK19lqPqxMO/JpvdCZdPtNdxY3kvom3jTvkAbOvQvD8wuD0G8BYR0IGAGYDlzqWJOh/ybn2g==} - engines: {node: '>=4.0'} - peerDependencies: - eslint: ^3 || ^4 || ^5 || ^6 || ^7 || ^8 - - eslint-plugin-react-hooks@4.6.2: - resolution: {integrity: sha512-QzliNJq4GinDBcD8gPB5v0wh6g8q3SUi6EFF0x8N/BL9PoVs0atuGc47ozMRyOWAKdwaZ5OnbOEa3WR+dSGKuQ==} - engines: {node: '>=10'} - peerDependencies: - eslint: ^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 - - eslint-plugin-react@7.35.1: - resolution: {integrity: sha512-B5ok2JgbaaWn/zXbKCGgKDNL2tsID3Pd/c/yvjcpsd9HQDwyYc/TQv3AZMmOvrJgCs3AnYNUHRCQEMMQAYJ7Yg==} - engines: {node: '>=4'} - peerDependencies: - eslint: ^3 || ^4 || ^5 || ^6 || ^7 || ^8 || ^9.7 - - eslint-scope@8.0.2: - resolution: {integrity: sha512-6E4xmrTw5wtxnLA5wYL3WDfhZ/1bUBGOXV0zQvVRDOtrR8D0p6W7fs3JweNYhwRYeGvd/1CKX2se0/2s7Q/nJA==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - - eslint-visitor-keys@3.4.3: - resolution: {integrity: sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==} - engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} - - eslint-visitor-keys@4.0.0: - resolution: {integrity: sha512-OtIRv/2GyiF6o/d8K7MYKKbXrOUBIK6SfkIRM4Z0dY3w+LiQ0vy3F57m0Z71bjbyeiWFiHJ8brqnmE6H6/jEuw==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - - eslint@9.9.1: - resolution: {integrity: sha512-dHvhrbfr4xFQ9/dq+jcVneZMyRYLjggWjk6RVsIiHsP8Rz6yZ8LvZ//iU4TrZF+SXWG+JkNF2OyiZRvzgRDqMg==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - hasBin: true - peerDependencies: - jiti: '*' - peerDependenciesMeta: - jiti: - optional: true - - espree@10.1.0: - resolution: {integrity: sha512-M1M6CpiE6ffoigIOWYO9UDP8TMUw9kqb21tf+08IgDYjCsOvCuDt4jQcZmoYxx+w7zlKw9/N0KXfto+I8/FrXA==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - - esprima@4.0.1: - resolution: {integrity: sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==} - engines: {node: '>=4'} - hasBin: true - - esquery@1.6.0: - resolution: {integrity: sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==} - engines: {node: '>=0.10'} - - esrecurse@4.3.0: - resolution: {integrity: sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==} - engines: {node: '>=4.0'} - - estraverse@5.3.0: - resolution: {integrity: sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==} - engines: {node: '>=4.0'} - - esutils@2.0.3: - resolution: {integrity: sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==} - engines: {node: '>=0.10.0'} - - eventemitter3@5.0.1: - resolution: {integrity: sha512-GWkBvjiSZK87ELrYOSESUYeVIc9mvLLf/nXalMOS5dYrgZq9o5OVkbZAVM06CVxYsCwH9BDZFPlQTlPA1j4ahA==} - - exifr@7.1.3: - resolution: {integrity: sha512-g/aje2noHivrRSLbAUtBPWFbxKdKhgj/xr1vATDdUXPOFYJlQ62Ft0oy+72V6XLIpDJfHs6gXLbBLAolqOXYRw==} - - expand-template@2.0.3: - resolution: {integrity: sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg==} - engines: {node: '>=6'} - - extend-shallow@2.0.1: - resolution: {integrity: sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==} - engines: {node: '>=0.10.0'} - - fast-deep-equal@3.1.3: - resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==} - - fast-glob@3.3.2: - resolution: {integrity: sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==} - engines: {node: '>=8.6.0'} - - fast-json-stable-stringify@2.1.0: - resolution: {integrity: sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==} - - fast-levenshtein@2.0.6: - resolution: {integrity: sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==} - - fastq@1.17.1: - resolution: {integrity: sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==} - - feed@4.2.2: - resolution: {integrity: sha512-u5/sxGfiMfZNtJ3OvQpXcvotFpYkL0n9u9mM2vkui2nGo8b4wvDkJ8gAkYqbA8QpGyFCv3RK0Z+Iv+9veCS9bQ==} - engines: {node: '>=0.4.0'} - - file-entry-cache@8.0.0: - resolution: {integrity: sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==} - engines: {node: '>=16.0.0'} - - fill-range@7.1.1: - resolution: {integrity: sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==} - engines: {node: '>=8'} - - find-up@5.0.0: - resolution: {integrity: sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==} - engines: {node: '>=10'} - - flat-cache@4.0.1: - resolution: {integrity: sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==} - engines: {node: '>=16'} - - flatted@3.3.1: - resolution: {integrity: sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==} - - for-each@0.3.3: - resolution: {integrity: sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==} - - foreground-child@3.3.0: - resolution: {integrity: sha512-Ld2g8rrAyMYFXBhEqMz8ZAHBi4J4uS1i/CxGMDnjyFWddMXLVcDp051DZfu+t7+ab7Wv6SMqpWmyFIj5UbfFvg==} - engines: {node: '>=14'} - - fs-constants@1.0.0: - resolution: {integrity: sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==} - - fsevents@2.3.3: - resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} - engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} - os: [darwin] - - function-bind@1.1.2: - resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} - - function.prototype.name@1.1.6: - resolution: {integrity: sha512-Z5kx79swU5P27WEayXM1tBi5Ze/lbIyiNgU3qyXUOf9b2rgXYyF9Dy9Cx+IQv/Lc8WCG6L82zwUPpSS9hGehIg==} - engines: {node: '>= 0.4'} - - functions-have-names@1.2.3: - resolution: {integrity: sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==} - - get-intrinsic@1.2.4: - resolution: {integrity: sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==} - engines: {node: '>= 0.4'} - - get-symbol-description@1.0.2: - resolution: {integrity: sha512-g0QYk1dZBxGwk+Ngc+ltRH2IBp2f7zBkBMBJZCDerh6EhlhSR6+9irMCuT/09zD6qkarHUSn529sK/yL4S27mg==} - engines: {node: '>= 0.4'} - - get-tsconfig@4.8.0: - resolution: {integrity: sha512-Pgba6TExTZ0FJAn1qkJAjIeKoDJ3CsI2ChuLohJnZl/tTU8MVrq3b+2t5UOPfRa4RMsorClBjJALkJUMjG1PAw==} - - github-from-package@0.0.0: - resolution: {integrity: sha512-SyHy3T1v2NUXn29OsWdxmK6RwHD+vkj3v8en8AOBZ1wBQ/hCAQ5bAQTD02kW4W9tUp/3Qh6J8r9EvntiyCmOOw==} - - glob-parent@5.1.2: - resolution: {integrity: sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==} - engines: {node: '>= 6'} - - glob-parent@6.0.2: - resolution: {integrity: sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==} - engines: {node: '>=10.13.0'} - - glob@10.3.10: - resolution: {integrity: sha512-fa46+tv1Ak0UPK1TOy/pZrIybNNt4HCv7SDzwyfiOZkvZLEbjsZkJBPtDHVshZjbecAoAGSC20MjLDG/qr679g==} - engines: {node: '>=16 || 14 >=14.17'} - hasBin: true - - globals@14.0.0: - resolution: {integrity: sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==} - engines: {node: '>=18'} - - globalthis@1.0.4: - resolution: {integrity: sha512-DpLKbNU4WylpxJykQujfCcwYWiV/Jhm50Goo0wrVILAv5jOr9d+H+UR3PhSCD2rCCEIg0uc+G+muBTwD54JhDQ==} - engines: {node: '>= 0.4'} - - globby@11.1.0: - resolution: {integrity: sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==} - engines: {node: '>=10'} - - gopd@1.0.1: - resolution: {integrity: sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==} - - graceful-fs@4.2.11: - resolution: {integrity: sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==} - - gray-matter@4.0.3: - resolution: {integrity: sha512-5v6yZd4JK3eMI3FqqCouswVqwugaA9r4dNZB1wwcmrD02QkV5H0y7XBQW8QwQqEaZY1pM9aqORSORhJRdNK44Q==} - engines: {node: '>=6.0'} - - has-bigints@1.0.2: - resolution: {integrity: sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ==} - - has-flag@4.0.0: - resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} - engines: {node: '>=8'} - - has-property-descriptors@1.0.2: - resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==} - - has-proto@1.0.3: - resolution: {integrity: sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==} - engines: {node: '>= 0.4'} - - has-symbols@1.0.3: - resolution: {integrity: sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==} - engines: {node: '>= 0.4'} - - has-tostringtag@1.0.2: - resolution: {integrity: sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==} - engines: {node: '>= 0.4'} - - hasown@2.0.2: - resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} - engines: {node: '>= 0.4'} - - hls.js@1.5.15: - resolution: {integrity: sha512-6cD7xN6bycBHaXz2WyPIaHn/iXFizE5au2yvY5q9aC4wfihxAr16C9fUy4nxh2a3wOw0fEgLRa9dN6wsYjlpNg==} - - ieee754@1.2.1: - resolution: {integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==} - - ignore@5.3.2: - resolution: {integrity: sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==} - engines: {node: '>= 4'} - - immutable@4.3.7: - resolution: {integrity: sha512-1hqclzwYwjRDFLjcFxOM5AYkkG0rpFPpr1RLPMEuGczoS7YA8gLhy8SWXYRAA/XwfEHpfo3cw5JGioS32fnMRw==} - - import-fresh@3.3.0: - resolution: {integrity: sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==} - engines: {node: '>=6'} - - imurmurhash@0.1.4: - resolution: {integrity: sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==} - engines: {node: '>=0.8.19'} - - inherits@2.0.4: - resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} - - ini@1.3.8: - resolution: {integrity: sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==} - - internal-slot@1.0.7: - resolution: {integrity: sha512-NGnrKwXzSms2qUUih/ILZ5JBqNTSa1+ZmP6flaIp6KmSElgE9qdndzS3cqjrDovwFdmwsGsLdeFgB6suw+1e9g==} - engines: {node: '>= 0.4'} - - is-arguments@1.1.1: - resolution: {integrity: sha512-8Q7EARjzEnKpt/PCD7e1cgUS0a6X8u5tdSiMqXhojOdoV9TsMsiO+9VLC5vAmO8N7/GmXn7yjR8qnA6bVAEzfA==} - engines: {node: '>= 0.4'} - - is-array-buffer@3.0.4: - resolution: {integrity: sha512-wcjaerHw0ydZwfhiKbXJWLDY8A7yV7KhjQOpb83hGgGfId/aQa4TOvwyzn2PuswW2gPCYEL/nEAiSVpdOj1lXw==} - engines: {node: '>= 0.4'} - - is-arrayish@0.3.2: - resolution: {integrity: sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==} - - is-async-function@2.0.0: - resolution: {integrity: sha512-Y1JXKrfykRJGdlDwdKlLpLyMIiWqWvuSd17TvZk68PLAOGOoF4Xyav1z0Xhoi+gCYjZVeC5SI+hYFOfvXmGRCA==} - engines: {node: '>= 0.4'} - - is-bigint@1.0.4: - resolution: {integrity: sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==} - - is-binary-path@2.1.0: - resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==} - engines: {node: '>=8'} - - is-boolean-object@1.1.2: - resolution: {integrity: sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==} - engines: {node: '>= 0.4'} - - is-bun-module@1.1.0: - resolution: {integrity: sha512-4mTAVPlrXpaN3jtF0lsnPCMGnq4+qZjVIKq0HCpfcqf8OC1SM5oATCIAPM5V5FN05qp2NNnFndphmdZS9CV3hA==} - - is-callable@1.2.7: - resolution: {integrity: sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==} - engines: {node: '>= 0.4'} - - is-core-module@2.15.1: - resolution: {integrity: sha512-z0vtXSwucUJtANQWldhbtbt7BnL0vxiFjIdDLAatwhDYty2bad6s+rijD6Ri4YuYJubLzIJLUidCh09e1djEVQ==} - engines: {node: '>= 0.4'} - - is-data-view@1.0.1: - resolution: {integrity: sha512-AHkaJrsUVW6wq6JS8y3JnM/GJF/9cf+k20+iDzlSaJrinEo5+7vRiteOSwBhHRiAyQATN1AmY4hwzxJKPmYf+w==} - engines: {node: '>= 0.4'} - - is-date-object@1.0.5: - resolution: {integrity: sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==} - engines: {node: '>= 0.4'} - - is-extendable@0.1.1: - resolution: {integrity: sha512-5BMULNob1vgFX6EjQw5izWDxrecWK9AM72rugNr0TFldMOi0fj6Jk+zeKIt0xGj4cEfQIJth4w3OKWOJ4f+AFw==} - engines: {node: '>=0.10.0'} - - is-extglob@2.1.1: - resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} - engines: {node: '>=0.10.0'} - - is-finalizationregistry@1.0.2: - resolution: {integrity: sha512-0by5vtUJs8iFQb5TYUHHPudOR+qXYIMKtiUzvLIZITZUjknFmziyBJuLhVRc+Ds0dREFlskDNJKYIdIzu/9pfw==} - - is-fullwidth-code-point@3.0.0: - resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==} - engines: {node: '>=8'} - - is-generator-function@1.0.10: - resolution: {integrity: sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==} - engines: {node: '>= 0.4'} - - is-glob@4.0.3: - resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==} - engines: {node: '>=0.10.0'} - - is-map@2.0.3: - resolution: {integrity: sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw==} - engines: {node: '>= 0.4'} - - is-negative-zero@2.0.3: - resolution: {integrity: sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==} - engines: {node: '>= 0.4'} - - is-network-error@1.1.0: - resolution: {integrity: sha512-tUdRRAnhT+OtCZR/LxZelH/C7QtjtFrTu5tXCA8pl55eTUElUHT+GPYV8MBMBvea/j+NxQqVt3LbWMRir7Gx9g==} - engines: {node: '>=16'} - - is-number-object@1.0.7: - resolution: {integrity: sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==} - engines: {node: '>= 0.4'} - - is-number@7.0.0: - resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==} - engines: {node: '>=0.12.0'} - - is-path-inside@3.0.3: - resolution: {integrity: sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==} - engines: {node: '>=8'} - - is-regex@1.1.4: - resolution: {integrity: sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==} - engines: {node: '>= 0.4'} - - is-set@2.0.3: - resolution: {integrity: sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg==} - engines: {node: '>= 0.4'} - - is-shared-array-buffer@1.0.3: - resolution: {integrity: sha512-nA2hv5XIhLR3uVzDDfCIknerhx8XUKnstuOERPNNIinXG7v9u+ohXF67vxm4TPTEPU6lm61ZkwP3c9PCB97rhg==} - engines: {node: '>= 0.4'} - - is-string@1.0.7: - resolution: {integrity: sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==} - engines: {node: '>= 0.4'} - - is-symbol@1.0.4: - resolution: {integrity: sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==} - engines: {node: '>= 0.4'} - - is-typed-array@1.1.13: - resolution: {integrity: sha512-uZ25/bUAlUY5fR4OKT4rZQEBrzQWYV9ZJYGGsUmEJ6thodVJ1HX64ePQ6Z0qPWP+m+Uq6e9UugrE38jeYsDSMw==} - engines: {node: '>= 0.4'} - - is-weakmap@2.0.2: - resolution: {integrity: sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w==} - engines: {node: '>= 0.4'} - - is-weakref@1.0.2: - resolution: {integrity: sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==} - - is-weakset@2.0.3: - resolution: {integrity: sha512-LvIm3/KWzS9oRFHugab7d+M/GcBXuXX5xZkzPmN+NxihdQlZUQ4dWuSV1xR/sq6upL1TJEDrfBgRepHFdBtSNQ==} - engines: {node: '>= 0.4'} - - isarray@2.0.5: - resolution: {integrity: sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==} - - isexe@2.0.0: - resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} - - iterator.prototype@1.1.2: - resolution: {integrity: sha512-DR33HMMr8EzwuRL8Y9D3u2BMj8+RqSE850jfGu59kS7tbmPLzGkZmVSfyCFSDxuZiEY6Rzt3T2NA/qU+NwVj1w==} - - jackspeak@2.3.6: - resolution: {integrity: sha512-N3yCS/NegsOBokc8GAdM8UcmfsKiSS8cipheD/nivzr700H+nsMOxJjQnvwOcRYVuFkdH0wGUvW2WbXGmrZGbQ==} - engines: {node: '>=14'} - - jose@4.15.9: - resolution: {integrity: sha512-1vUQX+IdDMVPj4k8kOxgUqlcK518yluMuGZwqlr44FS1ppZB/5GWh4rZG89erpOBOJjU/OBsnCVFfapsRz6nEA==} - - jose@5.9.6: - resolution: {integrity: sha512-AMlnetc9+CV9asI19zHmrgS/WYsWUwCn2R7RzlbJWD7F9eWYUTGyBmU9o6PxngtLGOiDGPRu+Uc4fhKzbpteZQ==} - - js-base64@3.7.7: - resolution: {integrity: sha512-7rCnleh0z2CkXhH67J8K1Ytz0b2Y+yxTPL+/KOJoa20hfnVQ/3/T6W/KflYI4bRHRagNeXeU2bkNGI3v1oS/lw==} - - js-tokens@4.0.0: - resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} - - js-yaml@3.14.1: - resolution: {integrity: sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==} - hasBin: true - - js-yaml@4.1.0: - resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==} - hasBin: true - - json-buffer@3.0.1: - resolution: {integrity: sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==} - - json-schema-traverse@0.4.1: - resolution: {integrity: sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==} - - json-stable-stringify-without-jsonify@1.0.1: - resolution: {integrity: sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==} - - json5@1.0.2: - resolution: {integrity: sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==} - hasBin: true - - jsx-ast-utils@3.3.5: - resolution: {integrity: sha512-ZZow9HBI5O6EPgSJLUb8n2NKgmVWTwCvHGwFuJlMjvLFqlGG6pjirPhtdsseaLZjSibD8eegzmYpUZwoIlj2cQ==} - engines: {node: '>=4.0'} - - jwt-decode@4.0.0: - resolution: {integrity: sha512-+KJGIyHgkGuIq3IEBNftfhW/LfWhXUIY6OmyVWjliu5KH1y0fw7VQ8YndE2O4qZdMSd9SqbnC8GOcZEy0Om7sA==} - engines: {node: '>=18'} - - keyv@4.5.4: - resolution: {integrity: sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==} - - kind-of@6.0.3: - resolution: {integrity: sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==} - engines: {node: '>=0.10.0'} - - language-subtag-registry@0.3.23: - resolution: {integrity: sha512-0K65Lea881pHotoGEa5gDlMxt3pctLi2RplBb7Ezh4rRdLEOtgi7n4EwK9lamnUCkKBqaeKRVebTq6BAxSkpXQ==} - - language-tags@1.0.9: - resolution: {integrity: sha512-MbjN408fEndfiQXbFQ1vnd+1NoLDsnQW41410oQBXiyXDMYH5z505juWa4KUE1LqxRC7DgOgZDbKLxHIwm27hA==} - engines: {node: '>=0.10'} - - levn@0.4.1: - resolution: {integrity: sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==} - engines: {node: '>= 0.8.0'} - - loadjs@4.3.0: - resolution: {integrity: sha512-vNX4ZZLJBeDEOBvdr2v/F+0aN5oMuPu7JTqrMwp+DtgK+AryOlpy6Xtm2/HpNr+azEa828oQjOtWsB6iDtSfSQ==} - - locate-path@6.0.0: - resolution: {integrity: sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==} - engines: {node: '>=10'} - - lodash.merge@4.6.2: - resolution: {integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==} - - lodash@4.17.21: - resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==} - - loose-envify@1.4.0: - resolution: {integrity: sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==} - hasBin: true - - lru-cache@10.4.3: - resolution: {integrity: sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==} - - lru-cache@6.0.0: - resolution: {integrity: sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==} - engines: {node: '>=10'} - - lunarphase-js@2.0.3: - resolution: {integrity: sha512-zTr/UWbxQ1lyKgaBnyJ/DvKCPONhZQcwmQ8PuF2g2QwdkF8JkhgPe8QlDroxSjZrfpg/9x6jQ6lFtBkSXXf1oQ==} - - map-obj@4.3.0: - resolution: {integrity: sha512-hdN1wVrZbb29eBGiGjJbeP8JbKjq1urkHJ/LIP/NY48MZ1QVXUsQBV1G1zvYFHn1XE06cwjBsOI2K3Ulnj1YXQ==} - engines: {node: '>=8'} - - media-chrome@3.2.5: - resolution: {integrity: sha512-tTsgS7x77Bn4p/wca/Si/7A+Q3z9DzKq0SOkroQvrNMXBVyQasMayDcsKg5Ur5NGsymZfttnJi7tXvVr/tPj8g==} - - media-tracks@0.3.3: - resolution: {integrity: sha512-9P2FuUHnZZ3iji+2RQk7Zkh5AmZTnOG5fODACnjhCVveX1McY3jmCRHofIEI+yTBqplz7LXy48c7fQ3Uigp88w==} - - memoize-one@6.0.0: - resolution: {integrity: sha512-rkpe71W0N0c0Xz6QD0eJETuWAJGnJ9afsl1srmwPrI+yBCkge5EycXXbYRyvL29zZVUWQCY7InPRCv3GDXuZNw==} - - merge2@1.4.1: - resolution: {integrity: sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==} - engines: {node: '>= 8'} - - micromatch@4.0.8: - resolution: {integrity: sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==} - engines: {node: '>=8.6'} - - mime-match@1.0.2: - resolution: {integrity: sha512-VXp/ugGDVh3eCLOBCiHZMYWQaTNUHv2IJrut+yXA6+JbLPXHglHwfS/5A5L0ll+jkCY7fIzRJcH6OIunF+c6Cg==} - - mimic-response@3.1.0: - resolution: {integrity: sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==} - engines: {node: '>=10'} - - minimatch@3.1.2: - resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==} - - minimatch@9.0.3: - resolution: {integrity: sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==} - engines: {node: '>=16 || 14 >=14.17'} - - minimatch@9.0.5: - resolution: {integrity: sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==} - engines: {node: '>=16 || 14 >=14.17'} - - minimist@1.2.8: - resolution: {integrity: sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==} - - minipass@7.1.2: - resolution: {integrity: sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==} - engines: {node: '>=16 || 14 >=14.17'} - - mkdirp-classic@0.5.3: - resolution: {integrity: sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==} - - ms@2.1.2: - resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==} - - ms@2.1.3: - resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} - - multiformats@13.2.2: - resolution: {integrity: sha512-RWI+nyf0q64vyOxL8LbKtjJMki0sogRL/8axvklNtiTM0iFCVtHwME9w6+0P1/v4dQvsIg8A45oT3ka1t/M/+A==} - - mux-embed@5.2.1: - resolution: {integrity: sha512-NukHw91xeEVDBeXVDBpi2BvXNix7gSuvdtyvOph5yR/ROn1hHbTlcYWoKQyCyJX9frsF00UROEul+S8wPzU3aQ==} - - namespace-emitter@2.0.1: - resolution: {integrity: sha512-N/sMKHniSDJBjfrkbS/tpkPj4RAbvW3mr8UAzvlMHyun93XEm83IAvhWtJVHo+RHn/oO8Job5YN4b+wRjSVp5g==} - - nanoid@3.3.7: - resolution: {integrity: sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==} - engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1} - hasBin: true - - nanoid@5.0.7: - resolution: {integrity: sha512-oLxFY2gd2IqnjcYyOXD8XGCftpGtZP2AbHbOkthDkvRywH5ayNtPVy9YlOPcHckXzbLTCHpkb7FB+yuxKV13pQ==} - engines: {node: ^18 || >=20} - hasBin: true - - napi-build-utils@1.0.2: - resolution: {integrity: sha512-ONmRUqK7zj7DWX0D9ADe03wbwOBZxNAfF20PlGfCWQcD3+/MakShIHrMqx9YwPTfxDdF1zLeL+RGZiR9kGMLdg==} - - natural-compare@1.4.0: - resolution: {integrity: sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==} - - next-auth@4.24.10: - resolution: {integrity: sha512-8NGqiRO1GXBcVfV8tbbGcUgQkAGsX4GRzzXXea4lDikAsJtD5KiEY34bfhUOjHLvr6rT6afpcxw2H8EZqOV6aQ==} - peerDependencies: - '@auth/core': 0.34.2 - next: ^12.2.5 || ^13 || ^14 || ^15 - nodemailer: ^6.6.5 - react: ^17.0.2 || ^18 - react-dom: ^17.0.2 || ^18 - peerDependenciesMeta: - '@auth/core': - optional: true - nodemailer: - optional: true - - next-goatcounter@1.0.5: - resolution: {integrity: sha512-t2khpwRk1sDeZwth8wg0SF6MKAQ4lrUJ0qKFETmeEkIudcrc6hbMSvmhiVc8XrpTl13S4F2wWRH248FSyY8Gvw==} - peerDependencies: - next: '>14.0.0' - react: '>18.0.0' - react-dom: '>18.0.0' - - next@14.2.7: - resolution: {integrity: sha512-4Qy2aK0LwH4eQiSvQWyKuC7JXE13bIopEQesWE0c/P3uuNRnZCQanI0vsrMLmUQJLAto+A+/8+sve2hd+BQuOQ==} - engines: {node: '>=18.17.0'} - hasBin: true - peerDependencies: - '@opentelemetry/api': ^1.1.0 - '@playwright/test': ^1.41.2 - react: ^18.2.0 - react-dom: ^18.2.0 - sass: ^1.3.0 - peerDependenciesMeta: - '@opentelemetry/api': - optional: true - '@playwright/test': - optional: true - sass: - optional: true - - nextjs-cors@2.2.0: - resolution: {integrity: sha512-FZu/A+L59J4POJNqwXYyCPDvsLDeu5HjSBvytzS6lsrJeDz5cmnH45zV+VoNic0hjaeER9xGaiIjZIWzEHnxQg==} - peerDependencies: - next: ^8.1.1-canary.54 || ^9.0.0 || ^10.0.0-0 || ^11.0.0 || ^12.0.0 || ^13.0.0 || ^14.0.0 - - nextjs-toploader@3.6.15: - resolution: {integrity: sha512-DMxO0BiT+NbGU7E9A8U2+0e7rRbG9qozbk19DFXhKRc9wuhgTPR/u8zr2TevsbFtVxDb9mf8YHFz+oBgjMC+HA==} - peerDependencies: - next: '>= 6.0.0' - react: '>= 16.0.0' - react-dom: '>= 16.0.0' - - node-abi@3.67.0: - resolution: {integrity: sha512-bLn/fU/ALVBE9wj+p4Y21ZJWYFjUXLXPi/IewyLZkx3ApxKDNBWCKdReeKOtD8dWpOdDCeMyLh6ZewzcLsG2Nw==} - engines: {node: '>=10'} - - node-addon-api@5.1.0: - resolution: {integrity: sha512-eh0GgfEkpnoWDq+VY8OyvYhFEzBk6jIYbRKdIlyTiAXIVJ8PyBaKb0rp7oDtoddbdoHWhq8wwr+XZ81F1rpNdA==} - - nodemailer@6.9.16: - resolution: {integrity: sha512-psAuZdTIRN08HKVd/E8ObdV6NO7NTBY3KsC30F7M4H1OnmLCUNaS56FpYxyb26zWLSyYF9Ozch9KYHhHegsiOQ==} - engines: {node: '>=6.0.0'} - - normalize-path@3.0.0: - resolution: {integrity: sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==} - engines: {node: '>=0.10.0'} - - nprogress@0.2.0: - resolution: {integrity: sha512-I19aIingLgR1fmhftnbWWO3dXc0hSxqHQHQb3H8m+K3TnEn/iSeTZZOyvKXWqQESMwuUVnatlCnZdLBZZt2VSA==} - - oauth@0.9.15: - resolution: {integrity: sha512-a5ERWK1kh38ExDEfoO6qUHJb32rd7aYmPHuyCu3Fta/cnICvYmgd2uhuKXvPD+PXB+gCEYYEaQdIRAjCOwAKNA==} - - object-assign@4.1.1: - resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} - engines: {node: '>=0.10.0'} - - object-hash@2.2.0: - resolution: {integrity: sha512-gScRMn0bS5fH+IuwyIFgnh9zBdo4DV+6GhygmWM9HyNJSgS0hScp1f5vjtm7oIIOiT9trXrShAkLFSc2IqKNgw==} - engines: {node: '>= 6'} - - object-inspect@1.13.2: - resolution: {integrity: sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==} - engines: {node: '>= 0.4'} - - object-is@1.1.6: - resolution: {integrity: sha512-F8cZ+KfGlSGi09lJT7/Nd6KJZ9ygtvYC0/UYYLI9nmQKLMnydpB9yvbv9K1uSkEu7FU9vYPmVwLg328tX+ot3Q==} - engines: {node: '>= 0.4'} - - object-keys@1.1.1: - resolution: {integrity: sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==} - engines: {node: '>= 0.4'} - - object.assign@4.1.5: - resolution: {integrity: sha512-byy+U7gp+FVwmyzKPYhW2h5l3crpmGsxl7X2s8y43IgxvG4g3QZ6CffDtsNQy1WsmZpQbO+ybo0AlW7TY6DcBQ==} - engines: {node: '>= 0.4'} - - object.entries@1.1.8: - resolution: {integrity: sha512-cmopxi8VwRIAw/fkijJohSfpef5PdN0pMQJN6VC/ZKvn0LIknWD8KtgY6KlQdEc4tIjcQ3HxSMmnvtzIscdaYQ==} - engines: {node: '>= 0.4'} - - object.fromentries@2.0.8: - resolution: {integrity: sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==} - engines: {node: '>= 0.4'} - - object.groupby@1.0.3: - resolution: {integrity: sha512-+Lhy3TQTuzXI5hevh8sBGqbmurHbbIjAi0Z4S63nthVLmLxfbj4T54a4CfZrXIrt9iP4mVAPYMo/v99taj3wjQ==} - engines: {node: '>= 0.4'} - - object.values@1.2.0: - resolution: {integrity: sha512-yBYjY9QX2hnRmZHAjG/f13MzmBzxzYgQhFrke06TTyKY5zSTEqkOeukBzIdVA3j3ulu8Qa3MbVFShV7T2RmGtQ==} - engines: {node: '>= 0.4'} - - oidc-token-hash@5.0.3: - resolution: {integrity: sha512-IF4PcGgzAr6XXSff26Sk/+P4KZFJVuHAJZj3wgO3vX2bMdNVp/QXTP3P7CEm9V1IdG8lDLY3HhiqpsE/nOwpPw==} - engines: {node: ^10.13.0 || >=12.0.0} - - once@1.4.0: - resolution: {integrity: sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==} - - openid-client@5.7.1: - resolution: {integrity: sha512-jDBPgSVfTnkIh71Hg9pRvtJc6wTwqjRkN88+gCFtYWrlP4Yx2Dsrow8uPi3qLr/aeymPF3o2+dS+wOpglK04ew==} - - optionator@0.9.4: - resolution: {integrity: sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==} - engines: {node: '>= 0.8.0'} - - p-limit@3.1.0: - resolution: {integrity: sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==} - engines: {node: '>=10'} - - p-locate@5.0.0: - resolution: {integrity: sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==} - engines: {node: '>=10'} - - p-queue@8.0.1: - resolution: {integrity: sha512-NXzu9aQJTAzbBqOt2hwsR63ea7yvxJc0PwN/zobNAudYfb1B7R08SzB4TsLeSbUCuG467NhnoT0oO6w1qRO+BA==} - engines: {node: '>=18'} - - p-retry@6.2.0: - resolution: {integrity: sha512-JA6nkq6hKyWLLasXQXUrO4z8BUZGUt/LjlJxx8Gb2+2ntodU/SS63YZ8b0LUTbQ8ZB9iwOfhEPhg4ykKnn2KsA==} - engines: {node: '>=16.17'} - - p-timeout@6.1.2: - resolution: {integrity: sha512-UbD77BuZ9Bc9aABo74gfXhNvzC9Tx7SxtHSh1fxvx3jTLLYvmVhiQZZrJzqqU0jKbN32kb5VOKiLEQI/3bIjgQ==} - engines: {node: '>=14.16'} - - parent-module@1.0.1: - resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} - engines: {node: '>=6'} - - path-exists@4.0.0: - resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==} - engines: {node: '>=8'} - - path-key@3.1.1: - resolution: {integrity: sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==} - engines: {node: '>=8'} - - path-parse@1.0.7: - resolution: {integrity: sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==} - - path-scurry@1.11.1: - resolution: {integrity: sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==} - engines: {node: '>=16 || 14 >=14.18'} - - path-type@4.0.0: - resolution: {integrity: sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==} - engines: {node: '>=8'} - - picocolors@1.1.0: - resolution: {integrity: sha512-TQ92mBOW0l3LeMeyLV6mzy/kWr8lkd/hp3mTg7wYK7zJhuBStmGMBG0BdeDZS/dZx1IukaX6Bk11zcln25o1Aw==} - - picomatch@2.3.1: - resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==} - engines: {node: '>=8.6'} - - plyr@3.7.8: - resolution: {integrity: sha512-yG/EHDobwbB/uP+4Bm6eUpJ93f8xxHjjk2dYcD1Oqpe1EcuQl5tzzw9Oq+uVAzd2lkM11qZfydSiyIpiB8pgdA==} - - possible-typed-array-names@1.0.0: - resolution: {integrity: sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==} - engines: {node: '>= 0.4'} - - postcss-value-parser@4.2.0: - resolution: {integrity: sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==} - - postcss@8.4.31: - resolution: {integrity: sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ==} - engines: {node: ^10 || ^12 || >=14} - - postcss@8.4.38: - resolution: {integrity: sha512-Wglpdk03BSfXkHoQa3b/oulrotAkwrlLDRSOb9D0bN86FdRyE9lppSp33aHNPgBa0JKCoB+drFLZkQoRRYae5A==} - engines: {node: ^10 || ^12 || >=14} - - preact-render-to-string@5.2.3: - resolution: {integrity: sha512-aPDxUn5o3GhWdtJtW0svRC2SS/l8D9MAgo2+AWml+BhDImb27ALf04Q2d+AHqUUOc6RdSXFIBVa2gxzgMKgtZA==} - peerDependencies: - preact: '>=10' - - preact@10.23.2: - resolution: {integrity: sha512-kKYfePf9rzKnxOAKDpsWhg/ysrHPqT+yQ7UW4JjdnqjFIeNUnNcEJvhuA8fDenxAGWzUqtd51DfVg7xp/8T9NA==} - - prebuild-install@7.1.2: - resolution: {integrity: sha512-UnNke3IQb6sgarcZIDU3gbMeTp/9SSU1DAIkil7PrqG1vZlBtY5msYccSKSHDqa3hNg436IXK+SNImReuA1wEQ==} - engines: {node: '>=10'} - hasBin: true - - prelude-ls@1.2.1: - resolution: {integrity: sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==} - engines: {node: '>= 0.8.0'} - - pretty-format@3.8.0: - resolution: {integrity: sha512-WuxUnVtlWL1OfZFQFuqvnvs6MiAGk9UNsBostyBOB0Is9wb5uRESevA6rnl/rkksXaGX3GzZhPup5d6Vp1nFew==} - - prop-types@15.8.1: - resolution: {integrity: sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==} - - property-expr@2.0.6: - resolution: {integrity: sha512-SVtmxhRE/CGkn3eZY1T6pC8Nln6Fr/lu1mKSgRud0eC73whjGfoAogbn78LkD8aFL0zz3bAFerKSnOl7NlErBA==} - - pump@3.0.0: - resolution: {integrity: sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==} - - punycode@2.3.1: - resolution: {integrity: sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==} - engines: {node: '>=6'} - - qs@6.13.0: - resolution: {integrity: sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==} - engines: {node: '>=0.6'} - - queue-microtask@1.2.3: - resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==} - - quick-lru@5.1.1: - resolution: {integrity: sha512-WuyALRjWPDGtt/wzJiadO5AXY+8hZ80hVpe6MyivgraREW751X3SbhRvG3eLKOYN+8VEvqLcf3wdnt44Z4S4SA==} - engines: {node: '>=10'} - - rangetouch@2.0.1: - resolution: {integrity: sha512-sln+pNSc8NGaHoLzwNBssFSf/rSYkqeBXzX1AtJlkJiUaVSJSbRAWJk+4omsXkN+EJalzkZhWQ3th1m0FpR5xA==} - - rc@1.2.8: - resolution: {integrity: sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==} - hasBin: true - - react-data-table-component@7.6.2: - resolution: {integrity: sha512-nHe7040fmtrJyQr/ieGrTfV0jBflYGK4sLokC6/AFOv3ThjmA9WzKz8Z8/2wMxzRqLU+Rn0CVFg+8+frKLepWQ==} - peerDependencies: - react: '>= 16.8.3' - styled-components: '>= 5.0.0' - - react-dom@18.3.1: - resolution: {integrity: sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==} - peerDependencies: - react: ^18.3.1 - - react-hook-form@7.53.0: - resolution: {integrity: sha512-M1n3HhqCww6S2hxLxciEXy2oISPnAzxY7gvwVPrtlczTM/1dDadXgUxDpHMrMTblDOcm/AXtXxHwZ3jpg1mqKQ==} - engines: {node: '>=18.0.0'} - peerDependencies: - react: ^16.8.0 || ^17 || ^18 || ^19 - - react-is@16.13.1: - resolution: {integrity: sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==} - - react-loading-skeleton@3.4.0: - resolution: {integrity: sha512-1oJEBc9+wn7BbkQQk7YodlYEIjgeR+GrRjD+QXkVjwZN7LGIcAFHrx4NhT7UHGBxNY1+zax3c+Fo6XQM4R7CgA==} - peerDependencies: - react: '>=16.8.0' - - react-toastify@10.0.5: - resolution: {integrity: sha512-mNKt2jBXJg4O7pSdbNUfDdTsK9FIdikfsIE/yUCxbAEXl4HMyJaivrVFcn3Elvt5xvCQYhUZm+hqTIu1UXM3Pw==} - peerDependencies: - react: '>=18' - react-dom: '>=18' - - react@18.3.1: - resolution: {integrity: sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ==} - engines: {node: '>=0.10.0'} - - readable-stream@3.6.2: - resolution: {integrity: sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==} - engines: {node: '>= 6'} - - readdirp@3.6.0: - resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==} - engines: {node: '>=8.10.0'} - - reflect.getprototypeof@1.0.6: - resolution: {integrity: sha512-fmfw4XgoDke3kdI6h4xcUz1dG8uaiv5q9gcEwLS4Pnth2kxT+GZ7YehS1JTMGBQmtV7Y4GFGbs2re2NqhdozUg==} - engines: {node: '>= 0.4'} - - regenerator-runtime@0.14.1: - resolution: {integrity: sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==} - - regexp.prototype.flags@1.5.2: - resolution: {integrity: sha512-NcDiDkTLuPR+++OCKB0nWafEmhg/Da8aUPLPMQbK+bxKKCm1/S5he+AqYa4PlMCVBalb4/yxIRub6qkEx5yJbw==} - engines: {node: '>= 0.4'} - - resolve-from@4.0.0: - resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==} - engines: {node: '>=4'} - - resolve-pkg-maps@1.0.0: - resolution: {integrity: sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==} - - resolve@1.22.8: - resolution: {integrity: sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==} - hasBin: true - - resolve@2.0.0-next.5: - resolution: {integrity: sha512-U7WjGVG9sH8tvjW5SmGbQuui75FiyjAX72HX15DwBBwF9dNiQZRQAg9nnPhYy+TUnE0+VcrttuvNI8oSxZcocA==} - hasBin: true - - retry@0.13.1: - resolution: {integrity: sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg==} - engines: {node: '>= 4'} - - reusify@1.0.4: - resolution: {integrity: sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==} - engines: {iojs: '>=1.0.0', node: '>=0.10.0'} - - run-parallel@1.2.0: - resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} - - safe-array-concat@1.1.2: - resolution: {integrity: sha512-vj6RsCsWBCf19jIeHEfkRMw8DPiBb+DMXklQ/1SGDHOMlHdPUkZXFQ2YdplS23zESTijAcurb1aSgJA3AgMu1Q==} - engines: {node: '>=0.4'} - - safe-buffer@5.2.1: - resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} - - safe-regex-test@1.0.3: - resolution: {integrity: sha512-CdASjNJPvRa7roO6Ra/gLYBTzYzzPyyBXxIMdGW3USQLyjWEls2RgW5UBTXaQVp+OrpeCK3bLem8smtmheoRuw==} - engines: {node: '>= 0.4'} - - sass@1.77.8: - resolution: {integrity: sha512-4UHg6prsrycW20fqLGPShtEvo/WyHRVRHwOP4DzkUrObWoWI05QBSfzU71TVB7PFaL104TwNaHpjlWXAZbQiNQ==} - engines: {node: '>=14.0.0'} - hasBin: true - - sax@1.4.1: - resolution: {integrity: sha512-+aWOz7yVScEGoKNd4PA10LZ8sk0A/z5+nXQG5giUO5rprX9jgYsTdov9qCchZiPIZezbZH+jRut8nPodFAX4Jg==} - - scheduler@0.23.2: - resolution: {integrity: sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==} - - section-matter@1.0.0: - resolution: {integrity: sha512-vfD3pmTzGpufjScBh50YHKzEu2lxBWhVEHsNGoEXmCmn2hKGfeNLYMzCJpe8cD7gqX7TJluOVpBkAequ6dgMmA==} - engines: {node: '>=4'} - - semver@6.3.1: - resolution: {integrity: sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==} - hasBin: true - - semver@7.6.3: - resolution: {integrity: sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==} - engines: {node: '>=10'} - hasBin: true - - set-function-length@1.2.2: - resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==} - engines: {node: '>= 0.4'} - - set-function-name@2.0.2: - resolution: {integrity: sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==} - engines: {node: '>= 0.4'} - - shallow-equal@3.1.0: - resolution: {integrity: sha512-pfVOw8QZIXpMbhBWvzBISicvToTiM5WBF1EeAUZDDSb5Dt29yl4AYbyywbJFSEsRUMr7gJaxqCdr4L3tQf9wVg==} - - shallowequal@1.1.0: - resolution: {integrity: sha512-y0m1JoUZSlPAjXVtPPW70aZWfIL/dSP7AFkRnniLCrK/8MDKog3TySTBmckD+RObVxH0v4Tox67+F14PdED2oQ==} - - sharp@0.30.7: - resolution: {integrity: sha512-G+MY2YW33jgflKPTXXptVO28HvNOo9G3j0MybYAHeEmby+QuD2U98dT6ueht9cv/XDqZspSpIhoSW+BAKJ7Hig==} - engines: {node: '>=12.13.0'} - - sharp@0.33.5: - resolution: {integrity: sha512-haPVm1EkS9pgvHrQ/F3Xy+hgcuMV0Wm9vfIBSiwZ05k+xgb0PkBQpGsAA/oWdDobNaZTH5ppvHtzCFbnSEwHVw==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - - shebang-command@2.0.0: - resolution: {integrity: sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==} - engines: {node: '>=8'} - - shebang-regex@3.0.0: - resolution: {integrity: sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==} - engines: {node: '>=8'} - - side-channel@1.0.6: - resolution: {integrity: sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==} - engines: {node: '>= 0.4'} - - signal-exit@4.1.0: - resolution: {integrity: sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==} - engines: {node: '>=14'} - - simple-concat@1.0.1: - resolution: {integrity: sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==} - - simple-get@4.0.1: - resolution: {integrity: sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA==} - - simple-swizzle@0.2.2: - resolution: {integrity: sha512-JA//kQgZtbuY83m+xT+tXJkmJncGMTFT+C+g2h2R9uxkYIrE2yy9sgmcLhCnw57/WSD+Eh3J97FPEDFnbXnDUg==} - - slash@3.0.0: - resolution: {integrity: sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==} - engines: {node: '>=8'} - - slugify@1.6.6: - resolution: {integrity: sha512-h+z7HKHYXj6wJU+AnS/+IH8Uh9fdcX1Lrhg1/VMdf9PwoBQXFcXiAdsy2tSK0P6gKwJLXp02r90ahUCqHk9rrw==} - engines: {node: '>=8.0.0'} - - source-map-js@1.2.0: - resolution: {integrity: sha512-itJW8lvSA0TXEphiRoawsCksnlf8SyvmFzIhltqAHluXd88pkCd+cXJVHTDwdCr0IzwptSm035IHQktUu1QUMg==} - engines: {node: '>=0.10.0'} - - sprintf-js@1.0.3: - resolution: {integrity: sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==} - - stop-iteration-iterator@1.0.0: - resolution: {integrity: sha512-iCGQj+0l0HOdZ2AEeBADlsRC+vsnDsZsbdSiH1yNSjcfKM7fdpCMfqAL/dwF5BLiw/XhRft/Wax6zQbhq2BcjQ==} - engines: {node: '>= 0.4'} - - streamsearch@1.1.0: - resolution: {integrity: sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg==} - engines: {node: '>=10.0.0'} - - string-width@4.2.3: - resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} - engines: {node: '>=8'} - - string-width@5.1.2: - resolution: {integrity: sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==} - engines: {node: '>=12'} - - string.prototype.includes@2.0.0: - resolution: {integrity: sha512-E34CkBgyeqNDcrbU76cDjL5JLcVrtSdYq0MEh/B10r17pRP4ciHLwTgnuLV8Ay6cgEMLkcBkFCKyFZ43YldYzg==} - - string.prototype.matchall@4.0.11: - resolution: {integrity: sha512-NUdh0aDavY2og7IbBPenWqR9exH+E26Sv8e0/eTe1tltDGZL+GtBkDAnnyBtmekfK6/Dq3MkcGtzXFEd1LQrtg==} - engines: {node: '>= 0.4'} - - string.prototype.repeat@1.0.0: - resolution: {integrity: sha512-0u/TldDbKD8bFCQ/4f5+mNRrXwZ8hg2w7ZR8wa16e8z9XpePWl3eGEcUD0OXpEH/VJH/2G3gjUtR3ZOiBe2S/w==} - - string.prototype.trim@1.2.9: - resolution: {integrity: sha512-klHuCNxiMZ8MlsOihJhJEBJAiMVqU3Z2nEXWfWnIqjN0gEFS9J9+IxKozWWtQGcgoa1WUZzLjKPTr4ZHNFTFxw==} - engines: {node: '>= 0.4'} - - string.prototype.trimend@1.0.8: - resolution: {integrity: sha512-p73uL5VCHCO2BZZ6krwwQE3kCzM7NKmis8S//xEC6fQonchbum4eP6kR4DLEjQFO3Wnj3Fuo8NM0kOSjVdHjZQ==} - - string.prototype.trimstart@1.0.8: - resolution: {integrity: sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==} - engines: {node: '>= 0.4'} - - string_decoder@1.3.0: - resolution: {integrity: sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==} - - strip-ansi@6.0.1: - resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} - engines: {node: '>=8'} - - strip-ansi@7.1.0: - resolution: {integrity: sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==} - engines: {node: '>=12'} - - strip-bom-string@1.0.0: - resolution: {integrity: sha512-uCC2VHvQRYu+lMh4My/sFNmF2klFymLX1wHJeXnbEJERpV/ZsVuonzerjfrGpIGF7LBVa1O7i9kjiWvJiFck8g==} - engines: {node: '>=0.10.0'} - - strip-bom@3.0.0: - resolution: {integrity: sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==} - engines: {node: '>=4'} - - strip-json-comments@2.0.1: - resolution: {integrity: sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==} - engines: {node: '>=0.10.0'} - - strip-json-comments@3.1.1: - resolution: {integrity: sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==} - engines: {node: '>=8'} - - styled-components@6.1.13: - resolution: {integrity: sha512-M0+N2xSnAtwcVAQeFEsGWFFxXDftHUD7XrKla06QbpUMmbmtFBMMTcKWvFXtWxuD5qQkB8iU5gk6QASlx2ZRMw==} - engines: {node: '>= 16'} - peerDependencies: - react: '>= 16.8.0' - react-dom: '>= 16.8.0' - - styled-jsx@5.1.1: - resolution: {integrity: sha512-pW7uC1l4mBZ8ugbiZrcIsiIvVx1UmTfw7UkC3Um2tmfUq9Bhk8IiyEIPl6F8agHgjzku6j0xQEZbfA5uSgSaCw==} - engines: {node: '>= 12.0.0'} - peerDependencies: - '@babel/core': '*' - babel-plugin-macros: '*' - react: '>= 16.8.0 || 17.x.x || ^18.0.0-0' - peerDependenciesMeta: - '@babel/core': - optional: true - babel-plugin-macros: - optional: true - - stylis@4.3.2: - resolution: {integrity: sha512-bhtUjWd/z6ltJiQwg0dUfxEJ+W+jdqQd8TbWLWyeIJHlnsqmGLRFFd8e5mA0AZi/zx90smXRlN66YMTcaSFifg==} - - supports-color@7.2.0: - resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==} - engines: {node: '>=8'} - - supports-preserve-symlinks-flag@1.0.0: - resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==} - engines: {node: '>= 0.4'} - - tapable@2.2.1: - resolution: {integrity: sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ==} - engines: {node: '>=6'} - - tar-fs@2.1.1: - resolution: {integrity: sha512-V0r2Y9scmbDRLCNex/+hYzvp/zyYjvFbHPNgVTKfQvVrb6guiE/fxP+XblDNR011utopbkex2nM4dHNV6GDsng==} - - tar-stream@2.2.0: - resolution: {integrity: sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==} - engines: {node: '>=6'} - - text-table@0.2.0: - resolution: {integrity: sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==} - - tiny-case@1.0.3: - resolution: {integrity: sha512-Eet/eeMhkO6TX8mnUteS9zgPbUMQa4I6Kkp5ORiBD5476/m+PIRiumP5tmh5ioJpH7k51Kehawy2UDfsnxxY8Q==} - - to-regex-range@5.0.1: - resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} - engines: {node: '>=8.0'} - - toposort@2.0.2: - resolution: {integrity: sha512-0a5EOkAUp8D4moMi2W8ZF8jcga7BgZd91O/yabJCFY8az+XSzeGyTKs0Aoo897iV1Nj6guFq8orWDS96z91oGg==} - - ts-api-utils@1.3.0: - resolution: {integrity: sha512-UQMIo7pb8WRomKR1/+MFVLTroIvDVtMX3K6OUir8ynLyzB8Jeriont2bTAtmNPa1ekAgN7YPDyf6V+ygrdU+eQ==} - engines: {node: '>=16'} - peerDependencies: - typescript: '>=4.2.0' - - tsconfig-paths@3.15.0: - resolution: {integrity: sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg==} - - tslib@2.6.2: - resolution: {integrity: sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==} - - tslib@2.7.0: - resolution: {integrity: sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA==} - - tunnel-agent@0.6.0: - resolution: {integrity: sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==} - - type-check@0.4.0: - resolution: {integrity: sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==} - engines: {node: '>= 0.8.0'} - - type-fest@1.4.0: - resolution: {integrity: sha512-yGSza74xk0UG8k+pLh5oeoYirvIiWo5t0/o3zHHAO2tRDiZcxWP7fywNlXhqb6/r6sWvwi+RsyQMWhVLe4BVuA==} - engines: {node: '>=10'} - - type-fest@2.19.0: - resolution: {integrity: sha512-RAH822pAdBgcNMAfWnCBU3CFZcfZ/i1eZjwFU/dsLKumyuuP3niueg2UAukXYF0E2AAoc82ZSSf9J0WQBinzHA==} - engines: {node: '>=12.20'} - - typed-array-buffer@1.0.2: - resolution: {integrity: sha512-gEymJYKZtKXzzBzM4jqa9w6Q1Jjm7x2d+sh19AdsD4wqnMPDYyvwpsIc2Q/835kHuo3BEQ7CjelGhfTsoBb2MQ==} - engines: {node: '>= 0.4'} - - typed-array-byte-length@1.0.1: - resolution: {integrity: sha512-3iMJ9q0ao7WE9tWcaYKIptkNBuOIcZCCT0d4MRvuuH88fEoEH62IuQe0OtraD3ebQEoTRk8XCBoknUNc1Y67pw==} - engines: {node: '>= 0.4'} - - typed-array-byte-offset@1.0.2: - resolution: {integrity: sha512-Ous0vodHa56FviZucS2E63zkgtgrACj7omjwd/8lTEMEPFFyjfixMZ1ZXenpgCFBBt4EC1J2XsyVS2gkG0eTFA==} - engines: {node: '>= 0.4'} - - typed-array-length@1.0.6: - resolution: {integrity: sha512-/OxDN6OtAk5KBpGb28T+HZc2M+ADtvRxXrKKbUwtsLgdoxgX13hyy7ek6bFRl5+aBs2yZzB0c4CnQfAtVypW/g==} - engines: {node: '>= 0.4'} - - typescript@5.5.4: - resolution: {integrity: sha512-Mtq29sKDAEYP7aljRgtPOpTvOfbwRWlS6dPRzwjdE+C0R4brX/GUyhHSecbHMFLNBLcJIPt9nl9yG5TZ1weH+Q==} - engines: {node: '>=14.17'} - hasBin: true - - unbox-primitive@1.0.2: - resolution: {integrity: sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==} - - undici-types@6.19.8: - resolution: {integrity: sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==} - - uri-js@4.4.1: - resolution: {integrity: sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==} - - url-polyfill@1.1.12: - resolution: {integrity: sha512-mYFmBHCapZjtcNHW0MDq9967t+z4Dmg5CJ0KqysK3+ZbyoNOWQHksGCTWwDhxGXllkWlOc10Xfko6v4a3ucM6A==} - - use-sync-external-store@1.2.2: - resolution: {integrity: sha512-PElTlVMwpblvbNqQ82d2n6RjStvdSoNe9FG28kNfz3WiXilJm4DdNkEzRhCZuIDwY8U08WVihhGR5iRqAwfDiw==} - peerDependencies: - react: ^16.8.0 || ^17.0.0 || ^18.0.0 - - util-deprecate@1.0.2: - resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==} - - uuid@8.3.2: - resolution: {integrity: sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==} - hasBin: true - - vary@1.1.2: - resolution: {integrity: sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==} - engines: {node: '>= 0.8'} - - which-boxed-primitive@1.0.2: - resolution: {integrity: sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==} - - which-builtin-type@1.1.4: - resolution: {integrity: sha512-bppkmBSsHFmIMSl8BO9TbsyzsvGjVoppt8xUiGzwiu/bhDCGxnpOKCxgqj6GuyHE0mINMDecBFPlOm2hzY084w==} - engines: {node: '>= 0.4'} - - which-collection@1.0.2: - resolution: {integrity: sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw==} - engines: {node: '>= 0.4'} - - which-typed-array@1.1.15: - resolution: {integrity: sha512-oV0jmFtUky6CXfkqehVvBP/LSWJ2sy4vWMioiENyJLePrBO/yKyV9OyJySfAKosh+RYkIl5zJCNZ8/4JncrpdA==} - engines: {node: '>= 0.4'} - - which@2.0.2: - resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==} - engines: {node: '>= 8'} - hasBin: true - - wildcard@1.1.2: - resolution: {integrity: sha512-DXukZJxpHA8LuotRwL0pP1+rS6CS7FF2qStDDE1C7DDg2rLud2PXRMuEDYIPhgEezwnlHNL4c+N6MfMTjCGTng==} - - word-wrap@1.2.5: - resolution: {integrity: sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==} - engines: {node: '>=0.10.0'} - - wrap-ansi@7.0.0: - resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} - engines: {node: '>=10'} - - wrap-ansi@8.1.0: - resolution: {integrity: sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==} - engines: {node: '>=12'} - - wrappy@1.0.2: - resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==} - - xml-js@1.6.11: - resolution: {integrity: sha512-7rVi2KMfwfWFl+GpPg6m80IVMWXLRjO+PxTq7V2CDhoGak0wzYzFgUY2m4XJ47OGdXd8eLE8EmwfAmdjw7lC1g==} - hasBin: true - - yallist@4.0.0: - resolution: {integrity: sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==} - - yocto-queue@0.1.0: - resolution: {integrity: sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==} - engines: {node: '>=10'} - - yup@1.4.0: - resolution: {integrity: sha512-wPbgkJRCqIf+OHyiTBQoJiP5PFuAXaWiJK6AmYkzQAh5/c2K9hzSApBZG5wV9KoKSePF7sAxmNSvh/13YHkFDg==} - -snapshots: - - '@babel/runtime@7.26.0': - dependencies: - regenerator-runtime: 0.14.1 - - '@dicebear/adventurer-neutral@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/adventurer@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/avataaars-neutral@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/avataaars@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/big-ears-neutral@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/big-ears@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/big-smile@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/bottts-neutral@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/bottts@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/collection@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/adventurer': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/adventurer-neutral': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/avataaars': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/avataaars-neutral': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/big-ears': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/big-ears-neutral': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/big-smile': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/bottts': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/bottts-neutral': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/core': 9.2.2 - '@dicebear/croodles': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/croodles-neutral': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/dylan': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/fun-emoji': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/glass': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/icons': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/identicon': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/initials': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/lorelei': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/lorelei-neutral': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/micah': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/miniavs': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/notionists': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/notionists-neutral': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/open-peeps': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/personas': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/pixel-art': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/pixel-art-neutral': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/rings': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/shapes': 9.2.2(@dicebear/core@9.2.2) - '@dicebear/thumbs': 9.2.2(@dicebear/core@9.2.2) - - '@dicebear/core@9.2.2': - dependencies: - '@types/json-schema': 7.0.15 - - '@dicebear/croodles-neutral@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/croodles@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/dylan@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/fun-emoji@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/glass@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/icons@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/identicon@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/initials@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/lorelei-neutral@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/lorelei@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/micah@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/miniavs@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/notionists-neutral@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/notionists@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/open-peeps@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/personas@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/pixel-art-neutral@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/pixel-art@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/rings@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/shapes@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@dicebear/thumbs@9.2.2(@dicebear/core@9.2.2)': - dependencies: - '@dicebear/core': 9.2.2 - - '@edge-runtime/cookies@4.1.1': {} - - '@emnapi/runtime@1.2.0': - dependencies: - tslib: 2.7.0 - optional: true - - '@emotion/is-prop-valid@1.2.2': - dependencies: - '@emotion/memoize': 0.8.1 - - '@emotion/memoize@0.8.1': {} - - '@emotion/unitless@0.8.1': {} - - '@eslint-community/eslint-utils@4.4.0(eslint@9.9.1)': - dependencies: - eslint: 9.9.1 - eslint-visitor-keys: 3.4.3 - - '@eslint-community/regexpp@4.11.0': {} - - '@eslint/config-array@0.18.0': - dependencies: - '@eslint/object-schema': 2.1.4 - debug: 4.3.6 - minimatch: 3.1.2 - transitivePeerDependencies: - - supports-color - - '@eslint/eslintrc@3.1.0': - dependencies: - ajv: 6.12.6 - debug: 4.3.6 - espree: 10.1.0 - globals: 14.0.0 - ignore: 5.3.2 - import-fresh: 3.3.0 - js-yaml: 4.1.0 - minimatch: 3.1.2 - strip-json-comments: 3.1.1 - transitivePeerDependencies: - - supports-color - - '@eslint/js@9.9.1': {} - - '@eslint/object-schema@2.1.4': {} - - '@fortawesome/fontawesome-common-types@6.6.0': {} - - '@fortawesome/fontawesome-free@6.6.0': {} - - '@fortawesome/fontawesome-svg-core@6.6.0': - dependencies: - '@fortawesome/fontawesome-common-types': 6.6.0 - - '@fortawesome/free-brands-svg-icons@6.6.0': - dependencies: - '@fortawesome/fontawesome-common-types': 6.6.0 - - '@fortawesome/free-solid-svg-icons@6.6.0': - dependencies: - '@fortawesome/fontawesome-common-types': 6.6.0 - - '@fortawesome/react-fontawesome@0.2.2(@fortawesome/fontawesome-svg-core@6.6.0)(react@18.3.1)': - dependencies: - '@fortawesome/fontawesome-svg-core': 6.6.0 - prop-types: 15.8.1 - react: 18.3.1 - - '@hookform/error-message@2.0.1(react-dom@18.3.1(react@18.3.1))(react-hook-form@7.53.0(react@18.3.1))(react@18.3.1)': - dependencies: - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - react-hook-form: 7.53.0(react@18.3.1) - - '@hookform/resolvers@3.9.0(react-hook-form@7.53.0(react@18.3.1))': - dependencies: - react-hook-form: 7.53.0(react@18.3.1) - - '@humanwhocodes/module-importer@1.0.1': {} - - '@humanwhocodes/retry@0.3.0': {} - - '@img/sharp-darwin-arm64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-darwin-arm64': 1.0.4 - optional: true - - '@img/sharp-darwin-x64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-darwin-x64': 1.0.4 - optional: true - - '@img/sharp-libvips-darwin-arm64@1.0.4': - optional: true - - '@img/sharp-libvips-darwin-x64@1.0.4': - optional: true - - '@img/sharp-libvips-linux-arm64@1.0.4': - optional: true - - '@img/sharp-libvips-linux-arm@1.0.5': - optional: true - - '@img/sharp-libvips-linux-s390x@1.0.4': - optional: true - - '@img/sharp-libvips-linux-x64@1.0.4': - optional: true - - '@img/sharp-libvips-linuxmusl-arm64@1.0.4': - optional: true - - '@img/sharp-libvips-linuxmusl-x64@1.0.4': - optional: true - - '@img/sharp-linux-arm64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linux-arm64': 1.0.4 - optional: true - - '@img/sharp-linux-arm@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linux-arm': 1.0.5 - optional: true - - '@img/sharp-linux-s390x@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linux-s390x': 1.0.4 - optional: true - - '@img/sharp-linux-x64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linux-x64': 1.0.4 - optional: true - - '@img/sharp-linuxmusl-arm64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linuxmusl-arm64': 1.0.4 - optional: true - - '@img/sharp-linuxmusl-x64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linuxmusl-x64': 1.0.4 - optional: true - - '@img/sharp-wasm32@0.33.5': - dependencies: - '@emnapi/runtime': 1.2.0 - optional: true - - '@img/sharp-win32-ia32@0.33.5': - optional: true - - '@img/sharp-win32-x64@0.33.5': - optional: true - - '@isaacs/cliui@8.0.2': - dependencies: - string-width: 5.1.2 - string-width-cjs: string-width@4.2.3 - strip-ansi: 7.1.0 - strip-ansi-cjs: strip-ansi@6.0.1 - wrap-ansi: 8.1.0 - wrap-ansi-cjs: wrap-ansi@7.0.0 - - '@logto/client@2.8.1': - dependencies: - '@logto/js': 4.2.0 - '@silverhand/essentials': 2.9.2 - camelcase-keys: 7.0.2 - jose: 5.9.6 - - '@logto/js@4.2.0': - dependencies: - '@silverhand/essentials': 2.9.2 - camelcase-keys: 7.0.2 - - '@logto/next@3.7.1(next@14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8))': - dependencies: - '@edge-runtime/cookies': 4.1.1 - '@logto/node': 2.5.8 - cookie: 0.6.0 - next: 14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8) - - '@logto/node@2.5.8': - dependencies: - '@logto/client': 2.8.1 - '@silverhand/essentials': 2.9.2 - js-base64: 3.7.7 - - '@mux/blurhash@0.1.2': - dependencies: - blurhash: 1.1.5 - sharp: 0.30.7 - - '@mux/mux-player-react@2.9.1(@types/react-dom@18.3.0)(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@mux/mux-player': 2.9.1 - '@mux/playback-core': 0.25.2 - prop-types: 15.8.1 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - '@types/react-dom': 18.3.0 - - '@mux/mux-player@2.9.1': - dependencies: - '@mux/mux-video': 0.20.2 - '@mux/playback-core': 0.25.2 - media-chrome: 3.2.5 - - '@mux/mux-video@0.20.2': - dependencies: - '@mux/playback-core': 0.25.2 - castable-video: 1.0.10 - custom-media-element: 1.3.2 - media-tracks: 0.3.3 - - '@mux/playback-core@0.25.2': - dependencies: - hls.js: 1.5.15 - mux-embed: 5.2.1 - - '@next/env@14.2.7': {} - - '@next/eslint-plugin-next@14.2.7': - dependencies: - glob: 10.3.10 - - '@next/swc-darwin-arm64@14.2.7': - optional: true - - '@next/swc-darwin-x64@14.2.7': - optional: true - - '@next/swc-linux-arm64-gnu@14.2.7': - optional: true - - '@next/swc-linux-arm64-musl@14.2.7': - optional: true - - '@next/swc-linux-x64-gnu@14.2.7': - optional: true - - '@next/swc-linux-x64-musl@14.2.7': - optional: true - - '@next/swc-win32-arm64-msvc@14.2.7': - optional: true - - '@next/swc-win32-ia32-msvc@14.2.7': - optional: true - - '@next/swc-win32-x64-msvc@14.2.7': - optional: true - - '@noble/hashes@1.5.0': {} - - '@nodelib/fs.scandir@2.1.5': - dependencies: - '@nodelib/fs.stat': 2.0.5 - run-parallel: 1.2.0 - - '@nodelib/fs.stat@2.0.5': {} - - '@nodelib/fs.walk@1.2.8': - dependencies: - '@nodelib/fs.scandir': 2.1.5 - fastq: 1.17.1 - - '@nolyfill/is-core-module@1.0.39': {} - - '@panva/hkdf@1.2.1': {} - - '@paralleldrive/cuid2@2.2.2': - dependencies: - '@noble/hashes': 1.5.0 - - '@pkgjs/parseargs@0.11.0': - optional: true - - '@react-hookz/deep-equal@1.0.4': {} - - '@react-hookz/web@24.0.4(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@react-hookz/deep-equal': 1.0.4 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - - '@rtsao/scc@1.1.0': {} - - '@rushstack/eslint-patch@1.10.4': {} - - '@silverhand/essentials@2.9.2': {} - - '@swc/counter@0.1.3': {} - - '@swc/helpers@0.5.5': - dependencies: - '@swc/counter': 0.1.3 - tslib: 2.7.0 - - '@tanstack/query-core@5.53.3': {} - - '@tanstack/react-query@5.53.3(react@18.3.1)': - dependencies: - '@tanstack/query-core': 5.53.3 - react: 18.3.1 - - '@tanstack/react-table@8.20.5(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@tanstack/table-core': 8.20.5 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - - '@tanstack/table-core@8.20.5': {} - - '@transloadit/prettier-bytes@0.3.4': {} - - '@types/json-schema@7.0.15': {} - - '@types/json5@0.0.29': {} - - '@types/lodash@4.17.7': {} - - '@types/node@22.5.2': - dependencies: - undici-types: 6.19.8 - - '@types/prop-types@15.7.12': {} - - '@types/qs@6.9.15': {} - - '@types/react-dom@18.3.0': - dependencies: - '@types/react': 18.3.5 - - '@types/react@18.3.5': - dependencies: - '@types/prop-types': 15.7.12 - csstype: 3.1.3 - - '@types/retry@0.12.2': {} - - '@types/stylis@4.2.5': {} - - '@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4)': - dependencies: - '@typescript-eslint/scope-manager': 7.2.0 - '@typescript-eslint/types': 7.2.0 - '@typescript-eslint/typescript-estree': 7.2.0(typescript@5.5.4) - '@typescript-eslint/visitor-keys': 7.2.0 - debug: 4.3.6 - eslint: 9.9.1 - optionalDependencies: - typescript: 5.5.4 - transitivePeerDependencies: - - supports-color - - '@typescript-eslint/scope-manager@7.2.0': - dependencies: - '@typescript-eslint/types': 7.2.0 - '@typescript-eslint/visitor-keys': 7.2.0 - - '@typescript-eslint/types@7.2.0': {} - - '@typescript-eslint/typescript-estree@7.2.0(typescript@5.5.4)': - dependencies: - '@typescript-eslint/types': 7.2.0 - '@typescript-eslint/visitor-keys': 7.2.0 - debug: 4.3.6 - globby: 11.1.0 - is-glob: 4.0.3 - minimatch: 9.0.3 - semver: 7.6.3 - ts-api-utils: 1.3.0(typescript@5.5.4) - optionalDependencies: - typescript: 5.5.4 - transitivePeerDependencies: - - supports-color - - '@typescript-eslint/visitor-keys@7.2.0': - dependencies: - '@typescript-eslint/types': 7.2.0 - eslint-visitor-keys: 3.4.3 - - '@uppy/aws-s3-multipart@4.0.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/aws-s3': 4.1.0(@uppy/core@4.2.0) - transitivePeerDependencies: - - '@uppy/core' - - '@uppy/aws-s3@4.1.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/companion-client': 4.1.0(@uppy/core@4.2.0) - '@uppy/core': 4.2.0 - '@uppy/utils': 6.0.2 - - '@uppy/box@3.1.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/companion-client': 4.1.0(@uppy/core@4.2.0) - '@uppy/core': 4.2.0 - '@uppy/provider-views': 4.0.1(@uppy/core@4.2.0) - '@uppy/utils': 6.0.2 - preact: 10.23.2 - - '@uppy/companion-client@4.1.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/core': 4.2.0 - '@uppy/utils': 6.0.2 - namespace-emitter: 2.0.1 - p-retry: 6.2.0 - - '@uppy/core@4.2.0': - dependencies: - '@transloadit/prettier-bytes': 0.3.4 - '@uppy/store-default': 4.1.0 - '@uppy/utils': 6.0.2 - lodash: 4.17.21 - mime-match: 1.0.2 - namespace-emitter: 2.0.1 - nanoid: 5.0.7 - preact: 10.23.2 - - '@uppy/dashboard@4.1.0(@uppy/core@4.2.0)': - dependencies: - '@transloadit/prettier-bytes': 0.3.4 - '@uppy/core': 4.2.0 - '@uppy/informer': 4.1.0(@uppy/core@4.2.0) - '@uppy/provider-views': 4.0.1(@uppy/core@4.2.0) - '@uppy/status-bar': 4.0.3(@uppy/core@4.2.0) - '@uppy/thumbnail-generator': 4.0.0(@uppy/core@4.2.0) - '@uppy/utils': 6.0.2 - classnames: 2.5.1 - lodash: 4.17.21 - memoize-one: 6.0.0 - nanoid: 5.0.7 - preact: 10.23.2 - shallow-equal: 3.1.0 - - '@uppy/drag-drop@4.0.2(@uppy/core@4.2.0)': - dependencies: - '@uppy/core': 4.2.0 - '@uppy/utils': 6.0.2 - preact: 10.23.2 - - '@uppy/dropbox@4.1.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/companion-client': 4.1.0(@uppy/core@4.2.0) - '@uppy/core': 4.2.0 - '@uppy/provider-views': 4.0.1(@uppy/core@4.2.0) - '@uppy/utils': 6.0.2 - preact: 10.23.2 - - '@uppy/facebook@4.1.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/companion-client': 4.1.0(@uppy/core@4.2.0) - '@uppy/core': 4.2.0 - '@uppy/provider-views': 4.0.1(@uppy/core@4.2.0) - '@uppy/utils': 6.0.2 - preact: 10.23.2 - - '@uppy/file-input@4.0.1(@uppy/core@4.2.0)': - dependencies: - '@uppy/core': 4.2.0 - '@uppy/utils': 6.0.2 - preact: 10.23.2 - - '@uppy/google-drive@4.1.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/companion-client': 4.1.0(@uppy/core@4.2.0) - '@uppy/core': 4.2.0 - '@uppy/provider-views': 4.0.1(@uppy/core@4.2.0) - '@uppy/utils': 6.0.2 - preact: 10.23.2 - - '@uppy/google-photos@0.3.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/companion-client': 4.1.0(@uppy/core@4.2.0) - '@uppy/core': 4.2.0 - '@uppy/provider-views': 4.0.1(@uppy/core@4.2.0) - '@uppy/utils': 6.0.2 - preact: 10.23.2 - - '@uppy/informer@4.1.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/core': 4.2.0 - '@uppy/utils': 6.0.2 - preact: 10.23.2 - - '@uppy/instagram@4.1.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/companion-client': 4.1.0(@uppy/core@4.2.0) - '@uppy/core': 4.2.0 - '@uppy/provider-views': 4.0.1(@uppy/core@4.2.0) - '@uppy/utils': 6.0.2 - preact: 10.23.2 - - '@uppy/onedrive@4.1.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/companion-client': 4.1.0(@uppy/core@4.2.0) - '@uppy/core': 4.2.0 - '@uppy/provider-views': 4.0.1(@uppy/core@4.2.0) - '@uppy/utils': 6.0.2 - preact: 10.23.2 - - '@uppy/progress-bar@4.0.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/core': 4.2.0 - '@uppy/utils': 6.0.2 - preact: 10.23.2 - - '@uppy/provider-views@4.0.1(@uppy/core@4.2.0)': - dependencies: - '@uppy/core': 4.2.0 - '@uppy/utils': 6.0.2 - classnames: 2.5.1 - nanoid: 5.0.7 - p-queue: 8.0.1 - preact: 10.23.2 - - '@uppy/react@4.0.2(@uppy/core@4.2.0)(@uppy/dashboard@4.1.0(@uppy/core@4.2.0))(@uppy/drag-drop@4.0.2(@uppy/core@4.2.0))(@uppy/file-input@4.0.1(@uppy/core@4.2.0))(@uppy/progress-bar@4.0.0(@uppy/core@4.2.0))(@uppy/status-bar@4.0.3(@uppy/core@4.2.0))(react@18.3.1)': - dependencies: - '@uppy/core': 4.2.0 - '@uppy/utils': 6.0.2 - react: 18.3.1 - use-sync-external-store: 1.2.2(react@18.3.1) - optionalDependencies: - '@uppy/dashboard': 4.1.0(@uppy/core@4.2.0) - '@uppy/drag-drop': 4.0.2(@uppy/core@4.2.0) - '@uppy/file-input': 4.0.1(@uppy/core@4.2.0) - '@uppy/progress-bar': 4.0.0(@uppy/core@4.2.0) - '@uppy/status-bar': 4.0.3(@uppy/core@4.2.0) - - '@uppy/remote-sources@2.2.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/box': 3.1.0(@uppy/core@4.2.0) - '@uppy/core': 4.2.0 - '@uppy/dashboard': 4.1.0(@uppy/core@4.2.0) - '@uppy/dropbox': 4.1.0(@uppy/core@4.2.0) - '@uppy/facebook': 4.1.0(@uppy/core@4.2.0) - '@uppy/google-drive': 4.1.0(@uppy/core@4.2.0) - '@uppy/google-photos': 0.3.0(@uppy/core@4.2.0) - '@uppy/instagram': 4.1.0(@uppy/core@4.2.0) - '@uppy/onedrive': 4.1.0(@uppy/core@4.2.0) - '@uppy/unsplash': 4.1.0(@uppy/core@4.2.0) - '@uppy/url': 4.1.0(@uppy/core@4.2.0) - '@uppy/zoom': 3.1.0(@uppy/core@4.2.0) - - '@uppy/status-bar@4.0.3(@uppy/core@4.2.0)': - dependencies: - '@transloadit/prettier-bytes': 0.3.4 - '@uppy/core': 4.2.0 - '@uppy/utils': 6.0.2 - classnames: 2.5.1 - preact: 10.23.2 - - '@uppy/store-default@4.1.0': {} - - '@uppy/thumbnail-generator@4.0.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/core': 4.2.0 - '@uppy/utils': 6.0.2 - exifr: 7.1.3 - - '@uppy/unsplash@4.1.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/companion-client': 4.1.0(@uppy/core@4.2.0) - '@uppy/core': 4.2.0 - '@uppy/provider-views': 4.0.1(@uppy/core@4.2.0) - '@uppy/utils': 6.0.2 - preact: 10.23.2 - - '@uppy/url@4.1.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/companion-client': 4.1.0(@uppy/core@4.2.0) - '@uppy/core': 4.2.0 - '@uppy/utils': 6.0.2 - nanoid: 5.0.7 - preact: 10.23.2 - - '@uppy/utils@6.0.2': - dependencies: - lodash: 4.17.21 - preact: 10.23.2 - - '@uppy/zoom@3.1.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/companion-client': 4.1.0(@uppy/core@4.2.0) - '@uppy/core': 4.2.0 - '@uppy/provider-views': 4.0.1(@uppy/core@4.2.0) - '@uppy/utils': 6.0.2 - preact: 10.23.2 - - acorn-jsx@5.3.2(acorn@8.12.1): - dependencies: - acorn: 8.12.1 - - acorn@8.12.1: {} - - ajv@6.12.6: - dependencies: - fast-deep-equal: 3.1.3 - fast-json-stable-stringify: 2.1.0 - json-schema-traverse: 0.4.1 - uri-js: 4.4.1 - - ansi-regex@5.0.1: {} - - ansi-regex@6.0.1: {} - - ansi-styles@4.3.0: - dependencies: - color-convert: 2.0.1 - - ansi-styles@6.2.1: {} - - anymatch@3.1.3: - dependencies: - normalize-path: 3.0.0 - picomatch: 2.3.1 - - argparse@1.0.10: - dependencies: - sprintf-js: 1.0.3 - - argparse@2.0.1: {} - - aria-query@5.1.3: - dependencies: - deep-equal: 2.2.3 - - array-buffer-byte-length@1.0.1: - dependencies: - call-bind: 1.0.7 - is-array-buffer: 3.0.4 - - array-includes@3.1.8: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-abstract: 1.23.3 - es-object-atoms: 1.0.0 - get-intrinsic: 1.2.4 - is-string: 1.0.7 - - array-union@2.1.0: {} - - array.prototype.findlast@1.2.5: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-abstract: 1.23.3 - es-errors: 1.3.0 - es-object-atoms: 1.0.0 - es-shim-unscopables: 1.0.2 - - array.prototype.findlastindex@1.2.5: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-abstract: 1.23.3 - es-errors: 1.3.0 - es-object-atoms: 1.0.0 - es-shim-unscopables: 1.0.2 - - array.prototype.flat@1.3.2: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-abstract: 1.23.3 - es-shim-unscopables: 1.0.2 - - array.prototype.flatmap@1.3.2: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-abstract: 1.23.3 - es-shim-unscopables: 1.0.2 - - array.prototype.tosorted@1.1.4: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-abstract: 1.23.3 - es-errors: 1.3.0 - es-shim-unscopables: 1.0.2 - - arraybuffer.prototype.slice@1.0.3: - dependencies: - array-buffer-byte-length: 1.0.1 - call-bind: 1.0.7 - define-properties: 1.2.1 - es-abstract: 1.23.3 - es-errors: 1.3.0 - get-intrinsic: 1.2.4 - is-array-buffer: 3.0.4 - is-shared-array-buffer: 1.0.3 - - ast-types-flow@0.0.8: {} - - available-typed-arrays@1.0.7: - dependencies: - possible-typed-array-names: 1.0.0 - - axe-core@4.10.0: {} - - axobject-query@3.1.1: - dependencies: - deep-equal: 2.2.3 - - balanced-match@1.0.2: {} - - base64-js@1.5.1: {} - - binary-extensions@2.3.0: {} - - bl@4.1.0: - dependencies: - buffer: 5.7.1 - inherits: 2.0.4 - readable-stream: 3.6.2 - - blurhash@1.1.5: {} - - brace-expansion@1.1.11: - dependencies: - balanced-match: 1.0.2 - concat-map: 0.0.1 - - brace-expansion@2.0.1: - dependencies: - balanced-match: 1.0.2 - - braces@3.0.3: - dependencies: - fill-range: 7.1.1 - - buffer@5.7.1: - dependencies: - base64-js: 1.5.1 - ieee754: 1.2.1 - - bulma@1.0.2: {} - - busboy@1.6.0: - dependencies: - streamsearch: 1.1.0 - - call-bind@1.0.7: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - set-function-length: 1.2.2 - - callsites@3.1.0: {} - - camelcase-keys@7.0.2: - dependencies: - camelcase: 6.3.0 - map-obj: 4.3.0 - quick-lru: 5.1.1 - type-fest: 1.4.0 - - camelcase@6.3.0: {} - - camelize@1.0.1: {} - - caniuse-lite@1.0.30001655: {} - - castable-video@1.0.10: - dependencies: - custom-media-element: 1.3.2 - - chalk@4.1.2: - dependencies: - ansi-styles: 4.3.0 - supports-color: 7.2.0 - - chokidar@3.6.0: - dependencies: - anymatch: 3.1.3 - braces: 3.0.3 - glob-parent: 5.1.2 - is-binary-path: 2.1.0 - is-glob: 4.0.3 - normalize-path: 3.0.0 - readdirp: 3.6.0 - optionalDependencies: - fsevents: 2.3.3 - - chownr@1.1.4: {} - - classnames@2.5.1: {} - - client-only@0.0.1: {} - - clsx@2.1.1: {} - - color-convert@2.0.1: - dependencies: - color-name: 1.1.4 - - color-name@1.1.4: {} - - color-string@1.9.1: - dependencies: - color-name: 1.1.4 - simple-swizzle: 0.2.2 - - color@4.2.3: - dependencies: - color-convert: 2.0.1 - color-string: 1.9.1 - - concat-map@0.0.1: {} - - cookie@0.6.0: {} - - cookie@0.7.1: {} - - core-js@3.38.1: {} - - cors@2.8.5: - dependencies: - object-assign: 4.1.1 - vary: 1.1.2 - - cross-spawn@7.0.3: - dependencies: - path-key: 3.1.1 - shebang-command: 2.0.0 - which: 2.0.2 - - css-color-keywords@1.0.0: {} - - css-to-react-native@3.2.0: - dependencies: - camelize: 1.0.1 - css-color-keywords: 1.0.0 - postcss-value-parser: 4.2.0 - - csstype@3.1.3: {} - - custom-event-polyfill@1.0.7: {} - - custom-media-element@1.3.2: {} - - damerau-levenshtein@1.0.8: {} - - data-view-buffer@1.0.1: - dependencies: - call-bind: 1.0.7 - es-errors: 1.3.0 - is-data-view: 1.0.1 - - data-view-byte-length@1.0.1: - dependencies: - call-bind: 1.0.7 - es-errors: 1.3.0 - is-data-view: 1.0.1 - - data-view-byte-offset@1.0.0: - dependencies: - call-bind: 1.0.7 - es-errors: 1.3.0 - is-data-view: 1.0.1 - - date-fns-tz@3.1.3(date-fns@3.6.0): - dependencies: - date-fns: 3.6.0 - - date-fns@3.6.0: {} - - dayjs@1.11.13: {} - - debug@3.2.7: - dependencies: - ms: 2.1.3 - - debug@4.3.6: - dependencies: - ms: 2.1.2 - - decompress-response@6.0.0: - dependencies: - mimic-response: 3.1.0 - - deep-equal@2.2.3: - dependencies: - array-buffer-byte-length: 1.0.1 - call-bind: 1.0.7 - es-get-iterator: 1.1.3 - get-intrinsic: 1.2.4 - is-arguments: 1.1.1 - is-array-buffer: 3.0.4 - is-date-object: 1.0.5 - is-regex: 1.1.4 - is-shared-array-buffer: 1.0.3 - isarray: 2.0.5 - object-is: 1.1.6 - object-keys: 1.1.1 - object.assign: 4.1.5 - regexp.prototype.flags: 1.5.2 - side-channel: 1.0.6 - which-boxed-primitive: 1.0.2 - which-collection: 1.0.2 - which-typed-array: 1.1.15 - - deep-extend@0.6.0: {} - - deep-is@0.1.4: {} - - deepmerge@4.3.1: {} - - define-data-property@1.1.4: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - gopd: 1.0.1 - - define-properties@1.2.1: - dependencies: - define-data-property: 1.1.4 - has-property-descriptors: 1.0.2 - object-keys: 1.1.1 - - detect-libc@2.0.3: {} - - dir-glob@3.0.1: - dependencies: - path-type: 4.0.0 - - doctrine@2.1.0: - dependencies: - esutils: 2.0.3 - - dotenv@16.4.5: {} - - eastasianwidth@0.2.0: {} - - emoji-regex@8.0.0: {} - - emoji-regex@9.2.2: {} - - end-of-stream@1.4.4: - dependencies: - once: 1.4.0 - - enhanced-resolve@5.17.1: - dependencies: - graceful-fs: 4.2.11 - tapable: 2.2.1 - - es-abstract@1.23.3: - dependencies: - array-buffer-byte-length: 1.0.1 - arraybuffer.prototype.slice: 1.0.3 - available-typed-arrays: 1.0.7 - call-bind: 1.0.7 - data-view-buffer: 1.0.1 - data-view-byte-length: 1.0.1 - data-view-byte-offset: 1.0.0 - es-define-property: 1.0.0 - es-errors: 1.3.0 - es-object-atoms: 1.0.0 - es-set-tostringtag: 2.0.3 - es-to-primitive: 1.2.1 - function.prototype.name: 1.1.6 - get-intrinsic: 1.2.4 - get-symbol-description: 1.0.2 - globalthis: 1.0.4 - gopd: 1.0.1 - has-property-descriptors: 1.0.2 - has-proto: 1.0.3 - has-symbols: 1.0.3 - hasown: 2.0.2 - internal-slot: 1.0.7 - is-array-buffer: 3.0.4 - is-callable: 1.2.7 - is-data-view: 1.0.1 - is-negative-zero: 2.0.3 - is-regex: 1.1.4 - is-shared-array-buffer: 1.0.3 - is-string: 1.0.7 - is-typed-array: 1.1.13 - is-weakref: 1.0.2 - object-inspect: 1.13.2 - object-keys: 1.1.1 - object.assign: 4.1.5 - regexp.prototype.flags: 1.5.2 - safe-array-concat: 1.1.2 - safe-regex-test: 1.0.3 - string.prototype.trim: 1.2.9 - string.prototype.trimend: 1.0.8 - string.prototype.trimstart: 1.0.8 - typed-array-buffer: 1.0.2 - typed-array-byte-length: 1.0.1 - typed-array-byte-offset: 1.0.2 - typed-array-length: 1.0.6 - unbox-primitive: 1.0.2 - which-typed-array: 1.1.15 - - es-define-property@1.0.0: - dependencies: - get-intrinsic: 1.2.4 - - es-errors@1.3.0: {} - - es-get-iterator@1.1.3: - dependencies: - call-bind: 1.0.7 - get-intrinsic: 1.2.4 - has-symbols: 1.0.3 - is-arguments: 1.1.1 - is-map: 2.0.3 - is-set: 2.0.3 - is-string: 1.0.7 - isarray: 2.0.5 - stop-iteration-iterator: 1.0.0 - - es-iterator-helpers@1.0.19: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-abstract: 1.23.3 - es-errors: 1.3.0 - es-set-tostringtag: 2.0.3 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - globalthis: 1.0.4 - has-property-descriptors: 1.0.2 - has-proto: 1.0.3 - has-symbols: 1.0.3 - internal-slot: 1.0.7 - iterator.prototype: 1.1.2 - safe-array-concat: 1.1.2 - - es-object-atoms@1.0.0: - dependencies: - es-errors: 1.3.0 - - es-set-tostringtag@2.0.3: - dependencies: - get-intrinsic: 1.2.4 - has-tostringtag: 1.0.2 - hasown: 2.0.2 - - es-shim-unscopables@1.0.2: - dependencies: - hasown: 2.0.2 - - es-to-primitive@1.2.1: - dependencies: - is-callable: 1.2.7 - is-date-object: 1.0.5 - is-symbol: 1.0.4 - - escape-string-regexp@4.0.0: {} - - eslint-config-next@14.2.7(eslint@9.9.1)(typescript@5.5.4): - dependencies: - '@next/eslint-plugin-next': 14.2.7 - '@rushstack/eslint-patch': 1.10.4 - '@typescript-eslint/parser': 7.2.0(eslint@9.9.1)(typescript@5.5.4) - eslint: 9.9.1 - eslint-import-resolver-node: 0.3.9 - eslint-import-resolver-typescript: 3.6.3(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-node@0.3.9)(eslint-plugin-import@2.30.0(eslint@9.9.1))(eslint@9.9.1) - eslint-plugin-import: 2.30.0(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-typescript@3.6.3(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-node@0.3.9)(eslint-plugin-import@2.30.0(eslint@9.9.1))(eslint@9.9.1))(eslint@9.9.1) - eslint-plugin-jsx-a11y: 6.9.0(eslint@9.9.1) - eslint-plugin-react: 7.35.1(eslint@9.9.1) - eslint-plugin-react-hooks: 4.6.2(eslint@9.9.1) - optionalDependencies: - typescript: 5.5.4 - transitivePeerDependencies: - - eslint-import-resolver-webpack - - eslint-plugin-import-x - - supports-color - - eslint-import-resolver-node@0.3.9: - dependencies: - debug: 3.2.7 - is-core-module: 2.15.1 - resolve: 1.22.8 - transitivePeerDependencies: - - supports-color - - eslint-import-resolver-typescript@3.6.3(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-node@0.3.9)(eslint-plugin-import@2.30.0(eslint@9.9.1))(eslint@9.9.1): - dependencies: - '@nolyfill/is-core-module': 1.0.39 - debug: 4.3.6 - enhanced-resolve: 5.17.1 - eslint: 9.9.1 - eslint-module-utils: 2.9.0(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.6.3(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-node@0.3.9)(eslint-plugin-import@2.30.0(eslint@9.9.1))(eslint@9.9.1))(eslint@9.9.1) - fast-glob: 3.3.2 - get-tsconfig: 4.8.0 - is-bun-module: 1.1.0 - is-glob: 4.0.3 - optionalDependencies: - eslint-plugin-import: 2.30.0(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-typescript@3.6.3(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-node@0.3.9)(eslint-plugin-import@2.30.0(eslint@9.9.1))(eslint@9.9.1))(eslint@9.9.1) - transitivePeerDependencies: - - '@typescript-eslint/parser' - - eslint-import-resolver-node - - eslint-import-resolver-webpack - - supports-color - - eslint-module-utils@2.9.0(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.6.3(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-node@0.3.9)(eslint-plugin-import@2.30.0(eslint@9.9.1))(eslint@9.9.1))(eslint@9.9.1): - dependencies: - debug: 3.2.7 - optionalDependencies: - '@typescript-eslint/parser': 7.2.0(eslint@9.9.1)(typescript@5.5.4) - eslint: 9.9.1 - eslint-import-resolver-node: 0.3.9 - eslint-import-resolver-typescript: 3.6.3(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-node@0.3.9)(eslint-plugin-import@2.30.0(eslint@9.9.1))(eslint@9.9.1) - transitivePeerDependencies: - - supports-color - - eslint-plugin-import@2.30.0(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-typescript@3.6.3(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-node@0.3.9)(eslint-plugin-import@2.30.0(eslint@9.9.1))(eslint@9.9.1))(eslint@9.9.1): - dependencies: - '@rtsao/scc': 1.1.0 - array-includes: 3.1.8 - array.prototype.findlastindex: 1.2.5 - array.prototype.flat: 1.3.2 - array.prototype.flatmap: 1.3.2 - debug: 3.2.7 - doctrine: 2.1.0 - eslint: 9.9.1 - eslint-import-resolver-node: 0.3.9 - eslint-module-utils: 2.9.0(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.6.3(@typescript-eslint/parser@7.2.0(eslint@9.9.1)(typescript@5.5.4))(eslint-import-resolver-node@0.3.9)(eslint-plugin-import@2.30.0(eslint@9.9.1))(eslint@9.9.1))(eslint@9.9.1) - hasown: 2.0.2 - is-core-module: 2.15.1 - is-glob: 4.0.3 - minimatch: 3.1.2 - object.fromentries: 2.0.8 - object.groupby: 1.0.3 - object.values: 1.2.0 - semver: 6.3.1 - tsconfig-paths: 3.15.0 - optionalDependencies: - '@typescript-eslint/parser': 7.2.0(eslint@9.9.1)(typescript@5.5.4) - transitivePeerDependencies: - - eslint-import-resolver-typescript - - eslint-import-resolver-webpack - - supports-color - - eslint-plugin-jsx-a11y@6.9.0(eslint@9.9.1): - dependencies: - aria-query: 5.1.3 - array-includes: 3.1.8 - array.prototype.flatmap: 1.3.2 - ast-types-flow: 0.0.8 - axe-core: 4.10.0 - axobject-query: 3.1.1 - damerau-levenshtein: 1.0.8 - emoji-regex: 9.2.2 - es-iterator-helpers: 1.0.19 - eslint: 9.9.1 - hasown: 2.0.2 - jsx-ast-utils: 3.3.5 - language-tags: 1.0.9 - minimatch: 3.1.2 - object.fromentries: 2.0.8 - safe-regex-test: 1.0.3 - string.prototype.includes: 2.0.0 - - eslint-plugin-react-hooks@4.6.2(eslint@9.9.1): - dependencies: - eslint: 9.9.1 - - eslint-plugin-react@7.35.1(eslint@9.9.1): - dependencies: - array-includes: 3.1.8 - array.prototype.findlast: 1.2.5 - array.prototype.flatmap: 1.3.2 - array.prototype.tosorted: 1.1.4 - doctrine: 2.1.0 - es-iterator-helpers: 1.0.19 - eslint: 9.9.1 - estraverse: 5.3.0 - hasown: 2.0.2 - jsx-ast-utils: 3.3.5 - minimatch: 3.1.2 - object.entries: 1.1.8 - object.fromentries: 2.0.8 - object.values: 1.2.0 - prop-types: 15.8.1 - resolve: 2.0.0-next.5 - semver: 6.3.1 - string.prototype.matchall: 4.0.11 - string.prototype.repeat: 1.0.0 - - eslint-scope@8.0.2: - dependencies: - esrecurse: 4.3.0 - estraverse: 5.3.0 - - eslint-visitor-keys@3.4.3: {} - - eslint-visitor-keys@4.0.0: {} - - eslint@9.9.1: - dependencies: - '@eslint-community/eslint-utils': 4.4.0(eslint@9.9.1) - '@eslint-community/regexpp': 4.11.0 - '@eslint/config-array': 0.18.0 - '@eslint/eslintrc': 3.1.0 - '@eslint/js': 9.9.1 - '@humanwhocodes/module-importer': 1.0.1 - '@humanwhocodes/retry': 0.3.0 - '@nodelib/fs.walk': 1.2.8 - ajv: 6.12.6 - chalk: 4.1.2 - cross-spawn: 7.0.3 - debug: 4.3.6 - escape-string-regexp: 4.0.0 - eslint-scope: 8.0.2 - eslint-visitor-keys: 4.0.0 - espree: 10.1.0 - esquery: 1.6.0 - esutils: 2.0.3 - fast-deep-equal: 3.1.3 - file-entry-cache: 8.0.0 - find-up: 5.0.0 - glob-parent: 6.0.2 - ignore: 5.3.2 - imurmurhash: 0.1.4 - is-glob: 4.0.3 - is-path-inside: 3.0.3 - json-stable-stringify-without-jsonify: 1.0.1 - levn: 0.4.1 - lodash.merge: 4.6.2 - minimatch: 3.1.2 - natural-compare: 1.4.0 - optionator: 0.9.4 - strip-ansi: 6.0.1 - text-table: 0.2.0 - transitivePeerDependencies: - - supports-color - - espree@10.1.0: - dependencies: - acorn: 8.12.1 - acorn-jsx: 5.3.2(acorn@8.12.1) - eslint-visitor-keys: 4.0.0 - - esprima@4.0.1: {} - - esquery@1.6.0: - dependencies: - estraverse: 5.3.0 - - esrecurse@4.3.0: - dependencies: - estraverse: 5.3.0 - - estraverse@5.3.0: {} - - esutils@2.0.3: {} - - eventemitter3@5.0.1: {} - - exifr@7.1.3: {} - - expand-template@2.0.3: {} - - extend-shallow@2.0.1: - dependencies: - is-extendable: 0.1.1 - - fast-deep-equal@3.1.3: {} - - fast-glob@3.3.2: - dependencies: - '@nodelib/fs.stat': 2.0.5 - '@nodelib/fs.walk': 1.2.8 - glob-parent: 5.1.2 - merge2: 1.4.1 - micromatch: 4.0.8 - - fast-json-stable-stringify@2.1.0: {} - - fast-levenshtein@2.0.6: {} - - fastq@1.17.1: - dependencies: - reusify: 1.0.4 - - feed@4.2.2: - dependencies: - xml-js: 1.6.11 - - file-entry-cache@8.0.0: - dependencies: - flat-cache: 4.0.1 - - fill-range@7.1.1: - dependencies: - to-regex-range: 5.0.1 - - find-up@5.0.0: - dependencies: - locate-path: 6.0.0 - path-exists: 4.0.0 - - flat-cache@4.0.1: - dependencies: - flatted: 3.3.1 - keyv: 4.5.4 - - flatted@3.3.1: {} - - for-each@0.3.3: - dependencies: - is-callable: 1.2.7 - - foreground-child@3.3.0: - dependencies: - cross-spawn: 7.0.3 - signal-exit: 4.1.0 - - fs-constants@1.0.0: {} - - fsevents@2.3.3: - optional: true - - function-bind@1.1.2: {} - - function.prototype.name@1.1.6: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-abstract: 1.23.3 - functions-have-names: 1.2.3 - - functions-have-names@1.2.3: {} - - get-intrinsic@1.2.4: - dependencies: - es-errors: 1.3.0 - function-bind: 1.1.2 - has-proto: 1.0.3 - has-symbols: 1.0.3 - hasown: 2.0.2 - - get-symbol-description@1.0.2: - dependencies: - call-bind: 1.0.7 - es-errors: 1.3.0 - get-intrinsic: 1.2.4 - - get-tsconfig@4.8.0: - dependencies: - resolve-pkg-maps: 1.0.0 - - github-from-package@0.0.0: {} - - glob-parent@5.1.2: - dependencies: - is-glob: 4.0.3 - - glob-parent@6.0.2: - dependencies: - is-glob: 4.0.3 - - glob@10.3.10: - dependencies: - foreground-child: 3.3.0 - jackspeak: 2.3.6 - minimatch: 9.0.5 - minipass: 7.1.2 - path-scurry: 1.11.1 - - globals@14.0.0: {} - - globalthis@1.0.4: - dependencies: - define-properties: 1.2.1 - gopd: 1.0.1 - - globby@11.1.0: - dependencies: - array-union: 2.1.0 - dir-glob: 3.0.1 - fast-glob: 3.3.2 - ignore: 5.3.2 - merge2: 1.4.1 - slash: 3.0.0 - - gopd@1.0.1: - dependencies: - get-intrinsic: 1.2.4 - - graceful-fs@4.2.11: {} - - gray-matter@4.0.3: - dependencies: - js-yaml: 3.14.1 - kind-of: 6.0.3 - section-matter: 1.0.0 - strip-bom-string: 1.0.0 - - has-bigints@1.0.2: {} - - has-flag@4.0.0: {} - - has-property-descriptors@1.0.2: - dependencies: - es-define-property: 1.0.0 - - has-proto@1.0.3: {} - - has-symbols@1.0.3: {} - - has-tostringtag@1.0.2: - dependencies: - has-symbols: 1.0.3 - - hasown@2.0.2: - dependencies: - function-bind: 1.1.2 - - hls.js@1.5.15: {} - - ieee754@1.2.1: {} - - ignore@5.3.2: {} - - immutable@4.3.7: {} - - import-fresh@3.3.0: - dependencies: - parent-module: 1.0.1 - resolve-from: 4.0.0 - - imurmurhash@0.1.4: {} - - inherits@2.0.4: {} - - ini@1.3.8: {} - - internal-slot@1.0.7: - dependencies: - es-errors: 1.3.0 - hasown: 2.0.2 - side-channel: 1.0.6 - - is-arguments@1.1.1: - dependencies: - call-bind: 1.0.7 - has-tostringtag: 1.0.2 - - is-array-buffer@3.0.4: - dependencies: - call-bind: 1.0.7 - get-intrinsic: 1.2.4 - - is-arrayish@0.3.2: {} - - is-async-function@2.0.0: - dependencies: - has-tostringtag: 1.0.2 - - is-bigint@1.0.4: - dependencies: - has-bigints: 1.0.2 - - is-binary-path@2.1.0: - dependencies: - binary-extensions: 2.3.0 - - is-boolean-object@1.1.2: - dependencies: - call-bind: 1.0.7 - has-tostringtag: 1.0.2 - - is-bun-module@1.1.0: - dependencies: - semver: 7.6.3 - - is-callable@1.2.7: {} - - is-core-module@2.15.1: - dependencies: - hasown: 2.0.2 - - is-data-view@1.0.1: - dependencies: - is-typed-array: 1.1.13 - - is-date-object@1.0.5: - dependencies: - has-tostringtag: 1.0.2 - - is-extendable@0.1.1: {} - - is-extglob@2.1.1: {} - - is-finalizationregistry@1.0.2: - dependencies: - call-bind: 1.0.7 - - is-fullwidth-code-point@3.0.0: {} - - is-generator-function@1.0.10: - dependencies: - has-tostringtag: 1.0.2 - - is-glob@4.0.3: - dependencies: - is-extglob: 2.1.1 - - is-map@2.0.3: {} - - is-negative-zero@2.0.3: {} - - is-network-error@1.1.0: {} - - is-number-object@1.0.7: - dependencies: - has-tostringtag: 1.0.2 - - is-number@7.0.0: {} - - is-path-inside@3.0.3: {} - - is-regex@1.1.4: - dependencies: - call-bind: 1.0.7 - has-tostringtag: 1.0.2 - - is-set@2.0.3: {} - - is-shared-array-buffer@1.0.3: - dependencies: - call-bind: 1.0.7 - - is-string@1.0.7: - dependencies: - has-tostringtag: 1.0.2 - - is-symbol@1.0.4: - dependencies: - has-symbols: 1.0.3 - - is-typed-array@1.1.13: - dependencies: - which-typed-array: 1.1.15 - - is-weakmap@2.0.2: {} - - is-weakref@1.0.2: - dependencies: - call-bind: 1.0.7 - - is-weakset@2.0.3: - dependencies: - call-bind: 1.0.7 - get-intrinsic: 1.2.4 - - isarray@2.0.5: {} - - isexe@2.0.0: {} - - iterator.prototype@1.1.2: - dependencies: - define-properties: 1.2.1 - get-intrinsic: 1.2.4 - has-symbols: 1.0.3 - reflect.getprototypeof: 1.0.6 - set-function-name: 2.0.2 - - jackspeak@2.3.6: - dependencies: - '@isaacs/cliui': 8.0.2 - optionalDependencies: - '@pkgjs/parseargs': 0.11.0 - - jose@4.15.9: {} - - jose@5.9.6: {} - - js-base64@3.7.7: {} - - js-tokens@4.0.0: {} - - js-yaml@3.14.1: - dependencies: - argparse: 1.0.10 - esprima: 4.0.1 - - js-yaml@4.1.0: - dependencies: - argparse: 2.0.1 - - json-buffer@3.0.1: {} - - json-schema-traverse@0.4.1: {} - - json-stable-stringify-without-jsonify@1.0.1: {} - - json5@1.0.2: - dependencies: - minimist: 1.2.8 - - jsx-ast-utils@3.3.5: - dependencies: - array-includes: 3.1.8 - array.prototype.flat: 1.3.2 - object.assign: 4.1.5 - object.values: 1.2.0 - - jwt-decode@4.0.0: {} - - keyv@4.5.4: - dependencies: - json-buffer: 3.0.1 - - kind-of@6.0.3: {} - - language-subtag-registry@0.3.23: {} - - language-tags@1.0.9: - dependencies: - language-subtag-registry: 0.3.23 - - levn@0.4.1: - dependencies: - prelude-ls: 1.2.1 - type-check: 0.4.0 - - loadjs@4.3.0: {} - - locate-path@6.0.0: - dependencies: - p-locate: 5.0.0 - - lodash.merge@4.6.2: {} - - lodash@4.17.21: {} - - loose-envify@1.4.0: - dependencies: - js-tokens: 4.0.0 - - lru-cache@10.4.3: {} - - lru-cache@6.0.0: - dependencies: - yallist: 4.0.0 - - lunarphase-js@2.0.3: {} - - map-obj@4.3.0: {} - - media-chrome@3.2.5: {} - - media-tracks@0.3.3: {} - - memoize-one@6.0.0: {} - - merge2@1.4.1: {} - - micromatch@4.0.8: - dependencies: - braces: 3.0.3 - picomatch: 2.3.1 - - mime-match@1.0.2: - dependencies: - wildcard: 1.1.2 - - mimic-response@3.1.0: {} - - minimatch@3.1.2: - dependencies: - brace-expansion: 1.1.11 - - minimatch@9.0.3: - dependencies: - brace-expansion: 2.0.1 - - minimatch@9.0.5: - dependencies: - brace-expansion: 2.0.1 - - minimist@1.2.8: {} - - minipass@7.1.2: {} - - mkdirp-classic@0.5.3: {} - - ms@2.1.2: {} - - ms@2.1.3: {} - - multiformats@13.2.2: {} - - mux-embed@5.2.1: {} - - namespace-emitter@2.0.1: {} - - nanoid@3.3.7: {} - - nanoid@5.0.7: {} - - napi-build-utils@1.0.2: {} - - natural-compare@1.4.0: {} - - next-auth@4.24.10(next@14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8))(nodemailer@6.9.16)(react-dom@18.3.1(react@18.3.1))(react@18.3.1): - dependencies: - '@babel/runtime': 7.26.0 - '@panva/hkdf': 1.2.1 - cookie: 0.7.1 - jose: 4.15.9 - next: 14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8) - oauth: 0.9.15 - openid-client: 5.7.1 - preact: 10.23.2 - preact-render-to-string: 5.2.3(preact@10.23.2) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - uuid: 8.3.2 - optionalDependencies: - nodemailer: 6.9.16 - - next-goatcounter@1.0.5(next@14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8))(react-dom@18.3.1(react@18.3.1))(react@18.3.1): - dependencies: - next: 14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - - next@14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8): - dependencies: - '@next/env': 14.2.7 - '@swc/helpers': 0.5.5 - busboy: 1.6.0 - caniuse-lite: 1.0.30001655 - graceful-fs: 4.2.11 - postcss: 8.4.31 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - styled-jsx: 5.1.1(react@18.3.1) - optionalDependencies: - '@next/swc-darwin-arm64': 14.2.7 - '@next/swc-darwin-x64': 14.2.7 - '@next/swc-linux-arm64-gnu': 14.2.7 - '@next/swc-linux-arm64-musl': 14.2.7 - '@next/swc-linux-x64-gnu': 14.2.7 - '@next/swc-linux-x64-musl': 14.2.7 - '@next/swc-win32-arm64-msvc': 14.2.7 - '@next/swc-win32-ia32-msvc': 14.2.7 - '@next/swc-win32-x64-msvc': 14.2.7 - sass: 1.77.8 - transitivePeerDependencies: - - '@babel/core' - - babel-plugin-macros - - nextjs-cors@2.2.0(next@14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8)): - dependencies: - cors: 2.8.5 - next: 14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8) - - nextjs-toploader@3.6.15(next@14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8))(react-dom@18.3.1(react@18.3.1))(react@18.3.1): - dependencies: - next: 14.2.7(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(sass@1.77.8) - nprogress: 0.2.0 - prop-types: 15.8.1 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - - node-abi@3.67.0: - dependencies: - semver: 7.6.3 - - node-addon-api@5.1.0: {} - - nodemailer@6.9.16: - optional: true - - normalize-path@3.0.0: {} - - nprogress@0.2.0: {} - - oauth@0.9.15: {} - - object-assign@4.1.1: {} - - object-hash@2.2.0: {} - - object-inspect@1.13.2: {} - - object-is@1.1.6: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - - object-keys@1.1.1: {} - - object.assign@4.1.5: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - has-symbols: 1.0.3 - object-keys: 1.1.1 - - object.entries@1.1.8: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-object-atoms: 1.0.0 - - object.fromentries@2.0.8: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-abstract: 1.23.3 - es-object-atoms: 1.0.0 - - object.groupby@1.0.3: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-abstract: 1.23.3 - - object.values@1.2.0: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-object-atoms: 1.0.0 - - oidc-token-hash@5.0.3: {} - - once@1.4.0: - dependencies: - wrappy: 1.0.2 - - openid-client@5.7.1: - dependencies: - jose: 4.15.9 - lru-cache: 6.0.0 - object-hash: 2.2.0 - oidc-token-hash: 5.0.3 - - optionator@0.9.4: - dependencies: - deep-is: 0.1.4 - fast-levenshtein: 2.0.6 - levn: 0.4.1 - prelude-ls: 1.2.1 - type-check: 0.4.0 - word-wrap: 1.2.5 - - p-limit@3.1.0: - dependencies: - yocto-queue: 0.1.0 - - p-locate@5.0.0: - dependencies: - p-limit: 3.1.0 - - p-queue@8.0.1: - dependencies: - eventemitter3: 5.0.1 - p-timeout: 6.1.2 - - p-retry@6.2.0: - dependencies: - '@types/retry': 0.12.2 - is-network-error: 1.1.0 - retry: 0.13.1 - - p-timeout@6.1.2: {} - - parent-module@1.0.1: - dependencies: - callsites: 3.1.0 - - path-exists@4.0.0: {} - - path-key@3.1.1: {} - - path-parse@1.0.7: {} - - path-scurry@1.11.1: - dependencies: - lru-cache: 10.4.3 - minipass: 7.1.2 - - path-type@4.0.0: {} - - picocolors@1.1.0: {} - - picomatch@2.3.1: {} - - plyr@3.7.8: - dependencies: - core-js: 3.38.1 - custom-event-polyfill: 1.0.7 - loadjs: 4.3.0 - rangetouch: 2.0.1 - url-polyfill: 1.1.12 - - possible-typed-array-names@1.0.0: {} - - postcss-value-parser@4.2.0: {} - - postcss@8.4.31: - dependencies: - nanoid: 3.3.7 - picocolors: 1.1.0 - source-map-js: 1.2.0 - - postcss@8.4.38: - dependencies: - nanoid: 3.3.7 - picocolors: 1.1.0 - source-map-js: 1.2.0 - - preact-render-to-string@5.2.3(preact@10.23.2): - dependencies: - preact: 10.23.2 - pretty-format: 3.8.0 - - preact@10.23.2: {} - - prebuild-install@7.1.2: - dependencies: - detect-libc: 2.0.3 - expand-template: 2.0.3 - github-from-package: 0.0.0 - minimist: 1.2.8 - mkdirp-classic: 0.5.3 - napi-build-utils: 1.0.2 - node-abi: 3.67.0 - pump: 3.0.0 - rc: 1.2.8 - simple-get: 4.0.1 - tar-fs: 2.1.1 - tunnel-agent: 0.6.0 - - prelude-ls@1.2.1: {} - - pretty-format@3.8.0: {} - - prop-types@15.8.1: - dependencies: - loose-envify: 1.4.0 - object-assign: 4.1.1 - react-is: 16.13.1 - - property-expr@2.0.6: {} - - pump@3.0.0: - dependencies: - end-of-stream: 1.4.4 - once: 1.4.0 - - punycode@2.3.1: {} - - qs@6.13.0: - dependencies: - side-channel: 1.0.6 - - queue-microtask@1.2.3: {} - - quick-lru@5.1.1: {} - - rangetouch@2.0.1: {} - - rc@1.2.8: - dependencies: - deep-extend: 0.6.0 - ini: 1.3.8 - minimist: 1.2.8 - strip-json-comments: 2.0.1 - - react-data-table-component@7.6.2(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)): - dependencies: - deepmerge: 4.3.1 - react: 18.3.1 - styled-components: 6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - - react-dom@18.3.1(react@18.3.1): - dependencies: - loose-envify: 1.4.0 - react: 18.3.1 - scheduler: 0.23.2 - - react-hook-form@7.53.0(react@18.3.1): - dependencies: - react: 18.3.1 - - react-is@16.13.1: {} - - react-loading-skeleton@3.4.0(react@18.3.1): - dependencies: - react: 18.3.1 - - react-toastify@10.0.5(react-dom@18.3.1(react@18.3.1))(react@18.3.1): - dependencies: - clsx: 2.1.1 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - - react@18.3.1: - dependencies: - loose-envify: 1.4.0 - - readable-stream@3.6.2: - dependencies: - inherits: 2.0.4 - string_decoder: 1.3.0 - util-deprecate: 1.0.2 - - readdirp@3.6.0: - dependencies: - picomatch: 2.3.1 - - reflect.getprototypeof@1.0.6: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-abstract: 1.23.3 - es-errors: 1.3.0 - get-intrinsic: 1.2.4 - globalthis: 1.0.4 - which-builtin-type: 1.1.4 - - regenerator-runtime@0.14.1: {} - - regexp.prototype.flags@1.5.2: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-errors: 1.3.0 - set-function-name: 2.0.2 - - resolve-from@4.0.0: {} - - resolve-pkg-maps@1.0.0: {} - - resolve@1.22.8: - dependencies: - is-core-module: 2.15.1 - path-parse: 1.0.7 - supports-preserve-symlinks-flag: 1.0.0 - - resolve@2.0.0-next.5: - dependencies: - is-core-module: 2.15.1 - path-parse: 1.0.7 - supports-preserve-symlinks-flag: 1.0.0 - - retry@0.13.1: {} - - reusify@1.0.4: {} - - run-parallel@1.2.0: - dependencies: - queue-microtask: 1.2.3 - - safe-array-concat@1.1.2: - dependencies: - call-bind: 1.0.7 - get-intrinsic: 1.2.4 - has-symbols: 1.0.3 - isarray: 2.0.5 - - safe-buffer@5.2.1: {} - - safe-regex-test@1.0.3: - dependencies: - call-bind: 1.0.7 - es-errors: 1.3.0 - is-regex: 1.1.4 - - sass@1.77.8: - dependencies: - chokidar: 3.6.0 - immutable: 4.3.7 - source-map-js: 1.2.0 - - sax@1.4.1: {} - - scheduler@0.23.2: - dependencies: - loose-envify: 1.4.0 - - section-matter@1.0.0: - dependencies: - extend-shallow: 2.0.1 - kind-of: 6.0.3 - - semver@6.3.1: {} - - semver@7.6.3: {} - - set-function-length@1.2.2: - dependencies: - define-data-property: 1.1.4 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - gopd: 1.0.1 - has-property-descriptors: 1.0.2 - - set-function-name@2.0.2: - dependencies: - define-data-property: 1.1.4 - es-errors: 1.3.0 - functions-have-names: 1.2.3 - has-property-descriptors: 1.0.2 - - shallow-equal@3.1.0: {} - - shallowequal@1.1.0: {} - - sharp@0.30.7: - dependencies: - color: 4.2.3 - detect-libc: 2.0.3 - node-addon-api: 5.1.0 - prebuild-install: 7.1.2 - semver: 7.6.3 - simple-get: 4.0.1 - tar-fs: 2.1.1 - tunnel-agent: 0.6.0 - - sharp@0.33.5: - dependencies: - color: 4.2.3 - detect-libc: 2.0.3 - semver: 7.6.3 - optionalDependencies: - '@img/sharp-darwin-arm64': 0.33.5 - '@img/sharp-darwin-x64': 0.33.5 - '@img/sharp-libvips-darwin-arm64': 1.0.4 - '@img/sharp-libvips-darwin-x64': 1.0.4 - '@img/sharp-libvips-linux-arm': 1.0.5 - '@img/sharp-libvips-linux-arm64': 1.0.4 - '@img/sharp-libvips-linux-s390x': 1.0.4 - '@img/sharp-libvips-linux-x64': 1.0.4 - '@img/sharp-libvips-linuxmusl-arm64': 1.0.4 - '@img/sharp-libvips-linuxmusl-x64': 1.0.4 - '@img/sharp-linux-arm': 0.33.5 - '@img/sharp-linux-arm64': 0.33.5 - '@img/sharp-linux-s390x': 0.33.5 - '@img/sharp-linux-x64': 0.33.5 - '@img/sharp-linuxmusl-arm64': 0.33.5 - '@img/sharp-linuxmusl-x64': 0.33.5 - '@img/sharp-wasm32': 0.33.5 - '@img/sharp-win32-ia32': 0.33.5 - '@img/sharp-win32-x64': 0.33.5 - - shebang-command@2.0.0: - dependencies: - shebang-regex: 3.0.0 - - shebang-regex@3.0.0: {} - - side-channel@1.0.6: - dependencies: - call-bind: 1.0.7 - es-errors: 1.3.0 - get-intrinsic: 1.2.4 - object-inspect: 1.13.2 - - signal-exit@4.1.0: {} - - simple-concat@1.0.1: {} - - simple-get@4.0.1: - dependencies: - decompress-response: 6.0.0 - once: 1.4.0 - simple-concat: 1.0.1 - - simple-swizzle@0.2.2: - dependencies: - is-arrayish: 0.3.2 - - slash@3.0.0: {} - - slugify@1.6.6: {} - - source-map-js@1.2.0: {} - - sprintf-js@1.0.3: {} - - stop-iteration-iterator@1.0.0: - dependencies: - internal-slot: 1.0.7 - - streamsearch@1.1.0: {} - - string-width@4.2.3: - dependencies: - emoji-regex: 8.0.0 - is-fullwidth-code-point: 3.0.0 - strip-ansi: 6.0.1 - - string-width@5.1.2: - dependencies: - eastasianwidth: 0.2.0 - emoji-regex: 9.2.2 - strip-ansi: 7.1.0 - - string.prototype.includes@2.0.0: - dependencies: - define-properties: 1.2.1 - es-abstract: 1.23.3 - - string.prototype.matchall@4.0.11: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-abstract: 1.23.3 - es-errors: 1.3.0 - es-object-atoms: 1.0.0 - get-intrinsic: 1.2.4 - gopd: 1.0.1 - has-symbols: 1.0.3 - internal-slot: 1.0.7 - regexp.prototype.flags: 1.5.2 - set-function-name: 2.0.2 - side-channel: 1.0.6 - - string.prototype.repeat@1.0.0: - dependencies: - define-properties: 1.2.1 - es-abstract: 1.23.3 - - string.prototype.trim@1.2.9: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-abstract: 1.23.3 - es-object-atoms: 1.0.0 - - string.prototype.trimend@1.0.8: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-object-atoms: 1.0.0 - - string.prototype.trimstart@1.0.8: - dependencies: - call-bind: 1.0.7 - define-properties: 1.2.1 - es-object-atoms: 1.0.0 - - string_decoder@1.3.0: - dependencies: - safe-buffer: 5.2.1 - - strip-ansi@6.0.1: - dependencies: - ansi-regex: 5.0.1 - - strip-ansi@7.1.0: - dependencies: - ansi-regex: 6.0.1 - - strip-bom-string@1.0.0: {} - - strip-bom@3.0.0: {} - - strip-json-comments@2.0.1: {} - - strip-json-comments@3.1.1: {} - - styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1): - dependencies: - '@emotion/is-prop-valid': 1.2.2 - '@emotion/unitless': 0.8.1 - '@types/stylis': 4.2.5 - css-to-react-native: 3.2.0 - csstype: 3.1.3 - postcss: 8.4.38 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - shallowequal: 1.1.0 - stylis: 4.3.2 - tslib: 2.6.2 - - styled-jsx@5.1.1(react@18.3.1): - dependencies: - client-only: 0.0.1 - react: 18.3.1 - - stylis@4.3.2: {} - - supports-color@7.2.0: - dependencies: - has-flag: 4.0.0 - - supports-preserve-symlinks-flag@1.0.0: {} - - tapable@2.2.1: {} - - tar-fs@2.1.1: - dependencies: - chownr: 1.1.4 - mkdirp-classic: 0.5.3 - pump: 3.0.0 - tar-stream: 2.2.0 - - tar-stream@2.2.0: - dependencies: - bl: 4.1.0 - end-of-stream: 1.4.4 - fs-constants: 1.0.0 - inherits: 2.0.4 - readable-stream: 3.6.2 - - text-table@0.2.0: {} - - tiny-case@1.0.3: {} - - to-regex-range@5.0.1: - dependencies: - is-number: 7.0.0 - - toposort@2.0.2: {} - - ts-api-utils@1.3.0(typescript@5.5.4): - dependencies: - typescript: 5.5.4 - - tsconfig-paths@3.15.0: - dependencies: - '@types/json5': 0.0.29 - json5: 1.0.2 - minimist: 1.2.8 - strip-bom: 3.0.0 - - tslib@2.6.2: {} - - tslib@2.7.0: {} - - tunnel-agent@0.6.0: - dependencies: - safe-buffer: 5.2.1 - - type-check@0.4.0: - dependencies: - prelude-ls: 1.2.1 - - type-fest@1.4.0: {} - - type-fest@2.19.0: {} - - typed-array-buffer@1.0.2: - dependencies: - call-bind: 1.0.7 - es-errors: 1.3.0 - is-typed-array: 1.1.13 - - typed-array-byte-length@1.0.1: - dependencies: - call-bind: 1.0.7 - for-each: 0.3.3 - gopd: 1.0.1 - has-proto: 1.0.3 - is-typed-array: 1.1.13 - - typed-array-byte-offset@1.0.2: - dependencies: - available-typed-arrays: 1.0.7 - call-bind: 1.0.7 - for-each: 0.3.3 - gopd: 1.0.1 - has-proto: 1.0.3 - is-typed-array: 1.1.13 - - typed-array-length@1.0.6: - dependencies: - call-bind: 1.0.7 - for-each: 0.3.3 - gopd: 1.0.1 - has-proto: 1.0.3 - is-typed-array: 1.1.13 - possible-typed-array-names: 1.0.0 - - typescript@5.5.4: {} - - unbox-primitive@1.0.2: - dependencies: - call-bind: 1.0.7 - has-bigints: 1.0.2 - has-symbols: 1.0.3 - which-boxed-primitive: 1.0.2 - - undici-types@6.19.8: {} - - uri-js@4.4.1: - dependencies: - punycode: 2.3.1 - - url-polyfill@1.1.12: {} - - use-sync-external-store@1.2.2(react@18.3.1): - dependencies: - react: 18.3.1 - - util-deprecate@1.0.2: {} - - uuid@8.3.2: {} - - vary@1.1.2: {} - - which-boxed-primitive@1.0.2: - dependencies: - is-bigint: 1.0.4 - is-boolean-object: 1.1.2 - is-number-object: 1.0.7 - is-string: 1.0.7 - is-symbol: 1.0.4 - - which-builtin-type@1.1.4: - dependencies: - function.prototype.name: 1.1.6 - has-tostringtag: 1.0.2 - is-async-function: 2.0.0 - is-date-object: 1.0.5 - is-finalizationregistry: 1.0.2 - is-generator-function: 1.0.10 - is-regex: 1.1.4 - is-weakref: 1.0.2 - isarray: 2.0.5 - which-boxed-primitive: 1.0.2 - which-collection: 1.0.2 - which-typed-array: 1.1.15 - - which-collection@1.0.2: - dependencies: - is-map: 2.0.3 - is-set: 2.0.3 - is-weakmap: 2.0.2 - is-weakset: 2.0.3 - - which-typed-array@1.1.15: - dependencies: - available-typed-arrays: 1.0.7 - call-bind: 1.0.7 - for-each: 0.3.3 - gopd: 1.0.1 - has-tostringtag: 1.0.2 - - which@2.0.2: - dependencies: - isexe: 2.0.0 - - wildcard@1.1.2: {} - - word-wrap@1.2.5: {} - - wrap-ansi@7.0.0: - dependencies: - ansi-styles: 4.3.0 - string-width: 4.2.3 - strip-ansi: 6.0.1 - - wrap-ansi@8.1.0: - dependencies: - ansi-styles: 6.2.1 - string-width: 5.1.2 - strip-ansi: 7.1.0 - - wrappy@1.0.2: {} - - xml-js@1.6.11: - dependencies: - sax: 1.4.1 - - yallist@4.0.0: {} - - yocto-queue@0.1.0: {} - - yup@1.4.0: - dependencies: - property-expr: 2.0.6 - tiny-case: 1.0.3 - toposort: 2.0.2 - type-fest: 2.19.0 diff --git a/services/next/public/fonts/nunito-v26-latin-regular.woff2 b/services/next/public/fonts/nunito-v26-latin-regular.woff2 deleted file mode 100644 index 99439c135ebd42eb92bf9a61c57980c7b5af816f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 16292 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 diff --git a/services/next/public/fonts/overpass-v13-latin-regular.woff2 b/services/next/public/fonts/overpass-v13-latin-regular.woff2 deleted file mode 100644 index afd295f42f6247e08da37dde0db5c62ad1e74483..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 16996 zcmV)5K*_&%Pew8T0RR91077H{5C8xG0I2`~073u&0RR9100000000000000000000 z0000Qfo2nvCdMEC>?gkk^zHUcCAi9iG(1%yloi*O7F z8^)Ih#M^gKKz9I{I#-v!D1wax!qE6h8;P(PoYYv8|Nqa)jWMJ~^!|G&D1%+BO?8|u zCWxKL#6mixg)G4*E;zWM(^6XQ%A1rQ{!ZH(R)>ZRQG4mz#B?856aK^esEj6doS&!fDl3mF+7S8AcPPwR56ASilN02AT~-j62yR^ z47wS_pngB{Sq2f2X$-@%Sl3~(H1spdVn-Gm)0xh+CV%Ptd{6lQKEZjey}yAs1KWp0ed<&}-7p>ayqc)9fXo+Haa>^Bo|bL-0N~;Idx4$1-VdX+ zh{hNc?P-!u4zNrTSpN>e+&bm4FaaH)Naa8x2Q1lG6e~t;W5m|vh!G2FWC=Y6BF@5U zjLi2#(XYS^0093>_l-)cD>19J`gKYss0XQnR6#5)J6GnUTEv3bBDRP<#1^p!vmbfG z|qbtv*1=6+Vac}zPI)Pqdi_i$_^o;kTD276YXVw zm+fZR0br1Y7eoioe|vLwZt6^^MeU$k5sn~C6$;Jqf9I#Y46}v439Su%aUa*Zk8lW$ z+MyQ}6{E3LOQ7HEAu~@_FTb!Mgdk$2$NHW4&)1XCKufNQ8ZjbbbY~+XBD(+k-rLtw zo#|Ls4wRwf@oR_JWqdju6}VZdQ&->*APr*`+rhu$2uu^$0*nO&4y+8Og3$<`XYCF! zM1ndX-vq5yZIOj_umZvIZ-B!ywJOLBT_ZE($_G>nnT_1Yu!AxVR7(E)c#a zA{HYOU&P-ZkxG$}P$Vn@QN<$h2}n{hl9Ga?r6HLHq@V!VxCJRJM9RvLib|x~g49?M z+b*Q332~en2hWe5o2q+n%>;GeNt(i2)d3CxL&;6JbzbLLy%G2yl-5~*|E0iI1AGLa z>r@~x_`x65)qvmc)o2iZCH0K~A1IV%BVZnEZICYFf&q!n6+%Lg`)VLdhiXqsatHm5 zjBthUW>sL^JsjuqOTB!MYSTi*R z>T`7g)Q9S1jnrE;0P5A%1?u_LUVEykz5;=IOx36dRI$2Cnba*R{lU6UCDxd#tg;$d zsVV|>eB}Yu!8is`{%aEZxw3O9zbLAF49cfD2g-Xn4a!S-B`5N1j^%K^k^P`tnVq1V zn{Dj#*P8ZHmO4N=B+Iif&1r;Zvh@v8lL`a^bvQ=fI`j|7 zZ`rT@5R0exWju}9xEr@(GM~pK)JNj<4S7clf!rK@AeYCXaE2pVq5+U208%RR0#x?E z%Fkq$iNCy~98;saIuKX0p#%UJeA|D!Cd&f=V%SX$E8%!JqnW3umgKAVfjPyt{CW33 zw~EXR)dPZ>+7AG+m>huuBAtPa!YBYURVK}zZcn4ifvF^Gyc__^vRnoru?#R3Kr9CE z&ustN`WbUZC?F>s&1udMW!)taU13?A|iS*f*VmZ>E6g>}LA5O|6uK9sM&bapijlZPuJ zq9Hma)~QKajizatS=l+Jd1&LN&32&?%I1|?Z5!D&57le1>+W+nhys2CegM7#z63r7 zJ_6nWUIShLz5gP}j$dCeDDLVo7S&aYzVkP1cC%~u-xan!ENqJd-DbLtD3{;yo+03P7tCSKV zWHL(&S^BPfnMqv^3oK;Y$5NRzw-YZs(8|w8NeFN=UmmLawO~2oUnubI$;CJf7%5A+ zGv)!dL*%ehd!+4(K5IXM?ug%^-p#EwooVOm07r1lrcUKBYMn`uBVPuOkS1P86fHSF zLzP~*)pX2JaW+9TmmYcxO;CgYj*T*EX?WshzF+@_C|B!J{~Z%j%x!*ECOGwKj=_M|96vlmf{BpnE3HVeht_4v7S;XL<{ z3?GIfku<@85U7u8>%{of#B2c0e%db!Z}`@Y`N8}FRpvGKbQ(%eC56&LV9OXOoci(9 zR}5@goG8%UL+qNVI5)o$;4;f|VE#27Ok+@72(F`m))Z)6aOP<*Mb)w` zZf&jy5X)LJMFh2P`#f2!^r4*#S|s!%daCdMfz3;Rat2M-g=j=zm0*yNp-X(&0+dr1 zG(TozRf72%RokNAw(cUZ!bZ;0(&3J~FJ3&$VFIxIrJ}+DDZwH=P|K!5jF9M<&LCVD zb=*n!*usl*e0yHr4H#tDEPE#gX$Rp#M4@JTh=HPXB!EV|7?|1tTXKlN10t6=L9x!G z4Pe|Ht>rc&Ai}DJ*^|XtUQN=XXfb{ZQVS1Om|MY?;c1A?l~nTT&=fUlH!Bfj2L@;s z%T_==4HW?4+A=cX48nM%n<}00RwNG~YRWhVCw&wTMuFe~AmY=2tej#*ccG75iYeor zl%rS(>YU2Oq6ljj$UqzhBm_)MLf44fE%XSAYX%zwMQxk6X>DNovH1(t5Y2CW#5bTLVxlsk(T$`)MiVA^wiW&=kHTl}0%(%~ec0Yq%hYu7Oypi11)&{{1Qita2a zo&z0&QmJgccTORuKPxY|!w@1NV@YZq8I)-5rdH0Xiio)N)-lIPW4LPB!nPbjrWI3# zyT@|fi;9}q_T_-GjbgBlOWUY27^8`(R-hWtvN}1P0}oJ%X*;=gxyzpq7>#T&C`pS) z>L;UO;shm4n@ds!UculToeETkym#3EeD#k-SFSin;f_KpgqMe5yutF_5ElG3;pR;u5%J3|17_3@9Cg2xn z*v*G7iHB3(x=2mPD)3ZG!}r^Ov-6jic^QsqCE{d`H<2t>!9mHqbFFu-W$sH9Py@v; zz(P>lAft)~vACN-Yal4CzUH^mPOKn(h32Sqgms`>7|r+r)8MfBNbyeWN(tI86-v{= z%6!$LT%u$}uCZtUF^GczOIhfI@dXe42GD0Xf*H)l$LPW zlgiMndws&+aD)IjCYKESR}|BAN$d}FBa8pDoy{T5Oe#{1Io^QnXHaQv8vKdaAT0Tg zuhw!_y~&?dGJTd+LWs)tvyToE|44U-%`!aV11mN^lQ%66euS34;0B$ZT}%hZBi)3r zT6_pQKaz*VlHZ&t4jjiYKOuDGw710!oY_C%8=1VaVS&(&2-cSd0pd^ha-h7Xxaf4F z+?#B@S_u6dg)Yf5f@HJP=c(fGf7+0KcNwlJ$5Pq z)Hx&e-u3Rm7+A`Lx(kJkI}HxE<(_*eYu$=T$UGnj4QX_UB*5hw(F9^5@|Fn=!iN?i zmJgI^@iw;kwumq8MlcAF50ZXbCP&hhl?g6RD*qW-m)a75H2@LO+n%&~>9u?%C;&q+ z9{LLdNDK$?a~cSQFAyS$c8EWMs{(njh2%0o*n#)m9k_%-T+%PrgNiJf8p( z{Zc8=Qe0*TCd2}08dTM}kY2=jRsj-iWh8=OavtGjRuu)H-jn}}+g4Tj7XUL;|4k73 zh;#CcXv6-~b-;NXx4TCmEz}y$tY|*^qdN(GrDg^6C4FJ!_g>DAzVC}!NV8L$`*P88 zF*FtBtzPE5410P;XFxjtIIQUy+uWb(DJ_Mxg;RT3&$ny6_LW)L#i0>fRew)bcG2nW zF*T$LD~bJ@-{fLm%HD6ZGCv+C;|=jY!7QBO0~iU31V-RN_UrlV3@9J2Rd(sjphWn0m*z`9y3*0!jgUD#<9 zN2hiBRybGcpfjafa4AGk2at|COwA;($Er^BQ=39%~y}A zgp?t-WsqC1i^w(KNv7cLQ%K{bjz5*S^59EIb05xEkf{{HGG5cKrkO8mDX4ujE8{hd z?3MH1Hmn!wuhj7nfiJ*$N5Di@Ya{&iXs^JQdwew-VJx*gSZ{SOhY!4WAxp*+SCC1Hb@H(hwUz(s5LI3{DpXs?jm+r&UFqPCQ|E*PG)XVQE z!sN$XV*oqZD$vC5Ow<1_b!R%VLlYqpo7435eOx%Q!MZrWfXjsmlQSA}VsKoUZ3ICit3(9S(;2k;iC8XLLavz3x?H>WVx2E$!KZV3onu@$79%~a1W{xHi1D33uFWaF{C8n|-tKTE%pGv1 znDQ^>Lgp$0`&AB!_@vyc^I`7;&e4J1gO9Jz$|NwTNdjWW$B1E8GWRP-3St5XL+1An z+92SPW5`NEmSL0tJs&lKR~P#pEUqFxG*H2)6*fUV!AQ}x6MN>s&fskieT$ z_hT#Jq$D_#(?^YoXM82R4la7Ik0OMh+5|AP@eROF)}f86h*5yD-|oS!v^TH{8YH&N z#d*aF1_ht$gdF)Hh!;B0Y@ro`*9Ew&fW%jDgHvF6sD#D|FnlrxiB<8B3O^V-qLYV4 zC1ECNVsb75q76nT8t9F0CxI)&SEoE-ILr9WRbaLoZ=GlKCb)vz^`1}vF91V^V?VwN zT?MEEV+UsALbRv_#s|LnGadg)r40f2W$PAlZ&pD(bbTy*pjKcLs8h~3K#7R){2}1j zL^%y?K)oVA_`=0%wVXr|*^n3VX3po+LO6l5aR1pDHG~>Qji*GEgz}>TsYoi0(mLCm zM}1v=5BqldS^Ie`_wF)ApeV!{%vNgH5b%mK*BwU*De-P-Oy)4+g*zZNcj{EGm5{>P@zFPvWTD(aQt<=q#pd+dv^pMHlr4Xvl{<%=WbcHYX3 z7sC^sE2Ga>`!D#9uRZivc(CLf&ph`aFnHgbDUaN8JD7-$7;wk8&`)S_|LH8BU>}(E zVQ}DS1WNchE#Cay-TqGDh2Ja$2{+96!?OPZ@Q?4a$`AJVJlODGKPJ@AVZlF)Mo_r$ zoqHiO;%~li#tCQLMTa3U(J-)ZxO26F8*f5hM0orJ39w4Az#u?sHKj;V;>c{Lwnie2 zwbn_ZlOf%DdU-Z5%9U!XQbmfDD+wBu*{RkJb?mA%3Q^H-T;TWWXBK_r6QBARg1UaA z09h7L_5hg;#{L8_{RsfQ0Z`8dXt2(w)H9*Lp)g|T{B=+c8uB=Heha`QaVLqG z!&F8Bhr<=6ahxJNn5Go2AwNa>Jq_*Z4y-C?=S4wrUSZdu2svoP*aP~dckJ4&bJKnOsI#$`t1vEX)(;w{Fw-D(}M^wH=)-OO&D^xMZ_h@obo z8N24A8DF2*CLz`N-jo9Mq-s*dk+)mJ;SkZQOa>;k)I#S6`6iohi&4@TrW=hv`^%mx zNCld5(xg~BXmkJ(dv1SO<`oX^hmj;b43U!c5_I)uRR!87nl zn_e^X?b&ElNZQ9>89yGhmL%8Mru-OHG)sC&X_ow{M$!2cd#9L*;P(VxJ5o@3qwG&A zFZuvR16M#0+BkSj=Stx*Pmv*se|%MZTummvzMtNd-^5~l3z3j#hq+?WVc88` zq0}SZy{3hcI-Hubn`hZ})6-mrU%oTLtXAR)>!L)pF#x>I^q@AzjZ%W0GPA<>92j)PIt}AsWF61+0VwlM&w4})~Ae0jc)Tk(y z)dJ_)E<;)){$@gGHGcpgU``@Dv^#!KQI4UTo@vXJlLV_}YiVtWc7-kLF~K62oT9#w z&O&)9V#U5Yx-KeTd}g#vOC>aK!*ZW&EuD&#cg(SQyK)!^4m<1Un#mI=oe>VI?bzpZb7|y*=u{%dqDXQ zUmZ1Ba@slTFGw^4VV5Q;MMvjScIhqMUEN+xvw=uAk3IvbJOvWZW1HETXdJ&18NHThS(SSzf_)MYZeI zp)r1kiCyyMY@6Zq>_d)8=act(6UlN4A4YJOI3WX0r#^i<9WHt}A&Y=PO6IOAmsc6W zG#wv2eq4E@!zY@Aio#t(4ZC{J3pxIwMTU~O)?W)G5kuJlWlXW4mAwjl%#ZVqXbz}U z-1x#qO?ecWudLvQ!~EF)(RuENTLC(60)FP>oL6TRn<^H~ip%$_+SoVA=r-}bh&iC1 zLCUNUeR$S;k-bcjWWo)Y2LOjGQ#HNz_LyPPwMzeHT$>*vAuDbzR-vW3Z9+S zp_Ko*|Rr!ShqAAq_0V(ruEACnrB^Xx+(aEy($R$huAx5AU})DAaN$VQ>!nyKE{ zbF@>9nd*4fPif}n$0n%IqrjWJ<@z&&vl^6TFF&?eSXyk<+8fIvhP1UHI5(3{T;nMf zHs7oPugn-S9m3!MPjkFL;6v4fgml5FUrfm|X?1mm18i6qSt^q(iW8m|dE?&7+w{$w zb)Lh=D)EosQGqT{!>_;nK@j~K8}KL6hN5Bq)ILP|yyR>tZ3ftAg9(jT8n^TWDw0F( zYK^ffJ;l6VuQHt$n9as|bzIFiw~uEHzae7$#(=oX}-6jr_51_tp!x+<(A}N zV{?2!TN}rh8QUJK1?B78x>37sML(RyYioMQvSrB!I>+Z4ifBt7%D7kqZ%oBbY(O|N zR2x$k3lX^-RymULM&ntycBQw<-P!HloXJf=>446+1%O7o=j6&p=H5<2&8m;)>O5(t+=exAiVO+K|2gF?6 z^ebuzeQM`rkFyy30POrb=yTrwO7@@VfbZ+cmP0|Fd#{G{E1mC4m+4$O`ldcQ`hV@r ztXuzM{VQ3Abkkl%=)VaeMzp+08iz#0sEB4zQnKU*a|1$^zC!`fL0{iNXuuHI68yeo z(dcSs8`;k&rvKPa&amvi>h^V44$T3v-bufhPDrRb()iMTfe+0tpAV~{;h@c;A~RMa z7*XZqq5Z{?=&+)u1bz6hb9Yy1W{78o94;n-tB{fB>bhHTOZHYHfZ_JD@J~%sIgoWgw zqL>K(;(|o6%Cy5*OX{LdA2Hy5$wyi(Hr!wJ*1|YfOGlE_4g{Tj7LoZpOg%}$ino7% zz@M~kov>Ta<+WlNF+b&+4w(N$@}53jKszX8AVynD3n6~Oc_&VI(m$0M`SIJN(V_l| zJW@_2NP(1{r`0!Yy^_8S*|T^ zB&?x2QKYOCu#rr*0ZwtmZ|=gDXAwI@+T#5;)4`WJwWgz{+B;x3WcGQL^!Ggyv*cjk z)!oVtCT>PIx!={puU9l2#{k4uOZ!pVx4#|OMChrt#SPn{e>>{Rem% zY8;$rgFmuP-_fvcqV4Fn7?KsT!Rl?xJ{wDW!}_xBg3M?fXvhJT;jZZDsSqfqn_dmT5E&nNsFdewZYXSZxgrSM zGqI4Ub8bs0mMZe`gJI~vpmtL5d?f$ify0MP#`MG8F=4&9hIrndK&=}rSDVR6D#!xY zA)--Ia$kOtw0jpe6I1Q@HbE_u_}m&VL%S z7mjOnJyPD_E;_I@nrWFi)nf@|I8Wp&sX88~Z8|fjL{F8b11O~SYDwOm+9uWAi$!W# zOWGRiCdI83TvZ*1B`xqObVOP-rCQAT;XDdTxg0@GlX9T&cT?tnms+~V$mJA#J)4Vb zs-V-zYDz+53Wrc%P6dIGrmF_|*TEfmpRZQb7o5t@e=uzq;5@f771vlqp&=E}Z&`Wu z(}aqg)ttzBIhm>)ZUwKhhx>+S6osYYkTl4}jtTSzx1iCerq6w!v%$+9jJmqzvr&m+ zT0@F@w(?j_&X=(X(ZUIxmRuWohiQHzk1`f3=VE;QQRP(XJqH~LUe|K!U_NH5JO!A5 zC36>beE^~w!K@jTch~_(7-svGaC4S5~H#%;9Ef8 zV!tsy2L>zvtDSUx%O+Kp5Nlp`c;>bxn-H)F4TK{CpbOieH|ng{2a(BAEMRi-=`JJk zZdq;I{ryRpxO@8zutxyL+~B~pXyzYnh+%QtC|=kA+yFU!^6Zse!)BRAVi2M-(BWnD zY_m+CU7M9%)Lf=OIcre>LLbLEy8qP=B(p@D-yoLcJU}6LzDoX_lrLP-H8x$*$QCBY zJM+sGZNr$hTm8(JRVU4xo^`U&Mc#WY;gR+NPpI>2c|iDaQmw z7hW%YYY+9Gvuf)!s*uWnYvQ8MvlxE^J#ycuhiSp1vpI@C4TdTkm)g8lNvKv@- znb~rApYFIed%Qv44CW$%RcO)|97nf z!ADy34#~oa$_imAu62CjYP9-P(~XKVI9x~rLCCf*vhrw({shDgVai%Ks?`!ILqD?3 zz!EWX6{W_8U>w&cdq)Xb6t$yr7P+$)d4L2hC?6lTF15)JYTC#9t!-1Td78#lhUk5oqC z%LM4CQawrGFzS$1nJE|M6)J@FEt8Aj3`DG`P&JtPOo_V~rvLi>qv3;55m-eAZu|IHZ=8w7jed^(%n=A28Rv^)DAA(H0Gyi{R=BcFY>A;dh`&>M9pi*P=_UC~M zE~L{ryPq9z9Zbyk-se~IYd@4vDRj*^=u^ya9gK;s z2=X()^1{~+phXbhK<4K35Ofj++;%Zve}C)d^^fK~*FWC8b^ZPBZdqP$eN$eyEW4}<{Ri_Dw{U`2oQ*Wuol6&O zzJMB6@lyGv*~lsT`ShI47crwHJP@7wqbgpxm5QTO0?QK6)IhmYJvymo1rhwTd|ELz ztFyjR+mP#+x;@@o^V_t!=~9;DK#5v9vqzf^V+A!!cvb6Y?`A+eWcQcJ*pV@0%IjB6 zBeex3eKm$mW3sZgOeoT;WhreupOmrK(E5*k17#=qvMH&6H_q_O^)ES~R+pW4ECY+F z*F;d^oKbUO@Iq?$pC21QJS1;VfEsI)QzJ-?QoFG;`x%_(q$F&$hQJm!W14ot=;wYO zXNFCt41}-@mrCVV%WJBh{ip{i5Y=9~sxx}N;L+|9!itZ91L`eLTLy1$sX99pms5I5 zrMOsHQ+nf0QSAh$8jG)D>Bk1E*)>?Amct@f)nGyTl6UmMxrUxghsVo?^}SQ1G=h5o zXaWLA!(B!3=`wxoDUp8@mE$dkXXO|V2}(*23v!HEaBn%4Gb@GGoYa>UpT4JPm|*Hr zL^X?uuhbLSY#lM7lE@;cbtuL$u|~xa>I<_o^@T!Cr3$1##!EKd(Y4=$v~~<)q-AIG zb1zg@tFJ#OGK}145&kytVNxISZPih>9*5Vk=)_7rj>*yC@Rf8nR-?lL+fk~(_JO{y>>)b{!OgOU6aP3GvavNIC|g35^TE1q4Q7U?EsIyawvZ>XZtL zoAlYC5t$C(6GS-mcoR5#*2FvzQNBV_Eje2n|2Y~Y+hjJPzL2GBc!9Bu)HE!Hz`#Y5 zf`kP@foHDNCPGf#CD?c+l@SpYB(2#53T*h5oVe~(gW}zdfmS=0wYWqrERh6{r;_4OWHJgxC4!T0b##7HeXV~rj-Gxv?8K%$1W;t78|zfj2-4=_ zFjjbWb5(9$LsfQ2a8g|bUQSUBcj_vTG%5;4Wug!aCJe=-#%&LO1`Ylc7KDloh9Y9X zzc=WPD5BXp5Xv;c&X=OXqDsls1ho#!7HC-Tj+A!!r>ng-M=@k7C#Q#srjwBIBog=@ z0((Dggwp^o?W3vm@Ng<48b)K_%SNMvo%Tac!3ZyoYX==4g`|?gVX3TSa%fGmXd~Q# z1x4Q^P&Y*=wjlGx2!~bB<`{B_%!q+edZEw(+CCzI6chp*i3rD&LV|cO);KLLM0jNU z0R_Knqc{im9E5Qkw_}{>Q0p9w71@uNv&`(NMs!U7^4p!`FN$eQ6RfQc{WbI)Cs84Sm>1hZ03LhxZA59fKbgJ~+lns$fUBmhMgf~qBLmYgz4nygty@0~_b?b?pEr|oL{ z+8MtfK&M68(!MzAUm;U+1)<6vPqgl`Mq(u#*@vg^_gnNREtIXAvVn|}^!48vSIhwC zytESZmYg@I6Foygu0@aHJ4LIuWH7CSyL!b6w7U?oDGOg#9f96Q_m1pgH-?8A;`Wxv z7`9@#ZI>-PLB>(MfAStYv;e2;@aQTq%vGa+XYs`kE3Ni)PL<1p5S{^DqK{q@_1tkq zPh23USSzfKPGkHq^GNu-w=zz{MWLufSqUQ{z^;q02Av2zEZPcXczq{5z@*0=$awC+ zs&mz-6WIlR3;h3V&jx-=|MH()Ah=TEDL+;Sn8;br_HcQUN3ZbreT0qx%(v|QpKBt4 z@DhCdNcam=f%TP3E9Ko6eFrDk&UWzMjA(dJtn;(d-3iEKmoTP|P3X_cNIq#S&n3%9 z5Yon~K3uy{w87(#tw9W7@D53X?#c&6PtC)WzDfFX-FWq%e>Ixpn>Q%kw)>T?6um_8 z$)RctCrK2K&cKJ2CM2=N;D1Q*94~9ONWvcIDB=dwv&hX%9=EPHAw9j5!fHq^=`iME zEu%*4bG=MU$=mjar7Q{g&@jp5z7Yz>7(%n`W>$x}#SCxk)M-}*zG2$&S zObkrK9mY|&61M1YR7Db*qWlr#lTeo|5Y?LFB|XQo|EUqbEJTXa<(T*oCZRcoxs|cC zjgVB9gO8RiAhtzn=_5~PLx$S_gGMA08Y7uo{w5-Wp-j|B99VtYZeW>weqO$8U;woH zkY(}$MYnaUhkB@hbiR!3H<(n~fKAWuD%i@XimFLOV(2M!&pI00r!JjLg{LAQC^gh2 zdq}08(%k0`kFy_FY1ogVHqM}#SvAPY1Y1AE*0)K) zkzauo!kn6dBTswGJ*(agM{WFnQplwqSM!>uPiC$9+^)==m00!WdMc)1$*BqPx|*{) zXLV=P1QFF1HyP$q*i&6-brw7>*6Iz7%boXAY0d^KgF#(YQ&ZpO&t$av*6a1!nwpw_ zo$6?bu1(&Jjp~ZJy1F($W@3v^ojohLmO<#?r4zbp*~v-OY(iH$kI+%g0AJ6|HE+5! z>6t$P^G-G9=9V?93TP~OBa!Neb!b8qB**JgEBBrrvmMr!s?lhJWhn}!XiqZI{@r%(BqKH^;srS=A zHxRl4R=Tk)$tRN%|1p6VH^O_utHXPOcm6$(WD!{>cmM6#<6Z6D<9%a7avA*|>dL3f zb$fK0X^jr-=1$K}gV^sYDu{SIuDCd-aQGC(RV``K%hehZT~7b$PTjbcs*#Z(L{p zm;_`zh(41v^Pg#!26kMmyJ$3A+~@Vl>(j!#KT)Z;BOfi<7Mq{BhCKaXWLZR9Y)N2B z9|(cK&QDLny&LpW)g+CZU`D&B2@*Amn^YQ0$Z@SeL_4eTk}?$65KWBn$cRRG2uKl8 zF~n%Mj2O7PfEWo2krXvTZ&-#(;+7#od9ne9-B%_?M8l9f^0;iEc!|F6SH^zMicP}R zKQR{G+t|FPFZ|B26D+UTujz)-AbO0%G+j{-WBeMWvAY8r$-q?0cY58rz~EWirY+EU z;G=V8;qpJM)?Q(nv*4y`?_@2f=CW6ttolW;{5_BpJvH<+TWdr@WA1Vy)-lAa6`O3q zRU@BP`@|SoRZah7-GQMlL#&y!1Ux|usG~n&*xsks-=qNAO0?+)G9_327W4OQt$%08 zkU$K4X;BkCz`sL|I=X-F-u~#fI|teBvet)M=j*fJ?DX0Cu4rpWD8dSwNQ+M^QDiAK zdYJ@?CZIw-1p3~8OhIuNz~$Xxs25ss(EqnO8wySXIAAowh`%;R0Cu%zGw}&0hFIef zVRY-s+xyFT2axaDi+Ym?o!r3)9~#3iE{^HvON$ufvZvjj1R(B7NGSp#K_C$#Y3xTY zQ4yy|A|?3^3$OAW3fR`~>w9EdzzN{|&J>yMrrU5B}1uND#2l!D1@XdT2oS2asA03J@ZmEI*YvUi+eT~j~ zj7al%DnOrf*y4Y{*Ao*GXczzepVt2DFmX{vHaw|154gND|28~nJfr4n#5}L16s8p8%m$bMmx>icxpkr4Caq{OY`}F%=8s^Or>@)tUT5`n)fAK|=EJy|FXz3zim~Pc>I30dGN)+#(6j%R&+OPo)rF zR0{Ehl0wfmQ7Mtv*{W1!Yh`OitK5}{pU7QY-%%ek6GzL;5dtqlBKwsy+Q)1H>^&#@ z@TQtR$KOF&+bP@oNLl+sW&8dsb3Rzs@v+#?^Brh{4T2s}R=$n9THr^9kDhc&S&AX%okK}A=Z%N{AH$T{yn@W#h z`hNFS@l!sU_e(tGqwSh2c(mpceUOq^*GmZY=h?(jl>EBj8cli1$4ZT-Jdmd)J%xc9 zF98ORlpZTlFF3!)Q#hVaK`5)-I5YRO1e8A93(UDjUn-t?pYCf+Bqd%;F<_q!J-g=6 zFE-sv;y504F8z31BPQs>1V`T{u=E-Bf0qB>nn0QLjzt4kT)OGouaq=cMz zVC$bwlS`JZu3@SK)QB!m)w{EB9p%s_9lcy{a^v}!ZS7mB!!P8{GBdJkg!dvaU1&o0 z@U>7S5VJwMkD+Rg)wLN%=keLy(>;JGN2Ugrxjn111%8%um8o`{(`}$!joG~%3k^-S z#dba9goS(fHZ~ao^|gQPQu+QG_Fmo8fdT!Jo$0=k5f0gl>+bUUz^vU(19}U4oxA$? z!1iyQw@hw*Q~!#s+uDY1>;i2KiBqJKYR?`!hv=b85{=QY(4ZbLHb13hZ1t#AMcQ-F z)JqFdk$$BO5o5=vjVd~(m5JCrhlncC34`=@&m?3(*^^6`UH2z-+4Z@vtF(NJ*sdHu zHwm|OjHNT_vn4l}(RB90N6{$?Mo#F8VHPo+b)=#SY8%>3NKXmb77e1#{cJ+W^yiyW z8B1vmC15AzFlZ-d_eFsH*2P(D`BR_E2@Qd!$nhYcfzO}&5zbqe@H-P{+y#8{@FAb= z`yY=mejD~bhan!U2l_C8fN$<=gZ66Z|9%_`@xO4EZFfyhrt+)lOzf|rx}!XBVusP= z)}=jWQc9ultlea)phir0mU2;jYsJ(@+E-iZCv_5>W{b{p>PuDcuV!J1>y;&++}4Lw zPq7L6E}_EHskF~A4+h!Ww&H7TvFiFceV0lN)uMyxQZ}t8YK`op>2lR}DE`#Nuxc~A zPdb}V9n{V zzDdxx-#dUreXn)2G?#j*Y|}kzXt8Qzs-Z6$0($cc**)c8pHzfV| z>-w%%7^Dj)slh>o(QsnAiVpd4NK`(g$vKMrl{jap@-9Bko4qH1f( zC3C(o#g~Fims>Kirly15%CnMwad`!*j?r&A1~rq;(%*>b5VmuUb88(Xrj*oimZ`LB zVNxa%O$S?lFTZ>5;w)p27;1~0o(D)cY5f&@9fZ^PvXQPebf#5S58V+mS0}^Tr@M+N z)m_DHrNDNq0Uik_{2zKuwxC|90KiGV1waDG)h7s z%i4fuGdvq2fN0d&4c7yQ8VkTZLBFq$xcYg04A0{FIF-}u6DVEl(}>tPy;AZyZ&KoC zqHA{JFjZi9FX%7ffT-t(m zg(}L_D?UU?4~Xy*W=f^*$Yz%-?+H=WC&k0_t__A!o=~^WuzNYn0`t8e82rCt|5Rd3 zr&R)#3i3IrGE;83Em(+9GSmKJ#vNgSGfBDl<&^Har@{k=op4g5uSHqSJ|*dLF#nQu2JnT}P8FQlq%ncNO(tVz!OF`EXvf`ED%K|^%5D?c3V3>r0S z(d2cnIAw!x809kQkSAZOJ^p650%jX+@|6jjgMpZc1y@{k%mB+p%0i{FH*t`N{kO-N z@sJC0&5F3iJ^Xku8UZc>(f#5=H(C_$EQ*yZhkkpf;v2vCCm^d5n4m}#oREaN;K#@k z7P+7NobV_T5oID1mDN!tIx&e&T;h|EHSi>6ZPZq9v693;5Z>hH4cG0%MdxO1tm7IQ zr=6j}#<4}AL~4_6IO|`|dC$8>Tyhx=$sdbTy94$+=+#IHt*sWdxrmWud-a44&$pJ0 zqn>ba>J%^{!?mJRcmzZwWE501bcq<4SlBp{a3xETDylk@$d;PLuS;K zHI%W76jchSC}6-z9PqQ2Sd51C44)Ci9-_=DKW6jFX_vLN>Y27Mc6d42znrdAl--f zmAjL5a!FRCbZJd8QC$<-|dX6s*ID$oHO-~rtT z8w?z$pry9*xZ>w%@fGr;q}T5ImyzTyin(D2-Id)2P}LGM_A<+$S@ zmjlPqpdJRiA2awH+3SFNfgdBX45f|X*4J>42U?)L-O=(*d0yMY@;sh1Tvxwna<_dCEIu$m?U z7kXSfHTKxISI-W|xf_97fyEBInN^C+eg=BK0_=gS-I)&oU-gWqs%Y9f2%zCO&jDE? z!qvb7+Wp#}?$v$)IKZ`2fGdH&07h0Fw+{ePk>B$1f%ArP*Lv$3IIbJG1LW&Usg1Vn z(6nU95biuU+UEcVbglMd{{=WU?!J4=L1i(ZT=YzsN=%rFf{6L=h`U=Kn^CSPX;6v= z<(lfsJ6Bw`t9j2F6{~)5#GI+w_aCsVM-g!v_c1r`Q-D2iTYyVEk=#*r%n*T9e)kFV zi?H1mHkBOSDB5f&IlNhN%(gB^=o29?V2hXrXFO0_l%maoaT$vPGZxcw7Spp9ld=}= z8H;A+c97$7e0_Blst>E$)``f!i?A3)Un7mEB~K2iA8OG)0XQI%HK6|tT;RooE5{50 zSiiE42-|Fjl?9t610`NA*sLkq2GD*FNb$~4D!Gv^Xp{xEHi9&$?UrIjF2j+H8IEbp za%j$CTt*o#cLL(Q>UvdelCngI*){;&4)RaH)>x1qS_#*JSzG~AjgFLJN2f&>R{FgT%P}x67aDDN(u3+;} zUx5dEi>xZz^g4p2R4Sl@1v##6VZenEN^xj|#Rpq*e6Tsk^sHM4Cwx*pz!K0f`ui#1 zauiFkh31!D8dTlWXxIP_h-K~i+ziYs?;9eps|zLW>MrnLUx5up2M1-0qbPVuVAczA zQvIUa^?zHmq!1=&Ek4we<;)Qc%+6V6n;oNj!{ADAb9CEF0+%XyPH$;>@x{98UW}Pr(>Y%H!4`V8aFSV89KQ9q(5gWM{NwIeSEoX<5tF z>*>0;^1tfGj8t8uE$s}7reOeiuK|vTWex68L)EiAj&N^Jf!n$USXHpib%VeT7T~?% z;939^09uP>&BdQ-ZQ!()46RC;5eV<-D2P=5g#NrF$Rbcn6%H*gEvu^yNwil02gI^A zsJ^dt?oi#HSMnvky}O@B`%2{T{nB(`={V@}R0 zTj?RLO#vI9`{5~sPaDI_vi3?`4@XYcc;Fi%a>7tf?stTHdh&c{_kgjAwU3%j0XA)0 z63B7ed~kx>^)~%K_?txjF+S-BvLwQT{WdF#4wtp%IK4Sbqbh?Iz%zts+&0~p-xTCx zaqO+hy;qxt%)!nH2G14^F7Vl*TuX`Ycz=;AI(ymW5Hg_Z`Aq|MZ>;Dj7n^Wq%(kRo z&CN}~*)16^Z_61U+L0s?T@7|ci*u6*UsBN5x@>5oIy01$G`ONn7hVSZ?O-KAurtSH zEefJoRzc`&gD${tcGnexv2I4>cZ`xP3{-Y0_;d0WTLERu_f$WdwrkX?@A*}oeBqyio*CbAk?n~ol?K;iLDXtoovfkRaEE5g-SH23YO+n~CGOc1veH-EL@5 z0ADq2nntSurT*2+z^F;}TGbVhl>u1-pe6yECZg*V&-F@%I4Myt*!p{i11}aGW@jy? z8j$0;HV$lHvOo@zM>7ug)t&VNT(^jpy6wF*SO0$dwW)PZeXU|gqrbWe?3zPzrs)y=_I zO*L6HlYnmnGwbQB!IC~#TWj+gGKQ0v+aSt{6}R;mT(yiX62vq$xHo!U8W$78$*W4j z(K(B;p5YZ0*~Ek02+Yu)7|?x9om^ZOz{YBJOvv!Bflm%*z}_NxcrEE%Ew1NkgG4D_ zv2m^I7?E8YT)n@RNcM&l>jXItOrPp8_I#bZ(GgB)P-Xx_lDRgrO*DgbH55v}`SOGb zT)k~ujh?Lo;L^z_qe>-R!MVV-!06~PN!D?{2u}`_j4RX_(q~AD!m&9;);lBx*kHHLv@_P@D7Jj1c{JDT*_m3Gn$t(E zfdPFJEjcfZWX_7utHq1tpc4^&AmafJU9^@Ec;^9 zaU3@~bQ8KAF#`vFY*{yCaaY0YyJBP`O{%tD_Q?@3*2RSlVgS1{(~PY-Lxf_n zSiY;@ca=`8Y)h@Lu8V$pJab2bsISr1HiEJ3V;MJYETcz{qN%CT9GuVR*|WQg&dzOg zZr#T2-Fqk&OM!<3^RVd$B3cfLY+crju+fH_`)sam%g~}?20w`9;trEjF7iCKDy6lr z9){Ao_apUM$GZCW_7=>jP$s}B2IQWi>7*;#8tMYM>}Gf-e zzjY%!w(s=rK>*Y!xJgJ?YM?=AX=&lG!)J5Ci3^x>#Cw=H=|I}rTFK>dWHMIyxKgP^ zZ*L!4w{)`Nl~p|X#4|kq+>7kk;p)RQV=FqykwJL8AlyITaCVal^+}B(%TwLyvyz(Q zlL)YV{P8Fe<+u~Us^$vCk!6erhugaHX2>raXqX$HRovQR3^p!r%_V{yJ3ebb7UsVE zB5M9z-ujc1X=-jZ;8~W6fLZ~%yhTOt;qDi441n_&EMVbTA7$>mIgA+5lH5kiqM@O| zd=5BZG>0BGn~#3%L#$l+DtG_%ULN|@qwLw!mB_`>HtN182>hU5I6kMCoN=XAQhx-> z;}JPG@AST~dxI9KSQamr z#Q3@Ma+a}{;~El6`ut)6d3ia#{R2Gs##?mv^(DAQzHW{)q$pdtl1o3s z%vm#NXvo#m)DUGd8755V;E&#S9J3FZ#oD!Rux;D+SbZEfOzHc_ElOCBOX!bau2-cz zE<^4eri)e9$-rB->l?RQVJ`0xx`d>=>@yQSCCt0&JpIN@+QD z6>xCPmU&~KH+SWEtiM#ny;bgp$H}kQV569$9FvHiXca{>6_q06I2kj2G0NVIR@N)Mbl?Y<%@rN1sxsZ z%GPtOe$gHIa6wpai%EuXn;US;L@I+6mGUtNhbLLARRG&3lrwKmPSd?nefdydk&PuM zbipL<(ZAa0XdgRFi9Q1{Q$70rl5FRbU!Ui!hVTV}J94&u9;5#Dp zMeC4M!-KrlY|4~2nDA1*WM&*>RJd``C)e#X;i#jJyJ^4`*Of4)+tir{A7uR3G}^UuH{be?8@S{4pRjS`W)m?p^|vf+ z+hP0morWv_@$dhgmA`v6^;%k5X@e`9%_NHEXkYb<;)$X^i7ujZrMe8#A;K|nD1`vl z?vc20JMAY~wFU+Bf!l0{XY-}lJgVsbvJaZ(b!m`4`AMt`FZ6V>;~5&EFubRK!rKfd zi*Qc9i-pB5*YJvrFin{h&}iH#Rnw`=E9|!IF|VY{*M2@T{G^tNs zS#Zp}p=g)IiRiQ^H%R>@8#(73G8bJG;baNo8v?t+$znM9oPpge%%MUVJN9&wIsbez7hMRta~vmR zSVe)A2xm8l!^a19a(aGu*~d0v#$0-&YS*saY~QxS#GaFY4f@-?y}fMMuqpML+8EOu zALZCI>J!CMTX`09qG%45r;9Kt3Pl{iGrfg8Yf`@kBySskJMKx0e_G?Or90{;|NS$* zdgVX!^i$8fvD<8x%qPz$d*Mae{Pj3F#KIL|A+*itWJmaT-%d{N-(^77yDKA5@l+@j z%wU2mmWh0nrF!CrX0cepw(ascNn)CaPrSlbW*j{ROkXqy$}$FIX4C{N zThVgTN!Z46Wp+Iadm8;)a#&lmu~Mw(XhV$tjexxR)z`Uo@elaF-#%>s&@M>%Nyz04 zCu`73i#_XFsbJ<(1Z>LNj0+_WXZGzdcfO}-G&^_gNo`jymowZ}X=S9{DsThy-zw|Z zB-*oQkFmnF73~j4s=(BkIBVKueF*Fjn5f(d>4cVos46#j*dc+MmjSl(c``e8V2>R; zE48*%#xggR9JV`>Xo|-HEz0n_DGv+`@ZkNAaQpZE+xUrErx^n%+ON9P&;OLEdFhtL z6+bx=^nl<9V1>6eC!g879mlbGYyCz_woNt@KhC0EpfO`c8&MrEM*TpHJBHlU+{na9 z6H@!@jddH0WGMGR09chU2TJ$pPEow&z$8nm(s{}W01{GJvqrRJ**tYBY}jD7R{UTQ z4Q^x7qGVJ#=I}@(!MU`2}`&ZlnFccu+=-9?6k&-^-dcZ&U&74|e$Nw{78|gQhd< z;DeI7%!vD?FD*Bm>-)jP038Q%Kd{kJ9IdDZHoiQyr?1%HEro0-%6u(JE8#>dphd>U zlH;yilj74w%T~r>)$dmGw_p4R?z{I_MkKoi4dsqY9K7;C&Ru-5F}y^)G1dT(VQ?$v zm*b5mpZNL;8t(e3R&pF*>d?eB9udg!z`w0U{ z{SsFXTNE1|%7#@~=T?^yv_Cu|2vro|)}Vov<_DL+EL9|^niUO}wFut0akFtPd^aR; zbm%9RJdmvmFT%++ctLpyb8^66tl;ea&)PQD9d}Up{_Tk%Pn$l4D?WcQ2hTpJGN7U= zePY`VW5$f(vzJ}KpT7TniB&C?Z0@=HzVg^`oXSe;Gpz+WMNujPC;>35%YuwSq%|0@ zj{@9QZ(?21TbnAbVsVyXko=LIFp||-7>Sh2(f#35d34f26%`QzRzkpry=H|B^pn5& z=2Vc6IPyLG{Z(J$tg}z!z=;#EELUf1Zaw+^@8h5U?JJyq#)kt2SvCFU*H7@^{SPNv z%W?lzBoSb{DAU*YW5E?L<1_VY{%Xw=C z=|&I3GDRQ1W^e80MDi<;7wwD>eFk}~XdWb$P~6)x#z=#+`nDTu*{fyXy9)JkNn7Lj zn{P7ZB#`yBCr&!heCnb6K%Pe)`3>K_@p~qtL!Vzy9(Wun+&(UZE#CZR3UW3ofulM&Q^ig(PoJ1I zp(L=9%rMg+`d4|JeBX$6?rt4N-uI8Ax>tzHFLr+i1}J>@yKK7VS_(CyQgNiO^XjXw zao10OVFoEQxWn}@+f;%aQJqV5SqYK}27FRRtdkej^kY)*B+?-NveivMS=ila{9d81 zSpue%b8M#m`eD#M?V?4o;yQ^mn8(bYXHsZiUbcdt|Lm7M_2hFVNq%ps>?2560wtBB z6b0CJAIRvz)dTI8ClB@s4PITDlME-fwVP+}j6GXTn6m1!;hdj<=d5u0`x&_DM)R+6 z(IV6!GcC1We8PzfOm6l4_dUcdH{X$%K3gwkwB4`ksR9@zgDf6~WeP14Dk23D2~ULx zjm6?|u^3sNt5NT?%>3b^kV^G9`To{%{Cw1S3d#yicq5XfSk`KOSt*=>0s3#ek%8N8 zLp&Es@7FqZ_pUwOpg}sdv@~)OeUVpJy=DdsX3Ut%F$?A~fBrm^nUX}hZGU?5`^?mu|G4%)jS)ALmNgkk8CW%d z$VghaTr?O4niU1O2P7Y-dxPMbY(|AyOj{z@q~#n{$94%si2R9Q)cw#%;Mze_<_(h? z3g;q9V>wwrT|;RcqC{y7h1+uWyKgf*|Ppapqa4nMhV+ zW3nj!$VWfObI-oOuO57)5}`^m_F}lX)r+G`@Hi6nZ{B;!W|^WQx?9Bf)|D=d5z%-P zhTE;UV0>*c6w1&DTsO`jN~NmQ+CX?!+{gL;kpXcnQg14%2@__DMLxW3BOmMC;idn0 zGcz2URV!aJoVobs@6*>;mCRVC=H@2O{P=0KjciMn9w~h{Mssj<-H(K__ma(cfIaGQ zU)h!fY=x7x4xW%v<-ErvT@r0nT*D+BkrxLpISMfOks=N^7`*Dq6RIOmANIsEWLObj~>)^4e2b;_g zcU{5yG8P@PJ7c5G`BPx4o>Tz&^D7=V;hju9^QaWQ9xvM)anf9MZ2Fh0y~4cmjJK< zc}GF9NiuSt{#L%!ki!2Qx!OI9EYBsiU+7cZeFy=9NYV!-!Vx55_rXb z(6Wz~#j~y#_H4&?9R6?eOb$6TeHwLdPp`>63-(DGHH+|GDG~J3t0bRDzz*RJU~}|3 zVK+&e;aVJ3y)Dp)x~(k#Tqn$_P7L=t4xN!zT3wJY zq~!`%ToW*!DpJX7Y^3aX$@L4nw({qdYiBu{71#bCIcD~O%|TKszx*Sf zdqbjn8B*MnDx#ZTe2J1W;ee;T)>d(CWt$j7tKEY64OR%CH2tf|wr%5H1b=P26e!y@ zG1mJ_-m#^UUA|WIy8b#>`qpo}v5~&M{wk1LF1(NtpZlB(vf~gAx=3hG;;HByKqPuNm8zb9VQvq&m z9l^PuSZHPyROdXoQKzmwUHs~SN6nH)Ce3%<1~#nU#G!|U#iY4zqR7N!)HRE9T1b2s-O3HIg3NGHcLyD3kDky%vxw@XfPu!haFzmTH=YM0bY3K zsj0`jxp@nZ|N6;tUexa0dwAlvPgem<8RUyhJ%b=qPr3~50L7l`t~38eMnHaH$x>FW zT3ybIPBIMf#`?5qK1_=4a>Frz27vYVEk{@v*9X;0IJo+(4|p^vj>DOXl)*Osnt~5B zT4siRnq`=zbM_UPIvWvP(b11S@>?@t5Jwu65f@!d+volY)zsvk8zY)y%eX_0d#?Ky zT{qrX7T5aW`kx=)Z4A7$_WRp4Ab;!(Fv@YobBb6F)IlC`Q{<|zTLp02(q(?{JO;8O z4p?HgozkdG;KPZjbm&vJ1vNYrG`cjs`Q{dWc*mVaY{!uS`QnRd{j1NTnwm|-D-KxV z%KCo&1MI%;Is>u>{H{BH!BbB@Tb`4g_OR=|JJ2a{qLUBZGZ;TF_hIGy?FK6m-qY1x z)t&?qZEm_k;CbNB1M#~m5$#uOnxzcLxAj#(jGOwgpP08?$DkGM({CPs(&RdSX3=@E z>8~2(){8GTo!E8#bvXS4m9fln(-X>?`upj+{(4@0^>x1g;%f2*cb(2!+>7DfJd>P~ zmzEI~$6?jr?b8_jE~yJ*=Z$#}O&`L~T{SBa29^-u;=Qo(^hU+Rh|5&TMi2INaFq3!cuFyb1=smQKCfFT+m#<%c$duhAtPELArok@;xYUp0w zXX724Q|x|sgf1$MtHm;I!??%8DwbjaZZ%@sT>W7|)}#UJZ_^b#IIB3-n~0m_28irW zAszIKgCRdLLw!H`!ChSQwQup->NT;Pr9p1HJ?^`1x59{sk?TOC@O(;I}K6Q{>kUz_)FMPqB zgdSXJKqR^BaczS*oSWanSq9|tqDY~B(S4i~K^0#dT?YnwS{b zx@p0G44mJr_*$=Hp2TPXhN`GV1l!TCth zxk>T=sm|XD%~`8Vo$1`V&FDXui{o19PQ7{S&8^&a%MVS``n;o#VE%%69DK+uI>xor zGNQ#S!O-^E&zdu|mF$=dj+hW@G?eIrjBs9|%e42-Y^yuhC~!_rgoUMEV_@0wdN!hi z&DU1oCuFl6k&;o8n(5Yk#2kHumM%9TTo?it;`ONry#inD5gsm7-bWbxG&eVq@z$)c zf=duu>94j^-dYMOx-PU*WpINJh3>q3K3`L6_?@UlxdtJd&Cu4?#@O})7&~?hqeivS z*wlnmDA4i9LwsPqm67e?^=MBC1JBDH{~R-zt7A3 zSCOTO>e&{a+!k1tDF}E3?6kDx=s<;KDU({WYu7Hbz_-Y2A={xrDR#Du;q2a>M*K!W zuCy~8n9#+XIEIt;F?VK0FaS(+N_?`|O^Y}I?hU(MF<^IUOhhQ94^q~6XT#N?dF}2JMVU z6fXF%X^wj}+h~3dEe?>WiZrE<`eXzSv-9TsH>@0=E_QLOJrI67X`H?dbMX(9;uaLn z+!||i)qJhaU(K39uD_rkluf{gfl(xg=t_g!ubCOlAmxfuJeYCmwZlVEkt_+Hn7YRR z6m@O1ZSiWR(LA_^mMS2}0U8%x%pe@<6gb2w7;rPS0S#_j4}1yXt%m*q{;Nh$)CBk& z8#nXSjt)fB7NyMV>`wAl!%OWP#LdV*EaNWdwce@B`W+6CYZ*87230DY9Dgl|jE4E2 zk_f9Z4Q4RlFc0KvKm%Y$K+hp2pphYLA;$n3+@}xyW;WAsr-;p*Qteq5b-nrQn$B%} zrDGg!z7~H3Opfc&s!s9}w-3z-FWbs2N00_GG5g}KpL);Gap`eysdDw|1J~cCIeC@U zQ0C-pz*R~Z^$CF{HNjX5o zuS;Cf{Jf>$raIDmQ!?1@?_xn|m1`fBl2{O>!M#6t(iDoN zqOIT+f#ZNFL7f{;4hW1md7;cH3T9bi-qo-(W+jfe72l#X819YZ*5L7Jpe9`_>0Ax2 zFRm*fCjnXm>`?!i?`ME7DA+ZkCChgPQs0FYqCo-twJlrt^7szKcDj_CApd8snvD_D zF(B)o$;#EqtF=z1@|wCdO4Gk-qBc!T*E%N+@TBAHvf-r+$QemCn5wxq`il?tR^W3W zOKlWK_6=@Q`Je#S^!07qxU#(+u`=tFNKueKb*HcfL5|_(z>|BR66RXMCV@?kWe$mo z>LkzyBbup0Os5?mcg?hW5-_}qz#(?QgdyXtm4*EiK{3BrKj->1Xn-|c zyJH7ex_Q9M5ylB}4CVFr(>d60kT$#)SdK}zeO=r)u+%cSlDIbdyEi(xo@=K4;?0Fx zZ{;||&NIm=x&tru0U8v``g<|LHG)+c7hV%D|Aa^ay%&+ibB=vd6JAS zypBy;Oe|^Ab@1G{|Fw*{=&@!Nq6jCH`V6?U>_XYs9i&*+-%kR6qrnxa6>UzZy^axf z&YI)qBlL)Ee|CJl+{B zEAN=iqPWM&@{r`%;_jd-TPkBZ zFEfnb`ZO4j9k0_SI7R-b*k{02E<4gTai~`G`0Ii5+cnaY%rC8?J!m;c<=lV44%hkO#eY#)0nIPX3;Zzrk zbgr!`2Mr~!yc@+UlHnP$EJa)0F4%rU(g3fAqJSzh%ib}mH1D zu@j3mSS%;YDza=)8Frz}LMwJ;&E;r0$RELSM#QX3HCk*YJ0)h@1rE3K%y5c~c5JiX zi2=55Jsa941Hkuyf6}&Uk0b1h*JbY01t-nRA+%&eEh{xZ=BeMQztqlI;3O(Z8TuWAPlGY zeN?+&oxs69wc-gii~MOwzf74KjrGMTY1|(2xExk~9ziLkTZd>)FbmZV|Zy z_&SP>`e{3MSbI61_6xv1ZJ0C(v-*LV?<9Zc`pse`>0F;uLxqX?PvOoF>-r%MDIk`o_n~pb#zCUTd1MJhf0}sSxx{Eyr;V*%OB2x9v;<~uE9zYZK z4GqvVvFxQa-3{Czuv~$&Ke;wYZyUfqt(!0bHvv}sYlONMaJq#b|b-H z8e(`cxUW;h__enREKyK=yMXoSZ3oz=H64a|bHIB=%02a8fPj%g^c%v#_ETtWDz+9#$V3LUw z%Yy}U5QdZH)NR0W5qT7N%&dZ-q%SL3j&=s0PP0Gg9Sd-fRvj>gG2J`aIr4xOabz0E zd~YD*NQCL`JNvZzHLQBJ0J2Nq4MdiDnT^kj$U5tQ(S6R2T`YJT`=nm_y#P2$D@L|y zFf4^?1DS*{Q;^vRvpkn;ydVc)-sLBah}P|rb)%ae#3ChuB0?Ud*Mzr`PJs=GnZdDI zU@an>1iG~)Jio?*qwhrey#zRpmbQ*Sl&Gwua)OLNXa#8%_qSQhd;KKiaj7kkN9YIX zG07zEyAFD7^STEm^9$*(f`6CZrFR6StkLl<{UOl10RMs0{|f*B|NmTP{JYNOoaX=l N002ovPDHLkV1jjAs`dZ? diff --git a/services/next/public/images/.keep b/services/next/public/images/.keep deleted file mode 100644 index e69de29..0000000 diff --git a/services/next/public/images/cj_clippy.jpg b/services/next/public/images/cj_clippy.jpg deleted file mode 100644 index d1c871709a662ad970baff230eeb0aaac0d0033d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 301528 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%0DjIdK=y8$iQcN09xV#{Le#B-}*0c4|c_=&z%_ zi!Bq0N{#5y3ayMpruXdtD+T?(m7;D|b5QI#X7>P!+OYxz^>k&Uq({a3ErW--3ledL zaP3L&PfDv{fb8V#)hN{9dhan(%aMC)>F}*2i!JB+FL^Vu+~S;lbTXll)j*ZH0Jn|2 z`k*lJgbyhD2kAS{c*nkJ*YdasivGi53Et;g>RqBFi6_as@SH8$zuKTd{<$0kH)H4x zxMXCwq>tCRZ8GP)X021jv?19)VGljNaxpy5=ql^5TkhbG28pP;IFB%7_pSf7=^WsWOHjr z<5po2R79$XZ(HcZJX)y2GrbWi2RZ_Dx2{eD8@-XRkm>rjc)KQul#}gy%fYDhlpg++{Il_wWJ31p(2@k92N{V;7467{al zPuKe}&5I`rPx-NSNorBra8g>`4JFn4(6IhYLAWc_BhxKzBYx5owE|M;KzHj5PUMXXX6}EkflJCpv z8&UM%T!v}60#*6;F)9IzYS877EHg#K)eR*uy(zLQQi_yC8CQOAu=_N(xJ1g>(e}~4 zJjCQO(ApOH?NWlxCXY%btPH1hjSKo;I5GHzjEvXcG#hmRSN!-4He}7E!yHSS#_qhfc8ew|(?lUp76yAtpbv zc})YRFb<&+s7DQVbeof>A~8tp5gy6Mq;PmOaauD z-;nz;_KSg&^96?MwJUpMy9ftw2OU8^kAxt;EtgE z7eev^xys}%cUuC@(lbq9cw1aTD%(z~qGbY=V?~RdnJI}QK9IUm>}j6BtEfiOc4C9K zNCUWP8`zM!Uu@Gyy+qYdJ3=vCfR>2I^M7bSlw>)(y-X>;_R>;#qE4e0fyS{+`O7!S zs<74--Yv6}tcex8&zLLX^u|wEv$aYru3Qm?gua>{d^a0tH^;M{BLflelobY%WQNu) z;dVR*CFVL!P_txVl{6k-G4O_PHaLcwL}XBKZnKl*Y-?U6*#YLPuUi7*c9E!9T;;kyB|qAQ=ZSM>%hfky)g zc&7phrU(~<(CBbGIN2fp^P)9Kx`@U%^+01n_?@lI#cU-Iu7){|BX2>xkGS|+T*gS2 zy}fZ)dLbkoClxDax>Ib}=pU9~V@)J_sCr7q!u(j4J$i+UIi#35z-J8ywNvlr)_mfl zwZ^E$E*j8SCC0!TUutH1HRJK-H0}(-P$-i_J;OVF1r<}R=1!w>&Yd|{A?P{(rt5Ru zar@6EKN;4%S?cqbyt{(IO&nXyMh5`Kq@(?&{OK$jQXNw(T;ajbP_vlDW2ypODxRsC zZ_AvOCFFJM&{_tB+Qboy?>?WlH&`U`TktzC!26ed{JnA%mIBZH-%}q+mXYdmfv6cA zv3lamdim{`ZrM>2LmC3ucU?06UVTOsv?6P9akfF`Rs7S~pZjzZY(x3Xg${|XXp|KB z>F&T0MCFu(o~fMqB3+TE^H)CcVLa@Q3<_<#hYZ~#C9<&;Q&H=i3zx;GGRONOQ@aR> z2@6iMT1l$iA9v7Zs_6m+8t84I9-DvNKurZ)T-4$vMW>Az3y9%M2H9A+y{3Sxy&jum z%k%86PwIro^gklRvU2~u6rb_n3{JR|$;msju8f{(imzFsTO-`-po?a$ z*O@<4`8Hi6g>u|_fW?t`=A5ep!m>kZn6qWOY*p5sZ5UoEoZU;6526ItaP00i9Wv5aEKI}lO-k;v<++b^>}FX;jD z#+gsed@*n7=6IZ^dO`X_-r0|9^LyTu`c^YLgRx>&z<=q{=qm{&V})iOFS~47xcHMQh+?M z??3Y!)n|_W3%GmE!^huL-C60$Z(c7RQFfjIoZ0pkq`iN-l79QtN+c-mhXfR7;7dVy)`T9NTJ|g~3GE zo8{Pmn1$dsC?FPfhsf8bQ586R2Fpxr0%fhS@Y(03XJoWnhEPjeb#k_7bbGB77Luzg z&)1Gv?{`0j89ne5UwK%`{`pn_Ok7&V_h#RGnyth5I*aY!_hdu;Lw~yCXEb$@Bif zIp!jEj(uJl(uSqM(mAKjqYs4TZsPIgyx&G*Mn>u+BUZ0YwU+58JGsWGRoAn1eeN51 zfQ|>%8{zF~<0q;z^{nvo#KJlZAN~uti2{)&g_El#lGj|;Xg$;1ZXx}DXlSX_^)gC| zJ@V#2N7tfEOdIjzR$p4IEh1*JU&;~+yC+t1f3W~vwZM+_F=ULy{FK8f+isI0X#VTH z?R~U~byIJ-;vWJD$JVfcvBw>ed&F6tmVx(GMoALA0Ehg%%Oid2^~ahtI^Sj5`>Wj! zcI?c}Kdo`l{cy>(TOUsMmC!iZDYvy{>~{z{&prpEavGCgH=!-FBdtBn-~^?m;8snjM5uH4^Lgnzpn$r=H~ig-V7vMXc1i zFei2m5&<9rQlMAgW>(cRu1N*bzDzMBW&0SE#BV8j@brjH$UN)JZFGo*RPuB8@G&hq zM`-D#wxVYX;Fg{mT&2s$RtkXUO}EIzwP>1w%xG%=Iajc-Crw*v+2PCi(lboN0|`j< zOHOvJ%(rLL;HsuJq~0hGcBXkm`|ssjcvWXt!`-u3t=jN+jlF)~etHiOs2@{-%in`# zDbJVcq8J2lXI`lFnKMLKAz*$oEfD<37jAm2ZLSHv&?|Sx$#Lx`bQ)~1284$q zqiPd_KcCqv1wnfSgt2Bf0p7kAX%0R25!XQQ{tBn zo)`DdjN(o{{VdccLHRs-Ed@TeAw{DdOHJ52pm?q2NA>h9yv4~aw_GwXq-4l3O84Mf z_}4~ujd?5wNV~zH4V@~VykhoY%wL4fmfO-MCNt)&X8HWD)*q?3ig?ftbcvAV=nJ&v z#mPKxV4S9L)puDK-KeGY#=mTkYSWG7)9{nYzFgcx-gmkQSrT^?4gssoQ`Bzipqo*> zWbfX__>V^Ol51UVrum0VhrtDnLt~l2#Y}p*!PZEpngAJdV%w4;-Eq$L@%&2Tcq4}* zG;;Pu1clqmyiA#8ZbGXmaj+cSj50Xr(?$TO(dX0^P`VLS>phfHbs3)S?#7Oae@$rI!>@ zZc|9JU4A7jQuSuyYBVW~ueiVF(Sb0!v?Y&FCvFbWNTT<-R1BoX6o^TtKAnN8Pm84% zT$Djy1NhzXHd5oPR5M&A_S*+jD52y`!m~Gj!n=M(bS$IRdNiQ$^4x8uxR%ZMJRdX!zz(I?rC5_Xbe7_8a^?|#k1>Dg zt!c_ji$)S(v#Xcja745l5)bjoy^>tA*Rv@}v5D(z&kq;~7Z|i1xRJX)*=4Mqrx=ub z)9hH=`#U@hFyJyjB%cQ~_P3SbUgF3;jGH(metYr0V&u#fuhXH7M&Kr4P%Kt@(dLk2 zSgDhrdKKd25Q8nvN89kPfsLSf9loJnF!aDtro3Shy}#gziPUcQU5OR4?9ma)Or^=I z1|Eiw?$0~yV`d?V*ebs}VIkeQ%l>%l#SJWS@dn`}*2%4)$lofO$Gaxo!}A^$SwU9wHWyqH<-aW(aa zyax+6VIJXsJvcrIzy5KX2&b-R;`*1mHc2IRvfHNzEZC&bRM?C${GRw6ag_kd&!}9W zB>2M1|K?muAo<&QG!}&d5^r@n%-~oe#LUv;!(inin|)><1bM5o(^FzRXn0J!+jwv1 zdYsU=3*R@^TOBNSnT6WlBe2xmI!@t#-L&|j*X3Vb41^=s3t=lD+Uw!eOoSbRKBC^Q~JC5U4&7WT1U6!{- z-AF39E*R!7BxZJ$Dm_mdvu%EP3}Q1wYboC1{>^p9nX&d5>E8)>IyGw@82#Z~9`Q-a z!XNoYEke0>Kz08^Bi!v7;eig9l>NoHl%sn58aQV?q5mf*DXKJ+^6e`+*8(h=*TsU@ zABcR0T}vf_fL?S^WRHEN_()5mLvRMddwj8lSjo@4CTb$9r`^>~D;NO{KWdfE9tQ6> z4f87nn7ZX>cs7R?%JI<7{=^K;_#wD|ErKaei{i`8Hoq!@GyPLMzxelV^EW7in>q*Y z-AjGg2mR~a!e=>!@7sH<@kX)8VfajqMw#eeQe7mcVj2T?!!}a780e?_Z@_Vq8phR* zQHQ`cnCDlY8c-kX)Qy%0*ZMjf5F6dmU}cjF>Iv_{uHg=ftv&Awop(SUq$mDy04~!z z$<;zv?r5;aYq9IGM_A&hN7>>r6)Xbtgd~b?B`vSp@3Oh%PW#o%4V}xis=S{X5_7R6 zVL!d_shUsM?$eIa&VK(LDx&!&7~C|x<@`4o@R`@AZ8;f#*l)A~H^e%h;6SGh$h30k z0zzAzhtS{1t>gw55NBhF^qbLZ2->ZCc>yY68 zq4h)1(9nZ5z=*N3x{ak``fKs(rAez9?Sk-?;?`D&S>1(AY$w9Xs>9`CyyCXpsuRV8 zpfbCnakk;i+VQ#eK%(}?;S{C(Ebqe=I=NTM{j@-OAF;`qG5&OpT(o+%gngC72?(#fm3v6ulJJb z18Hx^fL;qD$M)nWvT-%YExMVD*IYV206;32Ahpu{!5XdY;) z)5+CX^ZoucGZJJJ=BeWSxX~_cJr-Ketk=z;NLazSo^|m<+YTX@c6;Gbp9@M#F#~Nudh>PMlq#55au4nKGXe-v%c?)>(i1NM5;lk1 z65%2KN|M|U>PB;l{o(2R77Ven^nHVn^wf zclVDr;qgN9_tPN1hWUeLwd+jTMk;P`_M;0RsL)a31ebV$m#UZaQJMPD9 z3j}|ZG!%HMaxX ztxgZ7R=LQu3UvN8uh(rO))OqPmh0K|Ck)Du0GNT)blRD9$6}{@kyx_B2rP>ALk{h} z{X9G|ifME-?1%*eDQqs~_r`KW#+CKAgIo${*Hl81WUouFV)iQ`Dx2yX(Cy4x>aE>p zSsDp1scRz2uF+ZUU4p3-bA$2`0#;`iR9k~`At>ToAwUq#mX{7!4G62JwTO013^3e! z+H)shoO6T%lDYODjQ(vfW@|k0J>jM5v7g?I)3$E`Nn*Ej(hoTm%5_)e-(UaSMQAg(FZ>D2Me z8Q(SpM*D&AaLsqL3ccUJ8+QjCyR3lBu)djnfQXnk5W z8f!{ZSc1RBBm-le_f{?jj|Yd$iqW#Sx}eV#IWTe{4!!n z_~)Z>uYELLu3-M9)LlZ%JhTZV`nfSD%e{=U-;!emGU+K{!5W|H2K48cr^S)|VGw zvuEq6J6NxFMzvBYHl!^%iqe*9l49z9Uhv*+hjWYN_qmPsDV|T+80@X~J4XZA07w0* zn70FNefKKWy(l#muq(Pw9?wBILTdI>!N2WdhC>A{qIzU>f|hJ@mfRouP4~1_ziDwv zHO1_ky=3Ws6aa=H8-9vR25vEt*P&Poy%Ak9i=Ui@)3~wv1&73KgQH~3Gpj%=K2L=2 z-ys=0e2jE{9CWWFmb2l&t2GBPIdNWo(}1B)JTNY0q3IeS1it~k&A z;wEryP^o9r9xAme47>Rl{%C#r@U!q#YV#o(r7IoGBs*FY%C%sR7`7esxJ+k3l8>vU zZ!!PeY?Pz$y>jPsWf|eUV@({;$7Zw33t)V?OHafm1fGJW~6Wl7hfT=sye%}=KA^=kQ0IHdX8fc|%%PJnM>)m)~Wa?e&Z;t83Lt!;dK4H#Bv zL^-dh8C%XWT9BgOR!c(4KQhjUCuV#VY5fsqJdjJ+`K(po%0)xvjdsos{X^VV^mwWD z*OaM;O=su#2=`g&F^JmoS4(%_Bl|Rg-KNoDgV}0hBIF)*Fe6V>Erg^0BE{}~&dhD$ zA@|v?S!z9-Xs5Q7aquu4&jME;GMk-a^NEnY%OzAGxM+;nc{{awNd%H!D4ZZRwXYWv z<~5T>651oNHZKV^&pX|h~0CQR9(mN!m%*I)qyAr2%g+d3R= z+Zj zhO`jM$%Me+Bfl2o>vTkosy!I(mHo$jmUaQ712ib z{rv+*?8aj0=-{(qNm_&)*JNr1`@fyvB(NQtP>5|UHFl)SH9FIyqIIz?V1Mwpw8mxg zNUuQ=V}zAj9-ZbiVKR({DEPGkL4Str4hzdE_|e`mVoT{um{l^Hmcqq!_i9u_fV4l! zA@+07}lYFqMDv#3CKs17g|({4UD7AK&`Ec7_mn@A7AbOA_WQknyt=THzE^<2+Law zva;AXrZ-&mlSyePk{EybO;8>9zrB@QH+w?18tw(qLM>$_XpXRu!sQ50$dLGMm2J%5 z3b7ZkT8G$PYc7`^{=uy3&i|opV8?1(?c6?Z|BhI&Qo4^S@fu2t{K(YPlS-f;ma#AL zKeXy(1ExM6YY(M$aELM0!j)kxdzHM;(SRRNZcO5=!%*aPwty8aMyvf2V6MBAvqH3r zz;K8=Eo<!H*a1Y#q^52PCzn4_QFO$uSjYaR=dY7Y}P2L64oCPRc`DhxY zyzP4Lfq%OD^VrT()+G1m8TUu~v~W!S`%g#rGDYhqE<)d*2zH8gU<{$wcE_qAhlD@t z#uXIBg$lZGUfS2cC%v&wpGG^+115|$WfTGNj+N6DP&*~#9^Ex5qBbUtjEUq`kMeNsqhcLnRPIm(X5|J zniHN|^SeeLz$7zUAE^zmgOG!Djd0rOx+%>#f2&wZryYRSErW zzPw61Bb=0nq9O87Vx7-2WDQPZ$~`b;JKHH5_x5bV?aY0)IZ4#2?#oTS#ytnjjWMTK zOK*EI$vZ>s6%M6uEMI%8W8#mtQ9||_FO@o7ib^5hV~uyaLt{+SVgB8*mEUI=m1|*6 zL>Re#T9Fpay_&ywJ*BpakqYah+e8nSteDsFQ%$Q_!lC!daoaUJMd&-(zkmKwD8WsFRP`|R3*wx z$3If0d^G>6{~4?!X?9X9$gKK&+5FeST%zk zbXp8NPf_`?CafJlG>+TSF~|sGp+e8fpk1d((KK4LozCPZ4I(_VuLnq<144zfXDktX zUJn*uzRCN%?tD7WIRe^&*&2@WSxb5$wxL-Fj*J()5P#1|kNiK6=Abhb$>w*UgQvp0 zdmcGz{YT;;T!SE|V1ZCxscCnX8*U$a-{5sR|xo1SeiE$!QgpT3Em_ z*Ax`tEZ2;SRjK6Bd!3x#Ew%(ykbA)3wdpD7ZmDO`lqp2w-?OZMOr~ zTs^$y@uS>4S+! zZv7|lUr`I1*+`{ns<(aPDxrbaPoM3n5yAG@rE`B;T}QT@7rFV7?fzaFF3TSH;04#j zN3A%I0H3X_OnE|^&GH9#p~&VYK!TK4bf|EtMEhF-A+bN~Diie#Dd=B9nw&&(4TD;7 zhM$WN8bx75MTQc{J0z`&{5 zhmvN06V6ah|6Och6{4EK`>u{{;))ygTl4%BOp;@_&fJB7!?SlQb2fNV0>&m*6e-MO z(}msduZTpdHkVU13|yjSUNS~unq0Xk%RMn~B13e-7Wm5{&0PGi-gV?cf3@dzgT+pO zl!W7`*s4fUF=&(m9N)Z16erIu3#$NeF9q6Ke2zq@3&L@gA)!EAIYWVKN)ek8n{If zW9Hf>um=2b%PUpLpauTNo*no;pFww`f`4=25FoC=HILOt`st{ood&+$#BXa^??f4H zX4Fnt;v$#_UBxb@i)V|gw_cX))CWk}Gkt}Yx{D(Yd1Z~L`qvfQ^#0j>T8rf`WK^Ak zS_geq9XMvDp$Br8;JSaeWXgLqU8rJ;7fxTOiyrHC-2-l{Z=LjOBjt29Q4}#6bvjGi zPct73T_Hg(!ixnL)wKb~#PYZVsn?OD+T!9rwnVap^B?Cr-p?R-aD8`~i1|w_+b;=j zci-S{@&mqY7T}Y9to4uz<%~ELCL#K87Vcw`6jkI^B$)TwfwpjSd0PDCZa&lzV28Xf%skCJ$;2kD8;bqg^aXyf$?l|3iCQ z@~6I9^{?i~3D1;o!YU~x<wCax|H!nha|#X+U62xD z>67`|SHSW@4mY%sX#o{riOlqe1+~|!h5BhK#&oLYglLAh}FM5WMs8j66Kj%8>_w}nqrjhp7 zsZOObLY4uHHq%b!%coiy_HGK{IvXxzvD`|T1IRZDqKYFjQRbE@P|1looupFNmveJW zy}#7+euU=a6hnl3uP+9vc?EIQa2kHjhEG-&_zV0&5k0)&?(gCVBV-n(yz_fN=o06Co zwrJIL=>V#`A$EDkKFA$%BVuvs|5uh})LO6G@;R~AO+vEcNko&a?S-P_s3_HNW=6`% zY~+4;e9$o;y6V&VA3_zWMKvkl_hYN%Zz zNX|P3E4R51ggMA(I-%*-A!?MxnUl9!%_&YUkBF`iHeXUrbfcia_}N1y5`j-~_!bw_*U-G~^S)x%_H{|jGl&-X{%5g& zGA)#ue-Ee4o(R3aMM+eSs;M_0zSUaDSWCthdvHfv$1wqE)nmSVXKP;Dd9Qid7rJFN zQ73zRH10iipbQBp-SVcUQBAEFe#TsRtJFjPyBTfWV&NudGEARvebu=+>9uMw3tR5T z7q=D~&Ps@}`qJ%#o02ObK9ejAwB>`Ll7tfL$ar#W%+XS2`MVQ(6mI?b46@9p#9O%Nphvoo>fh6#^5aI$_gd=mxIY6Uvk% z6CI^e$nTn4ji#^rWZAQHLY+wShU~O044e&_|9hW7*kT`3koQ=TjHiMIK^$YnRE(h| z5$>$ma)GP>BdvQXuP5AVi)@KWIZD7TklINGP8*ILfPA1i46>UaZy z|5CbHLs8W%@alpz>;+<-U7_S30`J$^uR$0(TP6w=OVDBybAN(`25aFCo2y(WNMGqj z2)LXFJEF7oUEhzZex>-+P92(q)SYo#@#Qf&zJKdCQ3ufWckfrtdqXO0l!!q%X-nT+ zv<2dOpMNyn=q`}kleMJ=4YD|=@bznBu`(nm)P(UCN3dgz*TlgJQi)1E(E3eRUN_G5 zcu4aormstOtg2f6!Wyu@IVp7I^WNZWIG>)aSAOg+V)eAxP^2$C> zukBW{$FA_HFtdx#H4VeY%MTI-)08s4k+zL;TyF44CN}6%Lpg;y1qum7&ldp`k~HkLL!EWvt{GGEcMJ#VYhz-@dx2yBLlbzczeyN)Dj&^U z2Oq?7mal10_5r^Uc~nU2m$jBP6ut!k;TS}Vg+HGA$mja}-_x`SjMl>DbpKh8oxv zOnA$$f=XzfjK?Qcu&s}#dNg&kN8c8|_Uvm$E5^}zeVs6>bnW4!gXM*6KY*qVtYS!W zZe;@$-btZC(t`-xYRMhC8+_*IPkef(7ue|PA^*}TBk+xXVF`~{i?zhnWy4^2_(9Ez zz3B_53pB)@1O$b|eyv2vY$-j3hh`e0Ks&>=8D(*%1<{~_?%@=kKBK71)sm-Rh&J$z%|iT)TEZRU#9 zq66I!xt;M?2W$^73H)TgZ&)oxgnZgWb+9oZ$*OFWN@b0eF)hZ^BR_DBoI1>tANV|P zs`f}qyZI=5!ZON@N;_}Q=~IafTlViG9vSIy#hqpyBfV}^PuN`SOa?c0XFO_t9jO7i zGFh_@%K8VM_)O^l8dh5=$qXe*0(sNG&1bMyh|oOK4#QAmo1eO`29@u2rQ`Tv&w5>@ z9a2izy2P62-sjzB-z(Ro@0mPDTa0m}&GLC}xw;VrM)ESb<||G%I0pwJ z2kjUaF7hr4DSrP$qaB;gD&aFTQyrMmQ$oU7De<5D|A)3m=Hh+3u{J>lP+I3K{E@9? z3#TjD)w1ieh^g0gze#>SYZWjqz5gY8<)ZU7A#+hn6 zIJDd>Q&wdZFG-n{?loxTh4zE02GKVEfdaI_>W2g<7rAHS0QMPoJ#M>nCUmNe(-8Fd zY*#Lx)m8ULAAvT*$kL-x$3BHHe#dz^kh7wK{cPfkr8l1?11_afE*l?A_3#Fc#Eppa zm308pfXCXFnY!1gcd}$fl!|y4R*Rf>>xqAV_B_IsKQw^ z^o}$NaGo%;V8NfLmRRd@<-BJ1e&?K53>`%>aW;D1eW9mvQ2hO%VeOL^m*!LJ z;8adt_S@Of{suI8mytM~>ZQGDU{)`DTL-ZwM02&3 zsS`MQ++msgx|4A}CTUJx;>!Hr-KiWT2?)G?u_Zhe&x^>G+P6MDNbQ%j>S5Z6AH7Np zGVL=)5t{r*tKlsvCKArEWjdzxkH+mCQ&H?mzPV_)-^a!SkfTaQb++68hdGU!;iDM=NoB!v@O zp0&$7c%_7l(T2)u*Sd}Gum{L3!Y}(X-Vo_rTW|`oHapVy##v|^c0i`o3f1WPSUiYh zbwClfqNtL_&kqE*)bkr6hec=BCo7NF1)oG!Kkc^~Ay!_e9MLrG$k9yJGY+_-+rUWq z_p*WmY<;|sqM59}t5&f2?2KWY?CvVPCE(SHmMdk7+w;Q4M}+_HiFaojn~$!S1w3fY;wH5 zPF!0h1fZ}5*Z#5TdUNl$FH?UjM%#`myObumkIvcF@&+)3LPJ-mb!qDaYB76Pw*e*N zCNG08;{taO64`@-rE}}t6LfX#gV0BKo(Y+`&%<6Gqg%bGn+DZ3;stvNNfD>aV5-i3 ztx;^L4xTSpmYr}kFB*58UYYP~cF*eVF}^0+X&B?tDExZ3lf6j z(Lgc8zuZ1cF{~QAfeMSv$81$GDX%XnC*!XWeVVR`Cc^fGK%TZnT3$5f^5|f4Q@i}s zNZGY6m}LGf41fMM*rAg0qn#AfiEZf-PP!TR=u-OU55QFQqLXX(r$>scail+>IL!-6 zn_r0PQjL>^UXcfiO}Gj27wQ~nqmV;{u4U<&c+WJf}uV)dFjKk7Cq-ne2mg3VTcY%19>BTyRN#gSEI% zu*7@Yu!}Q-bH1|B>!bg2*IHD85DX>MJ3#{vd3|oooEM>pzOM~@Cs)j+c^edP(WbtM zA)xE1YCdo4Z>%r)k8e|Iuq&(z<`0zONp4Cp+xfjEvFFi8=1Ywv^|3}s3a0>8vc?aG zQ)NsjaKO)O0~@^Ce=i|XWJ15JM9WV*t$Wv2@-u|M2hZDSQ+gw|$PT1-57WjSm-IJU z_H>p;io;dEn8rgYtcebyt%JwPzkbkqNL=cht21%ci2p_@)v|MN6VI@+1=TaFmNk|uliWHb)=@p7TL;j$+^#u2`wlS zkhvV!kg_gh|MItu7+9tB%aR{?*4M`8LW#D$a-Dc)pJ6mRrixC}1(7ym;KZAk!@$vO z9jw!nrdwap_8=7|gLz*fQ#(~Io-JXoW2xzYrai*MKT1cExDBM+!SHsLCpML@A0Put=AZ<&1_4&VD~~y za0G<{@j5C0hbNQ1*I1Yw5G8uD)x7G5=j|fDmKTF+Xc;XC0ei+<{$hPcE|8LHFo8~y zU?b7}yy{nqgmXo@``4D_`Ww+$LSteBq$~R#(Vli^u~9Q$LY{sjp$Rb@w)V2r^nbtk zSe%~WB-SlZZmieo21jPW;C3!mYLRd7z66^GZ)V4ZVHPG)1~H$f0dDb|JwRR?{(5FJ zSX1PFBwt|nEl2X^JJE*Mjl!OW7U4hNcMe^+^&E9oI&`zeR*|o|*lGGO%e6oE_!~O5 zlxOYrYGnB(OfIZFU&yeg57s$Smhs|PZMb-{@#X@b^s{>6q z0Ll3n98;I}l`S~Ur-S88pkd#Ek#B3Q5Pfigt_jel5%3ap@8a5+j4?dI&o^m_7z2~$ zm&|XJ08^qccRj(I0@vpKk^!pXZWcNE*#`5%xSH-;!_I8kee0YJb9D+?-cyb0`Gv^^ zB)hwlj6vi{$qF{>q{4UlFN7Cci+E!l)UK$i z^jO%`1d2hJ7f`bIk`(LCEB7x5P z^1+>$x@q_7_g8vHTC691gsz;3Ps&hpAEV?(5^dvq9)p@0{rmzsh4a=PKqzal7L;om zRoAQQa7>H2>r)=u*~c;_NZJqziK1JFa8gHFLACCil_D^>s`F* zC097iLQ8jy@)^eQv_qofm78$2)=%4oEK2*ElDeW5Ggu4DjN3fZeZEO03w;bkqCk&~ z8Fx``aYho!i}_K4(O1#-*@WC?Dm!1QvTmw>?e=Ho1DL32c zB#0xh)_E}r$JdgGY=!nQ=;FUu#m?AzpH)qpm>hOidZ$oqF{L5^2nVlNpOwfc{AjTK zYOZ)aAXz$jYfxu|Vy*#1BX*Y{qX)@hYA1||8mNQmf)-uy0GxW~9c9 z!FakZ&IK^k3-ims&$Ktqm=y^v|B7l^@k0VDA49LOBl!dI&p(9?d|~h@mjH_jfvGz(AGn>v5rn+2 zQvt;are>OV2`sbyNNw4w`R+wS#Ky#}-lwvNG<|hP+qfKFOC`$e^>pS(lSo((en5=* zT$ngjEc;z%l=PQXlZ+3VLn6@`m{73(&+JSXAu!vCmM0N-J3&Oa(z1_$MYd4|(-%Kg zsMtGj)lKYfr@lQ+dF`g27QQ7w7DYrCa_9jN`uIDB(n>5=1@;sFgA9{yj{0+S$pQJ; zL?$skBZ6@S!>~$yF(ketCO6-tcdq7R%2#_z$PjPJzLJ9pDa#g%be=u3v}?Uw=evpX zXov<(P`D|EpQ};Nxbo827E*!?WG9tV8bD%xzV4TmIVuySn@72947HX})~2r(B@6T7 z#G7*ta(UQ+w_0n>4wJ=<{(G>c{l0LIlOW7GGZq^=O$#nsK{{LwDCXF(541fKT_XP) z=!NnH@Crs`$2zm^c_C?JLULX2OQ@LCAJWZ5x(k1<(Ej`XWi(U8d=9@33If9&%nibx zc@Z_=1E|{BNFP5*o3+^eXF1|5uH_cpAJGCPJiYr!0fBuLmeGArfH(}&@iKFP0Q87r z(O-RKbt~n;z2hS_7UIOxeiIWjfO!muGZ1+N!?>Ro zw5_SBA@RO2OpQDb(RMdBGt^3Gar+x~$WCXtb_Z*ih5@^n0>?_|NM^|E(O-Rh5L`41p+IJZSJTn>;s z^jg{XgS6E;`$L<+uh8EmnaB2S!`!ll*5C9y@>gvvlFaYkHxGKf8`|;29+vZn0BM!0j}E)Of~xc~wd1(-2G3@A zbjJcms;cVT-g`>x1qPB1MRGEdY?7aMdfetXd)Fxt{TfcM-R{@GDk=qe-G7pGlF00R zca0NTEm4UB4c4^eoa-Zg&o=6HN_-PEwniEes4P;uoreM5G(`JTZ?-I(xnXs!X!j!z zlbR=xXCdkj&2qNMoiqFL9s^RIpHOHquHbl6OJrt&;q)=RZ&|&5O||30oHjt zQ5qmiBDJ@;+<)Y*+GCo`e|XG2>zkf8cspO5nl^qXQC)@4w7-Ncyt#`kc4qdbH9p|4 z+oqD%@4bUZv0aUgVBiHb2O^78*ow0=jKfWJ3*2Y|bdKNB%P;beK)w$pJ9@GNEWTKY z#Co$0>b84Ca(Et+`HtkBxXseAwwamPmAL!X(0x0aN}!#!@54!k%?s@h7-5pRE4cuF zR@~M3y~=}D>Gm)1t6AC}XrGdOnqgMBdz0+b3|o=nG;F=3@X=GUIF;K!1&PCkj#2$@ zk~T{#w*7Yv+HG)esCI$nI9s&MnR~|_MvDbQ#&6u;6xPo-D*9sO%UWc+z;AbgqxqXQ z=!X9Q6+!+>k5Sp^8cv4&=zh8fatzR2#Wkn|)@Y$yc38jDMT8Xad0QrC_)Px*&1_n& z4gPE2HMMQDpJ*Q-wx4KsjPLx`)I|8m@6A+MBKC@^GyprbYJzi72&*H0@j3$T)Z|-} zdfq)VIKOMqZc4_zyb5%_Ppoo`Zr;GBSzz*u;+3AHjw=acm&Ak{Zt_dqJ5b_-fN=2? zL_o*~-g}SAil@p=XLBGQRaq_5OX3^(bXVt)H@kUGZGoR=jNo1~D)HKX6VM5w!bW99 zN1xIo!Q+p5y%-;8j}{(1$h5SA4GOBLV?pCZQfsG`kYl(bX0yAVIiAX?(;SXG)Ak?X zq47xD%y^@Mmdfm!c25@W&?bL1X6I!Tg0w- zmlfvfuxW3BT-2m^rm%HdDd-h(1I2Al&ta;inV-z%_Q<4jEh8Jl3Zm&nCXC&Ge}76} zBZFEulwtC7FP_<4t*S#S&lN$@GTmp&yFo{3gw?Meu@O8zs3&Qw?DmG3U&Q29;A#r0B^M&A z$Ix65C^9q4N!ey9=(_`ZTm6W|80y1`;%lm2^G1x(d%fH`5ow%ZxGX5p6(Mc%v6Am_ zSWZ>h#vQJkoRzKiu4#^w;1&+#$K6++;^V}fM%o9NlV(tZ+G^4&4@b$z&rQH&uiE4x zo0t_EDbg04iu(-i<|a6I9W3!z#$i$j6mLd(=70|>li2#{F`HT8BHMn0Eko?{vYI*&zpA4o!kzr zZs99Vi?T2{ccSYJrSSd3i%O-o%|ZJ{WQQdn#Q=%}&^gHf?nSm349?@ZR@-See`g*6 zU5SCZEl%Ney1bK4!#dsPp<_6d*5m+Hb`0UgPzq_GPK^4eqUR-OJU6OA7q#nZXH$T~ zpmYJZsyf}2?Q8mSY7u&C9>f&cq_J;oe}Q}tav>pYj6sUy(_o{K+y|( z)ol-S-Pt$XuCxFEj`+sK5^JJT?U|UBK8_F>GIiLH5a<+$=UU>Qo)ML9q z12tC~P-0ca9%Y?g{{1jo$Lifv;k`CS8nwcpsVwpewnosYY|?e3bu{wQHn2g7?>mBg z(mLxHDoT>sC$H2UmhSWmtm=%)^M>--m2!t^o|f6;>#)@_W_4UN%0A&x_p~517lW(Rn9a4M`s;u-guPXCPa2ywKvdoIdlxe7h<_Mh39YXL6ZSPyeV&R z^uuLw*pn7`@X=ABgL_FI9ehIBS!!gm8oNsYtyK4LX;BZmrM5`U)SYhTst$@WQEA?c zZp>+*#mkhFtbQ6f*;?P4rJg(8?p)-ioHum0oK^X7ZsuykYKsT4$}3cPJbSbeBk#-m zHs+{LAqQk{TO-@(k+XAgIxileRA-%j3>-VrbB#&WiS9IxPHp}NMAld2_WO!_PZVaLBGE*8eyrl5-&67tAT6@!F^|E;2 zblMrhO8fdOLcpxGyZRXb%2t3{gD;u;P1ccUGQeV!iH@~=B>+)trn z8@%^`BK;{Dyr zkb><__Z$~LYxXg#9$Ee`US;3wD`E4q`IqSSraQUfinH0=8`>C;N1^jmV0(_2iemPI z^RNBAE=whn%cFuC(yzRInEf=i$lj>hlju57_}p4TquLy{;af~_RI@gnQ_nXomhP6t&18;ZkLGnJQc&DMpJJ#4}3(dymSO8Zl6ck7$8;xrj!ZDzCW4)MHda<*wp9JwdUMx5xQ@|CZx){UGy)V8Pkjlgj`lcbQ= zN1M()`m`F9*zLfdT>s?(J0ojY^{y})IQT=o9 z)TuNIkhb@JNN6_kI~uI@9s^FqKzAe1nFLA)(Vv^u1>Vhz2G-=P&2%I)A(88r?luS zc&l6}LNOk$Z9GP!e4%M>Cm5P{cl4#M0E3t-l!$(hA=&$ym$79h3CYEI+c8 zk4cr^c7ZXSmVlim{oX4a_x}Kpa-OK_{(5$t0_K_g&Q+Dy&%smD1Dlvb9kG~K+1Crd zK!Q{FPJ+oATH%;FVYLpcO&&P6dJkc%Tq+7AxCR_j;oC!p{eZjU(JSWyH}jTMk{5n}%S?w$5|MOyHrv&+_D_v+na zONrdJtnz!7EF;;{vTsq_579GsNWk&(>9Rc>5Ps58Ukw5w)QdmG{y-saZ}~3Zv%Gi( zfaHh8RP7!o(>Y*h25GX!a|@M-ZT!_KL&HG7n8AODK+bY5y*IZ|S>$Ie2*jby8+JkH9=h5FCT8F+anksIr-?08f~4a zJbeMS<_)O!U-)%?+aSdCS3FuP#aEMQ2f0edv99?(Y4^S2koD>U8ZhuhMJ$%$Q2 zMTlbRQ8|NmK;98YUui|dtZ;9NJ{b4^0FYV_>pG*mCC*6qt(Jl5wI+dIeh|0fzR~KO zncS+wj_|KHW$NpqB=;(~-6YnCwZfp_RJMOox0&Iq%E&qC20wIg&^QcK_MXEq@!YY` zGp3ydr;CNr?+Q5kEdx-o;;nF@3OHOV8#&tX122!D$qV|gX=zW1{bn1dOISCQ#tfPh zq&WPNCXccOg;KeV#+B+<*3u-x=34~@hiCT1%WMdwC{?aUicnmfSieYRn;)37+6I!pMgHHaR&xkq>olFc6HP6^g@ zl*b@_3TkBfKTIqR{*tJJy*kSL7_xpI?W&B0J% z5cuoCXdb%(QrX|tk)$WW{<9ofbO^_r7`*Tav9M9C6a!0J+^rAwV&EwDr9C7+J= z`1E*YF$g=+vCnW^q@8WUhKr-xs5&;nI)fLU0R}Kppim7;@hZ<}2MD7tJ}0G&fByhQ zzfEU1(59wLqdHBT2Of+NXhtBaP;lSqk>BbnmxXdX8N8s;km)|vD|x6f9q!d& zQ`%4J*huk0xtvc+19;kFKRs~of1ozR!?!Lj{5=gy!)3g8st%sHvE5Bj_BI zer#rYUV*c;-W%0NLdNfimK|%7ia2#%GYgWLz6WvWT^8eueG-lfdY}{?BO;7d?=%5L zWo<73SDr4_YGgwxWeq+ZHIw+w*P-g3o%70Q#tNw|GG^nK0nl+g6(b9f~qz7H)p%0sWbs{FrU%mIDO;Cpt{^$ z@?1&G`X%`Nqj6PkV==>{zVkq-lQ%cfC~{Y1J`UH4cIL+iAoiMv3XOL|GcPc!<8A|! zxOnsiKULVb{DvDVJx1bqo%+l@>eUS@T@j~_IW=C8_;-3W3Zu>|9f7`EzSYi-&`8anFt(H&O%4yQY-%;>U&vxzcT68ox zwQQb%v^$oUu+OPgnbctFuw*syTp95yg8}IntFu+@?QaV692`5fYE|1lKYwy0l1biu zCb8n)&*ZQ=ud|bct;58BXIaE>SB_-nxfu@D?(|sj9MgdX8Q=uj&-*pV^&O8xhVM>q zb_an>2>6nByBb}MC)l8N%R=iYeQyxv(cPrq~ zjp?r95o>@T<_hKD1liB~HLAf@wOw$*b(I6OVnXXI8sqF(G&jXl)1;5uthUDI6qir8 z{{RRPzoe>?D*3rwJ(={sY>Z%h=jYPXQxDwW?gN4i8?+X<0@Ii&){7`EgL`+g`f744 z>l7QoVi!~_g6f6U9`8w~Lf#I+TAze7Cy0y@~c~OiZ)+QZZQE-mIhQK1s}bOI*`J4(~>H zf@gLbWj}PL`aE{4R2;RsZ~AyRe|yU1U1+iL+0aCA@OEm;Yng$TmN?nxA9#Oc`*i*t zZb+OZZ9TmqwOCo+_{Y;hax%L!{!kn<2C1(Z5mjokQjNVyakF>mV| ziwZ4pp&9gaSm(b*y7pCt=kSN(suhL9&WlrS(>i+J1!|z)KjJ+YqBuKD--R>7PDjw$ z_}lYVr%Bfkvy%Kc*W;qtLD_Y!jUFOAbX4RPIh;N7S^%^$Lf}9!_jj^Q`Opt4gLvW)5whD%`!B ziOt8yLt^8J@aP0&?Luz*H*|(5&p$Q;xGJ8ru|MRQP;p*!6|*2!ep*atCxvK)3cL&K z@e1zt;nGnUPH{e+7^_xGWfnMh5U}%3#fRz9_wH5nxyOE!_=0bOE^z9FUHgK(_a6na z=K{Ld#{#Lc59o;0Ynn6E*Sa~Te;4m=xP)azpVJ$PrIXr*w)fl8>I0;$3OBf|HJ?)5CH%J0RjXD z0|5a60RR910096IAu&M^QDJc)FoBVwvBA;s;qgFF|Jncu0RaF3KOxgZK@Q~K^u7`n z)9or;)lL5ZuQ;ZrRnB`eId~LCIlQG~ACtVuQPlEPZQKkSO-~;{?=k`bwy@;#1MbEN zHa6@y>2N8`-1diDr#UdIxUCVHNZ4L&JO;qGJDrW@xni>)>(VNXNtKzVlph&F%A(=r zXyA34@a#?cJHq7z9ex>pp4t1K#(w5{^)dT}T8DlqhcfjhMDBY*(HDZQ5*w^U7*@ln18#;K>T{+1Xy7aD|PBAyGHhuK(TrN?B!%*dvb;PMs=4zv?O;YwU>)`4# z8kKNjT^AZt2b8dm4e3yqL|JgL!AlGm1aZr&)PS-pLmUB@2SM@~Q`S?TDQu_9_uMVb zU&hlw2XpfbIsX8h<7N=`C6Q6kJ|ML0r1g%B-Z->82z~yhIO=aS8u>V8Xz!VbuYg<+ z+C-!|KIyR$5p5i}0k{=tp~I4<@=AA{wT!HHR__WdRxPgRthV_UHc*W&?y&FWkV=>1|UuQ8fC zdB5gsN7DZQk_J5u`;;Zt80?BI*qT9tRA%5Hfv|PwDp!F!X@`xKoq0hHl9R|aY6D2i z%wzgD^~388#i^-zC}s{KAppRkrxTHhT(OvLrA19~R8ndH#Gv5XQU+I&!JIW>tX2W6 z0m5O9TqTIg2+R-`2;qtwWB4If4UDM8HV)P_u*JzdjgU3hP5%HKYV?_iJd*7QbA)Qf zM(a_w&OY!xc`;1!7?LV4A)c*jE;6l1`SKn9x zKpm(2?+->fWYjOO@l6`#>y;;?1PBd&rDhFSv_3_DIi-lrgV>wsv0{?`Kz>tC5g>g? z@}tVmaTXnU%5MF5tAO3ZhnzA}&>B!A&%WYkL-WyWbe`9bU&w}lVLE6if$k{9d6iwj7F~Bo4@MRe@3Yl|zJsYnH^W)rLp5H$+c zmvJE4m05`MGkd`S2Cl>)wYeHV3ILA_vZ~3;ko3EObl>a5^l9%91g@%^YMAtR$Q!M4 z?&s7cX$Han0G-K99yH`oJ1iZ_V?JfJe9gULBFJa2ewW=2(x$G){5|RjL>4O|LfiUA zUXfd~ZcZxj3d!hdi`$1K=g<6|B14})C5dQ#d-`GYh$tar&I{ryP6%FxkbG4r@@^!6 zaaCokRwF2Y;Y%J_FWLsldQ}z1NTU>78U`y5F(M|U)x1*CAm-Uy%!iWrNB%r8UXYza znc&d?DJ+groApZw$;-%GOyEO3Wwg+`{{U=O&1$8Pd1|~unkm3q((?R_W(67gkIZgy zsfSKxyP$iaN@2~C_*IWIBeNgY@+LquZO~gQmXxthGO6fEI9UJA3=c->*Bw6BBskQ zu-Z0Nw%J?F)aO~K44_mE@#L88{P;|?PMBZvs4z=m7aStzG1gna7H7ao2myd*SUrxh z>=QVqx|en~gYVd$$5x3&IwAU(0o-RrGOX>qca z>-vnHtmaCUTP%Fb4{Y>`8a8HG?kyd;z=|a%vfEn@(>fq4T(_1{sX7j(`yVG*Vy&U| zB|CqZ7?&7gTWJfi1!E8(yTHmyN=nK^L5IUuDKfz$ybg>rXcPn}SrL`Z5sbz%k_xlq z8vG~E6k!i!x8`Y&%SR7n5E;pM1797=aU8(u74#jfrXOTRlaL{L`;&B@IfP1S;>6w+ zt1YSLR^_epeDx>7R>k!CP<7AExW>d zrh9O5wbcYkwFfM8QvvR`g&Wb}Ea<@F$z8 zo}@17`w*{TqL|1~p;Wwsk?742Pam6{)1PNf0**+jmb1KciU<(R!L`rH9;MEvPv!UG z4v>bShM`S3K|w+AJ#p)gR9n2WEntxm_(nh>LxRq%Mb;3#%zPTa7SKzAp{py<^apd9l?637Fc<# zIA1rze49^*+m{Fil1qZfC2X*JC0m{&iByNRer3?TCE2|am#dG6gGYB_k4`!_{`13t6Dvd;_Vn;114*QJtoJj3dmwjbzu$mJTPS3^ zhu_gy>b$|k-WACG9`jj5l&P9g@FPzfXEcb@9^o94>`~T;PgyY=uu5OJR}yx05$gW{ zoGuJGITP6LF6&Esv!qJ3webK&8YJr860O>RHIxW*uZ`(k1|pFFXp-@bA|TjuJ9H0N>&1d|beE5g4RDh5cqp7n34dvAcIIpvd=6rt>0f`ROo*VCbHP{RDXFxLta@Gnr`J z3^GA%L(9Bk$fQh4iC_T=uBxwO1cv$#|iv) zfQBuo3BT{Q7S8^LugOLXX{CA`j*y3h2zh7<0B*!9wvWtVMsg3-6yonkxhe)$`F}_X z!RWo%gzCiOLl!+5NO8XlA`#wmn;nxuEQ3_1uM+uQIjA5Hj zx+Qq3k{LqfY23x&Ya!BUKbMsH5EyW+-&lKPf)QLB%au-235yWb^?P`Qyb4~Vu6qUY z+Hz^>7a+FUcKS>SY8I=kS3zQO_vsq7O)9FEnyew>0pz3= z-Twew=hNo<2wab@1JNDo(t49j#Ezo+KWI`VG|u>wv1C5iltWVyOR2BimU})1Yq@>I z+14@nVxac_0Fz&des*8ABK>Xt$J79j1)T6EcVg#on(zY!^#y{(<#&I6SO@B zj|K`0A|9{t7#nWiI)KaR6Ou&+4)Uews%bA0v^O407ZVl7@E*N9l}csJOcI-dXpS_H z%6)aLk7HI}{{TQ+%alT}zFfj`sk)@2Q7uC2#P8RrwLZ94HTn~yFZ995JOizc{<@y> z%DtvBMqM}|NWCR*S}01y{#S2_T))V|6*W4)b4n@O@u9m;QIK~KkzEU6-(Mk7bxii| z%B5YrhkI-7z|dOgN@^Q>b#D!)fdwK9+$5pZGEl4e30dQtY$IM>Cy*$KsVPmv^X2M3 zN61v>X9^&T5)Bj347v7(`Vb|ZpUAypHjCGYq5*BLG6J<3Vj|$vaP`;2Z&V8C!ZuzI znJlX)dhqZhBspt?vN*5pIas1qZ=eUU2o9yM-gO(bPPkL@L`SjG`|B6jH@AK>hBVpq zm&Eg4(R`k)EdU&t!W~Ypi)qwM^fS7La;0}70<@+Lw948Mxzamy{i2s9Rq+dDtblT1 zyU;qKJPfmG^H_zT_YT~I2E=qi2o?{vX*erT?O|%+h9;;uIuxjw1MWP_dbQJxI!cSyD8V=$3QW z$WRq%HDfl*{)UqEVWpGv4xw}!z5<59)s3E5wS_lcPj7D};*P`{^gam25A~7#>o%DW z+$c0SYIB^d&YvY><{w%%r$_SyK^q?$&Mf#;FV#+E$geyJhV8J4R;u`_DJRg$`QR;< z=}waGjSf$T;blmf4RO9@Zm}F!Jrb}_E3~2r=A@`Ja`H7YsXvK)prf6y(ZC?e7OrC=Iv}hlm6r;=iej%~|q>y7+fctL+623N@oo zD;ae5`Otlp0*h}b?AD9Z=4l*8xIySVV@fQ%`W(fteem9irg-Vw^#Df?-j6Px(mFqEWmgPE;!Av|ZZC~rfYjsNK z!Fm`-jSv}>&uS~bfcy5=N=1F%eWf~m!jjaP)qKP0qmvNS9_9(HzLy^$1XZ;>Cj)v` z0%B%$59VNrVBfJcr5Zn<11S9`KS%(AixYktBg=Je_}A8Eq{a(ANEU1ff9yz&a)H!x zy0Idv!SZ+=m0uR&BYX-v(eN(Dqu>i@n~>Lq>da~ZfHsSWYk-6_#9D-)tW{W520IMs z%chZ*lBF`RN#q=EvmEF-ykFy(=-=|mu?{W_20 z1mr+@VuI`iNE=$|c6&m+h1ShgqeY1T?y6${0HH~KCc|x`rmw^y&P@#wnT=nVWZeG% zA^da_qxg1%xm%sp+71Gt$yj~I{RVIGG0lCG{`pUd-~u8W;#EMYG)O-zGwyW%0J}4z z2fSBvwfOQ;k;kkWsQ1ph{+fO27FNhm`*8-XVguGym$HRlyYX_&J>b#RekwO&*3$Si zypMtThDmeAaGIn(y?k9iM(46(8&b11SxA;QuzSkEJLN)!pF!;r5n9qdI3Oa$#@Fzu z`EBL-2kIQ(sB>2r!VMo>$A}F>+U@$dfDttqa0^7*Z`moQnbq#WX>%Zj=Xb=+%2)kz z<4|Oq!e)h0WlF!n(3Ie zf~$g@OWIAU2yrMfjjnsJa;p!u*|+-!O(NdK{{SE4#isNB0H=n(0B{~45b@Pp{{W)E zX&)kg)XsRU=}YOBGz-IXT~RQTw@I)i%_9MYA8dQX*N4TA$BK|GLhD9bPwl(bxZHk zT6zUNdHD)WUM7^q_U9{yOtz78{v*IoV07?iTa53`=>|jCk)1vl1B%xgJlB*_fb<8k z$czR9lmX!p9bzt0G$g9HK?N^sHOwr{1RPUW^8I?iL9+8JK`vFi-ri1CryJy3ZV-ov z1SMV$-_)b5Rr9FFucIqs?q6cC`$yZKxo$jh)@(k9?s~s_WA0wX3b~+9W9>7(VDdp; z;V3pwz;C9qte$$tV? zNE0_RdcSnREA26$5dE_1_)dEaauIH`Cq`EjUT_;faVu+U;uaexNfhECiRpBV4Db!;+jQJ#w+r13uG*vm}}L}O2_Pt4V50Z>Fqbkl~vV>tLJZ~Vli>7{HklS zS4Ob;q6Z-HV+KI)(pfH7@)AE>%1A$5q7Xz@k2U^rDprdQy3T|j=z-F7oqgY==aOac zTvn@B@uUG?cgl`Lby3QCyG2}t4V@NrYvNqFa{NriIVK!JS!2PDG%EIfAJ`&H3_Qt+ z%fQ~>gX!h}0K>JzpdnGHDd<^$oB+&gPtKFHIPjRs00HtQRSPmw$&!UpxP7J(I^0fN z@`SCbzDZ|M{)gw^i6VgZbMG0s7Ot;><3sUBwZvGeN!iK9yn>OMN)5hyJZB!xN>FB| zIAdi&vyFpprXusSi_yuhl?NbF46+K#Rr7uy4U8UN6ZU~s>&RE((+NyLE<^K;x@H2m zV|D~pQ@gtc8CqrS-_x7}swKtJMrY~3hm665XyBshM(fv4S%-sHJ&>o;8G`D~IR`$n zss4nyaJlL}$zs=->qYej+J8bruMLm(TAio*Je%@1aoAAfFp!s(?1!@e^G02_4}PTl~j3iOo;9VD(} z+dbic$<}@2ix{wSq=c=lzD>RJI}y!x7iNzJY>N|USm->rOBy4No^czIYU;@JmB0eV+z5CAiYPE;%a<-+1{<`Y*0DF8QSwXgUxAOu2I>W(FF^-Tw7I*C zPe&Gu=k4W|nij_`YcHuR%~toX$Am9}<;#~ADC6T>m`#Lm*B1j#<#OJvqF1Xqc7IXc zgH%SRg4L`Kws>B-R&BS{;H(Q2!0t*dR^S$~$37LVW|Tc*s1M^<2_gIpscUG>s@-Kw zX&Q8tuRh0T$apYH9lpG#uKsZoR5&<*q2_GF?5kIQ7un%aoTSQ(eKu$67 z#ayaWqZng;>C4FlXf1-!@@^cU<24`mF)J$7@CfC&?dO>e9a%+9R@?PpD?w~E&g<{> zgAIXY@OUo>AVM)pddrA0L=D3dTQvgZrZ&GQwOXwK3K#m4=01cYrpbp)6I)}IrzATa zV5G#t5QI@h6d7{(FTxp~-$sn#3wKx0I3^>o`wB?xfZja>vD6B`E%p-bNsnOCrOg)>CAezE3!0I10#4**2<_)BxT(jeD-jmhwhupyu$Ts6hlH^!CN!BL z0LX^eip*vs8+DY}>^i!}EpOdDgA2S0Caegm+U1t>;A;zkalnL7Vu}nrkOfRVms8yN znG5o1Ql5+y^^~;AM80958FId(NCKRHj6RVZbdc_w7 zuPv?mm%(!7_`rAt6k{65cCZE`mi!HC^|t${LEV9 zTD~<9bPHAj5j&kHUa{d>NtiM@I+Lpng1|+W1(6`Ll5LWKGB`#?BWFM0_n68Ka?tI& z&|ooA+p1-K<>fnG7Lf+4;zAKc1`v554yK2yMiDkZJvZ?VGqB3JW%->JFn!bU70Q2s zUs1{CqLu6Zf3rV|BdTZx%|8VAe8uA1M( zyy^tI;y&g!QA&D&n6g}=;LDdTT)BQAFtJ69e49kZQR!peW)Sc*wDi~CpN8cwW(1)z z-F@Kn=U3e^!t<4!i1^G3M?>mAZgqT}a$*y;(VG$1z|AWhwuQhwRX}_Ux6jD0v}zbX z*t*ahqVNhDnO+*JQmyg{oLkw+dQ%m zV8MgsfEu|vKLm3r^#1Q%HN&iKrQUOL$W5Cwr^4$?^CDehiBgqMU_p6;3b{Ymo=TLZ zg4F#x%a<-(xpL+4E?ii#V#mruYDUO=vJ*hMLeM!~t@t?hmn69o;kncJnDF~OkXc5D zQvn4r2&YFCU>ZE#6^?F^8L0QBJQ$IAq7E*K!;*_%fqUTW{Y! zlCV4syf#^7mT8ueNW|+ysv8N4RHi_F0Jpxc)W3Qkadq?`iJ`VfH+EQuW6Tg>!Iv&y z1!a~Z??V*R_$xAlGqcsr4WA7P;C%Og&Tf&f&|5TIJSG1T>p zTU4^86nWYV^_n>Dp1%oJ0o74fYPAjpb`pH#)9p3c>9>H_Q!?%daLYYJWpMn!jU_$@ zz>PFDPv^7%04Ove!HC+ONYEHbYqxrgz9rYdN^3F(z&e{haSJNd-eY9!5V-(!bcUP< z#WVTXdG$p;vnX=xkIW60f<`e5Wt&7H_! zm+GaJJ13k$mo8rt<;#}@yIMFPgpM0-{{XLkQ%Wz*s|e{BAwh{!rApwbQHn=|XT)A@P%aGTG88BLS*RTUFnIf-k%7@`qr6Xave*%v__n zRK`lU!hdiK4l`2%ksZymxi4s{kq*F*31xW7@1l-vj`F(rJe(h25 zDhA$a_G`xF%aoWB`~T#-+u*gdK{qZYvLs;wUx)$l*%36EP}^a zwQBTyCbSGA>k;q;fYpI*!A563IT08E&5T<2c|X|+2Y`6*Q3j)~p&vm+%BAj8Z2Gi3qnJU6s@t#r8<)c5MAX&o5+g>? z{Lh@7J@tsuKJR&Ne~3SFssrb&ML`26Fc2U>jtP5?5tS-bNl_w1oTjMQ78Foj+JEx&rfEj4D(72zn3ULe<|J_J{q8qv!$&xMPK0YE zuM351ao36o$d#b7Cqr)s+Lutt!0}?54a)w;Ya8JvttcV)(VV4gD^{3&bnxQ~ner0M z*^Q;>@gu=QRG0FE=?-_9xel`O0B_{&Nv7w2Js`2LaF-F3~ry zcQ#Z6GtK7$1PE}kMw|zSQ6fizLuJy-0WA@7nwv8ur*aGs=Ccy6xY&o7nJ&MGU;}e? z$^QVxkQ$I778~5z4`%)#oMfl7!n;o5D+&b!Q>gLx881fxg4BxWsM_1$!YCy=F<;x; z%cKDERn*2M^u8kNHX1=LL#gyI0j?xqC|oP zip@tB&-^+cEI03eQu{{V}tT+G$3s~LL|;Ly>52vb4UV2i2q zXJC47;b&yld`4peJRbw3=;4|}&II%UEb`n*x=Shg)c)Brv0(TPrrSjD6Z}%wFC$<4C!!bE{m2qTP?W6HIA@wAcO^D zHf8!ZSHk@8Od7Z|2*#gfIelEypmucPmxz@~uA;-W;F+k~z!qx^E&&v}rSiHkaB@>G z&U=s3Z%7-K)%Ks?nqj%!a(|D7-(w&89=>DY$sjmmzsT_C1{RY@-w_CBT5B70){Ygj zK?ahNJZvm1rntDppv<$K0G&W$zo@r;WkXqll5VDU&c`UM4z%6fmw?9K%rwXHKZ}OV z9s5D_=TMoe6>e9_>DD3PC~nO^+JcL!ts$0)Z4Pjn9bUIfvYW4pBgr*?k$pC7yp7s;;MtC0iT_8an*Oa|FGRu+o zWf3z1h-(K1h1M;pf*ejy)*a(l+(9)!qmD3ER*Ft>UN0@X)AR4(6;SgZwm-kL;#cJw zl=2y(4ge!XXgO3a`?)ct_4a?olF(li1t6Vf9d#W$ww>Czzt@@D*q5VMM?chAUCO_Y zBx2_mgTp^p?~a)a)?sjXAvI`)AQteA0u9J?ES9J{bnTP*u4;mNsJgVvSvf zBQ1HK^m9C(xX|FeJrBkI0O8Xpa3gQ^@V)|jRpnEyv^VS{{ocy_5$CKW>)>pu;>zwn ziB#FDk3dA4c1;f_NI;Gg46@m9Djw_VY}UYyF!4tvZzx+C>a)1wau8>OTvx$k$-;%h zJDYwzyarx)1Qd4Q;9;%X25EzVUl&0rvQ{m1Jya__MC;U=f!p@ELp(yOp;g`3cKxPs z9;ZXfyOxv(0^dPX9Omia1OTg`K>6L^gK>>E zM;#G6A7j&qb@-<)`0WF<4zj?GP%RTDw?}U>YPCVw4xT}V$?h@BG<`~!<=jk6q$-sT zlE`xZ0AR)*^W-5A167$fx*_aynWi0cz5~l;&@!@_xAnSBeF>p7J}JMeVFCnn346k_ zrasfIE;tM(s+W#5A`kBKs_Y#?gcUi}z_T%%ifcF~x%<|?F+i~W+;yU;s4}6_$8-Fy zH+`rVU3cRYtQxQIaQ&3L^>ml@UTnEV8adk@t-C;22UDc*P~N79(n?-a9|L~y`woio zGj`O@}N)dWbMmsdC#wr-M2acwrCcH6b=d93K zRy|o|4R&2@X6o#Z21NdTBMAdBybTSSZ!yQNOp~}W%%*E-^kXX*%Ph3AxpC{Y`SkK= zt5*7Nem!^|8<#FD1BYb4hPoW)qk;Mn7527Izaa*6drx<1W)4g~u5bXLqrg^W8vg*o z%wE@BU-C0BEnT9($V2D&fdUXmmuYH6C2K7I0MLqIDp?HY8DoH2!j~+oP)UXyYNk0p@`PRHn%Y;Q8*PqsS62$Th}%f};;;X#MQ(}|Y~((9TmyHMjv0$GiH(Y3cP7Cqi*M_IEN_DHB! z-ZU(YYBy1bKq`*Yo5A6Dv?n4dLucD~Y^_=#C`5}hW{)kBx(R2OZ(a^4V^3SN;a84I zbULPpg>ErYj+ihyKxtzL_H#v1VgiauOkHWar;x^!jgUB(q-=Hl`uG+B1Dvyqs62Gw z@|`J(jpOJ7cmDuL2k2zGc6d^d>ZrO}1s=h)BE3ti!3dfT6!_Ob%PTn#M|+oOVD!Pj z9KVQuA21+5fj==a%^9egV4Q(ldqH!oVag^3XOt5jWv5OZmzH&7sb6Vvgm?syOi_gA zqb$GrRr>Hki>%cvd7@b%jH0klG_z^Yf`Q%T42}*sxs3gI(+|^sPQP$43dG!mh?Y9b zccSeXei7O*q^M}PBCBigWCA*j-dx|n{w4++nmxwe7hS{x8YJB}&fn%5eLyfVHAujrr{?ad_HSU2Y&q2m;M{y2K^35&EJSlvm$@ znAl4ueXL4%mZMQODe!Q0Bm+?=dD`K#yjB5r;|4YUqq)YF>JX7x!@JA0t+`U27!m?8 zV&mzA(Xy$rd9fXK28$%&H<;sp4?SKI?VdSTFXcv%mNt?0bl$iUl8Dnt0%#9 z8XN88O2YeI!OyI7#RR8DToDaz4zn=O;O_I4PIy={9UferSZu&h8$f{qXLs6XW@ILW z=GAAsqp4K`$GJpT^h-8YJ3gV7^#+F?N3JDL&oJDZ4v)VNHxy>bV+67>U)-;`tp zBOcMG78h2~wOxr(N1$u|@YM!o2ZgZJDXm-++W~tsCRKbH{7f)iB@D@=1i&>%ZZ8VW zulteyEPX)~@<{kW;}vvx0<>kDNl^4Vzvno5a8{5&&Tuy(r3^%-T075}%~Ppfp!<0g zdiUg$9_6FlJ@|tv5~gHKVo?NvUKV%A`+mV4o?ZZ3bimbYOVr|DgQUR(=cGa%osp5I zy;CarhL2v6vmKq-Xj;|sH-_aM91*N*KY%bGK$sTH^ck9SkymrrsWS=}%>vVu0ccK; z3jpg6vg}|ZiDs**+L|~;Z({zZwi#Nrg<7!W;k*xQ(Vw(ggV~f&8k2h8S~CbHj{uoj zr7GK6F6RZJ*p-81JigGudUcj19GO{)8(c$bNlki_qmLLA3mha&`wcEJhe^zFdbK~l zF!28J5)-)aP}M_WsPl=Y9*@tjg1SwjD*dO{6~_zB$G;3#q0#Fcw#UGliChJ&Wv#Qt z;P7eyWIXXx{!yBqq@=RYT9>Zrq%pEJO&_>JR@=} zd-!o5;U=wCvFb77DuNrk7Bp?Kx1i=b@G_1vXoANK%u?Kktmu8pn#?Ij1H`H>BH6&# zKP3It=>7RMQ&vZ2r>PGK=X<0YVh+hig5kq>JlVZqsk;0XE}yK^*!$o%!{<=PqI570 z8C~YmTOV!|mzRQ1pi7};K?DdDEWxBQ%cD*;_867{#xp!iX^$MPvdZc?dWSi)hQw}* zJI+ju?JwH~TokLKkuS>5aqiEH!-@VnLTO|lO#XCN5yk5rNMtUOuP)}$Ein0xy6NC` zVwx3>v#kuZ;2F$rbBRU#qku<)LAn?b@QlN+>$+Y`v1?VMFoT_QA6|-mf@?q0*{&JO z^WV4Mf;vah3j5JuS$S|nObD*GjAUA#2foEVas4?RL2(#hCapcIyFY585$RS5_{)2Ln_c$C%ARMFD4C zylTZ>UqlRxK|oDa9XJ&m6E6-9a}s1Pt(`8t&1Ex~`t|MPQwhIRyO57gkj=}eTOpA` z0S$)=hYjUPy?(Ikz%@Ga`Wz5SOC|`0lS#T0YukoMt7>72tEVN*A@jmK$NW!NK>`G) z4g)bJga%uxeMPdlW@)aJDa^}wPj=zcrS15EdC>4MM5XWvK)zSPssnYS1OD(-?IY}5 z2;u#OblxG8T&@%u-dem^%ZCOY4jQuYTfuaRRt%7;P_j4;M!{!W$@hOHU9u< zdFutLv%pN+OO6dAc{S`D8ks$D9WV;z`kn!&U0j^!+39tH%kruqq{MhkX8_fL-Wt|a zakJgs?PsCnu6)6HZ2I?SG8rr&Zkxo>9ucF%sxHy|&#&OD?rtIm=}D&U2muP{@|Px$ zvA+hj96j#thxp5xh%VNI5J5A0#kN#1M#)!fz0+nLZI?)>M4H|jPq)X?y68-((6_hD6%7oAtmCHlS zfmY;M9V4v18LAuwj1CB}b5C@WUQ;?3;rNEQ0XMR)C`-&kP9HKlK>q+izi<%gVe>AX zv^lXUOI)vo;PD!@ramR#@f@YWcNr-8l#vv&Uff;}jev>1>_tk@UE&~8YIt$>Sl|Bu zY~AJHyQ|oCpxT93`E9@g_OEXG{zxqz}e8x>yOHslii8@ z{{Xad)l)rQd>;kgKU|T#lC}L1a_m1O#4KHZM6pq5nS0AwTo%JGzXBh)H;R-Rj|a)g zSiQ-FKAsDR^P$Ko@43f_yH~{J7E7|gPRtwMm-@L7?T+=o5Ca{YTDJy0CUJ$4cmVNjTS}{UOFdeFoS#dW& zeIOJ8(3g>gH0u<(=H^RZ^hf(S5f9hAbgse(Ae|#w%VFyMFk5kiC06GulqM4l1wej$ z9R6j2ru9mgouqM!e^w3$c_+RmZ=5GViv3cbR~LiW!S5Dkf7MI5YI}GK1u`x(ru`t_ z?rO6;3<%J@*ZMbgR}(BjN7ymKM;}fKsv2SqM3*#9Zi2T-b|4 zxbgX1Aa?}6pZ4&&0696qfKTy$1)9CMsx{DaW0u|@-_(@c9UV!7E>6z}!*B_QT<~VG zYGj{;j=t^|?NC}PRhe0GmJvH(h;MWBF!I;@2gKDuL8M6paY7e|>*IB~Zn1EpL6-$1 zAa->9dc>7{43CibCFth0mbkF!#bX8_6Hs7r8s!{Wtv}-=&%_t#efTbwuLw3yHefyk z6`Xq|bZ(xASU_EO{{RQv2qBtgjXAufJwe*!dP<_WRY72ao=J1pCC1ji;rT!XG}1Lc zz)XCJXJ-#k9s>m{{Kh)pPN41sKxR5TqNW_llq@H`|;M>_%SGs z1gh|7Ex%p({{SabgA0epSc3hYPVwOJ>W=Gq;PA|Ke-jVi46tUl)VLzqD1Fb6BY=j< zMRtalbav`;n6VeWtPgWv3JktHM1eD6oP@d31!)M=#e=A*>;CO?Pj zFCA|&l&l0Z?}Phf>_?4_LMW2;_!H~h`->(uN+zEo4z^Y{2fSeG>DcsMpU64 z#O)se3FA&}gsy3Il2}7yyNv-yan38H+PnZcv#-D9zy$RQ#ET17H}vV@fr9YJfkIG# z7!WuZsw z5Q(BW{{RL`otQ|uI4m~!F$Y+A%fUub;qM;?hX~(tE9N; zG*WG!yB6Vg^pp!h-d!n`r$X~CS~tK_;nr*q?rc)YCg0hFJbQ2p9)C(<{Eumfmp;#V z?q9h0FN)%K!uaqm_0N1q&04E$ezA-K_4{I?*;3_MZ%qCnx7i*RB3^qcBDKDfR&Y3v z7lt2NdLp-t1(KWdh^Xne@J*5b01QzJC;6v=;J~(~3=?ywh2#C5NVpR*d821HlxnuR z8wV)x{*L|!g#fZ9;*1ZW;e7fVZzcr6vS@+hzV8B$0!46`911s`Xr1Rh3bmR9s$Bs@ zIArKDoTupeJd|9xF0QYCi<%{n2tlk6=3;JS8K@={A;`M^dET??K;(mqN{R#DpgL_$3- zLsO|*`hZ;`jO+S=7gc0IX6eN(#hs>sRGXgrLfYxF-lti2Y9M;N53XvCpZlb2`qa~v zgHiw*B1Inq$Fy{gjJWjpwI?#7JX*0E=hb?~qEdpFtWrcEu}*D^8piL@QHS*6&700z z09brNsH4#u{{X!HBah&D5L(l(A-+fvZP=s{Te)#@xW*|{Z`@72bb&1dvyFhXnEwFb zGyd^0sybE;bpHT^SiEsd-Ql;u@n`aNArxShK(=BWpYy@s@zsa1c)Tzl=4O~ZbotAt zp1~86Eh_Xeo;5Wi;HQ+Q!i~h4ibgnO;k*X(orn;H%v=QoB@hT;6QuPUdTHbeB5FXo zgVFEnhIL^EumQ|VW?VRR6?`#XbDrW);TZ?i30CXkI`JrCfw7mQeU)?C00BXy>gdgL zQZ%AlSJC|Xgcj7mZ@jO2+okDpxSoc6h-wnh-8Gew34+sgVx z01-_wrXxjQH?eH|!(Bzhq&-7!-lVb-g_MrD;=uHd<6QpEiFxn$FwEQ*()Z_r0ZDX2}z7cYxY9nY9q%LQoG2Ck*QDW~sD zz9{4)3KpHW^+8E64eKt$awg8sQPNx+IJ^dO=hRt0JbkH7V^Mu`#OBRbO6*0oslAu( zEImJLd<=Ul1Yyq$b_{$$zZ{hUJ=r*qV#x4#Jsb*OJA43iH)3N&+;Wy$TpW&3mNK_= z+lE-6c@bU(w^4(Z5ay2kh-h{Au2^AN-Q1NfbN-xILI;0|v0 zCFwLJ$YMWKkVSR5(EfUKVO5_djWlb9Z)Oj)qrA4vIB4cwYh# zTE$8fT8>mb4sLuXGjG(If8l$FzXd8dLi*0hj~t)UbyE_qwfTWWjKx7PpQN;86Rs_cIgf^I7afA4Mm*On6W1{a98Wyt)&0zW|Vz!&=gV)#YXeIBW{{Y!v zrUPPKbJdNdN1^_LX_P0_i%U+9Ww5^^Zrq-bq#dKOU6Vw7rVPz43=l=jir;wgIy&DG z<#{1qK6y(Pz{S)sL90z7GJD&oj8vbUDMSwR(64zOs`p5C*^dAHYS~ zQSrgy%q8gwTI-}Jb9dks%nxo1U@8G?w5%Lze(=x%phf|QL+>H=4G2BX4K!$W4LF_; z9aE~h5k+oGx|FgT9tKG?tQ&|TSlJLtM_Z)4fiUdXEBfHh%7&+X&WAEi|h{St%}XNHs+`;utQ@^}Ph z;P7M%SYJ`2scZz%7pWAvn~3tq(Q9P$TG`Ft?34$XWUp2JH6W;GN0S4JN=V0p#|49R z_w^cFDiBJFwR{BJ84_jx0PJc0S@o7S@tR@}x6&I@1+ZvvRTn576&;c#$(Ze{@IKrS zjdeLOvU+EN7&MrgM-3+149pb04mB%rDmaf533CG;Voh~WKEvvH6g(vAO~{RMcRl7Q zR89prU6U-=;sYupq#4%(vNO)2{sP; z{@8bGiP3W@+sD5E|i}cJ>Ckb%~&B2oCE;+12=dH~7yY(IhZ0a2^_dYFP z3gKycamy8*wTQ}l=DmX~unC7ptn0tDTqj_h$*LgNeVt%-Q0pu%3>l6?{{UO>XRI0D z2p`)$Noay={z{n-03|D;uI5k76*LzURZj&(UgJ~fD1IlTtBwBvk}qF#gW%Dua$>eS zL2)q9b>L}QQ8or=t2bpQ^UYq}AOHX`1awSO2_>4V;$J>}<|dKPmWTUzX7TQJ{^{Xo zD4MY?0j_(L!4L;4(3)Dnr83C!R(gMKKR-DtgQ&yE6hJYwx3|E)G&y$#4A};!^!sqJ zlB9Hmxf}F>+9nC4J0Tf3E7B_<0f=51z7GUmYuYz`p59Fk>HI%F`~i3zs$U)^bZk8d z{zZP}euMHQk-1Z#4+1GoBvRe_M^IZt3%?1W8FG7y@>P=jSL&|!8Fg#n>4QFs>I_g-D zQ{w%g3kc+R8`}?wv~>HQh@rs6z(EeCaupc5kR z)-`#xYn}SdUG|TuIsrv;nwWrU&1sI1hzKS`nw;6574mjq;{~Im_pI!! z({Z3KDXM<8g&Z~{BMdW02qwtQH`gT5Wezz4P&-D|523TI=*`#>B44E2Y zt8Wab!+x=IO?+>i2db?hp2-5MxqqFZQuGd2_zO;LZT&&Ub&lU~^>vZcDcPjLI zO9$LI4?g~R{sDiuMC`AJ)`2~I@Z6ZlI?ft9YD18(eYep zq;~LRx?J?6FTm{|CIss%!)J#GNNXX%4Ob|%nfa_=Xwwpq!33@hsB_yNPQH`nyaDnW zdOvqaX$1Pwt|^0twk$+8zY+fcV2!07Kwq=NymP>cJSmaeQ0wdHO<m0HyqK&#o<|_sa81XJF!Omck;>#)q z3Onii!&gjSc)@P}0H}n4*IL%!-D4yM%D3Jv)gYe7`s*{K+pM*&Q=s5SE@-qC#^ATB zzGT5#(Z^`vvv&=dkb?$cy9qu_2?Tg{OHqPI*AL)ygip-em&betBLN*oUrp}Scm;=8 zXhIL&KhZHwZT|O$)-eQW93f``@IDTyNAU*E=-I3n*YAdp1syV0`Xm?7$W z{mLQCQLN~WE=>hJrX}bCn|-bU+Iz`<1wM{#_Y}`rb^7n^2)SC0%+BWDF38IgjLwA; zm}knBn%fIqu|D6y(1tCGcL%sbL09)Efzuui#KIztpa7tw!KF+QaG+lzV2NX7!(N z$dNwZws_pjxC2QGVbBoQRJ{0m%hPcOFd=Igg4D%}GSXPfsHbL_FUEwRm;^UNSvyRZ z1A%x#E_K~rqW6o%2q_cD%>IMiJ)lMa8;r(hHN!-5&WjJs!OW@J!}B{dk|*gqayV)V zu{)z^69IcafapeTX=^l|m7lDNG+3j2aV{{S!worK4JZZE8L{v-ivZT7T)Z`uf- zs(+DR(gzwX^EohUv!uBj_9V8`HfjfF{G922q@tt5OP5OGF3|l+wG(?Um)@n-IYl2> zjE_dsva8+XRLdy8)*lImnS^|$(7xZ;$INpVvWtDj{ZYE4woQA(*UI7dvc4l{U>|f8 zzuNXWnP?195G+VCoCeEpPwim_ZD_%U32cD#<(iJp7*bhLnA%)Wxu*95ZtUgS217g! z&XjM`a$9%%@Vw+AK}qcs?-d`SruU{H=@`t<&FFD_Jcj+8q3H!)gtfJW{x@C=6D)S{ z6nfCS_h@dy;zyTSZ5nvta)FcZ{D8 z=q62!5)U{$;B$e_1DqakD4nr@YwxeWlbwz8m1*tzmJy{=X({C()a|4oftT-{_&~Iq z?n!t-FZxxO_|eL6ynfg^%W<`ntWw-XC~#)Ppk>t#jozL6@|w+`HP`g%5pXy-VAXHT zu3>*!Ia*WbCd3a_ex_7Bw_HI^NG32>XsrvDV)dO-D?7ewQMbBte61tg-_{VSF-1PQ zJ(INO`0HNdY4i$<)8@&HQfYzQf4!!Kar7UldUzeUPM@>{K$~;v zyb5W_ndVJ+fN1pW?KYS;Lax~zBD0YrXZC&PWZ@oEDuXi8ZK@^RE`1;jWJ7I1gIYe= zX|iK^N}oGD2LhPfU(3t8~`joY4c zU7&4%2EoGCUhy2-K>Clx&G1+3{{XBgm3_*7A*Pc(HlZw-Gbu+8BZMF4O6@G(b&LR8 z(<*`8E-o0IdDWd*u5f(f^NY>}p29kQy<<31H2Dnm7pm*~nVg1Ww3|c{qST|0K1K_L zhIB{b3L48TUgrkhzfcy10bjZXAIR7*Vg(*w=pe~T9Qnpif({H+O38MPyr^OYTYo|K z8nJ{GZ568y>m1GXz4n-&zk_xmNdExB5r=f3Hn20;PHZlU`cQHv-s1q4fbC)4|&==x$LlR?iD>1ABmn>f!jj|(&d`FNY}ngmDKKH zdNFx%qM2P{FpCnBJ5rnoF!_VikudcKm9zfECNV{@U=3c9)dKA~C7cqGYLhX=3{H7c z7tI-Q3Km19Hvk5Zq3&t*7xIW1J0Pt3`G|Ma$-3n*2Sse3^Q5`+K3q2=nJClIW>x0( zc)kl~P&exq_Jf-g)}U&wTCR{+2L|hu!0S!1!m*XF zxQ1fWCJ~pu;(ttWHxM5z!+ub<0zAQ1ZAPQv^>N1v-XV4L$@?X=P#yfAa^NDzrq9j6 zEm6D;*TZuZ)qs^$A&uFek`o{m-GjPv7|X_95g?<@GJ8Mb{d|!Pb#*tt9*~Za14K5n z4-%yBLJ(IeHvRDx5GbRnfGPy)qZ-0EKmoy(=F<~Ost|BgW^f*sv2d62yADJ+q3AzQ z8i%d|4(PI+eKRTWDJ27=^B!}ht@<{XSM7)ME?)zsf1X0i)yrMe0eO@1vysHuo0#KUAZ%|tR?k?vgRlWK?p^r{a$vMRb34^ae zxihnKpVlKQ^>vpZ8P@gwkUEc6u8%XA!E4+;Yw$6x$7zxkN!mv3E#CP*Ch!ma851H9 zM_7g#r4rT>GX3@X-~fO}QGu%!+FkmI)^jl^uQQaxpI_AbNgV*K-BzBO)%Q+W@R6>G5;g>`;!2Au-!M%)1a+MO6FPEl^A zfQ5*F{=FtBz9fBH={gt&oo4%8dU1;%1%$diI^O{)IXBP=^GyQZ3skUX#1JZ93j@L@ z2oXz(Q&_xF@tg$7cetPB#|=I}05i_RzoSs0?1p7Q;j{B#ip1 z@NwJ^r8!?!ce!&$Sxl#aG^ChjF35(ge>9f1EO#x)U_s&!Jx8Mycy*6%_4ONaYX1Po z`5%kR9HB3XPxJo(B}QLDHI0SROP3sTEihGDtSexiw4%=|4jLn{YJ^4+`(*b6R9E$rn==@F%$O<;*;kS*!Cu4A(@ zkr+*SJ$NL$9`15wX_bsC)YD$Xw7GSWOeNeVel5}K^kBhHEO+&qI1%d@b2LDf8uo^L zMIKwY{CBUk7X+1U?MrF-dcy&9ZwOT%X9tVIv@ScN<4b+iW11>V>~f4Z2lV!TQ~q=Z z6IUF=LN3dZeMk1X!<2L3yV!v3O3tA)nnRAIXf0V^L)gbV^^FeMY`TZhUI44`Ee!tv zBz5?f(u8#_pNrv8Vqswnh-ziP-iVmRxIIJCIV-UYWa7_-oE+0IA*N_^Je$`&`D^Sx zM}=H@iBdN%5quDTkNS?1>a`_#y?8J4=?dyP4h!}>OUe;aIf;L1SUa!KfsU3l@5KHi zd4SpuLsI_$gnW`)Rx2G@SgKw6@YQd>Aj=Cnig!bDm>qMNdSe@qj&F{}x-kpwHEGD< zCjQ|XkMZxv2;uVwcP6_ZZcyq3xwO^n2lsE8MavPn&I*jK;#|2}Q&?oj4rjN#$sbh> z{{UXBS5 z{{ZE{zJt_xon~~_=-1L}#8!wd+VfMii1^?^#Jzv8xc1H6*#HFzRzCPEs36lzY5;(le@v6Df*ww z8u8OLad;fIdt%Jx^hUZV9)s}$tSwn))Q5LDojd}XLouuJ%dy?-IV@!yitPYEwbseH z{9FtPeIvvyX;Znu5H9LAb@yJ-5nS@^68 zr5+6vA}sis#^zJKqjUS0lZqNTJO23f;5=H9ChM?9H5#2kp2<1F7cr0TP85@zoKXx*-Guv?bpKSa=?QTI*<4(49*53#rilp3(ju^%CRaTfhI{V6FB8M!s}F+#2D7dJ^>1E+zC7ke=FtV5!! zEkvj6f@;y&Y*HGN0(nX(sCXP(ZXor)%+6rOu$g8{v0JZd_l!w(A5WyGnZ)%M24DoC z_~14LWa*Q^uvS6ubo=|8kj|da?`=D*UvG)6 zUgueR6~MWcz6*z{9S{33r2+nit1x1R-uZB98D%WdGHjh-e@Z4|1exJSQGvozR?*Hb zz^e?UEW^77M%1=b8tSHqYzu}b`TqbHg^R?bpMLTZ0|_yk8WLiL_pUj?QGgcu-Hqm9 zT_50j`$vq{yIG@Ams}r*;tCnS*Js-OBbB1N@AE7t^*yx=Wtx~6b2oo;gdT(J;KP7p zF>N!I2X*zL##qLwk%b%d>Y8;R;hy0PL^EH@nZ^^Y(@fXiFe6$Bg>cs@Amqw5l)X0f zPy2T+286J>@1ry1@c#hRSw}^m;>=Rij1=!(7yvd6O&dCjJ>rc~Ye2qf)3h$Q^a$AC zh$5>t2BLutMCaW7~Vj zKN;*m!VMXq^G`Dm#|IN?OI?WDTFwVBs>Qf5YiW6`ixc3MbIA^ zHQW@q3s+(jJB9ZNrrq#)Lg`;<@MqfkW?=8cDn6bEPyA9@{{VSp&`1EePI~S?9-JQ# zW!xTJnR$*5rv#!{G`B9@U_dniKueg_Y6-dumpRmu;STQ+Rb#iosXY<|97;jxM=Hhn zt8W5pQ%J)Lk{p*9WGV!x6tb*oO-v}5sgs$@VCSQ(6j8*OEMq9B{U```zAp>HxYq*Z z`I_H<_kSc&vUYuOdtQ+Ow&iibVK4Mygq_iRiei_Lm$m-K-Z5A3~xhXSAyndJxNFNFjkPaAW2OlKeMFI z_bc97`H;*;fL^FfOfvzOCw;zQ9**1VOC(dFEe%^rTh0cTNl}Q65}Ds9JSxd_I;;Lg z9j<&;Zoyhqc$F!L*~-!ZmW06k~o54^gRuTz==&Jp}mK@ZU*N ze{c0aksyId)F}Zg5?d8sOSH7wjC6HN5;RWGv~b{wd^YER zP<%2=CA>5MIvvQz&R3er<#b@B0)UCA@xUp0*IfG5`pU!5s$=Z;a9cRPCT9Ju`b?sA z1U=p+BwTjM+KB9!CAve@BL$KgaG|3SR#&!YK`F|vx@%9N1nPloU3jm)7QT@ zjj`=Mlh9Ac_bfg`?}F{=w@1OW{M-T`xj-+llj(LA!VMSDy$ved)4z$dd< zhn6*{!uBbRd%81mr$o)%$FHKV*uR(s>bBKpF&)a*eOPS(mHz;VS_X5vAve9E(l!0> z8H6)+^vmW4_{VsgwvFiE>t<{dqo!$u3Ty04t=UrbSo=rERZ+}3zVNE6!$&jM2kSC) z;DjKEHPLi^R5+p%66zl+Vq^<9Hl|iwFsCSmgdM6NRs12lvu^w94l7VeJuUor1p$Zl`)t@>rdg8S*ARlp_Uc_36d#cd3=l$11WS}5q?rvZiEO`wutk^f{YWC z5W=<-UaVi*i$Xd->>pmU_96-WdP*ap`G1cNnW|8eA`=D})$Owt!tuf3^Pgn?f}s8J z((t2yAQXNj>yzwogliHX^mLo-=I71p9u+ROnQ;!L5aLx;oh?)-arP?u_|csi?ds?_v;_#LHISFGefZoZ z_5MpeW#F^*f5ky03P(hVy*ID@>j4M7je~KIX`tm~yC-j5>4|?z0S9=+nXgMEy4!bc zJ}Zs_-6g1M{fAa)%I&2tqxg%u-^h%U)grCf<$h^cEDbL=F3?IhqNgyv@ojT$w^Q{H z5K&{~GM1LKIm+!V>cztAe>Z45Wg6^1a}_u1f0=6Q=GdJBnfRM!t8Lgky*m#Eib+mx z$Gmfuq1Lwj`ocSazL9LTuAu#M`I>GesIsCr5kc%trrhZMW(*V1O=q`%B4jc4b}l{J z_?>n1@ZfMDYRto~IY)!XeMRJeQ6WNq56AiA9})pdRruR}>y}#?WzOUL;xPC+RzYf5 zlS=;pm%j%|w;g;I%DqGWE(-Md3Hm)IIN;Z;t^*Epf!CNJs@bQ3t~^0*!XUK?)8>gEY~4Kcer~r;|!e zuc%b%f1Z;%Z{`0062}o*yG5NXTWtH#ge|kBM0-q6m)42U2%|!zZ6Is+41(#l9smshfketJJF))&V@V8-OHB8b#_*VY_fjMgYA2Gv#iLDjGWPB^bq9|F95a2Li!IY-eEU^V<40waUhZ3C1 zr|Pe(@ZZu+PB#7Y_TVKE20F{QJkAC;k=vnT$x|Rw0qtBbc#fv{{Vp$SffBUo`_JT zuHVGQ=e|&d+`W%3&5`!4zK?GS9wkv-CsZP zEMgmN-8I+NcJKtFsL&uf)?aK#;u5M$o%j!gK9u;B4z@qXSf}38l8DA(84wfz)nc(@ zgQd87)7_iHk_BnY)*E+it}G2;T2Ee{{Xyn{0fgH@PsJmd}q$# zQ6fs0as^C_2{yH_hy{T_Y;eTIegpYTPJ}V}l-+IUe3IBXRUC236xJvFqxS|J>j5Lf zrQvFdXyb5r3?nWZb1XDM5QHLdXk#jW9uS*>u{MHJz^lGp9$;#Z&H*s-epBG<4sW}9 zviOzFGMGmx(M(Lhe4_cu>k}(kJ;wlVXtpIY+kpQ7iG+hsWAZ{PeOG1t%e;w~rgfQ4 z@PaY1z>A-&Z!OQK-Zy5OAyo>&{KTZ8{{SK*;~D|H_Ks~B!FogmGP}F1TipC~^B9ey zzoRhGp2XsyTY@oj{qN>HlS$oilnagKKPZ<50EJ}7g&t_;r|#3$nmr$LzYOxc_irU$ z^5tV>1%aT8GmhN5exM}^tF6bh08~?3w;?c9=w|P4xrjTy!1=$4TFSgmWLc{VrLlfdVk_sNL1aK;24o%h7xt z652<>j_Tgqz*3&F)AV{u7i=SxXICt_NS;f#wN1iDX+zA3c5t|SB^Ef`7%+*5u`G=6 zaK0cQL_ljr0TmN`oS(gYW~#~OTn~dpE3Yyv6#5O5ryi_8Y8OIxAo5?|=t7oS0xkGg zTq}d71;H4r8b9n3y-1wK3}9N^oeCzmesH#Q_@(y*5zPcI~{{rUAqRRMntF&g+RzeQ^cSN+U# zI)CIsCW1dZ6GyMx>Ts}S(_*1rZhs6`di%fUDX-4L4t`_u7#i6E%_ev@9Na2Ih-e@7 zlsF;e1xCM~q5eTj6H3}$Ii^sKa`_D0ifCg4m;8!Y@^l?I_f-0i#^b7aH|d|t0WPc4(e@?`=BT`ueW6Du;pRrL3^58tFk$+nO2)81V>@H{`yHakplWym#oO(AGC z=fa?Ryrm(XZxg?D+>5>KR&S!#&^VUTr?NM9q$wF%b20p9{a!eI{0jd7hZ9r0AqdJ2 znTW(CwHeQJgH99<^~>JlG1ON0H6#zx1CPNU)y}rX9u^j^j?pz*gO^{Lj(R`sdwN3p zGZm6(ic^~~W8od}o@2x~JSAQ}%N(#F;Kvg}Si002GQ z7BTwt5Cx?JNv1K$!)tVcXgmNG>yXw6p`qtd9s%hc5a zc)OC%nvPkw_Wi-p0BfUj`jhMHF#t?HZc%JQ4`uNBg3y6(I&nYnR2VG1hZZv>2JVAb z{{SF5l@`ZIM&G>DgzCid^c>z%w_aEGtX2O2;M8ttQ@MnNl2LC=O)2xNeThL_61!fF z2hX!UvWvj$bA4RQFDS-5IV*cZ2UpK&XS(y6CXEQoHsf<1|(sEbhg;8ta!zZ>f{w55V{vVgTusarv z*JPq;yeeXS$zQ+QfY&X^Mcv*eNltpTxe(}vvau!Ci(?gdFHPUm_7X77t7ek)-$IcM| zDAagR(Q518(^yTcEMf^sV+`NNBHR4K(x0{d_>F!18h-DD_(`qaJPC56(&EdDNMT<% z;xedr;b(zNIkOgBmF17u)!B|ZNlw*u)LJ6uVql3g~ z#!lV?f{JE|l9<|bKXX^5J!?CZ3ln&tL8Cuv9-5K7<42i-Tg))QEV)Q8)Z+gD3ncBt2ZYMNnQyH1{dcyq)KuIINKKZdS4p09&G*Z7Zxnw{lJl{)&%{c7?Buw_p5VE96!Wsg1* zAwgey{`8#6sypzjXx4hJ?v~ynn=Zc5LM1+g8x0uW1~ z8-;@S9e$##ZdqkVx4Ze*g}tF`*LoarPz}?YlpsK=Gzp<#XhINfctsBdF$6d#vx#00 zMp^}pc1wKpfE=pSYV(b!Gi9$=M@gmCk(S)agnN)#(BQ0za2;>(SJZe;l`MKf$~(|v zPU=_bQx5VA{v42wM*f-q0K<0CXu1r`=9}BmA9#+lX0r8-wC?2m#){Wu_%3F&2N};T zK<$)@DGkPXuJQM3^ZS=Y8b*b!62?RRS^JjUt5{+Jn%Q+DUlP2BHaN=UCSB7&(3miM z_TqSIcAIdkill&Qva2+AzLp41A00$A5 zz!r0l$N8TNH9PRG7p_86+q_DcQOO68D3p@+J_F4SZmb_MYfA^TeB99>W{I%i%2 z+YB~ccnL$j+=wW^Rx}CghVytFMQ{`ci$@P-!5Ze9un*4);^XUl=abfZ5|^>wU)GLE zR3tjzCIv2T>rO}C;Wd(Lt1`l!WgV`+_<=yOXcrx9dsJ#N;4AOaXH?ka+7x3F+G&|Z zyAplEf{YNC*2Uc)gu3S3c>K9ek~7I36l7%soKjbzzht&9S-mF7U5)J&oh>Uu+aV)tq$v3%XC=_0NrqF7o&?tMiB&gZ&S=taRYJd^!0whb_Kf zMg|9+W%rz~8bV}7>MwLi_~B+b68XnLw!5q@wB5lfzAwcX|^Epint?dCEsmu?`28$4Q8} zyCcnL{D8UiF58y;ks7hau3@)05THF%9C^;5lACrme1yb|H~Z06t!6mIk3ievp>3k(%r21R66N?Uqlc(w9!PAPaQ#fe zL0y@R(_S5TQd+exMA1}AJtqDdTFzgK{OTnUS9geM#=XyO1}dY>=E ztlO=XE)k!sva~yl{{S!4pvoa0qwg$Y;P6aUtmmfTD_l-KjxQNyg={XdB(&8SFl%tA zQM&XV=x!VEKh+`iswEWC;gG z;P;9F1&!RPmI67>XmYI7muHPxT*vJTy6$W9G(-)|#kg&e@?XSQuFd@m)z$iku16q_ zcg{WOaM5R)zn@RqJyG%$?YF-R^dZ4`n2^IlX83<{j>d*bTSN`GrhxY$>*mjKi+`#C z73oNNz~Y9vj=Wl^{z2%(0La7=og+-*Uh%-jNmyP;TEx{}J^Y_6RyA|nG>%8pfP}*XbVfw{Ib?v#?EuA zi|-VV!JE7$BgGygs%5>;UY+2#9>x#cpy`&&YgL!DvI~#);6Sc#LIoD4AZJ560v3?v z-E)ZgMKrCoL>xe%z6TS_?kim2q)A!<7)yw<`W(M3D;ncJz0wp_dL*0Oj%lM7V!~w3 z8G`97nX5fG5y9Zpuj}-bEMVmNlOg44mrG@^^4|(Ffd$UrDMdrh8ZI@!W$w%1J*pAu zmHgW_$MxEAKggSf**$N_o~;mmnNoE5Vu!M z8XpI@0>?TJx zF#C~JwYP;{?0lGGUtpS%ihnP6ID|AST(zU^!GuJ59y2eg=j7sV!BGDInIg6ZM?BwC z6YAh?Xv)rHI@7NUoaUjhcsvWF6^m)0OYXP!o&E$vQd~AmiILqm;MTLY>9zXyn~x}25GGsbeAL4gb?N~9ueX{Lt=)wO&BmL< z7SHKVtUS6YTqn8tdqAFoh#FX2?lau$z_RK+{{Z9&h3G9}`gHe$*AW`*b7j%>azW1I zSMNGY9tIKEdcGku21@BM?@#+?H!sc8!Q3VL53U*N4cA{(Qplur9s}95+7nsUQ}4j? z75pB2O+N$r?-0y$WBBVaKD;*fNi&kt-5n#<6715WtC#mMMVRirC-3FfL3|<*dyr&^`b$LcV$`)J9I8$w? zKK4h<(Cuee56q+}3LK9vs1)^ngt!)&uAX1y|Beg^f#j(d-2UV~$C;Y~vz zIq&8R0sb$Eu;xczJ$3t&qMzja$6b@^F>1`bc8iK#bS{_EE4Y9Pf`VX+J`vO2(nSE< zv%Ul7J7l(9mNZ6^U|iOpze$(5o2_hqePX`|idL{e;vN#XlBbMXaK#I4({%n*W*K|M z#w@7adcre#!K^X0X5`gquSn?MKyL_;Ir8!)-!}umO$Ls5OsC^U9@5Wd-Mkj(e-zRg z#Y{`f2R2qw#^2iIv?78`hDoMRr=$%C-@Dy|KM^UD82MSWe_29J_AieWZiU2;w6oIz)rFO0}?j-Iro}oyYM)LXf2=FxuK6U znRhLCss_5dL&67Wg+9I?w}BR>7Y7FY(@BC?%*h6sHzAYi~5UzcVq^QwmW4E`!1a(n^-b*Ahc^h&O z-V6i4Tf?Kt=P4Lx+ilR~%J~qrvCtSOs5mjua!==zv{mg z;7o3!BBZV);Hm6T0n+#08(_8BFmP;_>B#EhQTpM!9Zgwqx?!y*F;+uO4n5`E4PF`x zeL&Uzec-A=WyqSB>c;9Ia?|jAzA^0B9#Q2rB4!7Hj4)#(i`rA)WSrOOHUZjc8rG1F zRWp1a3b`u*cA1$kQ!nk0=c^ZA-K_Ffz1k z-PzgqsaF)8sak(@i(|LE{XdDAWlw_qJ);(C()Xivd<@QL^*KdQ>>v01PMCx&e58*1 z33c`x+1K4=4JV|?DtZ#_CeuAg;~(S~;&ni!_ZL28$`h1xV=?DVRqpD=aRF`ubBd$Q zJF%{Oh?j8D(i03|KA;s^*F7H3$t} zPE_q0yFKGwmqKGmg(q&e2#=!biokkIdOBPR-9C|c_~1Vf0rde#OB|wXnGj&Uul!1qS(T8VEpj1FlWNYgWpIp*V;CYmM&Szn4dxn5?MEmCX>0( zh1hcmrzooZ3>>?`qeeQ#wZfG()p~T4d#Z6lgHjxFeSWalfEln`9O@iqvqUx?k%lK* zlCiOh!U=2x5su_Kl$ zb?Te-+vtKvp1ED0EK!z=7?uZ({{S;FMp=8o$#8ey@jjarY3hS@$94i+g_`mP3WhDXMfV?29O8p1SZA2aH zzcVWu)Oii7=`buRvu{S;G>JqTS^o5aOrBUJq}7Nq@H{qOIe9FKP;G0K%5q=d0lJ~1 zHLV)WvSKwj-celv&K1Bnyt%}`ZMYEDWxh}0te|Ez8jywSs<;xRbz$u*2L=#yE@2dvV3n->;3ybcbYNmCKz$A!M-F#%MCcf#*sq-&6bVmAqwVcKP zv;P1giJ3pw&huDg_Ge!1y@er2h9*lT8ilt4ET2E#g2wW_i68-@!28Nzenl^&fup6?9 zn!Y9K-_YC1EdaM{XVla=CKdqDjzl#o)OZX^VW)!_#}GOXca(dw=;Y-W#yNaO=q+tY zVrtyp1_emWRL%;3TEoEZZ{p_n-;R(qq&cNI`^`8U36*J#x~(;L;BYu1u}P$R&?eoo z;*65u%MVB}F*(Xz9*t%*71LjT3*pT_g^A4ciDFCEh3B=oE%0jo3W=btlUE#hqrjzdi8I&(&I%pXFV_PkMi~KOwCD$eWPmu zj%BoUOd6~UF2CvP3>;M0cYN$wr{=}m1oUOnh<70cVH=T&v|8)ai>4c0U0z36x%gg_ z_}KTFG`gX}aOHx`(X$s&`VWQhdOE>twlZM(mU6Q;5b(6LfLyCwqOP?2G$XsQ?-qio&^xO9#a7(B zD)!J9S0qp^!7;#DNX3TMS67kNd+~3_==XyRIScCZda>}_4Q&S7(Ek8_6C8X7Xz1m; zY|8+nXV?9WU+Y?~Nwgk)-&zUpu5{Q_`7KfqteNjmlu@ynF)tPMi+`z=;Gz(IX-GkQ=q!+~cQ%4*vjN zBF}f?X=k}HUPk>qPL4@LfM%x=8?M()Q7Kh6{k>1r^iAK!!Qu{5M)h3BY1tBV`f~in z>NV%wL#=sWO-#I%1+ch(lyhNAn7nBRzE=u|vR}7^@Eiz{5vQESmFK*^S<`<9H*`Dr1+MSJp=u2_Ldy3bTijb!LH2sVsm$6MGe;-N!$;j)AJrC&Gm@UxX8`+Rp8Lkaa)E zTR|;qk6-kavy~kgkGvF{kg7XB`tUJY)oVeRx-~suD z)vSR&a?pLPpl=U^PX7Q5tTo>AeD4cl+i=!k4l_)#Z?^-9;wGU?5rvx&z)L$auxrnF zRoT;i7#}Cs!S+8M2sP`@BBbCX2CQzO=DbT50G*;VFGycqcWe5Ibc8!!S3+7nQmd;= z&>md~`V-x=h5rEc#KD#GrTG{=87wa`+F7moJ&)lAGwpdh@Ce{iC%g{ue2uA!FPas8 z*Fr3Bgx zJ;e^+y}_mp0HE5GU##pB;eUg1{{XOl*J+cO{{Vx$1)>DX`g@UFxGHPHTMRQ*50vnWmiHkVscNmP z(UQ%xoTDdq1#Qw;W$6iB$jq(`gg*%J9vX*{Hjs&#{HsY`MNK8!$}+T!Yp}z);rkzm zU}@pqqnxqO(}N&av_@!dkU)X46Kt#OA7GhLph`X4%Fp^7EBp^k=fU-U$&#PY@?7DG zT(uSH?e)?c8)C@&PUVn2=VLEzW6Yf7em%I8I7a^f)j9M%Wl0xi2hjfN!VFXj4Ypi#0|^RMmPvBjp*@&%wZ$*)_r#_?4=iFJgKV z{W{oI$mRLO8dtgB@h0^;q<6j#;yc~WV^(%Y=2rN1G;?wu<-rQ#3Ms36KAIHI1mM0iII8@Wj~3VKf{cA$6J8$OR}5!JH)G2@u^69 zVg6#K8zHIl0u!n9Jj>NfVENR^qpRQH8aL23`oF1Of!lo^P{Xy9V=GsJ8j}KoL&@M7po9Z- zsCbV6)u;xL!I{`=6(xOYA7M@>gYJGIC}J-2;FaCg)_RaFWo@rt=NOxA9hXhM{bQEZ zy3bZM6OwB9eFqZ9*C&1Bl&$3U6Y`R{bo@X14jrtg8G5``Ls<1c++fgB^QVc+8D6Ju zQkC5WKfEw?T5s()^u8AuCK{Si%U^YKzX{lIFve)~jG~eW?%a!gEdi3>ztRnZ>yF;8 zI7y$R8SxGRy+x`JvJMzkQms>uJFJ(Y`)PY)>SKV+JsWOFMx0&2BVqq zvxyu_T;5SCMH*x7T|IlA1Bt*&Ev1iGD%4C9nJZwaw})XJRH_->-Gl>T$O|@Y04OSK zM{@oiL=A7BM1yYn%sn&;m*1XzfJLHIT@syw-KXlpsj2lp^KG`~RzB}P4aFBu64q~|8jgCjc4F_1q`uL6KU8Al%o;MOAg=lA~rQ9OF3lG@j{TzpLr`#Hl_g7v;BGYX_O1@+!(^E7#m${_kI{!Hn6rtBsC zOW{S)WEN#P0#k%?Rp3-HJcjb100zWjumVx3_8()h)uLS3x@Y9`Ff`(j^R@_haIvLDTAt4+BbIf>W~6vV9n)irlpkBr69A#g2;}`cFR| zScTK-%s&itaN_>}bHA=p88RcU0rPI@CDNx;D(2A+Mbcd?7<{4(Sj*fT%sp=K?eq%R z)hiW8)bKC50#mjZ@{t?UXAK~v&)@Oj98iftm6*1Lj@xnc9sBSnHg}J*_JP(X{L!mc zI|^4i9Ca{-$8b#wk64yU;BX{j=sWM5jMMg)yotRA}!y4 zmu$4(VXP15)+U;_zgOySG3}ZBf8;XJG*7e|SB|bQ#d{w_TBe|$j=$X}%dkb$7CR?l z@=ykhYMM(@9rvp)O_R@S(_d+W$J!oD=bSMo0p1MNmwFn6%{JTz)^c@!^IwR)ok(Pw z5a5RpAv5hsj@gvz3dFq2oamSn5MNor#wS3LxR}( zq|R&S-as@K)gBJmiDP<_%5w5ZvXn!QpYNx?1Kc#Zzw7`)$7!gQWY@{lj?MfZ?=T!R z=ev{ e6!l%*kkA^?IqBq0tqLoWTd{_Dh2H)>*97$L;tf^*UwP6rKLiEOmg@MCMV zPw1;YY`4v*8+Qgh*frit{2?@%V)n(9#>U5$Q4p)%9sLV^B?@p=LkB`CrPVx@E8&0R ztMOSqL}W+3!B@Nk7+G}qw z{b_^JVA;#am*D%H=VHv7RiP|=77c?m#Q0C+@*v3$38RS_S*)Xq%^S1+o+1gDNr_?) z7e463bTZ>W90zBeZ#Tky{U7-7Aqk4DIPbJdBCb`3gy3*s?{GpFti?3PX!D9t9wg}tyBovB>njD;4aUBs=B3-Ij_v;dy3J(o4!$C5l`2%IR)l5gEG@RPtFHlWl>0y5 zm;eAM4ZrZ#_ResSPQAS;a}(^`CWGyC8|6Gi?oet4b}fS@&D2a+BqSW}lHfG&h~)!uHGG4=N);14Is zEjg86WV&>B9t*D2Q*yrAa6pg(zrx*i;Cx2_VpOS65DZ@G{{WmFY;}J6l~fIPTd)2K z^KAbBjJ|UgQP>Q&MXKNO92WxGbwAscDpaXbrE@Lh8j|W2CgtjPfs)~#_pBno`Z2;g>gcXW4wsNm2vApy--PJXatg9)RLJ;CPm&i)X|yYcr1sve{kHPdItyMzD=1a6s$T}7ZO^+C@#=R?P87+&`gsm~P`+S0 z5qE70G?#6@1BS9ut!rDu{DEFr9@u!&BUovrApw;rclW~oaHP)@0F^#NDBo&o42ZXQ0-df4af1$Mtja;3XBS2?&J2%;@OEvX~q(9*cOUr~*{= z3z2nf#8norgyMMIgvh?XGQ;>CW$hg-KG75eO-I03d#Ru|n9N=vJwS8cvR}P+4{2aF z=^_@D>%hq9zQoB4g2n3>g09){{{S|o_xs=kmt)g+?)~9oYXgVGL79c%6p0d3aaivy zJ+fM-(f|Mh0C~(%wjTB6{{W4GH%@)&*?pm6F2OOc2l-^mTXScZAFe z0Kw&jv`RN;lG*&e3&77oxb%a7OofXmZaOkq>_PdBsbX4to&z|T&sjjWp@+s08nZz= zEP9;kMrP0+zw>~GV_P~t@ERAZ?>#0YMpbkoV;)LV9^8#{qWkN zBYmEa3)UaPy&p(`4U85;k@I9|`ERNJ0KOxcaqveb%)Jlx2DPd;YxMcm8su|@;hul_ zbitCfaKC8?z!w8-$LGWXEmK^0VbhBHdPH*xX7&D(h zPp8NB90byfp{}fPx0ki)E7bSj zXvfI#ZtBOj{{R;sDB zS92J@U43}2C0#vRK!G@-vqO;Rf4ljb#4#TP#yR(X2)?}5|exO5Cn*Xi)-EWaVLb^>}^@d-oeE|U2W8)!34&{S z&8(PmcpA-9q5vqBDpa)Cj`Od~(?|{gdeRO3W{E%^Fygg8^RqB5T?v)4xnH#WM}>)J z=EvuPrAn1DT_&WWqP2$-Iv&9?K5jeip4Bxn;61#m`A=vOfiXA-f#JcK&rHY4Nd5AC zFsv*+%pb2GUEeIO{2@O)F_x_LoV?ZUF32nV5q~el1lEMtn-R#`UN@1Y)FFz|5rg>N z=VxfVssnfyuTgHBg=x_6Q=D(fH25`LMyOcLLfSe>Lka!z7xVnRrLAqtk!vOUw!dB*dCL zLlYKQy%K=;7njw|k)6~2JXZdn`q^H_YHokI!KkvE+o9?Eofik#O31?xV427vm8ffd z6h>VTa?LVzeD`L9_2VLCpDVPc05ZjL4-rqihI!^?F`5fYnv;^xe-BiCO^o#{n#(rR z-gf0R)52Tes~Z|}seXoCI?CR?xz_T6HUA(6w&!^CYz3`wx4&;QVc?-Iy3fjT zkJ~Q1Ggx&KroVcW&OhYs=$x(;y$)=Bc9D?Oz^OCV zPrvz@CQ-B}ca=kDb>e;(-SHlFL%>7FtV7)v`4l4vOn3PipKHWvdVSr9i(Tgx7YWHG zVJ#`Do~y&@Qt67jE4Z^LNg7iQ%=Tq`Q9d4+vUtI9dY2)a}9a~%4(#p6-fAJ2hL=)<3+Kp>7p--Y5QdriCb^_sC`L4PDFMPK00a<^&91HH}Vl}SYbG>K8rVljWLJ;vg^eTEpQER<1Uk~&&PNv=lG9J{UB6K-1M1DW5TeO%jZ}U>4?2QkEpAZap%9!hV~kK%P#AC zvim*WYMbo?Q-AB#m4GX{_Vm(TKl{N0LV7$0CR+P*Ooz_zz}XXN^h#;wb$wy9xu33~MuCi$0^EM*LT-}_Ksq5o>h4^a+opnvd>m$BXlod)rq2X+IIY(XgMeqyA6*? zUuUNqQ>wAv86$^?vMW4a$;(e@^D6prbO_h5RC<5OcuEdV0ZGw~)#^DGpO=W>D?Fo# z;5=8c0ISs1$fQ0|)2v?FSm%IWWM>HO)utAXbk3 zK92C(0|QlE8Lsg%*tu+&1^W+A*Q|E+MtQY8AHNHdQDP-|dCpsIhCc($w`GRV2iY-HKK7(Cx$wSdXJ|dK0HfZ0B7%!lJX>&7LZK1JHtCBkRDQT2^v|Ue=ETSYe%o zu&eL%L+$mijqR!Sl;Sq}4EwXAVFTg}3I70;Y3nJCx{KFuNU@vB8CpY-cpW0<<{JM1 zExtS-^;84aGy21)NncZqh9oE0b^XSE#5nd2U{Z2kxcp+49$CPIAqaSDHT&s5I6wk} zND&~8BGS7qpX@Mr1hB0d0$z5d&AI!!`*<%kayhb~yXN?5vV=<*S-yGi;%4p`+=pgJ zD@m98XRS4BxwO@IGn_j?Ywba#YC{Kj=t&IafcLfx0a`aaNm~=i>R8T;=b3BFE3Gs= zWkaPycUCAKrF-l~wX|zH@q_QaF{kODF6o~q*ktgUdF82!-4Ns zuf$At!1+s}kdhvRfuw{+r5?DtWkJrnnFcK_pZK2N7~}d|Nlqk8c^m zR(V7?5QHJ&#PhELp3qHq7$CXOSB9(symZ^@`tRXzY{~Dp_iq(E;60D`iOzpx=$i7k z$Wpmm%kR+w*wGJWR+(LxgkM1g_L^GFB$Hm2dr!?yVSsFD?Wd=Tl%bFQPDKUmd!X-f z0jDwaL?p3Zz)XTeYeDdxfB{r;mu*9Dp3Vn=@L03cXc3^>Z;4qWPgT-aQ5l%^=i{NT zVbl6gUIay#ZZj*Ve%awWPXv8Ngh!^z4SP)}Wtc%_8C^BjT&}-hbb}kPWtF5k2a!~G z?R0+O#1Hl&<$UK~SXAzK{z?$^Wb#k?n90la4Sn_Tx@DeO2tp8tgDqEcc?VVoHBRM= z4@fp9irO-(Dm)DAhv*5E(_MLZGat-uV)Z;-R9^>$$xO6Tw*F`tFOy$#pSZxUe7@TJ zOq&ftdoWcyT=&}$O$vdn^r`w07W$uFI+klFfAV7tDt*6=Wd>~e6F=w7j%1HjrM?lA zo1+`<;&_S5ctb$~#Qd>YwuZxzz;zDh=w+FKEFI4N7gPW?NH(zhmx2l?G0toW(?QL4 zAXHmPZkD$1{)9 zZg}P2Zt>)YL*}JBtN~HaNGjhA74Js>0G|kW`ijT9$)jtYW3#Clb+7&d^2SGAy8M9+ z)%%?9Er8*Ab@85hqK-)a0O3l8p4KB-DOl9eGY>c=Rq`z__&k6zrdJcr zH6^szhN!N(F8=@@5`$?jEr6pV5Ont4ANea)y-UXQECo_(J_?wA>AbtvXHAtod_;oR zLJ})*c^U(!XIS|NwR{>j3i_D*I1`6G(YLOVkRZ}Kabu+gUu{foqs?|6nm*$arcea0 z1=@Q502dT#?Y^g1q`tA4hI4fz*V4(wyuDEk{Cj8wKGB<27)+?S7*K_ zO$Wp|J}Z~iZw{OKzN^A{3peCowNvT~O)Y%xpnf4)V>9DLg?VIQev2d<89bT%kB zA&S+oLJ@VNg5W5b^AuEc{8Ni6OY?KrE zd1dMIH6Wky2fSa2o5?*-*hT5CKC_x`IkR0sn}&#F)pvG2tH_ zmaHoF{imP2Tx!Qb{ln>ahhJAuAVc6TbUM0q#?#yjM{{o=>>Y)l?k{*vrm9m9BZ!A# zNi}d6TtAYaF*;r+|^qgF)38 z@#yGYguqe3<+%;wXKu^ScYe^2fM^~J#5B&XSg~8=kAY~5eh11P7|US?C!8~r#U?KN zJ-9jxD1>N1dgVl+I>q7MbTE#m)&&&+fFr2H?JqDdh=sRl{KBY?iY=WPKT8+2I|V;! zgVf~t#}Sb~KgyLpwiI}5JdD2dif3>`Mjngo&IS_0r>nu0^9mh#Ke~A&JQk#+SL5J6 z@V|J}Za+z;k;vr1^*2>@uc;Wp9L@X7AR2QHwFw~X==5R$*Mz$odirZ}3eQK>Fob>K4) z89OCI_dpG5`E=7)7cIfR$^F)210u%eQ_!8Hm{)mo=_p*!HDbk%#E#uqqwPm_9nvuu z25z2Iy8G)`no9<1#cm_f%L zVIz1FCf-Y?+xU3cx+Xa0_La7^I`oX4DC*zTb_%s-Ep5@(J#sA6J9CJE29eu|)J;Qv zZsZybZ51TD?;~Jp!nZ||J{W4ybI7BH~J%rA3Yp)~zgp6~l9ng0M-?oX5OEwj%bdpkkEfGzZIhgXjrK*N24NwnM@M5V4dNb>lsb7f;SIU=6Ic1% zkKD2dx-X9t7HE&z;cI<#on|rgeLNaq@Ht!qK7@2D@3G!yhdWz==G<{0jMu1(gLe*i zJ3NGLQ+URM({;)&nA2yGz=2vXWAS_s0%%TpcvSd`?SG+_qqdJz_3H@6d9JI^$4CDF z7N;N*k=&NR4r6Q7(m5(Bj+_m?^gWZM>xr33s0b-VJjjf@`4<*z_GJQ(G_l7(tj<5Xr`XTQ9kN~9Q?eQw&=o10#&x%aRW3XyqMxrg z2k*y!SCiPBdR&muFcQ<90~V78n_i`z4;8ResQ8-xp!^@XWIgCUJe8M34KJuVBLw58 zGE?bz+Bll{=_7J>N9r}oC00Wkf8bTtqvlkFqaK^^Fzq-aQVSbsy+csu;qE|%`to?zUIp5JV*(QHyk($Lt#CwVds4hVQ#xsbMCt~o?v4GVK=_SeVe2O;4&w+wrwB@s5T3!SW zZYCwaMk|~B6~1@o_ydO+e~wpGwLseIdXN#MDK#osEO={c9Kk`96~7>rsqW5mj*B@G z?w*wcTbDqTBp4iRZtSchC|!o!6Q|pMn*AecriQ#cKQL-te^l)&Kkt6U8AxFjOj7eG zwkNOMQG*gzMLxbMKYMj9(RfK+JSmE^8N<(PVb@chEev};Q`c`@UN=jeaF|#>sysZA zp?wYaJidi1g^1@aOJDq|?VdK2IrOd8T^`oMjGn5pJ~l~r^&m4ibwZ^K119qJny&Ei z&$euqvdC2n`jq4pDqN|3iN<9L6$TS!tk~tjpE&b$V3Y4d565+GRr7wy*s4$tTRiUq zi6icP`kN(3Xwc6E6E-m5(g%dgD&-(PB<=TPh$4kyRj>A^hryV`!Q!In5)Z*c{)OI% zIN@y*36QjwOTEorO@pwGxM~^^dpoIB4zt_6!m3o!C7U`568Cv~`s*gbN900g!O5Br zKYws<3D67r4N zP&Je18#}^hdk%orJC^j*Sjo75BZBL~ag)6n8)8+BY4(LwZxD8#8RT3&F#;oIIE=-_ zYcGSG4&6w-%l7Y;e@VVa>!Pwc`xx+O>iFQ=KDCBI5Q_&8v2&O zdmnBclHvP#RUXQ~<|R3(U?p5Q<2);c9hOLu=tgtaDp2rB|D`y#L1N_5M_~Qb<}Qw9 ztfWTWGLCYY^Gq{}<=oVND9mxCIv{zxN?b!g{nYiUzsl>(3h#Ytc2dMw&Cj7>!hMMb zf`g#ow~{6m5(zAJc~1V$+kXMf+(nn15|eJ0IH@vi>`cDD0g z5uBCx`l|A1Qn2~pCd|_oCqofq{3%gwIiz|nKxpQz&
    8@02mB?7;(#)(GITL${+9)a{_-5sO`?S3`0(~X}C?9KO1 z3FE7Ht4oQeFD?iz2yRAiIx#75@NM>={U1sr%ZT5axj%*9{{@T|_oNpIx~#p78A=L< zck5cOwz=)XhMrRHl2q>(>}YfHdZl$6%Kj1<64?+7Tu6am_|;bXdQ zgKRpCV^H&t+q22sr@_|v z*O3QUi=7hsRukY;yuh>b9%jdo}4@ zE!MM!s%&R;8DalAWeDpV_SD$5oB2Kwg^-|K6s(_llN~x;R^9BYwBq*US=)i314Z%MgR4o z5-yJicRHy2i*BQmlt;8m+{mC)H_H4XF=uvk**x)Hi^M&cKi-a0s=BgPElW43O>4b) zF?>e?y)-cqJ(taMqj{^ix03A)oL#O=fr!F9rBdYazJ-Ab)7Y>bmk~S4O|wN~@A}2h z2hM<}XLhFkS@;61GOut1Hct8~DcF*ceLQTrV?kbXWTJ~`B-U69plYQ^(Gt%an3lK? zT1}=+`9NKyMr9ry$8WEn5TKAky>-M}7B^8e-FK^1ycY6n!0FFpcww}!bs5?(sTdQc z_pB8$rPr3pa}8GI7J~7Ae&Fpo!C3-y$3?ws-lE&c4*^ zo1|2k@WDUBDB%tLi@iVIQ28I{ye9v+Fztb6C>V{FiOoLr?1ph38(gTLGY0nr4o52T z&n$g!uD%U~)bLSR?7&;zs_PDVq5mkUo%suhW~ykI2TxvdWh&Q`i(fC&hkex6%_Ato zm?$XzJZl@lBI3I4H0)_+%?^&Z8p+-wK zaSiOX>~mL}ymI#7djwXvpF?1V4@vSzl&_bf-o>5Uqz|0bBx(eS=$&YoN}E60N;0D( z6<|WR0+^WSHdjB1@MqF2n`k24K;UZN-vqO*Bl9BFxuTzBDQ4+N6jg>7vIgXo$`98L zYCn|15ESih?nQXqtkp0MqNk7M4t!VBpHJ4|=TxKc5K_`(P;7B63|}716B2mk@d_Us z4hFj3++2zHt;Iu1=Ju4rypB5 zrzuHapWAhSTJF1m*UoU06AWqtogyBZ9003{pf)o zt(er@Z3sI0MI6?jB&kzmKs80)L%1lXg7@!BZJVdPk52;7Jost~o_el|71#3sF^7m? zBSsI2kn_2b0F;N|yw>3{9T$(S!y0;$9@{LXt|zOClhf zSs2TNkN3TC>YBgV#VhqIilv%we(Yk~wF*OKQrbyI?^b(G%3nZS8Ogxb?k)XwAiJe) zl-$5b*KujQja?&_zEcMM9VVgaWjIoN;8N3m>-As2xcv>z$g^KRwcMBP$)DAU;XPO9 zKmP)_j>Mlm1>;GalP&}AAqb6T-`jFu4Pa`8WqfP1jNzD}ppyUiK**`8#z{SdNr+k% zLzHZBL3>htntm8$WsNISX(H>PFfP?4ZZR-?9K4m!ihIp_)ol=JFV*=(8%Utcb9DBg zC$OA8fuS8bzrO;*>>Qza@3)QD`%SNs3E6(YoTX#5ChLsE$ePDI2ETi^o2}c8v1V92q~t#TasE!U?6h*-SOZErdT1q!!*#B4Qb%@C4|}0U zK0`p3*`Ms9(u|0mJCtbf_2^$fBzw&W(vZPZNXsq~+%Bjdxp^ghI2(a!VOdzQ1Q@(& z)q|mH^FW{i9>--gF{$>4M%rBcSj%G`DgbG;UAl*vMA0`ZKt_-GRYrVhP^qLFJ`@Cf zT8WEFXfQ_7`C`Q)wyPHioebs3Y#Uh-CzU(W3J;b0M1SJ7H9V{AXygh+nV`34TY|AQ5F1N9po;@sM<3_1ZUAFa(C*8s)7P?Xm>W>aiv| zMkCrjj&Wc``#OF6=F~2=4QSyHrV0ezombSmxwA&h`1@UK@=<6WphP!txHcr8-&A<+IX^@ zx4ahLy6}QY!d30v5NsxbSr5gk?0y6=#8@Zhx7zLR!D0e*;dt`&Uem@$dR`_)GIg9i zj_#8HKCFXYJVzj~ue1H9#c@{B=kalh*Q7MfpEVwr+#KpF54hE#!=got!%gABbH%lq z@3=3M*lk{ZSOs;ciBZ$@^}cqvC8hG+=jv7RwNm5a%`*&#C3lUE-6e{%ii!JgIiiY( zy-G2yzTYFYiAn^SHX~6hy5P;HF9>+w1#QGwSjRDr88zMW zA8vo*6NoZBAbu+yzNmc#1M_VXf`s)L&ubONjgcT!G4WH9%r`!IO-l6ePe5 zI8Cd;%!_JVr7=xTJv$HhDo zYwaHkK(wr2b1sl%OuHXr3)sgRYfWtAPabT_|s#fve6HFMk2Kv}H(@QpAWzJd~4#e2(SgF{rq~n^;&%{#NcnwIF=C ze8XV=5#kFVN?2-?9)jhLl&kBDUmfX%Dya}IY#;DVrSX{t8u2%DB@^R-{m)_mrVY7; zP;-?4n}#R4c=I!tT4rU3Y{1>S=OyIadF_^!&_g2Rs7yNJdcvKFZu z%_h5Q8jS6k9*Y_pW81cNim$&j_IMd5=l=W!*vE+7vjy2#dEz+5f1cJrtQP1?q z(9gc&r$d$6?{udIm9Iv{l}dcP+rE8>Ae0%A=7{KGDShbtar-8WLX&e0GFOb>-+V0C zER^|YM7Qz<`@lI~aTu}!-zu_MMIZPn z$PX}HW_kuGp!D2B(+jU9f?W5K)ZI=7dn&NO>8Op02~J-`TM29sW}Y3gbflXj>y%cF zM6KNlu?*7+7F?d^Zw1cp0T8MSg0l1i6cV6{e ztW}8aI?e*;1Sf{DW&>+Tx_3zl%8|C~`uN91-~KAXPYSk=PS|oLq3f?)%}7@UN{u88 zxUuUa`Js1Y-8yHSjgp`7M{2EI}Q6q&*$LnX;h};p6@8xtnp~U9$*14m8CQ39-=mC;!4ZaSL?ly@tuZ+ z&)Ln-E~+@mAv!V2x^3$j@Ak*|E(m8fWI{HA zH|DVLp0N3GgaT;hyY^CBm1dB972rkJ&~gyJv(Q)NZHOS78tG7?GDYR&>%t-E=h=T& zO7jW(CBCrVEfz5TIF*y_KOFuS5Eg$~Tr|~a{=El2`VRNQXk#aqP}4Qy-FL|^$vYh4 zrtO=&)p;ORwN5z))A7-;=y+@X_Xt#@*2mZzY)~Q$Suy<=SGpb;$!P+C!peE9BsN}D7r6}*BlVw%#dBYamr60WYdL8(( zutu&RZx8o_|1JYC5#RvuuuxI}02cvyg=oFASp6V`Eo(;>hGWg&4aH5~DY8NPHrMk( zbei4$GbwZrMo_TukK$f3wk*HOSQV!taam$Fr1htv;b%i^<_QD z8zlDPLOL)l@NypSmh$Jq(K?$m06LhNQ3SmyixMgxS5sxwSn`O{;Xz7vysJWJ(vY&d z(5VNDdH3$RZg_be8YBmIKb`-3KWrWI00+Inz?PK|zQ&n&mqk~Cv?DLXLeJG_>u+PW9F;%=9C|HDhBbr6+8}6Q_Pg>sCQmxBg zCnDGC>dDh{;ANO~z*TV;+$cvJ_(4+m-Wf z)X)C?DgiLrpGyLMyisT zbK@fKU}3j^0S^EtfE0d(28E>q0?r~Lwa^?Gcm;WePKxi(6>4}SF9+HQ@qL~}q};Q= zyILiO1cU5Xs_!%Q^|ix+!4~FJMTmqShCWHNWa=zAg@1DlfNg}v4mua*+!SXL6(h=s z{xri4bj!0{r@B5lI5+K)VXz<@C67BA^!)R{nuiTX3q!*kjImzVHoCntn0^FPm<$i* zxurl<;^=hpzXUNsmSTNvN%(+;#9pSZ3+Gy{s@X1Nf@*VcqF);Usm!A$T z7@we`B3bkRD1GqBvT@VD=1_nZZ1YnJKZa+R4!kXZl`}~paY$VW*cSvl;_># z

    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/v9jQX!l=}lr|=}-hugqRf|bSD8&fD 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

    {e9>*03wERwsD2hs(^p05QPjjZqe^ z^F1k-VC7>wMK5n0etLmp(07Ujqc`KsNW_Ud+ddH~gQYDnbV!lKT2ba-Qmwin*+B0K ze({(4C6WH6s9o4R8+1tGF)A^MZxG^@qLQ@in%?WK!5H;5VzyKzgWhN6uzIcuVpT<+ z$}u)M=B!2?@#Ow;4QL;zoQnW=fP%3^W$hRE{{S(hxww>@8U>dvf|SDTHO%fO;Wol=o#Cp|myA1G#jJ zkyC2!^}pdd3>YxzTx(i%C(|(P1AH)4OYs3(8@KV14gIka?zFY%Q!R)Fg{yw=W7~r@ zg#J~!cV1xkD+CPuA4mleS_y){iPl`^FvHTlOWB5t@P?{85qOAe1E`VJS%Y`w_!%8o`?kbW8My`ZxsHe%a<-(S#f%?{u;X1-ZcBeYOJkuU&-w; zs;#O-ei+&t?CvFg_Il`&n<%v1&xw!HfA(Y6Rja&CVqmm44(|Oz%MPOSP^?|8Q6fZ+ zsdazZDYgYSHh*6o2Sy;M1x9fcRe$G0`((s{4-RfhcflJ35gYQcVGG>^#7& zU9;Jm*H<&)l;JvP4+OMF4gnBSp>vq{=w+#fuVqvHN^==myML;SS-2YaMd%bn=F<`} z?-0fa)d_5;SQ73d&;mql8MMD?+Kg>GGaaDfZ)(@z{LC{gt;NkZ67C|!Q>R1H&RBMf zAHn@6@9Msz^p8t}E)1)}1N(b#Djpi7;>_P;ROSE|^N+kmuu&QZ{zV13!ejFdIit$( z4cx~zyXCt4X$0I~w8L`i^av0pSI6%%m|%lBl^Tg2pfxCZ0OZdyk|uUwN}_DKbrZPK zOb9O&ykGCLXq5z>!T~3j7XgmqC|-jBtg^*FTBEyW5>{v%m=|tJcytfvKbBu?;#H zW=2@6y&Z?dR8@7m{ZI%QruZp~05UXv$Iu1`sE<@PFG1anAZ;hJ9Us|4f&C?4BhaCY z22Y8v+&mJLYJq>aU>G$p{wA#kZG9Ue&7{0Ox}gHW8ika!34*!pHkItS7n8Xzn2iD= z3>_e{p<-M@EK{6gSBGuJX^MiO&}tAf3|cd2F4EbkM?r4idHh$z(j7`56U$fIxeTD? zuu=GnYhs=T58%Fo^xsxAWdUpF(+ApVd8R2Sm)N#dJ)IV4LHm2TdPhyiOPA51J30SDjX_RfV*}7;!lV?lc-yR|7Yek93%lcUV03C~R z&rV3-+baW7FL3W=q}4SJImQ0q&?yU4&_no~Gh5D~x6C$j_Js?Wks}gYPbV>~+`iK1 z$!{%cGoJY zsty3rRLaJSz9R`__%3C_aq9m2K;o6G2QF7t-Kx$R`3+9wY&s^QU(6$cIuSD+^|l~7 zKCw*{k4=A4vL;N~uJtbN#)h#S#CnOQ5ZGg8zYs2;iB*rpKLQ0E52Z*l$Vlxt6gJxu zh|!uBt-<{$Rbbb-IQ}c>KTA+*Kp)`vx3nnQ%Y0vvqqKZЊz{f~P?@(Pn>+Tt3 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⪚+kOv9^AbDhvk~MO_<>Ffe<XQvY@2UEdp& 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 A#{d8T diff --git a/services/next/public/images/default-thumbnail.webp b/services/next/public/images/default-thumbnail.webp deleted file mode 100644 index d468e1550ec6e84e3e761c3e5119bff94154a26e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 27268 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 diff --git a/services/next/public/images/projekt-melody.jpg b/services/next/public/images/projekt-melody.jpg deleted file mode 100644 index 41916fa3768fd0529734e64fb57f8e5ab153ebef..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 134526 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%0A)eECV+`4H>9@P74TTE)R1~TqPV- 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

    1=I6< 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 P@G>1CM&R)M-`f8HJgPzT diff --git a/services/next/public/images/projektmelody-thumbnail.webp b/services/next/public/images/projektmelody-thumbnail.webp deleted file mode 100644 index dfb8073fb7905d5cd3059b66a46c0e0e0a40d428..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1356 zcmV-S1+)56Nk&FQ1pok7MM6+kP&gns1pojr6#$(9DnI~106uLjlSU*Wp`k8#>ga$C ziD&?wGQEFt>=Nqj=+$8$FZ9p6>fxFwl07pL8V zr&lbSZ#7=~ZiZJ|55`TYh_fMoungy(GqZO>;=~qdcISl&;^r_(!Q#`JS8Zr$Q~>__ zpC=52Sx6W;ePns+Dq>aZI7vS-W=jyHm6F>;5lS%}@z-p*5QRF^4Z208cT_(?H^f6c z51H~Qo$edi2A4Yflb_hq0sVx}{*>*n{vQ~EPYgYnZFn_6{0BS)ms1U;KQC<gA2Um(TKH6+ zQF84>a6TXkAT@4EXR^JEmE}+_qgd>=A4hs@h}eS#IA6JO1}c*{fKY ztYb!KY#PLo9)&NE*us)dg$5OrHfcouhsm37ZRYy0_&xE?f0 zSq6mVP~ay_?3?$^!M%T!Iwg1GO&@l1DL}6u=ikY&!eD~u807=>RDdmV@Ls2wT&NCE zMk>v*k}IoX-;^>UD!_ye9i#3*}^JEx# zttRC}DaDF%?ex6UFp1TFa`MJTIF{Hee;nKASad~Ul%8a|U0om#&GxQ^Jo=(4_ddmA8gr7#46n$1qKJ%|?$lmC=vA1qc+y8*Y!{Kh&5FF0xB(f( z3pK;dHi+I4`qFTQhy(^?l&M_9YUQeoI1^-MNIFz^m92uDU7*pIX~3N;x2VVL5`6>G zhv?d5g3R#4QBmeS8X9>|?L~Gz=(ch_+}aMVIb_whqAg|K;-dlHI=1)=sZB;0nigew z#yKJXtakAEjl_lXC$sGm^uFmz0BnyDH3FUj7%KpnslIN!5s*2zw;B9$4IiI|hRn!1 zm>2@TKiMr0F%aBtr`xTcY6A*D%Z6fMPsb zJ@~n*a>8DO-6=;F@8T^%8?-C7m}u$0pqaF{^6e`#rH0#FWqkesb3bbJM{~T&$U?j# z1#t=N29zh}cJLAF7&1dE*@cn%j@p@^!#FbAslzzwmBz>H~!TYgoiB9)-;HI`jpu_N~}oT0%=W6wkd`>JXjel z$wVOxc6j*DkC+mYSI;zKf9GGA2r`by>bp-xI%+@Azwz7u?Ij&#?2L;LdIF8@fFokZ Oop8Dpc{wbwDu4iP1)B!| diff --git a/services/next/public/images/vercel.svg b/services/next/public/images/vercel.svg deleted file mode 100644 index 08d0cd1..0000000 --- a/services/next/public/images/vercel.svg +++ /dev/null @@ -1,3 +0,0 @@ - - - \ No newline at end of file diff --git a/services/next/test-jwe-decrypt.js b/services/next/test-jwe-decrypt.js deleted file mode 100644 index 58d8ce8..0000000 --- a/services/next/test-jwe-decrypt.js +++ /dev/null @@ -1,18 +0,0 @@ -// const { jwtDecrypt } = require('jose'); -// const dotenv = require('dotenv') -// dotenv.config({ -// path: '../../.env.development' -// }) - -// async function decryptJWE(jwe, secret) { -// const key = new TextEncoder().encode(secret); // Encode the secret to Uint8Array -// const { payload, protectedHeader } = await jwtDecrypt(jwe, key); -// console.log('Decrypted payload:', new TextDecoder().decode(payload)); -// console.log('Header:', protectedHeader); -// } - -// const jwe = '375hO4PPr4GCghunRf52r7tlG7HAFLanZ7D5blc0QDamW3nXp_terShYROf7bpPak8N1k29cF_e_hdDZNeI42ZaFQInhjCOBLrYDpcxbvjALNhkwFNPz_wj1IPqDehjMz1vowdN9LdrwJup9a8Q.HbClJBjeMIyzT3vrwRdM0Q'; - - -// const secret = process.env.NEXTAUTH_SECRET; -// decryptJWE(jwe, secret).catch(console.error); diff --git a/services/next/tsconfig.json b/services/next/tsconfig.json deleted file mode 100644 index f991314..0000000 --- a/services/next/tsconfig.json +++ /dev/null @@ -1,41 +0,0 @@ -{ - "compilerOptions": { - "target": "ES2015", - "lib": [ - "dom", - "dom.iterable", - "esnext" - ], - "allowJs": true, - "skipLibCheck": true, - "strict": true, - "noEmit": true, - "esModuleInterop": true, - "module": "esnext", - "moduleResolution": "bundler", - "resolveJsonModule": true, - "isolatedModules": true, - "jsx": "preserve", - "incremental": true, - "plugins": [ - { - "name": "next" - } - ], - "paths": { - "@/*": [ - "./*" - ] - } - }, - "include": [ - "next-env.d.ts", - "**/*.ts", - "**/*.tsx", - ".next/types/**/*.ts", - "app/components/auth.tsx.old" - ], - "exclude": [ - "node_modules" - ] -} \ No newline at end of file diff --git a/services/next/tsconfig.json.old b/services/next/tsconfig.json.old deleted file mode 100644 index 4b44d95..0000000 --- a/services/next/tsconfig.json.old +++ /dev/null @@ -1,51 +0,0 @@ -{ - "compilerOptions": { - "baseUrl": ".", - "paths": { - "@/components/*": [ - "app/components/*" - ], - "@/lib/*": [ - "app/lib/*" - ], - "@/assets/*": [ - "assets/*" - ] - }, - "target": "es5", - "lib": [ - "dom", - "dom.iterable", - "esnext" - ], - "allowJs": true, - "skipLibCheck": true, - "strict": true, - "forceConsistentCasingInFileNames": true, - "noEmit": true, - "esModuleInterop": true, - "module": "esnext", - "moduleResolution": "node", - "resolveJsonModule": true, - "isolatedModules": true, - "jsx": "preserve", - "incremental": true, - "plugins": [ - { - "name": "next" - } - ] - }, - "include": [ - "next-env.d.ts", - "**/*.ts", - "**/*.tsx", - "dist/types/**/*.ts", - ".next/types/**/*.ts", - "./app", - "./assets" - ], - "exclude": [ - "node_modules" - ] -} \ No newline at end of file diff --git a/services/scout/.dockerignore b/services/scout/.dockerignore deleted file mode 100644 index f46ca59..0000000 --- a/services/scout/.dockerignore +++ /dev/null @@ -1,4 +0,0 @@ -# STOP! This .dockerignore is probably not the .dockerignore you are looking for. -# The dockerignore in the ROOT of the Docker context is the .dockerignore that docker uses. -# We are using a monorepo and the docker build context is the root of this git repo. -# thus, see ../../.dockerignore diff --git a/services/scout/.gitignore b/services/scout/.gitignore deleted file mode 100644 index 25b08c6..0000000 --- a/services/scout/.gitignore +++ /dev/null @@ -1,146 +0,0 @@ -lib/ - -# Created by https://www.toptal.com/developers/gitignore/api/node -# Edit at https://www.toptal.com/developers/gitignore?templates=node - -### Node ### -# Logs -logs -*.log -npm-debug.log* -yarn-debug.log* -yarn-error.log* -lerna-debug.log* -.pnpm-debug.log* - -# Diagnostic reports (https://nodejs.org/api/report.html) -report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json - -# Runtime data -pids -*.pid -*.seed -*.pid.lock - -# Directory for instrumented libs generated by jscoverage/JSCover -lib-cov - -# Coverage directory used by tools like istanbul -coverage -*.lcov - -# nyc test coverage -.nyc_output - -# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) -.grunt - -# Bower dependency directory (https://bower.io/) -bower_components - -# node-waf configuration -.lock-wscript - -# Compiled binary addons (https://nodejs.org/api/addons.html) -build/Release - -# Dependency directories -node_modules/ -jspm_packages/ - -# Snowpack dependency directory (https://snowpack.dev/) -web_modules/ - -# TypeScript cache -*.tsbuildinfo - -# Optional npm cache directory -.npm - -# Optional eslint cache -.eslintcache - -# Optional stylelint cache -.stylelintcache - -# Microbundle cache -.rpt2_cache/ -.rts2_cache_cjs/ -.rts2_cache_es/ -.rts2_cache_umd/ - -# Optional REPL history -.node_repl_history - -# Output of 'npm pack' -*.tgz - -# Yarn Integrity file -.yarn-integrity - -# dotenv environment variable files -.env -.env.development.local -.env.test.local -.env.production.local -.env.local - -# parcel-bundler cache (https://parceljs.org/) -.cache -.parcel-cache - -# Next.js build output -.next -out - -# Nuxt.js build / generate output -.nuxt -dist - -# Gatsby files -.cache/ -# Comment in the public line in if your project uses Gatsby and not Next.js -# https://nextjs.org/blog/next-9-1#public-directory-support -# public - -# vuepress build output -.vuepress/dist - -# vuepress v2.x temp and cache directory -.temp - -# Docusaurus cache and generated files -.docusaurus - -# Serverless directories -.serverless/ - -# FuseBox cache -.fusebox/ - -# DynamoDB Local files -.dynamodb/ - -# TernJS port file -.tern-port - -# Stores VSCode versions used for testing VSCode extensions -.vscode-test - -# yarn v2 -.yarn/cache -.yarn/unplugged -.yarn/build-state.yml -.yarn/install-state.gz -.pnp.* - -### Node Patch ### -# Serverless Webpack directories -.webpack/ - -# Optional stylelint cache - -# SvelteKit build / generate output -.svelte-kit - -# End of https://www.toptal.com/developers/gitignore/api/node diff --git a/services/scout/.mocharc.json b/services/scout/.mocharc.json deleted file mode 100644 index 7771cc1..0000000 --- a/services/scout/.mocharc.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "$schema": "https://json.schemastore.org/mocharc.json", - "extension": ["ts"], - "require": "tsx", - "spec": "src/**/*.spec.ts" -} diff --git a/services/scout/README.md b/services/scout/README.md deleted file mode 100644 index 7bf8e7f..0000000 --- a/services/scout/README.md +++ /dev/null @@ -1,47 +0,0 @@ -# scout - -Vtuber data acquisition. Anything having to do with external WWW data acquisition goes in this module. - -## Features - - * [x] Ingests going live notification e-mails - * [ ] Sends `start_recording` signals to @futureporn/capture - * [x] Fetches vtuber data from platform - * [x] image - * [x] themeColor - * [x] displayName - * [x] Platform Support - * [x] fansly - * [x] chaturbate - - -## Design requirements - - * [ ] get watched channels list from Strapi - * [ ] every 3 mins, watch/unwatch based on channels list - * [ ] watch important sources for go-live notifications - * [ ] CB tab - * [ ] email inbox - * [ ] alerts realtime server when watched room goes live - * [ ] logs chat messages - * [ ] throws errors when unable to connect - * [ ] runs browser headless - * [ ] runs in the cloud - * [x] runs in k8s cluster - - - -## Puppeteer - -For when we get to the point where we need it, here are the packages we used with success during past testing. - - "puppeteer": "^22.7.1", - "puppeteer-extra": "^3.3.6", - "puppeteer-extra-plugin-repl": "^2.3.3", - "puppeteer-extra-plugin-stealth": "^2.11.2" - - -## Adding API routes - -* Routes are specified in ./src/api.yml -* Route handlers are specified in ./src/app.js \ No newline at end of file diff --git a/services/scout/node.d.ts b/services/scout/node.d.ts deleted file mode 100644 index a9bed81..0000000 --- a/services/scout/node.d.ts +++ /dev/null @@ -1,4 +0,0 @@ -interface ImportMeta { - dirname: string; - url: string; -} \ No newline at end of file diff --git a/services/scout/package.json b/services/scout/package.json deleted file mode 100644 index e631ccf..0000000 --- a/services/scout/package.json +++ /dev/null @@ -1,85 +0,0 @@ -{ - "name": "@futureporn/scout", - "type": "module", - "version": "4.0.1", - "description": "vtuber data acquisition", - "exports": { - "./*.ts": "./src/*.ts" - }, - "scripts": { - "build": "tsup", - "test": "pnpm run test.unit && pnpm run test.integration", - "test.unit": "mocha --require ts-node/register src/**/*.spec.ts -g unit", - "test.integration": "mocha --require ts-node/register src/**/*.spec.ts -g integration", - "dev": "nodemon --ext js,ts,json,yaml --exec \"node --import tsx --disable-warning=ExperimentalWarning ./src/index.ts\"", - "start": "node ./dist/index.cjs", - "clean": "rm -rf dist", - "superclean": "rm -rf node_modules && rm -rf pnpm-lock.yaml && rm -rf dist" - }, - "keywords": [], - "author": "@CJ_Clippy", - "license": "Unlicense", - "dependencies": { - "@aws-sdk/client-s3": "^3.637.0", - "@aws-sdk/lib-storage": "^3.637.0", - "@aws-sdk/s3-request-presigner": "^3.637.0", - "@fastify/static": "^7.0.4", - "@fastify/swagger": "^8.15.0", - "@fastify/swagger-ui": "^4.1.0", - "@futureporn/types": "workspace:*", - "@paralleldrive/cuid2": "^2.2.2", - "@temporalio/client": "^1.11.1", - "@temporalio/worker": "^1.11.1", - "@temporalio/workflow": "^1.11.1", - "@tsconfig/node20": "^20.1.4", - "@types/imapflow": "^1.0.19", - "@types/node": "^22.5.2", - "@types/pg": "^8.11.8", - "cheerio": "1.0.0", - "concurrently": "^8.2.2", - "date-fns": "^3.6.0", - "dotenv": "^16.4.5", - "fastify": "^4.28.1", - "fastq": "^1.17.1", - "faye": "^1.4.0", - "htmlparser2": "^9.1.0", - "imapflow": "^1.0.164", - "js-yaml": "^4.1.0", - "limiter": "2.1.0", - "mailparser": "^3.7.1", - "node-fetch": "^3.3.2", - "openapi-backend": "^5.10.6", - "p-retry": "^6.2.0", - "pg": "8.12.0", - "prevvy": "^8.0.1", - "qs": "^6.13.0", - "rate-limiter-flexible": "^5.0.3", - "sharp": "^0.33.5", - "slugify": "^1.6.6", - "swagger-editor-dist": "^4.13.1", - "swagger-ui-dist": "^5.17.14", - "ts-json-schema-generator": "^2.3.0", - "tsx": "^4.19.0", - "typescript-json-schema": "^0.65.1", - "xpath": "^0.0.34" - }, - "packageManager": "pnpm@9.6.0", - "devDependencies": { - "@babel/preset-env": "^7.25.4", - "@babel/preset-typescript": "^7.24.7", - "@futureporn/utils": "workspace:^", - "@types/chai": "^4.3.19", - "@types/cheerio": "^0.22.35", - "@types/mailparser": "^3.4.4", - "@types/mocha": "^10.0.7", - "@types/sinon": "^17.0.3", - "chai": "^5.1.1", - "esmock": "^2.6.7", - "mocha": "^10.7.3", - "nodemon": "^3.1.4", - "sinon": "^18.0.0", - "ts-node": "^10.9.2", - "tsup": "^8.2.4", - "typescript": "^5.5.4" - } -} diff --git a/services/scout/pnpm-lock.yaml b/services/scout/pnpm-lock.yaml deleted file mode 100644 index 0efa8d4..0000000 --- a/services/scout/pnpm-lock.yaml +++ /dev/null @@ -1,8278 +0,0 @@ -lockfileVersion: '9.0' - -settings: - autoInstallPeers: true - excludeLinksFromLockfile: false - -importers: - - .: - dependencies: - '@aws-sdk/client-s3': - specifier: ^3.637.0 - version: 3.637.0 - '@aws-sdk/lib-storage': - specifier: ^3.637.0 - version: 3.637.0(@aws-sdk/client-s3@3.637.0) - '@aws-sdk/s3-request-presigner': - specifier: ^3.637.0 - version: 3.637.0 - '@fastify/static': - specifier: ^7.0.4 - version: 7.0.4 - '@fastify/swagger': - specifier: ^8.15.0 - version: 8.15.0 - '@fastify/swagger-ui': - specifier: ^4.1.0 - version: 4.1.0 - '@futureporn/types': - specifier: workspace:* - version: link:../../packages/types - '@paralleldrive/cuid2': - specifier: ^2.2.2 - version: 2.2.2 - '@temporalio/client': - specifier: ^1.11.1 - version: 1.11.1 - '@temporalio/worker': - specifier: ^1.11.1 - version: 1.11.1(esbuild@0.23.1) - '@temporalio/workflow': - specifier: ^1.11.1 - version: 1.11.1 - '@tsconfig/node20': - specifier: ^20.1.4 - version: 20.1.4 - '@types/imapflow': - specifier: ^1.0.19 - version: 1.0.19 - '@types/node': - specifier: ^22.5.2 - version: 22.5.2 - '@types/pg': - specifier: ^8.11.8 - version: 8.11.8 - cheerio: - specifier: 1.0.0 - version: 1.0.0 - concurrently: - specifier: ^8.2.2 - version: 8.2.2 - date-fns: - specifier: ^3.6.0 - version: 3.6.0 - dotenv: - specifier: ^16.4.5 - version: 16.4.5 - fastify: - specifier: ^4.28.1 - version: 4.28.1 - fastq: - specifier: ^1.17.1 - version: 1.17.1 - faye: - specifier: ^1.4.0 - version: 1.4.0 - htmlparser2: - specifier: ^9.1.0 - version: 9.1.0 - imapflow: - specifier: ^1.0.164 - version: 1.0.164 - js-yaml: - specifier: ^4.1.0 - version: 4.1.0 - limiter: - specifier: 2.1.0 - version: 2.1.0 - mailparser: - specifier: ^3.7.1 - version: 3.7.1 - node-fetch: - specifier: ^3.3.2 - version: 3.3.2 - openapi-backend: - specifier: ^5.10.6 - version: 5.10.6 - p-retry: - specifier: ^6.2.0 - version: 6.2.0 - pg: - specifier: 8.12.0 - version: 8.12.0 - prevvy: - specifier: ^8.0.1 - version: 8.0.1 - qs: - specifier: ^6.13.0 - version: 6.13.0 - rate-limiter-flexible: - specifier: ^5.0.3 - version: 5.0.3 - sharp: - specifier: ^0.33.5 - version: 0.33.5 - slugify: - specifier: ^1.6.6 - version: 1.6.6 - swagger-editor-dist: - specifier: ^4.13.1 - version: 4.13.1 - swagger-ui-dist: - specifier: ^5.17.14 - version: 5.17.14 - ts-json-schema-generator: - specifier: ^2.3.0 - version: 2.3.0 - tsx: - specifier: ^4.19.0 - version: 4.19.0 - typescript-json-schema: - specifier: ^0.65.1 - version: 0.65.1(@swc/core@1.7.23) - xpath: - specifier: ^0.0.34 - version: 0.0.34 - devDependencies: - '@babel/preset-env': - specifier: ^7.25.4 - version: 7.25.4(@babel/core@7.25.2) - '@babel/preset-typescript': - specifier: ^7.24.7 - version: 7.24.7(@babel/core@7.25.2) - '@futureporn/utils': - specifier: workspace:^ - version: link:../../packages/utils - '@types/chai': - specifier: ^4.3.19 - version: 4.3.19 - '@types/cheerio': - specifier: ^0.22.35 - version: 0.22.35 - '@types/mailparser': - specifier: ^3.4.4 - version: 3.4.4 - '@types/mocha': - specifier: ^10.0.7 - version: 10.0.7 - '@types/sinon': - specifier: ^17.0.3 - version: 17.0.3 - chai: - specifier: ^5.1.1 - version: 5.1.1 - esmock: - specifier: ^2.6.7 - version: 2.6.7 - mocha: - specifier: ^10.7.3 - version: 10.7.3 - nodemon: - specifier: ^3.1.4 - version: 3.1.4 - sinon: - specifier: ^18.0.0 - version: 18.0.0 - ts-node: - specifier: ^10.9.2 - version: 10.9.2(@swc/core@1.7.23)(@types/node@22.5.2)(typescript@5.5.4) - tsup: - specifier: ^8.2.4 - version: 8.2.4(@swc/core@1.7.23)(tsx@4.19.0)(typescript@5.5.4)(yaml@2.5.0) - typescript: - specifier: ^5.5.4 - version: 5.5.4 - -packages: - - '@ampproject/remapping@2.3.0': - resolution: {integrity: sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==} - engines: {node: '>=6.0.0'} - - '@apidevtools/json-schema-ref-parser@11.7.0': - resolution: {integrity: sha512-pRrmXMCwnmrkS3MLgAIW5dXRzeTv6GLjkjb4HmxNnvAKXN1Nfzp4KmGADBQvlVUcqi+a5D+hfGDLLnd5NnYxog==} - engines: {node: '>= 16'} - - '@aws-crypto/crc32@5.2.0': - resolution: {integrity: sha512-nLbCWqQNgUiwwtFsen1AdzAtvuLRsQS8rYgMuxCrdKf9kOssamGLuPwyTY9wyYblNr9+1XM8v6zoDTPPSIeANg==} - engines: {node: '>=16.0.0'} - - '@aws-crypto/crc32c@5.2.0': - resolution: {integrity: sha512-+iWb8qaHLYKrNvGRbiYRHSdKRWhto5XlZUEBwDjYNf+ly5SVYG6zEoYIdxvf5R3zyeP16w4PLBn3rH1xc74Rag==} - - '@aws-crypto/sha1-browser@5.2.0': - resolution: {integrity: sha512-OH6lveCFfcDjX4dbAvCFSYUjJZjDr/3XJ3xHtjn3Oj5b9RjojQo8npoLeA/bNwkOkrSQ0wgrHzXk4tDRxGKJeg==} - - '@aws-crypto/sha256-browser@5.2.0': - resolution: {integrity: sha512-AXfN/lGotSQwu6HNcEsIASo7kWXZ5HYWvfOmSNKDsEqC4OashTp8alTmaz+F7TC2L083SFv5RdB+qU3Vs1kZqw==} - - '@aws-crypto/sha256-js@5.2.0': - resolution: {integrity: sha512-FFQQyu7edu4ufvIZ+OadFpHHOt+eSTBaYaki44c+akjg7qZg9oOQeLlk77F6tSYqjDAFClrHJk9tMf0HdVyOvA==} - engines: {node: '>=16.0.0'} - - '@aws-crypto/supports-web-crypto@5.2.0': - resolution: {integrity: sha512-iAvUotm021kM33eCdNfwIN//F77/IADDSs58i+MDaOqFrVjZo9bAal0NK7HurRuWLLpF1iLX7gbWrjHjeo+YFg==} - - '@aws-crypto/util@5.2.0': - resolution: {integrity: sha512-4RkU9EsI6ZpBve5fseQlGNUWKMa1RLPQ1dnjnQoe07ldfIzcsGb5hC5W0Dm7u423KWzawlrpbjXBrXCEv9zazQ==} - - '@aws-sdk/client-s3@3.637.0': - resolution: {integrity: sha512-y6UC94fsMvhKbf0dzfnjVP1HePeGjplfcYfilZU1COIJLyTkMcUv4XcT4I407CGIrvgEafONHkiC09ygqUauNA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/client-sso-oidc@3.637.0': - resolution: {integrity: sha512-27bHALN6Qb6m6KZmPvRieJ/QRlj1lyac/GT2Rn5kJpre8Mpp+yxrtvp3h9PjNBty4lCeFEENfY4dGNSozBuBcw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.637.0 - - '@aws-sdk/client-sso@3.637.0': - resolution: {integrity: sha512-+KjLvgX5yJYROWo3TQuwBJlHCY0zz9PsLuEolmXQn0BVK1L/m9GteZHtd+rEdAoDGBpE0Xqjy1oz5+SmtsaRUw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/client-sts@3.637.0': - resolution: {integrity: sha512-xUi7x4qDubtA8QREtlblPuAcn91GS/09YVEY/RwU7xCY0aqGuFwgszAANlha4OUIqva8oVj2WO4gJuG+iaSnhw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/core@3.635.0': - resolution: {integrity: sha512-i1x/E/sgA+liUE1XJ7rj1dhyXpAKO1UKFUcTTHXok2ARjWTvszHnSXMOsB77aPbmn0fUp1JTx2kHUAZ1LVt5Bg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-env@3.620.1': - resolution: {integrity: sha512-ExuILJ2qLW5ZO+rgkNRj0xiAipKT16Rk77buvPP8csR7kkCflT/gXTyzRe/uzIiETTxM7tr8xuO9MP/DQXqkfg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-http@3.635.0': - resolution: {integrity: sha512-iJyRgEjOCQlBMXqtwPLIKYc7Bsc6nqjrZybdMDenPDa+kmLg7xh8LxHsu9088e+2/wtLicE34FsJJIfzu3L82g==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-ini@3.637.0': - resolution: {integrity: sha512-h+PFCWfZ0Q3Dx84SppET/TFpcQHmxFW8/oV9ArEvMilw4EBN+IlxgbL0CnHwjHW64szcmrM0mbebjEfHf4FXmw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.637.0 - - '@aws-sdk/credential-provider-node@3.637.0': - resolution: {integrity: sha512-yoEhoxJJfs7sPVQ6Is939BDQJZpZCoUgKr/ySse4YKOZ24t4VqgHA6+wV7rYh+7IW24Rd91UTvEzSuHYTlxlNA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-process@3.620.1': - resolution: {integrity: sha512-hWqFMidqLAkaV9G460+1at6qa9vySbjQKKc04p59OT7lZ5cO5VH5S4aI05e+m4j364MBROjjk2ugNvfNf/8ILg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-sso@3.637.0': - resolution: {integrity: sha512-Mvz+h+e62/tl+dVikLafhv+qkZJ9RUb8l2YN/LeKMWkxQylPT83CPk9aimVhCV89zth1zpREArl97+3xsfgQvA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-web-identity@3.621.0': - resolution: {integrity: sha512-w7ASSyfNvcx7+bYGep3VBgC3K6vEdLmlpjT7nSIHxxQf+WSdvy+HynwJosrpZax0sK5q0D1Jpn/5q+r5lwwW6w==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.621.0 - - '@aws-sdk/lib-storage@3.637.0': - resolution: {integrity: sha512-HiNGOP4a1QrCWwO1joKw4mCp19nLXoF9K52PislBaYDI35IlHC3DP6MeOg5zmElwtL1GtEHFBy5olfPWPsLyLg==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-s3': ^3.637.0 - - '@aws-sdk/middleware-bucket-endpoint@3.620.0': - resolution: {integrity: sha512-eGLL0W6L3HDb3OACyetZYOWpHJ+gLo0TehQKeQyy2G8vTYXqNTeqYhuI6up9HVjBzU9eQiULVQETmgQs7TFaRg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-expect-continue@3.620.0': - resolution: {integrity: sha512-QXeRFMLfyQ31nAHLbiTLtk0oHzG9QLMaof5jIfqcUwnOkO8YnQdeqzakrg1Alpy/VQ7aqzIi8qypkBe2KXZz0A==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-flexible-checksums@3.620.0': - resolution: {integrity: sha512-ftz+NW7qka2sVuwnnO1IzBku5ccP+s5qZGeRTPgrKB7OzRW85gthvIo1vQR2w+OwHFk7WJbbhhWwbCbktnP4UA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-host-header@3.620.0': - resolution: {integrity: sha512-VMtPEZwqYrII/oUkffYsNWY9PZ9xpNJpMgmyU0rlDQ25O1c0Hk3fJmZRe6pEkAJ0omD7kLrqGl1DUjQVxpd/Rg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-location-constraint@3.609.0': - resolution: {integrity: sha512-xzsdoTkszGVqGVPjUmgoP7TORiByLueMHieI1fhQL888WPdqctwAx3ES6d/bA9Q/i8jnc6hs+Fjhy8UvBTkE9A==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-logger@3.609.0': - resolution: {integrity: sha512-S62U2dy4jMDhDFDK5gZ4VxFdWzCtLzwbYyFZx2uvPYTECkepLUfzLic2BHg2Qvtu4QjX+oGE3P/7fwaGIsGNuQ==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-recursion-detection@3.620.0': - resolution: {integrity: sha512-nh91S7aGK3e/o1ck64sA/CyoFw+gAYj2BDOnoNa6ouyCrVJED96ZXWbhye/fz9SgmNUZR2g7GdVpiLpMKZoI5w==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-sdk-s3@3.635.0': - resolution: {integrity: sha512-RLdYJPEV4JL/7NBoFUs7VlP90X++5FlJdxHz0DzCjmiD3qCviKy+Cym3qg1gBgHwucs5XisuClxDrGokhAdTQw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-ssec@3.609.0': - resolution: {integrity: sha512-GZSD1s7+JswWOTamVap79QiDaIV7byJFssBW68GYjyRS5EBjNfwA/8s+6uE6g39R3ojyTbYOmvcANoZEhSULXg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-user-agent@3.637.0': - resolution: {integrity: sha512-EYo0NE9/da/OY8STDsK2LvM4kNa79DBsf4YVtaG4P5pZ615IeFsD8xOHZeuJmUrSMlVQ8ywPRX7WMucUybsKug==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/region-config-resolver@3.614.0': - resolution: {integrity: sha512-vDCeMXvic/LU0KFIUjpC3RiSTIkkvESsEfbVHiHH0YINfl8HnEqR5rj+L8+phsCeVg2+LmYwYxd5NRz4PHxt5g==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/s3-request-presigner@3.637.0': - resolution: {integrity: sha512-URRiEDZEICyfAXmXcXREQCsvZrapITAymvg46p1Xjnuv7PTnUB0SF18B2omPL0E5d/X+T3O9NKdtot+BqJbIWw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/signature-v4-multi-region@3.635.0': - resolution: {integrity: sha512-J6QY4/invOkpogCHjSaDON1hF03viPpOnsrzVuCvJMmclS/iG62R4EY0wq1alYll0YmSdmKlpJwHMWwGtqK63Q==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/token-providers@3.614.0': - resolution: {integrity: sha512-okItqyY6L9IHdxqs+Z116y5/nda7rHxLvROxtAJdLavWTYDydxrZstImNgGWTeVdmc0xX2gJCI77UYUTQWnhRw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sso-oidc': ^3.614.0 - - '@aws-sdk/types@3.609.0': - resolution: {integrity: sha512-+Tqnh9w0h2LcrUsdXyT1F8mNhXz+tVYBtP19LpeEGntmvHwa2XzvLUCWpoIAIVsHp5+HdB2X9Sn0KAtmbFXc2Q==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-arn-parser@3.568.0': - resolution: {integrity: sha512-XUKJWWo+KOB7fbnPP0+g/o5Ulku/X53t7i/h+sPHr5xxYTJJ9CYnbToo95mzxe7xWvkLrsNtJ8L+MnNn9INs2w==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-endpoints@3.637.0': - resolution: {integrity: sha512-pAqOKUHeVWHEXXDIp/qoMk/6jyxIb6GGjnK1/f8dKHtKIEs4tKsnnL563gceEvdad53OPXIt86uoevCcCzmBnw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-format-url@3.609.0': - resolution: {integrity: sha512-fuk29BI/oLQlJ7pfm6iJ4gkEpHdavffAALZwXh9eaY1vQ0ip0aKfRTiNudPoJjyyahnz5yJ1HkmlcDitlzsOrQ==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-locate-window@3.568.0': - resolution: {integrity: sha512-3nh4TINkXYr+H41QaPelCceEB2FXP3fxp93YZXB/kqJvX0U9j0N0Uk45gvsjmEPzG8XxkPEeLIfT2I1M7A6Lig==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-user-agent-browser@3.609.0': - resolution: {integrity: sha512-fojPU+mNahzQ0YHYBsx0ZIhmMA96H+ZIZ665ObU9tl+SGdbLneVZVikGve+NmHTQwHzwkFsZYYnVKAkreJLAtA==} - - '@aws-sdk/util-user-agent-node@3.614.0': - resolution: {integrity: sha512-15ElZT88peoHnq5TEoEtZwoXTXRxNrk60TZNdpl/TUBJ5oNJ9Dqb5Z4ryb8ofN6nm9aFf59GVAerFDz8iUoHBA==} - engines: {node: '>=16.0.0'} - peerDependencies: - aws-crt: '>=1.0.0' - peerDependenciesMeta: - aws-crt: - optional: true - - '@aws-sdk/xml-builder@3.609.0': - resolution: {integrity: sha512-l9XxNcA4HX98rwCC2/KoiWcmEiRfZe4G+mYwDbCFT87JIMj6GBhLDkAzr/W8KAaA2IDr8Vc6J8fZPgVulxxfMA==} - engines: {node: '>=16.0.0'} - - '@babel/code-frame@7.24.7': - resolution: {integrity: sha512-BcYH1CVJBO9tvyIZ2jVeXgSIMvGZ2FDRvDdOIVQyuklNKSsx+eppDEBq/g47Ayw+RqNFE+URvOShmf+f/qwAlA==} - engines: {node: '>=6.9.0'} - - '@babel/compat-data@7.25.4': - resolution: {integrity: sha512-+LGRog6RAsCJrrrg/IO6LGmpphNe5DiK30dGjCoxxeGv49B10/3XYGxPsAwrDlMFcFEvdAUavDT8r9k/hSyQqQ==} - engines: {node: '>=6.9.0'} - - '@babel/core@7.25.2': - resolution: {integrity: sha512-BBt3opiCOxUr9euZ5/ro/Xv8/V7yJ5bjYMqG/C1YAo8MIKAnumZalCN+msbci3Pigy4lIQfPUpfMM27HMGaYEA==} - engines: {node: '>=6.9.0'} - - '@babel/generator@7.25.6': - resolution: {integrity: sha512-VPC82gr1seXOpkjAAKoLhP50vx4vGNlF4msF64dSFq1P8RfB+QAuJWGHPXXPc8QyfVWwwB/TNNU4+ayZmHNbZw==} - engines: {node: '>=6.9.0'} - - '@babel/helper-annotate-as-pure@7.24.7': - resolution: {integrity: sha512-BaDeOonYvhdKw+JoMVkAixAAJzG2jVPIwWoKBPdYuY9b452e2rPuI9QPYh3KpofZ3pW2akOmwZLOiOsHMiqRAg==} - engines: {node: '>=6.9.0'} - - '@babel/helper-builder-binary-assignment-operator-visitor@7.24.7': - resolution: {integrity: sha512-xZeCVVdwb4MsDBkkyZ64tReWYrLRHlMN72vP7Bdm3OUOuyFZExhsHUUnuWnm2/XOlAJzR0LfPpB56WXZn0X/lA==} - engines: {node: '>=6.9.0'} - - '@babel/helper-compilation-targets@7.25.2': - resolution: {integrity: sha512-U2U5LsSaZ7TAt3cfaymQ8WHh0pxvdHoEk6HVpaexxixjyEquMh0L0YNJNM6CTGKMXV1iksi0iZkGw4AcFkPaaw==} - engines: {node: '>=6.9.0'} - - '@babel/helper-create-class-features-plugin@7.25.4': - resolution: {integrity: sha512-ro/bFs3/84MDgDmMwbcHgDa8/E6J3QKNTk4xJJnVeFtGE+tL0K26E3pNxhYz2b67fJpt7Aphw5XcploKXuCvCQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0 - - '@babel/helper-create-regexp-features-plugin@7.25.2': - resolution: {integrity: sha512-+wqVGP+DFmqwFD3EH6TMTfUNeqDehV3E/dl+Sd54eaXqm17tEUNbEIn4sVivVowbvUpOtIGxdo3GoXyDH9N/9g==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0 - - '@babel/helper-define-polyfill-provider@0.6.2': - resolution: {integrity: sha512-LV76g+C502biUK6AyZ3LK10vDpDyCzZnhZFXkH1L75zHPj68+qc8Zfpx2th+gzwA2MzyK+1g/3EPl62yFnVttQ==} - peerDependencies: - '@babel/core': ^7.4.0 || ^8.0.0-0 <8.0.0 - - '@babel/helper-member-expression-to-functions@7.24.8': - resolution: {integrity: sha512-LABppdt+Lp/RlBxqrh4qgf1oEH/WxdzQNDJIu5gC/W1GyvPVrOBiItmmM8wan2fm4oYqFuFfkXmlGpLQhPY8CA==} - engines: {node: '>=6.9.0'} - - '@babel/helper-module-imports@7.24.7': - resolution: {integrity: sha512-8AyH3C+74cgCVVXow/myrynrAGv+nTVg5vKu2nZph9x7RcRwzmh0VFallJuFTZ9mx6u4eSdXZfcOzSqTUm0HCA==} - engines: {node: '>=6.9.0'} - - '@babel/helper-module-transforms@7.25.2': - resolution: {integrity: sha512-BjyRAbix6j/wv83ftcVJmBt72QtHI56C7JXZoG2xATiLpmoC7dpd8WnkikExHDVPpi/3qCmO6WY1EaXOluiecQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0 - - '@babel/helper-optimise-call-expression@7.24.7': - resolution: {integrity: sha512-jKiTsW2xmWwxT1ixIdfXUZp+P5yURx2suzLZr5Hi64rURpDYdMW0pv+Uf17EYk2Rd428Lx4tLsnjGJzYKDM/6A==} - engines: {node: '>=6.9.0'} - - '@babel/helper-plugin-utils@7.24.8': - resolution: {integrity: sha512-FFWx5142D8h2Mgr/iPVGH5G7w6jDn4jUSpZTyDnQO0Yn7Ks2Kuz6Pci8H6MPCoUJegd/UZQ3tAvfLCxQSnWWwg==} - engines: {node: '>=6.9.0'} - - '@babel/helper-remap-async-to-generator@7.25.0': - resolution: {integrity: sha512-NhavI2eWEIz/H9dbrG0TuOicDhNexze43i5z7lEqwYm0WEZVTwnPpA0EafUTP7+6/W79HWIP2cTe3Z5NiSTVpw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0 - - '@babel/helper-replace-supers@7.25.0': - resolution: {integrity: sha512-q688zIvQVYtZu+i2PsdIu/uWGRpfxzr5WESsfpShfZECkO+d2o+WROWezCi/Q6kJ0tfPa5+pUGUlfx2HhrA3Bg==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0 - - '@babel/helper-simple-access@7.24.7': - resolution: {integrity: sha512-zBAIvbCMh5Ts+b86r/CjU+4XGYIs+R1j951gxI3KmmxBMhCg4oQMsv6ZXQ64XOm/cvzfU1FmoCyt6+owc5QMYg==} - engines: {node: '>=6.9.0'} - - '@babel/helper-skip-transparent-expression-wrappers@7.24.7': - resolution: {integrity: sha512-IO+DLT3LQUElMbpzlatRASEyQtfhSE0+m465v++3jyyXeBTBUjtVZg28/gHeV5mrTJqvEKhKroBGAvhW+qPHiQ==} - engines: {node: '>=6.9.0'} - - '@babel/helper-string-parser@7.24.8': - resolution: {integrity: sha512-pO9KhhRcuUyGnJWwyEgnRJTSIZHiT+vMD0kPeD+so0l7mxkMT19g3pjY9GTnHySck/hDzq+dtW/4VgnMkippsQ==} - engines: {node: '>=6.9.0'} - - '@babel/helper-validator-identifier@7.24.7': - resolution: {integrity: sha512-rR+PBcQ1SMQDDyF6X0wxtG8QyLCgUB0eRAGguqRLfkCA87l7yAP7ehq8SNj96OOGTO8OBV70KhuFYcIkHXOg0w==} - engines: {node: '>=6.9.0'} - - '@babel/helper-validator-option@7.24.8': - resolution: {integrity: sha512-xb8t9tD1MHLungh/AIoWYN+gVHaB9kwlu8gffXGSt3FFEIT7RjS+xWbc2vUD1UTZdIpKj/ab3rdqJ7ufngyi2Q==} - engines: {node: '>=6.9.0'} - - '@babel/helper-wrap-function@7.25.0': - resolution: {integrity: sha512-s6Q1ebqutSiZnEjaofc/UKDyC4SbzV5n5SrA2Gq8UawLycr3i04f1dX4OzoQVnexm6aOCh37SQNYlJ/8Ku+PMQ==} - engines: {node: '>=6.9.0'} - - '@babel/helpers@7.25.6': - resolution: {integrity: sha512-Xg0tn4HcfTijTwfDwYlvVCl43V6h4KyVVX2aEm4qdO/PC6L2YvzLHFdmxhoeSA3eslcE6+ZVXHgWwopXYLNq4Q==} - engines: {node: '>=6.9.0'} - - '@babel/highlight@7.24.7': - resolution: {integrity: sha512-EStJpq4OuY8xYfhGVXngigBJRWxftKX9ksiGDnmlY3o7B/V7KIAc9X4oiK87uPJSc/vs5L869bem5fhZa8caZw==} - engines: {node: '>=6.9.0'} - - '@babel/parser@7.25.6': - resolution: {integrity: sha512-trGdfBdbD0l1ZPmcJ83eNxB9rbEax4ALFTF7fN386TMYbeCQbyme5cOEXQhbGXKebwGaB/J52w1mrklMcbgy6Q==} - engines: {node: '>=6.0.0'} - hasBin: true - - '@babel/plugin-bugfix-firefox-class-in-computed-class-key@7.25.3': - resolution: {integrity: sha512-wUrcsxZg6rqBXG05HG1FPYgsP6EvwF4WpBbxIpWIIYnH8wG0gzx3yZY3dtEHas4sTAOGkbTsc9EGPxwff8lRoA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0 - - '@babel/plugin-bugfix-safari-class-field-initializer-scope@7.25.0': - resolution: {integrity: sha512-Bm4bH2qsX880b/3ziJ8KD711LT7z4u8CFudmjqle65AZj/HNUFhEf90dqYv6O86buWvSBmeQDjv0Tn2aF/bIBA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0 - - '@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression@7.25.0': - resolution: {integrity: sha512-lXwdNZtTmeVOOFtwM/WDe7yg1PL8sYhRk/XH0FzbR2HDQ0xC+EnQ/JHeoMYSavtU115tnUk0q9CDyq8si+LMAA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0 - - '@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining@7.24.7': - resolution: {integrity: sha512-+izXIbke1T33mY4MSNnrqhPXDz01WYhEf3yF5NbnUtkiNnm+XBZJl3kNfoK6NKmYlz/D07+l2GWVK/QfDkNCuQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.13.0 - - '@babel/plugin-bugfix-v8-static-class-fields-redefine-readonly@7.25.0': - resolution: {integrity: sha512-tggFrk1AIShG/RUQbEwt2Tr/E+ObkfwrPjR6BjbRvsx24+PSjK8zrq0GWPNCjo8qpRx4DuJzlcvWJqlm+0h3kw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0 - - '@babel/plugin-proposal-private-property-in-object@7.21.0-placeholder-for-preset-env.2': - resolution: {integrity: sha512-SOSkfJDddaM7mak6cPEpswyTRnuRltl429hMraQEglW+OkovnCzsiszTmsrlY//qLFjCpQDFRvjdm2wA5pPm9w==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-async-generators@7.8.4': - resolution: {integrity: sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-class-properties@7.12.13': - resolution: {integrity: sha512-fm4idjKla0YahUNgFNLCB0qySdsoPiZP3iQE3rky0mBUtMZ23yDJ9SJdg6dXTSDnulOVqiF3Hgr9nbXvXTQZYA==} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-class-static-block@7.14.5': - resolution: {integrity: sha512-b+YyPmr6ldyNnM6sqYeMWE+bgJcJpO6yS4QD7ymxgH34GBPNDM/THBh8iunyvKIZztiwLH4CJZ0RxTk9emgpjw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-dynamic-import@7.8.3': - resolution: {integrity: sha512-5gdGbFon+PszYzqs83S3E5mpi7/y/8M9eC90MRTZfduQOYW76ig6SOSPNe41IG5LoP3FGBn2N0RjVDSQiS94kQ==} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-export-namespace-from@7.8.3': - resolution: {integrity: sha512-MXf5laXo6c1IbEbegDmzGPwGNTsHZmEy6QGznu5Sh2UCWvueywb2ee+CCE4zQiZstxU9BMoQO9i6zUFSY0Kj0Q==} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-import-assertions@7.25.6': - resolution: {integrity: sha512-aABl0jHw9bZ2karQ/uUD6XP4u0SG22SJrOHFoL6XB1R7dTovOP4TzTlsxOYC5yQ1pdscVK2JTUnF6QL3ARoAiQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-import-attributes@7.25.6': - resolution: {integrity: sha512-sXaDXaJN9SNLymBdlWFA+bjzBhFD617ZaFiY13dGt7TVslVvVgA6fkZOP7Ki3IGElC45lwHdOTrCtKZGVAWeLQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-import-meta@7.10.4': - resolution: {integrity: sha512-Yqfm+XDx0+Prh3VSeEQCPU81yC+JWZ2pDPFSS4ZdpfZhp4MkFMaDC1UqseovEKwSUpnIL7+vK+Clp7bfh0iD7g==} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-json-strings@7.8.3': - resolution: {integrity: sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-jsx@7.24.7': - resolution: {integrity: sha512-6ddciUPe/mpMnOKv/U+RSd2vvVy+Yw/JfBB0ZHYjEZt9NLHmCUylNYlsbqCCS1Bffjlb0fCwC9Vqz+sBz6PsiQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-logical-assignment-operators@7.10.4': - resolution: {integrity: sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig==} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-nullish-coalescing-operator@7.8.3': - resolution: {integrity: sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ==} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-numeric-separator@7.10.4': - resolution: {integrity: sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug==} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-object-rest-spread@7.8.3': - resolution: {integrity: sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-optional-catch-binding@7.8.3': - resolution: {integrity: sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-optional-chaining@7.8.3': - resolution: {integrity: sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg==} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-private-property-in-object@7.14.5': - resolution: {integrity: sha512-0wVnp9dxJ72ZUJDV27ZfbSj6iHLoytYZmh3rFcxNnvsJF3ktkzLDZPy/mA17HGsaQT3/DQsWYX1f1QGWkCoVUg==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-top-level-await@7.14.5': - resolution: {integrity: sha512-hx++upLv5U1rgYfwe1xBQUhRmU41NEvpUvrp8jkrSCdvGSnM5/qdRMtylJ6PG5OFkBaHkbTAKTnd3/YyESRHFw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-typescript@7.25.4': - resolution: {integrity: sha512-uMOCoHVU52BsSWxPOMVv5qKRdeSlPuImUCB2dlPuBSU+W2/ROE7/Zg8F2Kepbk+8yBa68LlRKxO+xgEVWorsDg==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-syntax-unicode-sets-regex@7.18.6': - resolution: {integrity: sha512-727YkEAPwSIQTv5im8QHz3upqp92JTWhidIC81Tdx4VJYIte/VndKf1qKrfnnhPLiPghStWfvC/iFaMCQu7Nqg==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0 - - '@babel/plugin-transform-arrow-functions@7.24.7': - resolution: {integrity: sha512-Dt9LQs6iEY++gXUwY03DNFat5C2NbO48jj+j/bSAz6b3HgPs39qcPiYt77fDObIcFwj3/C2ICX9YMwGflUoSHQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-async-generator-functions@7.25.4': - resolution: {integrity: sha512-jz8cV2XDDTqjKPwVPJBIjORVEmSGYhdRa8e5k5+vN+uwcjSrSxUaebBRa4ko1jqNF2uxyg8G6XYk30Jv285xzg==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-async-to-generator@7.24.7': - resolution: {integrity: sha512-SQY01PcJfmQ+4Ash7NE+rpbLFbmqA2GPIgqzxfFTL4t1FKRq4zTms/7htKpoCUI9OcFYgzqfmCdH53s6/jn5fA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-block-scoped-functions@7.24.7': - resolution: {integrity: sha512-yO7RAz6EsVQDaBH18IDJcMB1HnrUn2FJ/Jslc/WtPPWcjhpUJXU/rjbwmluzp7v/ZzWcEhTMXELnnsz8djWDwQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-block-scoping@7.25.0': - resolution: {integrity: sha512-yBQjYoOjXlFv9nlXb3f1casSHOZkWr29NX+zChVanLg5Nc157CrbEX9D7hxxtTpuFy7Q0YzmmWfJxzvps4kXrQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-class-properties@7.25.4': - resolution: {integrity: sha512-nZeZHyCWPfjkdU5pA/uHiTaDAFUEqkpzf1YoQT2NeSynCGYq9rxfyI3XpQbfx/a0hSnFH6TGlEXvae5Vi7GD8g==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-class-static-block@7.24.7': - resolution: {integrity: sha512-HMXK3WbBPpZQufbMG4B46A90PkuuhN9vBCb5T8+VAHqvAqvcLi+2cKoukcpmUYkszLhScU3l1iudhrks3DggRQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.12.0 - - '@babel/plugin-transform-classes@7.25.4': - resolution: {integrity: sha512-oexUfaQle2pF/b6E0dwsxQtAol9TLSO88kQvym6HHBWFliV2lGdrPieX+WgMRLSJDVzdYywk7jXbLPuO2KLTLg==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-computed-properties@7.24.7': - resolution: {integrity: sha512-25cS7v+707Gu6Ds2oY6tCkUwsJ9YIDbggd9+cu9jzzDgiNq7hR/8dkzxWfKWnTic26vsI3EsCXNd4iEB6e8esQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-destructuring@7.24.8': - resolution: {integrity: sha512-36e87mfY8TnRxc7yc6M9g9gOB7rKgSahqkIKwLpz4Ppk2+zC2Cy1is0uwtuSG6AE4zlTOUa+7JGz9jCJGLqQFQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-dotall-regex@7.24.7': - resolution: {integrity: sha512-ZOA3W+1RRTSWvyqcMJDLqbchh7U4NRGqwRfFSVbOLS/ePIP4vHB5e8T8eXcuqyN1QkgKyj5wuW0lcS85v4CrSw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-duplicate-keys@7.24.7': - resolution: {integrity: sha512-JdYfXyCRihAe46jUIliuL2/s0x0wObgwwiGxw/UbgJBr20gQBThrokO4nYKgWkD7uBaqM7+9x5TU7NkExZJyzw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-duplicate-named-capturing-groups-regex@7.25.0': - resolution: {integrity: sha512-YLpb4LlYSc3sCUa35un84poXoraOiQucUTTu8X1j18JV+gNa8E0nyUf/CjZ171IRGr4jEguF+vzJU66QZhn29g==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0 - - '@babel/plugin-transform-dynamic-import@7.24.7': - resolution: {integrity: sha512-sc3X26PhZQDb3JhORmakcbvkeInvxz+A8oda99lj7J60QRuPZvNAk9wQlTBS1ZynelDrDmTU4pw1tyc5d5ZMUg==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-exponentiation-operator@7.24.7': - resolution: {integrity: sha512-Rqe/vSc9OYgDajNIK35u7ot+KeCoetqQYFXM4Epf7M7ez3lWlOjrDjrwMei6caCVhfdw+mIKD4cgdGNy5JQotQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-export-namespace-from@7.24.7': - resolution: {integrity: sha512-v0K9uNYsPL3oXZ/7F9NNIbAj2jv1whUEtyA6aujhekLs56R++JDQuzRcP2/z4WX5Vg/c5lE9uWZA0/iUoFhLTA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-for-of@7.24.7': - resolution: {integrity: sha512-wo9ogrDG1ITTTBsy46oGiN1dS9A7MROBTcYsfS8DtsImMkHk9JXJ3EWQM6X2SUw4x80uGPlwj0o00Uoc6nEE3g==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-function-name@7.25.1': - resolution: {integrity: sha512-TVVJVdW9RKMNgJJlLtHsKDTydjZAbwIsn6ySBPQaEAUU5+gVvlJt/9nRmqVbsV/IBanRjzWoaAQKLoamWVOUuA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-json-strings@7.24.7': - resolution: {integrity: sha512-2yFnBGDvRuxAaE/f0vfBKvtnvvqU8tGpMHqMNpTN2oWMKIR3NqFkjaAgGwawhqK/pIN2T3XdjGPdaG0vDhOBGw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-literals@7.25.2': - resolution: {integrity: sha512-HQI+HcTbm9ur3Z2DkO+jgESMAMcYLuN/A7NRw9juzxAezN9AvqvUTnpKP/9kkYANz6u7dFlAyOu44ejuGySlfw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-logical-assignment-operators@7.24.7': - resolution: {integrity: sha512-4D2tpwlQ1odXmTEIFWy9ELJcZHqrStlzK/dAOWYyxX3zT0iXQB6banjgeOJQXzEc4S0E0a5A+hahxPaEFYftsw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-member-expression-literals@7.24.7': - resolution: {integrity: sha512-T/hRC1uqrzXMKLQ6UCwMT85S3EvqaBXDGf0FaMf4446Qx9vKwlghvee0+uuZcDUCZU5RuNi4781UQ7R308zzBw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-modules-amd@7.24.7': - resolution: {integrity: sha512-9+pB1qxV3vs/8Hdmz/CulFB8w2tuu6EB94JZFsjdqxQokwGa9Unap7Bo2gGBGIvPmDIVvQrom7r5m/TCDMURhg==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-modules-commonjs@7.24.8': - resolution: {integrity: sha512-WHsk9H8XxRs3JXKWFiqtQebdh9b/pTk4EgueygFzYlTKAg0Ud985mSevdNjdXdFBATSKVJGQXP1tv6aGbssLKA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-modules-systemjs@7.25.0': - resolution: {integrity: sha512-YPJfjQPDXxyQWg/0+jHKj1llnY5f/R6a0p/vP4lPymxLu7Lvl4k2WMitqi08yxwQcCVUUdG9LCUj4TNEgAp3Jw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-modules-umd@7.24.7': - resolution: {integrity: sha512-3aytQvqJ/h9z4g8AsKPLvD4Zqi2qT+L3j7XoFFu1XBlZWEl2/1kWnhmAbxpLgPrHSY0M6UA02jyTiwUVtiKR6A==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-named-capturing-groups-regex@7.24.7': - resolution: {integrity: sha512-/jr7h/EWeJtk1U/uz2jlsCioHkZk1JJZVcc8oQsJ1dUlaJD83f4/6Zeh2aHt9BIFokHIsSeDfhUmju0+1GPd6g==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0 - - '@babel/plugin-transform-new-target@7.24.7': - resolution: {integrity: sha512-RNKwfRIXg4Ls/8mMTza5oPF5RkOW8Wy/WgMAp1/F1yZ8mMbtwXW+HDoJiOsagWrAhI5f57Vncrmr9XeT4CVapA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-nullish-coalescing-operator@7.24.7': - resolution: {integrity: sha512-Ts7xQVk1OEocqzm8rHMXHlxvsfZ0cEF2yomUqpKENHWMF4zKk175Y4q8H5knJes6PgYad50uuRmt3UJuhBw8pQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-numeric-separator@7.24.7': - resolution: {integrity: sha512-e6q1TiVUzvH9KRvicuxdBTUj4AdKSRwzIyFFnfnezpCfP2/7Qmbb8qbU2j7GODbl4JMkblitCQjKYUaX/qkkwA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-object-rest-spread@7.24.7': - resolution: {integrity: sha512-4QrHAr0aXQCEFni2q4DqKLD31n2DL+RxcwnNjDFkSG0eNQ/xCavnRkfCUjsyqGC2OviNJvZOF/mQqZBw7i2C5Q==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-object-super@7.24.7': - resolution: {integrity: sha512-A/vVLwN6lBrMFmMDmPPz0jnE6ZGx7Jq7d6sT/Ev4H65RER6pZ+kczlf1DthF5N0qaPHBsI7UXiE8Zy66nmAovg==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-optional-catch-binding@7.24.7': - resolution: {integrity: sha512-uLEndKqP5BfBbC/5jTwPxLh9kqPWWgzN/f8w6UwAIirAEqiIVJWWY312X72Eub09g5KF9+Zn7+hT7sDxmhRuKA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-optional-chaining@7.24.8': - resolution: {integrity: sha512-5cTOLSMs9eypEy8JUVvIKOu6NgvbJMnpG62VpIHrTmROdQ+L5mDAaI40g25k5vXti55JWNX5jCkq3HZxXBQANw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-parameters@7.24.7': - resolution: {integrity: sha512-yGWW5Rr+sQOhK0Ot8hjDJuxU3XLRQGflvT4lhlSY0DFvdb3TwKaY26CJzHtYllU0vT9j58hc37ndFPsqT1SrzA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-private-methods@7.25.4': - resolution: {integrity: sha512-ao8BG7E2b/URaUQGqN3Tlsg+M3KlHY6rJ1O1gXAEUnZoyNQnvKyH87Kfg+FoxSeyWUB8ISZZsC91C44ZuBFytw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-private-property-in-object@7.24.7': - resolution: {integrity: sha512-9z76mxwnwFxMyxZWEgdgECQglF2Q7cFLm0kMf8pGwt+GSJsY0cONKj/UuO4bOH0w/uAel3ekS4ra5CEAyJRmDA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-property-literals@7.24.7': - resolution: {integrity: sha512-EMi4MLQSHfd2nrCqQEWxFdha2gBCqU4ZcCng4WBGZ5CJL4bBRW0ptdqqDdeirGZcpALazVVNJqRmsO8/+oNCBA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-regenerator@7.24.7': - resolution: {integrity: sha512-lq3fvXPdimDrlg6LWBoqj+r/DEWgONuwjuOuQCSYgRroXDH/IdM1C0IZf59fL5cHLpjEH/O6opIRBbqv7ELnuA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-reserved-words@7.24.7': - resolution: {integrity: sha512-0DUq0pHcPKbjFZCfTss/pGkYMfy3vFWydkUBd9r0GHpIyfs2eCDENvqadMycRS9wZCXR41wucAfJHJmwA0UmoQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-shorthand-properties@7.24.7': - resolution: {integrity: sha512-KsDsevZMDsigzbA09+vacnLpmPH4aWjcZjXdyFKGzpplxhbeB4wYtury3vglQkg6KM/xEPKt73eCjPPf1PgXBA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-spread@7.24.7': - resolution: {integrity: sha512-x96oO0I09dgMDxJaANcRyD4ellXFLLiWhuwDxKZX5g2rWP1bTPkBSwCYv96VDXVT1bD9aPj8tppr5ITIh8hBng==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-sticky-regex@7.24.7': - resolution: {integrity: sha512-kHPSIJc9v24zEml5geKg9Mjx5ULpfncj0wRpYtxbvKyTtHCYDkVE3aHQ03FrpEo4gEe2vrJJS1Y9CJTaThA52g==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-template-literals@7.24.7': - resolution: {integrity: sha512-AfDTQmClklHCOLxtGoP7HkeMw56k1/bTQjwsfhL6pppo/M4TOBSq+jjBUBLmV/4oeFg4GWMavIl44ZeCtmmZTw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-typeof-symbol@7.24.8': - resolution: {integrity: sha512-adNTUpDCVnmAE58VEqKlAA6ZBlNkMnWD0ZcW76lyNFN3MJniyGFZfNwERVk8Ap56MCnXztmDr19T4mPTztcuaw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-typescript@7.25.2': - resolution: {integrity: sha512-lBwRvjSmqiMYe/pS0+1gggjJleUJi7NzjvQ1Fkqtt69hBa/0t1YuW/MLQMAPixfwaQOHUXsd6jeU3Z+vdGv3+A==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-unicode-escapes@7.24.7': - resolution: {integrity: sha512-U3ap1gm5+4edc2Q/P+9VrBNhGkfnf+8ZqppY71Bo/pzZmXhhLdqgaUl6cuB07O1+AQJtCLfaOmswiNbSQ9ivhw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-unicode-property-regex@7.24.7': - resolution: {integrity: sha512-uH2O4OV5M9FZYQrwc7NdVmMxQJOCCzFeYudlZSzUAHRFeOujQefa92E74TQDVskNHCzOXoigEuoyzHDhaEaK5w==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-unicode-regex@7.24.7': - resolution: {integrity: sha512-hlQ96MBZSAXUq7ltkjtu3FJCCSMx/j629ns3hA3pXnBXjanNP0LHi+JpPeA81zaWgVK1VGH95Xuy7u0RyQ8kMg==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/plugin-transform-unicode-sets-regex@7.25.4': - resolution: {integrity: sha512-qesBxiWkgN1Q+31xUE9RcMk79eOXXDCv6tfyGMRSs4RGlioSg2WVyQAm07k726cSE56pa+Kb0y9epX2qaXzTvA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0 - - '@babel/preset-env@7.25.4': - resolution: {integrity: sha512-W9Gyo+KmcxjGahtt3t9fb14vFRWvPpu5pT6GBlovAK6BTBcxgjfVMSQCfJl4oi35ODrxP6xx2Wr8LNST57Mraw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/preset-modules@0.1.6-no-external-plugins': - resolution: {integrity: sha512-HrcgcIESLm9aIR842yhJ5RWan/gebQUJ6E/E5+rf0y9o6oj7w0Br+sWuL6kEQ/o/AdfvR1Je9jG18/gnpwjEyA==} - peerDependencies: - '@babel/core': ^7.0.0-0 || ^8.0.0-0 <8.0.0 - - '@babel/preset-typescript@7.24.7': - resolution: {integrity: sha512-SyXRe3OdWwIwalxDg5UtJnJQO+YPcTfwiIY2B0Xlddh9o7jpWLvv8X1RthIeDOxQ+O1ML5BLPCONToObyVQVuQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 - - '@babel/regjsgen@0.8.0': - resolution: {integrity: sha512-x/rqGMdzj+fWZvCOYForTghzbtqPDZ5gPwaoNGHdgDfF2QA/XZbCBp4Moo5scrkAMPhB7z26XM/AaHuIJdgauA==} - - '@babel/runtime@7.25.6': - resolution: {integrity: sha512-VBj9MYyDb9tuLq7yzqjgzt6Q+IBQLrGZfdjOekyEirZPHxXWoTSGUTMrpsfi58Up73d13NfYLv8HT9vmznjzhQ==} - engines: {node: '>=6.9.0'} - - '@babel/template@7.25.0': - resolution: {integrity: sha512-aOOgh1/5XzKvg1jvVz7AVrx2piJ2XBi227DHmbY6y+bM9H2FlN+IfecYu4Xl0cNiiVejlsCri89LUsbj8vJD9Q==} - engines: {node: '>=6.9.0'} - - '@babel/traverse@7.25.6': - resolution: {integrity: sha512-9Vrcx5ZW6UwK5tvqsj0nGpp/XzqthkT0dqIc9g1AdtygFToNtTF67XzYS//dm+SAK9cp3B9R4ZO/46p63SCjlQ==} - engines: {node: '>=6.9.0'} - - '@babel/types@7.25.6': - resolution: {integrity: sha512-/l42B1qxpG6RdfYf343Uw1vmDjeNhneUXtzhojE7pDgfpEypmRhI6j1kr17XCVv4Cgl9HdAiQY2x0GwKm7rWCw==} - engines: {node: '>=6.9.0'} - - '@cspotcode/source-map-support@0.8.1': - resolution: {integrity: sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==} - engines: {node: '>=12'} - - '@emnapi/runtime@1.2.0': - resolution: {integrity: sha512-bV21/9LQmcQeCPEg3BDFtvwL6cwiTMksYNWQQ4KOxCZikEGalWtenoZ0wCiukJINlGCIi2KXx01g4FoH/LxpzQ==} - - '@esbuild/aix-ppc64@0.23.1': - resolution: {integrity: sha512-6VhYk1diRqrhBAqpJEdjASR/+WVRtfjpqKuNw11cLiaWpAT/Uu+nokB+UJnevzy/P9C/ty6AOe0dwueMrGh/iQ==} - engines: {node: '>=18'} - cpu: [ppc64] - os: [aix] - - '@esbuild/android-arm64@0.23.1': - resolution: {integrity: sha512-xw50ipykXcLstLeWH7WRdQuysJqejuAGPd30vd1i5zSyKK3WE+ijzHmLKxdiCMtH1pHz78rOg0BKSYOSB/2Khw==} - engines: {node: '>=18'} - cpu: [arm64] - os: [android] - - '@esbuild/android-arm@0.23.1': - resolution: {integrity: sha512-uz6/tEy2IFm9RYOyvKl88zdzZfwEfKZmnX9Cj1BHjeSGNuGLuMD1kR8y5bteYmwqKm1tj8m4cb/aKEorr6fHWQ==} - engines: {node: '>=18'} - cpu: [arm] - os: [android] - - '@esbuild/android-x64@0.23.1': - resolution: {integrity: sha512-nlN9B69St9BwUoB+jkyU090bru8L0NA3yFvAd7k8dNsVH8bi9a8cUAUSEcEEgTp2z3dbEDGJGfP6VUnkQnlReg==} - engines: {node: '>=18'} - cpu: [x64] - os: [android] - - '@esbuild/darwin-arm64@0.23.1': - resolution: {integrity: sha512-YsS2e3Wtgnw7Wq53XXBLcV6JhRsEq8hkfg91ESVadIrzr9wO6jJDMZnCQbHm1Guc5t/CdDiFSSfWP58FNuvT3Q==} - engines: {node: '>=18'} - cpu: [arm64] - os: [darwin] - - '@esbuild/darwin-x64@0.23.1': - resolution: {integrity: sha512-aClqdgTDVPSEGgoCS8QDG37Gu8yc9lTHNAQlsztQ6ENetKEO//b8y31MMu2ZaPbn4kVsIABzVLXYLhCGekGDqw==} - engines: {node: '>=18'} - cpu: [x64] - os: [darwin] - - '@esbuild/freebsd-arm64@0.23.1': - resolution: {integrity: sha512-h1k6yS8/pN/NHlMl5+v4XPfikhJulk4G+tKGFIOwURBSFzE8bixw1ebjluLOjfwtLqY0kewfjLSrO6tN2MgIhA==} - engines: {node: '>=18'} - cpu: [arm64] - os: [freebsd] - - '@esbuild/freebsd-x64@0.23.1': - resolution: {integrity: sha512-lK1eJeyk1ZX8UklqFd/3A60UuZ/6UVfGT2LuGo3Wp4/z7eRTRYY+0xOu2kpClP+vMTi9wKOfXi2vjUpO1Ro76g==} - engines: {node: '>=18'} - cpu: [x64] - os: [freebsd] - - '@esbuild/linux-arm64@0.23.1': - resolution: {integrity: sha512-/93bf2yxencYDnItMYV/v116zff6UyTjo4EtEQjUBeGiVpMmffDNUyD9UN2zV+V3LRV3/on4xdZ26NKzn6754g==} - engines: {node: '>=18'} - cpu: [arm64] - os: [linux] - - '@esbuild/linux-arm@0.23.1': - resolution: {integrity: sha512-CXXkzgn+dXAPs3WBwE+Kvnrf4WECwBdfjfeYHpMeVxWE0EceB6vhWGShs6wi0IYEqMSIzdOF1XjQ/Mkm5d7ZdQ==} - engines: {node: '>=18'} - cpu: [arm] - os: [linux] - - '@esbuild/linux-ia32@0.23.1': - resolution: {integrity: sha512-VTN4EuOHwXEkXzX5nTvVY4s7E/Krz7COC8xkftbbKRYAl96vPiUssGkeMELQMOnLOJ8k3BY1+ZY52tttZnHcXQ==} - engines: {node: '>=18'} - cpu: [ia32] - os: [linux] - - '@esbuild/linux-loong64@0.23.1': - resolution: {integrity: sha512-Vx09LzEoBa5zDnieH8LSMRToj7ir/Jeq0Gu6qJ/1GcBq9GkfoEAoXvLiW1U9J1qE/Y/Oyaq33w5p2ZWrNNHNEw==} - engines: {node: '>=18'} - cpu: [loong64] - os: [linux] - - '@esbuild/linux-mips64el@0.23.1': - resolution: {integrity: sha512-nrFzzMQ7W4WRLNUOU5dlWAqa6yVeI0P78WKGUo7lg2HShq/yx+UYkeNSE0SSfSure0SqgnsxPvmAUu/vu0E+3Q==} - engines: {node: '>=18'} - cpu: [mips64el] - os: [linux] - - '@esbuild/linux-ppc64@0.23.1': - resolution: {integrity: sha512-dKN8fgVqd0vUIjxuJI6P/9SSSe/mB9rvA98CSH2sJnlZ/OCZWO1DJvxj8jvKTfYUdGfcq2dDxoKaC6bHuTlgcw==} - engines: {node: '>=18'} - cpu: [ppc64] - os: [linux] - - '@esbuild/linux-riscv64@0.23.1': - resolution: {integrity: sha512-5AV4Pzp80fhHL83JM6LoA6pTQVWgB1HovMBsLQ9OZWLDqVY8MVobBXNSmAJi//Csh6tcY7e7Lny2Hg1tElMjIA==} - engines: {node: '>=18'} - cpu: [riscv64] - os: [linux] - - '@esbuild/linux-s390x@0.23.1': - resolution: {integrity: sha512-9ygs73tuFCe6f6m/Tb+9LtYxWR4c9yg7zjt2cYkjDbDpV/xVn+68cQxMXCjUpYwEkze2RcU/rMnfIXNRFmSoDw==} - engines: {node: '>=18'} - cpu: [s390x] - os: [linux] - - '@esbuild/linux-x64@0.23.1': - resolution: {integrity: sha512-EV6+ovTsEXCPAp58g2dD68LxoP/wK5pRvgy0J/HxPGB009omFPv3Yet0HiaqvrIrgPTBuC6wCH1LTOY91EO5hQ==} - engines: {node: '>=18'} - cpu: [x64] - os: [linux] - - '@esbuild/netbsd-x64@0.23.1': - resolution: {integrity: sha512-aevEkCNu7KlPRpYLjwmdcuNz6bDFiE7Z8XC4CPqExjTvrHugh28QzUXVOZtiYghciKUacNktqxdpymplil1beA==} - engines: {node: '>=18'} - cpu: [x64] - os: [netbsd] - - '@esbuild/openbsd-arm64@0.23.1': - resolution: {integrity: sha512-3x37szhLexNA4bXhLrCC/LImN/YtWis6WXr1VESlfVtVeoFJBRINPJ3f0a/6LV8zpikqoUg4hyXw0sFBt5Cr+Q==} - engines: {node: '>=18'} - cpu: [arm64] - os: [openbsd] - - '@esbuild/openbsd-x64@0.23.1': - resolution: {integrity: sha512-aY2gMmKmPhxfU+0EdnN+XNtGbjfQgwZj43k8G3fyrDM/UdZww6xrWxmDkuz2eCZchqVeABjV5BpildOrUbBTqA==} - engines: {node: '>=18'} - cpu: [x64] - os: [openbsd] - - '@esbuild/sunos-x64@0.23.1': - resolution: {integrity: sha512-RBRT2gqEl0IKQABT4XTj78tpk9v7ehp+mazn2HbUeZl1YMdaGAQqhapjGTCe7uw7y0frDi4gS0uHzhvpFuI1sA==} - engines: {node: '>=18'} - cpu: [x64] - os: [sunos] - - '@esbuild/win32-arm64@0.23.1': - resolution: {integrity: sha512-4O+gPR5rEBe2FpKOVyiJ7wNDPA8nGzDuJ6gN4okSA1gEOYZ67N8JPk58tkWtdtPeLz7lBnY6I5L3jdsr3S+A6A==} - engines: {node: '>=18'} - cpu: [arm64] - os: [win32] - - '@esbuild/win32-ia32@0.23.1': - resolution: {integrity: sha512-BcaL0Vn6QwCwre3Y717nVHZbAa4UBEigzFm6VdsVdT/MbZ38xoj1X9HPkZhbmaBGUD1W8vxAfffbDe8bA6AKnQ==} - engines: {node: '>=18'} - cpu: [ia32] - os: [win32] - - '@esbuild/win32-x64@0.23.1': - resolution: {integrity: sha512-BHpFFeslkWrXWyUPnbKm+xYYVYruCinGcftSBaa8zoF9hZO4BcSCFUvHVTtzpIY6YzUnYtuEhZ+C9iEXjxnasg==} - engines: {node: '>=18'} - cpu: [x64] - os: [win32] - - '@fastify/accept-negotiator@1.1.0': - resolution: {integrity: sha512-OIHZrb2ImZ7XG85HXOONLcJWGosv7sIvM2ifAPQVhg9Lv7qdmMBNVaai4QTdyuaqbKM5eO6sLSQOYI7wEQeCJQ==} - engines: {node: '>=14'} - - '@fastify/ajv-compiler@3.6.0': - resolution: {integrity: sha512-LwdXQJjmMD+GwLOkP7TVC68qa+pSSogeWWmznRJ/coyTcfe9qA05AHFSe1eZFwK6q+xVRpChnvFUkf1iYaSZsQ==} - - '@fastify/error@3.4.1': - resolution: {integrity: sha512-wWSvph+29GR783IhmvdwWnN4bUxTD01Vm5Xad4i7i1VuAOItLvbPAb69sb0IQ2N57yprvhNIwAP5B6xfKTmjmQ==} - - '@fastify/fast-json-stringify-compiler@4.3.0': - resolution: {integrity: sha512-aZAXGYo6m22Fk1zZzEUKBvut/CIIQe/BapEORnxiD5Qr0kPHqqI69NtEMCme74h+at72sPhbkb4ZrLd1W3KRLA==} - - '@fastify/merge-json-schemas@0.1.1': - resolution: {integrity: sha512-fERDVz7topgNjtXsJTTW1JKLy0rhuLRcquYqNR9rF7OcVpCa2OVW49ZPDIhaRRCaUuvVxI+N416xUoF76HNSXA==} - - '@fastify/send@2.1.0': - resolution: {integrity: sha512-yNYiY6sDkexoJR0D8IDy3aRP3+L4wdqCpvx5WP+VtEU58sn7USmKynBzDQex5X42Zzvw2gNzzYgP90UfWShLFA==} - - '@fastify/static@7.0.4': - resolution: {integrity: sha512-p2uKtaf8BMOZWLs6wu+Ihg7bWNBdjNgCwDza4MJtTqg+5ovKmcbgbR9Xs5/smZ1YISfzKOCNYmZV8LaCj+eJ1Q==} - - '@fastify/swagger-ui@4.1.0': - resolution: {integrity: sha512-Bqsd6VFQR7WoT6eRammOF8/gXf5GKywq2zYy8/3fj2rsZw43cmXdfsEKxVAmAwOW2Nv+dnyQaf5qM6kBqyXRlw==} - - '@fastify/swagger@8.15.0': - resolution: {integrity: sha512-zy+HEEKFqPMS2sFUsQU5X0MHplhKJvWeohBwTCkBAJA/GDYGLGUWQaETEhptiqxK7Hs0fQB9B4MDb3pbwIiCwA==} - - '@grpc/grpc-js@1.11.1': - resolution: {integrity: sha512-gyt/WayZrVPH2w/UTLansS7F9Nwld472JxxaETamrM8HNlsa+jSLNyKAZmhxI2Me4c3mQHFiS1wWHDY1g1Kthw==} - engines: {node: '>=12.10.0'} - - '@grpc/proto-loader@0.7.13': - resolution: {integrity: sha512-AiXO/bfe9bmxBjxxtYxFAXGZvMaN5s8kO+jBHAJCON8rJoB5YS/D6X7ZNc6XQkuHNmyl4CYaMI1fJ/Gn27RGGw==} - engines: {node: '>=6'} - hasBin: true - - '@img/sharp-darwin-arm64@0.33.5': - resolution: {integrity: sha512-UT4p+iz/2H4twwAoLCqfA9UH5pI6DggwKEGuaPy7nCVQ8ZsiY5PIcrRvD1DzuY3qYL07NtIQcWnBSY/heikIFQ==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm64] - os: [darwin] - - '@img/sharp-darwin-x64@0.33.5': - resolution: {integrity: sha512-fyHac4jIc1ANYGRDxtiqelIbdWkIuQaI84Mv45KvGRRxSAa7o7d1ZKAOBaYbnepLC1WqxfpimdeWfvqqSGwR2Q==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [x64] - os: [darwin] - - '@img/sharp-libvips-darwin-arm64@1.0.4': - resolution: {integrity: sha512-XblONe153h0O2zuFfTAbQYAX2JhYmDHeWikp1LM9Hul9gVPjFY427k6dFEcOL72O01QxQsWi761svJ/ev9xEDg==} - cpu: [arm64] - os: [darwin] - - '@img/sharp-libvips-darwin-x64@1.0.4': - resolution: {integrity: sha512-xnGR8YuZYfJGmWPvmlunFaWJsb9T/AO2ykoP3Fz/0X5XV2aoYBPkX6xqCQvUTKKiLddarLaxpzNe+b1hjeWHAQ==} - cpu: [x64] - os: [darwin] - - '@img/sharp-libvips-linux-arm64@1.0.4': - resolution: {integrity: sha512-9B+taZ8DlyyqzZQnoeIvDVR/2F4EbMepXMc/NdVbkzsJbzkUjhXv/70GQJ7tdLA4YJgNP25zukcxpX2/SueNrA==} - cpu: [arm64] - os: [linux] - - '@img/sharp-libvips-linux-arm@1.0.5': - resolution: {integrity: sha512-gvcC4ACAOPRNATg/ov8/MnbxFDJqf/pDePbBnuBDcjsI8PssmjoKMAz4LtLaVi+OnSb5FK/yIOamqDwGmXW32g==} - cpu: [arm] - os: [linux] - - '@img/sharp-libvips-linux-s390x@1.0.4': - resolution: {integrity: sha512-u7Wz6ntiSSgGSGcjZ55im6uvTrOxSIS8/dgoVMoiGE9I6JAfU50yH5BoDlYA1tcuGS7g/QNtetJnxA6QEsCVTA==} - cpu: [s390x] - os: [linux] - - '@img/sharp-libvips-linux-x64@1.0.4': - resolution: {integrity: sha512-MmWmQ3iPFZr0Iev+BAgVMb3ZyC4KeFc3jFxnNbEPas60e1cIfevbtuyf9nDGIzOaW9PdnDciJm+wFFaTlj5xYw==} - cpu: [x64] - os: [linux] - - '@img/sharp-libvips-linuxmusl-arm64@1.0.4': - resolution: {integrity: sha512-9Ti+BbTYDcsbp4wfYib8Ctm1ilkugkA/uscUn6UXK1ldpC1JjiXbLfFZtRlBhjPZ5o1NCLiDbg8fhUPKStHoTA==} - cpu: [arm64] - os: [linux] - - '@img/sharp-libvips-linuxmusl-x64@1.0.4': - resolution: {integrity: sha512-viYN1KX9m+/hGkJtvYYp+CCLgnJXwiQB39damAO7WMdKWlIhmYTfHjwSbQeUK/20vY154mwezd9HflVFM1wVSw==} - cpu: [x64] - os: [linux] - - '@img/sharp-linux-arm64@0.33.5': - resolution: {integrity: sha512-JMVv+AMRyGOHtO1RFBiJy/MBsgz0x4AWrT6QoEVVTyh1E39TrCUpTRI7mx9VksGX4awWASxqCYLCV4wBZHAYxA==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm64] - os: [linux] - - '@img/sharp-linux-arm@0.33.5': - resolution: {integrity: sha512-JTS1eldqZbJxjvKaAkxhZmBqPRGmxgu+qFKSInv8moZ2AmT5Yib3EQ1c6gp493HvrvV8QgdOXdyaIBrhvFhBMQ==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm] - os: [linux] - - '@img/sharp-linux-s390x@0.33.5': - resolution: {integrity: sha512-y/5PCd+mP4CA/sPDKl2961b+C9d+vPAveS33s6Z3zfASk2j5upL6fXVPZi7ztePZ5CuH+1kW8JtvxgbuXHRa4Q==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [s390x] - os: [linux] - - '@img/sharp-linux-x64@0.33.5': - resolution: {integrity: sha512-opC+Ok5pRNAzuvq1AG0ar+1owsu842/Ab+4qvU879ippJBHvyY5n2mxF1izXqkPYlGuP/M556uh53jRLJmzTWA==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [x64] - os: [linux] - - '@img/sharp-linuxmusl-arm64@0.33.5': - resolution: {integrity: sha512-XrHMZwGQGvJg2V/oRSUfSAfjfPxO+4DkiRh6p2AFjLQztWUuY/o8Mq0eMQVIY7HJ1CDQUJlxGGZRw1a5bqmd1g==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm64] - os: [linux] - - '@img/sharp-linuxmusl-x64@0.33.5': - resolution: {integrity: sha512-WT+d/cgqKkkKySYmqoZ8y3pxx7lx9vVejxW/W4DOFMYVSkErR+w7mf2u8m/y4+xHe7yY9DAXQMWQhpnMuFfScw==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [x64] - os: [linux] - - '@img/sharp-wasm32@0.33.5': - resolution: {integrity: sha512-ykUW4LVGaMcU9lu9thv85CbRMAwfeadCJHRsg2GmeRa/cJxsVY9Rbd57JcMxBkKHag5U/x7TSBpScF4U8ElVzg==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [wasm32] - - '@img/sharp-win32-ia32@0.33.5': - resolution: {integrity: sha512-T36PblLaTwuVJ/zw/LaH0PdZkRz5rd3SmMHX8GSmR7vtNSP5Z6bQkExdSK7xGWyxLw4sUknBuugTelgw2faBbQ==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [ia32] - os: [win32] - - '@img/sharp-win32-x64@0.33.5': - resolution: {integrity: sha512-MpY/o8/8kj+EcnxwvrP4aTJSWw/aZ7JIGR4aBeZkZw5B7/Jn+tY9/VNwtcoGmdT7GfggGIU4kygOMSbYnOrAbg==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [x64] - os: [win32] - - '@isaacs/cliui@8.0.2': - resolution: {integrity: sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==} - engines: {node: '>=12'} - - '@jridgewell/gen-mapping@0.3.5': - resolution: {integrity: sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==} - engines: {node: '>=6.0.0'} - - '@jridgewell/resolve-uri@3.1.2': - resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==} - engines: {node: '>=6.0.0'} - - '@jridgewell/set-array@1.2.1': - resolution: {integrity: sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==} - engines: {node: '>=6.0.0'} - - '@jridgewell/source-map@0.3.6': - resolution: {integrity: sha512-1ZJTZebgqllO79ue2bm3rIGud/bOe0pP5BjSRCRxxYkEZS8STV7zN84UBbiYu7jy+eCKSnVIUgoWWE/tt+shMQ==} - - '@jridgewell/sourcemap-codec@1.5.0': - resolution: {integrity: sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==} - - '@jridgewell/trace-mapping@0.3.25': - resolution: {integrity: sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==} - - '@jridgewell/trace-mapping@0.3.9': - resolution: {integrity: sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==} - - '@js-sdsl/ordered-map@4.4.2': - resolution: {integrity: sha512-iUKgm52T8HOE/makSxjqoWhe95ZJA1/G1sYsGev2JDKUSS14KAgg1LHb+Ba+IPow0xflbnSkOsZcO08C7w1gYw==} - - '@jsdevtools/ono@7.1.3': - resolution: {integrity: sha512-4JQNk+3mVzK3xh2rqd6RB4J46qUR19azEHBneZyTZM+c456qOrbbM/5xcR8huNCCcbVt7+UmizG6GuUvPvKUYg==} - - '@jsonjoy.com/base64@1.1.2': - resolution: {integrity: sha512-q6XAnWQDIMA3+FTiOYajoYqySkO+JSat0ytXGSuRdq9uXE7o92gzuQwQM14xaCRlBLGq3v5miDGC4vkVTn54xA==} - engines: {node: '>=10.0'} - peerDependencies: - tslib: '2' - - '@jsonjoy.com/json-pack@1.1.0': - resolution: {integrity: sha512-zlQONA+msXPPwHWZMKFVS78ewFczIll5lXiVPwFPCZUsrOKdxc2AvxU1HoNBmMRhqDZUR9HkC3UOm+6pME6Xsg==} - engines: {node: '>=10.0'} - peerDependencies: - tslib: '2' - - '@jsonjoy.com/util@1.3.0': - resolution: {integrity: sha512-Cebt4Vk7k1xHy87kHY7KSPLT77A7Ev7IfOblyLZhtYEhrdQ6fX4EoLq3xOQ3O/DRMEh2ok5nyC180E+ABS8Wmw==} - engines: {node: '>=10.0'} - peerDependencies: - tslib: '2' - - '@lukeed/ms@2.0.2': - resolution: {integrity: sha512-9I2Zn6+NJLfaGoz9jN3lpwDgAYvfGeNYdbAIjJOqzs4Tpc+VU3Jqq4IofSUBKajiDS8k9fZIg18/z13mpk1bsA==} - engines: {node: '>=8'} - - '@noble/hashes@1.5.0': - resolution: {integrity: sha512-1j6kQFb7QRru7eKN3ZDvRcP13rugwdxZqCjbiAVZfIJwgj2A65UmT4TgARXGlXgnRkORLTDTrO19ZErt7+QXgA==} - engines: {node: ^14.21.3 || >=16} - - '@nodelib/fs.scandir@2.1.5': - resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==} - engines: {node: '>= 8'} - - '@nodelib/fs.stat@2.0.5': - resolution: {integrity: sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==} - engines: {node: '>= 8'} - - '@nodelib/fs.walk@1.2.8': - resolution: {integrity: sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==} - engines: {node: '>= 8'} - - '@paralleldrive/cuid2@2.2.2': - resolution: {integrity: sha512-ZOBkgDwEdoYVlSeRbYYXs0S9MejQofiVYoTbKzy/6GQa39/q5tQU2IX46+shYnUkpEl3wc+J6wRlar7r2EK2xA==} - - '@pkgjs/parseargs@0.11.0': - resolution: {integrity: sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==} - engines: {node: '>=14'} - - '@protobufjs/aspromise@1.1.2': - resolution: {integrity: sha512-j+gKExEuLmKwvz3OgROXtrJ2UG2x8Ch2YZUxahh+s1F2HZ+wAceUNLkvy6zKCPVRkU++ZWQrdxsUeQXmcg4uoQ==} - - '@protobufjs/base64@1.1.2': - resolution: {integrity: sha512-AZkcAA5vnN/v4PDqKyMR5lx7hZttPDgClv83E//FMNhR2TMcLUhfRUBHCmSl0oi9zMgDDqRUJkSxO3wm85+XLg==} - - '@protobufjs/codegen@2.0.4': - resolution: {integrity: sha512-YyFaikqM5sH0ziFZCN3xDC7zeGaB/d0IUb9CATugHWbd1FRFwWwt4ld4OYMPWu5a3Xe01mGAULCdqhMlPl29Jg==} - - '@protobufjs/eventemitter@1.1.0': - resolution: {integrity: sha512-j9ednRT81vYJ9OfVuXG6ERSTdEL1xVsNgqpkxMsbIabzSo3goCjDIveeGv5d03om39ML71RdmrGNjG5SReBP/Q==} - - '@protobufjs/fetch@1.1.0': - resolution: {integrity: sha512-lljVXpqXebpsijW71PZaCYeIcE5on1w5DlQy5WH6GLbFryLUrBD4932W/E2BSpfRJWseIL4v/KPgBFxDOIdKpQ==} - - '@protobufjs/float@1.0.2': - resolution: {integrity: sha512-Ddb+kVXlXst9d+R9PfTIxh1EdNkgoRe5tOX6t01f1lYWOvJnSPDBlG241QLzcyPdoNTsblLUdujGSE4RzrTZGQ==} - - '@protobufjs/inquire@1.1.0': - resolution: {integrity: sha512-kdSefcPdruJiFMVSbn801t4vFK7KB/5gd2fYvrxhuJYg8ILrmn9SKSX2tZdV6V+ksulWqS7aXjBcRXl3wHoD9Q==} - - '@protobufjs/path@1.1.2': - resolution: {integrity: sha512-6JOcJ5Tm08dOHAbdR3GrvP+yUUfkjG5ePsHYczMFLq3ZmMkAD98cDgcT2iA1lJ9NVwFd4tH/iSSoe44YWkltEA==} - - '@protobufjs/pool@1.1.0': - resolution: {integrity: sha512-0kELaGSIDBKvcgS4zkjz1PeddatrjYcmMWOlAuAPwAeccUrPHdUqo/J6LiymHHEiJT5NrF1UVwxY14f+fy4WQw==} - - '@protobufjs/utf8@1.1.0': - resolution: {integrity: sha512-Vvn3zZrhQZkkBE8LSuW3em98c0FwgO4nxzv6OdSxPKJIEKY2bGbHn+mhGIPerzI4twdxaP8/0+06HBpwf345Lw==} - - '@rollup/rollup-android-arm-eabi@4.21.2': - resolution: {integrity: sha512-fSuPrt0ZO8uXeS+xP3b+yYTCBUd05MoSp2N/MFOgjhhUhMmchXlpTQrTpI8T+YAwAQuK7MafsCOxW7VrPMrJcg==} - cpu: [arm] - os: [android] - - '@rollup/rollup-android-arm64@4.21.2': - resolution: {integrity: sha512-xGU5ZQmPlsjQS6tzTTGwMsnKUtu0WVbl0hYpTPauvbRAnmIvpInhJtgjj3mcuJpEiuUw4v1s4BimkdfDWlh7gA==} - cpu: [arm64] - os: [android] - - '@rollup/rollup-darwin-arm64@4.21.2': - resolution: {integrity: sha512-99AhQ3/ZMxU7jw34Sq8brzXqWH/bMnf7ZVhvLk9QU2cOepbQSVTns6qoErJmSiAvU3InRqC2RRZ5ovh1KN0d0Q==} - cpu: [arm64] - os: [darwin] - - '@rollup/rollup-darwin-x64@4.21.2': - resolution: {integrity: sha512-ZbRaUvw2iN/y37x6dY50D8m2BnDbBjlnMPotDi/qITMJ4sIxNY33HArjikDyakhSv0+ybdUxhWxE6kTI4oX26w==} - cpu: [x64] - os: [darwin] - - '@rollup/rollup-linux-arm-gnueabihf@4.21.2': - resolution: {integrity: sha512-ztRJJMiE8nnU1YFcdbd9BcH6bGWG1z+jP+IPW2oDUAPxPjo9dverIOyXz76m6IPA6udEL12reYeLojzW2cYL7w==} - cpu: [arm] - os: [linux] - - '@rollup/rollup-linux-arm-musleabihf@4.21.2': - resolution: {integrity: sha512-flOcGHDZajGKYpLV0JNc0VFH361M7rnV1ee+NTeC/BQQ1/0pllYcFmxpagltANYt8FYf9+kL6RSk80Ziwyhr7w==} - cpu: [arm] - os: [linux] - - '@rollup/rollup-linux-arm64-gnu@4.21.2': - resolution: {integrity: sha512-69CF19Kp3TdMopyteO/LJbWufOzqqXzkrv4L2sP8kfMaAQ6iwky7NoXTp7bD6/irKgknDKM0P9E/1l5XxVQAhw==} - cpu: [arm64] - os: [linux] - - '@rollup/rollup-linux-arm64-musl@4.21.2': - resolution: {integrity: sha512-48pD/fJkTiHAZTnZwR0VzHrao70/4MlzJrq0ZsILjLW/Ab/1XlVUStYyGt7tdyIiVSlGZbnliqmult/QGA2O2w==} - cpu: [arm64] - os: [linux] - - '@rollup/rollup-linux-powerpc64le-gnu@4.21.2': - resolution: {integrity: sha512-cZdyuInj0ofc7mAQpKcPR2a2iu4YM4FQfuUzCVA2u4HI95lCwzjoPtdWjdpDKyHxI0UO82bLDoOaLfpZ/wviyQ==} - cpu: [ppc64] - os: [linux] - - '@rollup/rollup-linux-riscv64-gnu@4.21.2': - resolution: {integrity: sha512-RL56JMT6NwQ0lXIQmMIWr1SW28z4E4pOhRRNqwWZeXpRlykRIlEpSWdsgNWJbYBEWD84eocjSGDu/XxbYeCmwg==} - cpu: [riscv64] - os: [linux] - - '@rollup/rollup-linux-s390x-gnu@4.21.2': - resolution: {integrity: sha512-PMxkrWS9z38bCr3rWvDFVGD6sFeZJw4iQlhrup7ReGmfn7Oukrr/zweLhYX6v2/8J6Cep9IEA/SmjXjCmSbrMQ==} - cpu: [s390x] - os: [linux] - - '@rollup/rollup-linux-x64-gnu@4.21.2': - resolution: {integrity: sha512-B90tYAUoLhU22olrafY3JQCFLnT3NglazdwkHyxNDYF/zAxJt5fJUB/yBoWFoIQ7SQj+KLe3iL4BhOMa9fzgpw==} - cpu: [x64] - os: [linux] - - '@rollup/rollup-linux-x64-musl@4.21.2': - resolution: {integrity: sha512-7twFizNXudESmC9oneLGIUmoHiiLppz/Xs5uJQ4ShvE6234K0VB1/aJYU3f/4g7PhssLGKBVCC37uRkkOi8wjg==} - cpu: [x64] - os: [linux] - - '@rollup/rollup-win32-arm64-msvc@4.21.2': - resolution: {integrity: sha512-9rRero0E7qTeYf6+rFh3AErTNU1VCQg2mn7CQcI44vNUWM9Ze7MSRS/9RFuSsox+vstRt97+x3sOhEey024FRQ==} - cpu: [arm64] - os: [win32] - - '@rollup/rollup-win32-ia32-msvc@4.21.2': - resolution: {integrity: sha512-5rA4vjlqgrpbFVVHX3qkrCo/fZTj1q0Xxpg+Z7yIo3J2AilW7t2+n6Q8Jrx+4MrYpAnjttTYF8rr7bP46BPzRw==} - cpu: [ia32] - os: [win32] - - '@rollup/rollup-win32-x64-msvc@4.21.2': - resolution: {integrity: sha512-6UUxd0+SKomjdzuAcp+HAmxw1FlGBnl1v2yEPSabtx4lBfdXHDVsW7+lQkgz9cNFJGY3AWR7+V8P5BqkD9L9nA==} - cpu: [x64] - os: [win32] - - '@selderee/plugin-htmlparser2@0.11.0': - resolution: {integrity: sha512-P33hHGdldxGabLFjPPpaTxVolMrzrcegejx+0GxjrIb9Zv48D8yAIA/QTDR2dFl7Uz7urX8aX6+5bCZslr+gWQ==} - - '@sinonjs/commons@2.0.0': - resolution: {integrity: sha512-uLa0j859mMrg2slwQYdO/AkrOfmH+X6LTVmNTS9CqexuE2IvVORIkSpJLqePAbEnKJ77aMmCwr1NUZ57120Xcg==} - - '@sinonjs/commons@3.0.1': - resolution: {integrity: sha512-K3mCHKQ9sVh8o1C9cxkwxaOmXoAMlDxC1mYyHrjqOWEcBjYr76t96zL2zlj5dUGZ3HSw240X1qgH3Mjf1yJWpQ==} - - '@sinonjs/fake-timers@11.3.1': - resolution: {integrity: sha512-EVJO7nW5M/F5Tur0Rf2z/QoMo+1Ia963RiMtapiQrEWvY0iBUvADo8Beegwjpnle5BHkyHuoxSTW3jF43H1XRA==} - - '@sinonjs/samsam@8.0.0': - resolution: {integrity: sha512-Bp8KUVlLp8ibJZrnvq2foVhP0IVX2CIprMJPK0vqGqgrDa0OHVKeZyBykqskkrdxV6yKBPmGasO8LVjAKR3Gew==} - - '@sinonjs/text-encoding@0.7.3': - resolution: {integrity: sha512-DE427ROAphMQzU4ENbliGYrBSYPXF+TtLg9S8vzeA+OF4ZKzoDdzfL8sxuMUGS/lgRhM6j1URSk9ghf7Xo1tyA==} - - '@smithy/abort-controller@3.1.1': - resolution: {integrity: sha512-MBJBiidoe+0cTFhyxT8g+9g7CeVccLM0IOKKUMCNQ1CNMJ/eIfoo0RTfVrXOONEI1UCN1W+zkiHSbzUNE9dZtQ==} - engines: {node: '>=16.0.0'} - - '@smithy/chunked-blob-reader-native@3.0.0': - resolution: {integrity: sha512-VDkpCYW+peSuM4zJip5WDfqvg2Mo/e8yxOv3VF1m11y7B8KKMKVFtmZWDe36Fvk8rGuWrPZHHXZ7rR7uM5yWyg==} - - '@smithy/chunked-blob-reader@3.0.0': - resolution: {integrity: sha512-sbnURCwjF0gSToGlsBiAmd1lRCmSn72nu9axfJu5lIx6RUEgHu6GwTMbqCdhQSi0Pumcm5vFxsi9XWXb2mTaoA==} - - '@smithy/config-resolver@3.0.5': - resolution: {integrity: sha512-SkW5LxfkSI1bUC74OtfBbdz+grQXYiPYolyu8VfpLIjEoN/sHVBlLeGXMQ1vX4ejkgfv6sxVbQJ32yF2cl1veA==} - engines: {node: '>=16.0.0'} - - '@smithy/core@2.4.0': - resolution: {integrity: sha512-cHXq+FneIF/KJbt4q4pjN186+Jf4ZB0ZOqEaZMBhT79srEyGDDBV31NqBRBjazz8ppQ1bJbDJMY9ba5wKFV36w==} - engines: {node: '>=16.0.0'} - - '@smithy/credential-provider-imds@3.2.0': - resolution: {integrity: sha512-0SCIzgd8LYZ9EJxUjLXBmEKSZR/P/w6l7Rz/pab9culE/RWuqelAKGJvn5qUOl8BgX8Yj5HWM50A5hiB/RzsgA==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-codec@3.1.2': - resolution: {integrity: sha512-0mBcu49JWt4MXhrhRAlxASNy0IjDRFU+aWNDRal9OtUJvJNiwDuyKMUONSOjLjSCeGwZaE0wOErdqULer8r7yw==} - - '@smithy/eventstream-serde-browser@3.0.6': - resolution: {integrity: sha512-2hM54UWQUOrki4BtsUI1WzmD13/SeaqT/AB3EUJKbcver/WgKNaiJ5y5F5XXuVe6UekffVzuUDrBZVAA3AWRpQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-config-resolver@3.0.3': - resolution: {integrity: sha512-NVTYjOuYpGfrN/VbRQgn31x73KDLfCXCsFdad8DiIc3IcdxL+dYA9zEQPyOP7Fy2QL8CPy2WE4WCUD+ZsLNfaQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-node@3.0.5': - resolution: {integrity: sha512-+upXvnHNyZP095s11jF5dhGw/Ihzqwl5G+/KtMnoQOpdfC3B5HYCcDVG9EmgkhJMXJlM64PyN5gjJl0uXFQehQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-universal@3.0.5': - resolution: {integrity: sha512-5u/nXbyoh1s4QxrvNre9V6vfyoLWuiVvvd5TlZjGThIikc3G+uNiG9uOTCWweSRjv1asdDIWK7nOmN7le4RYHQ==} - engines: {node: '>=16.0.0'} - - '@smithy/fetch-http-handler@3.2.4': - resolution: {integrity: sha512-kBprh5Gs5h7ug4nBWZi1FZthdqSM+T7zMmsZxx0IBvWUn7dK3diz2SHn7Bs4dQGFDk8plDv375gzenDoNwrXjg==} - - '@smithy/hash-blob-browser@3.1.2': - resolution: {integrity: sha512-hAbfqN2UbISltakCC2TP0kx4LqXBttEv2MqSPE98gVuDFMf05lU+TpC41QtqGP3Ff5A3GwZMPfKnEy0VmEUpmg==} - - '@smithy/hash-node@3.0.3': - resolution: {integrity: sha512-2ctBXpPMG+B3BtWSGNnKELJ7SH9e4TNefJS0cd2eSkOOROeBnnVBnAy9LtJ8tY4vUEoe55N4CNPxzbWvR39iBw==} - engines: {node: '>=16.0.0'} - - '@smithy/hash-stream-node@3.1.2': - resolution: {integrity: sha512-PBgDMeEdDzi6JxKwbfBtwQG9eT9cVwsf0dZzLXoJF4sHKHs5HEo/3lJWpn6jibfJwT34I1EBXpBnZE8AxAft6g==} - engines: {node: '>=16.0.0'} - - '@smithy/invalid-dependency@3.0.3': - resolution: {integrity: sha512-ID1eL/zpDULmHJbflb864k72/SNOZCADRc9i7Exq3RUNJw6raWUSlFEQ+3PX3EYs++bTxZB2dE9mEHTQLv61tw==} - - '@smithy/is-array-buffer@2.2.0': - resolution: {integrity: sha512-GGP3O9QFD24uGeAXYUjwSTXARoqpZykHadOmA8G5vfJPK0/DC67qa//0qvqrJzL1xc8WQWX7/yc7fwudjPHPhA==} - engines: {node: '>=14.0.0'} - - '@smithy/is-array-buffer@3.0.0': - resolution: {integrity: sha512-+Fsu6Q6C4RSJiy81Y8eApjEB5gVtM+oFKTffg+jSuwtvomJJrhUJBu2zS8wjXSgH/g1MKEWrzyChTBe6clb5FQ==} - engines: {node: '>=16.0.0'} - - '@smithy/md5-js@3.0.3': - resolution: {integrity: sha512-O/SAkGVwpWmelpj/8yDtsaVe6sINHLB1q8YE/+ZQbDxIw3SRLbTZuRaI10K12sVoENdnHqzPp5i3/H+BcZ3m3Q==} - - '@smithy/middleware-content-length@3.0.5': - resolution: {integrity: sha512-ILEzC2eyxx6ncej3zZSwMpB5RJ0zuqH7eMptxC4KN3f+v9bqT8ohssKbhNR78k/2tWW+KS5Spw+tbPF4Ejyqvw==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-endpoint@3.1.0': - resolution: {integrity: sha512-5y5aiKCEwg9TDPB4yFE7H6tYvGFf1OJHNczeY10/EFF8Ir8jZbNntQJxMWNfeQjC1mxPsaQ6mR9cvQbf+0YeMw==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-retry@3.0.15': - resolution: {integrity: sha512-iTMedvNt1ApdvkaoE8aSDuwaoc+BhvHqttbA/FO4Ty+y/S5hW6Ci/CTScG7vam4RYJWZxdTElc3MEfHRVH6cgQ==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-serde@3.0.3': - resolution: {integrity: sha512-puUbyJQBcg9eSErFXjKNiGILJGtiqmuuNKEYNYfUD57fUl4i9+mfmThtQhvFXU0hCVG0iEJhvQUipUf+/SsFdA==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-stack@3.0.3': - resolution: {integrity: sha512-r4klY9nFudB0r9UdSMaGSyjyQK5adUyPnQN/ZM6M75phTxOdnc/AhpvGD1fQUvgmqjQEBGCwpnPbDm8pH5PapA==} - engines: {node: '>=16.0.0'} - - '@smithy/node-config-provider@3.1.4': - resolution: {integrity: sha512-YvnElQy8HR4vDcAjoy7Xkx9YT8xZP4cBXcbJSgm/kxmiQu08DwUwj8rkGnyoJTpfl/3xYHH+d8zE+eHqoDCSdQ==} - engines: {node: '>=16.0.0'} - - '@smithy/node-http-handler@3.1.4': - resolution: {integrity: sha512-+UmxgixgOr/yLsUxcEKGH0fMNVteJFGkmRltYFHnBMlogyFdpzn2CwqWmxOrfJELhV34v0WSlaqG1UtE1uXlJg==} - engines: {node: '>=16.0.0'} - - '@smithy/property-provider@3.1.3': - resolution: {integrity: sha512-zahyOVR9Q4PEoguJ/NrFP4O7SMAfYO1HLhB18M+q+Z4KFd4V2obiMnlVoUFzFLSPeVt1POyNWneHHrZaTMoc/g==} - engines: {node: '>=16.0.0'} - - '@smithy/protocol-http@4.1.0': - resolution: {integrity: sha512-dPVoHYQ2wcHooGXg3LQisa1hH0e4y0pAddPMeeUPipI1tEOqL6A4N0/G7abeq+K8wrwSgjk4C0wnD1XZpJm5aA==} - engines: {node: '>=16.0.0'} - - '@smithy/querystring-builder@3.0.3': - resolution: {integrity: sha512-vyWckeUeesFKzCDaRwWLUA1Xym9McaA6XpFfAK5qI9DKJ4M33ooQGqvM4J+LalH4u/Dq9nFiC8U6Qn1qi0+9zw==} - engines: {node: '>=16.0.0'} - - '@smithy/querystring-parser@3.0.3': - resolution: {integrity: sha512-zahM1lQv2YjmznnfQsWbYojFe55l0SLG/988brlLv1i8z3dubloLF+75ATRsqPBboUXsW6I9CPGE5rQgLfY0vQ==} - engines: {node: '>=16.0.0'} - - '@smithy/service-error-classification@3.0.3': - resolution: {integrity: sha512-Jn39sSl8cim/VlkLsUhRFq/dKDnRUFlfRkvhOJaUbLBXUsLRLNf9WaxDv/z9BjuQ3A6k/qE8af1lsqcwm7+DaQ==} - engines: {node: '>=16.0.0'} - - '@smithy/shared-ini-file-loader@3.1.4': - resolution: {integrity: sha512-qMxS4hBGB8FY2GQqshcRUy1K6k8aBWP5vwm8qKkCT3A9K2dawUwOIJfqh9Yste/Bl0J2lzosVyrXDj68kLcHXQ==} - engines: {node: '>=16.0.0'} - - '@smithy/signature-v4@4.1.0': - resolution: {integrity: sha512-aRryp2XNZeRcOtuJoxjydO6QTaVhxx/vjaR+gx7ZjaFgrgPRyZ3HCTbfwqYj6ZWEBHkCSUfcaymKPURaByukag==} - engines: {node: '>=16.0.0'} - - '@smithy/smithy-client@3.2.0': - resolution: {integrity: sha512-pDbtxs8WOhJLJSeaF/eAbPgXg4VVYFlRcL/zoNYA5WbG3wBL06CHtBSg53ppkttDpAJ/hdiede+xApip1CwSLw==} - engines: {node: '>=16.0.0'} - - '@smithy/types@3.3.0': - resolution: {integrity: sha512-IxvBBCTFDHbVoK7zIxqA1ZOdc4QfM5HM7rGleCuHi7L1wnKv5Pn69xXJQ9hgxH60ZVygH9/JG0jRgtUncE3QUA==} - engines: {node: '>=16.0.0'} - - '@smithy/url-parser@3.0.3': - resolution: {integrity: sha512-pw3VtZtX2rg+s6HMs6/+u9+hu6oY6U7IohGhVNnjbgKy86wcIsSZwgHrFR+t67Uyxvp4Xz3p3kGXXIpTNisq8A==} - - '@smithy/util-base64@3.0.0': - resolution: {integrity: sha512-Kxvoh5Qtt0CDsfajiZOCpJxgtPHXOKwmM+Zy4waD43UoEMA+qPxxa98aE/7ZhdnBFZFXMOiBR5xbcaMhLtznQQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-body-length-browser@3.0.0': - resolution: {integrity: sha512-cbjJs2A1mLYmqmyVl80uoLTJhAcfzMOyPgjwAYusWKMdLeNtzmMz9YxNl3/jRLoxSS3wkqkf0jwNdtXWtyEBaQ==} - - '@smithy/util-body-length-node@3.0.0': - resolution: {integrity: sha512-Tj7pZ4bUloNUP6PzwhN7K386tmSmEET9QtQg0TgdNOnxhZvCssHji+oZTUIuzxECRfG8rdm2PMw2WCFs6eIYkA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-buffer-from@2.2.0': - resolution: {integrity: sha512-IJdWBbTcMQ6DA0gdNhh/BwrLkDR+ADW5Kr1aZmd4k3DIF6ezMV4R2NIAmT08wQJ3yUK82thHWmC/TnK/wpMMIA==} - engines: {node: '>=14.0.0'} - - '@smithy/util-buffer-from@3.0.0': - resolution: {integrity: sha512-aEOHCgq5RWFbP+UDPvPot26EJHjOC+bRgse5A8V3FSShqd5E5UN4qc7zkwsvJPPAVsf73QwYcHN1/gt/rtLwQA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-config-provider@3.0.0': - resolution: {integrity: sha512-pbjk4s0fwq3Di/ANL+rCvJMKM5bzAQdE5S/6RL5NXgMExFAi6UgQMPOm5yPaIWPpr+EOXKXRonJ3FoxKf4mCJQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-defaults-mode-browser@3.0.15': - resolution: {integrity: sha512-FZ4Psa3vjp8kOXcd3HJOiDPBCWtiilLl57r0cnNtq/Ga9RSDrM5ERL6xt+tO43+2af6Pn5Yp92x2n5vPuduNfg==} - engines: {node: '>= 10.0.0'} - - '@smithy/util-defaults-mode-node@3.0.15': - resolution: {integrity: sha512-KSyAAx2q6d0t6f/S4XB2+3+6aQacm3aLMhs9aLMqn18uYGUepbdssfogW5JQZpc6lXNBnp0tEnR5e9CEKmEd7A==} - engines: {node: '>= 10.0.0'} - - '@smithy/util-endpoints@2.0.5': - resolution: {integrity: sha512-ReQP0BWihIE68OAblC/WQmDD40Gx+QY1Ez8mTdFMXpmjfxSyz2fVQu3A4zXRfQU9sZXtewk3GmhfOHswvX+eNg==} - engines: {node: '>=16.0.0'} - - '@smithy/util-hex-encoding@3.0.0': - resolution: {integrity: sha512-eFndh1WEK5YMUYvy3lPlVmYY/fZcQE1D8oSf41Id2vCeIkKJXPcYDCZD+4+xViI6b1XSd7tE+s5AmXzz5ilabQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-middleware@3.0.3': - resolution: {integrity: sha512-l+StyYYK/eO3DlVPbU+4Bi06Jjal+PFLSMmlWM1BEwyLxZ3aKkf1ROnoIakfaA7mC6uw3ny7JBkau4Yc+5zfWw==} - engines: {node: '>=16.0.0'} - - '@smithy/util-retry@3.0.3': - resolution: {integrity: sha512-AFw+hjpbtVApzpNDhbjNG5NA3kyoMs7vx0gsgmlJF4s+yz1Zlepde7J58zpIRIsdjc+emhpAITxA88qLkPF26w==} - engines: {node: '>=16.0.0'} - - '@smithy/util-stream@3.1.3': - resolution: {integrity: sha512-FIv/bRhIlAxC0U7xM1BCnF2aDRPq0UaelqBHkM2lsCp26mcBbgI0tCVTv+jGdsQLUmAMybua/bjDsSu8RQHbmw==} - engines: {node: '>=16.0.0'} - - '@smithy/util-uri-escape@3.0.0': - resolution: {integrity: sha512-LqR7qYLgZTD7nWLBecUi4aqolw8Mhza9ArpNEQ881MJJIU2sE5iHCK6TdyqqzcDLy0OPe10IY4T8ctVdtynubg==} - engines: {node: '>=16.0.0'} - - '@smithy/util-utf8@2.3.0': - resolution: {integrity: sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A==} - engines: {node: '>=14.0.0'} - - '@smithy/util-utf8@3.0.0': - resolution: {integrity: sha512-rUeT12bxFnplYDe815GXbq/oixEGHfRFFtcTF3YdDi/JaENIM6aSYYLJydG83UNzLXeRI5K8abYd/8Sp/QM0kA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-waiter@3.1.2': - resolution: {integrity: sha512-4pP0EV3iTsexDx+8PPGAKCQpd/6hsQBaQhqWzU4hqKPHN5epPsxKbvUTIiYIHTxaKt6/kEaqPBpu/ufvfbrRzw==} - engines: {node: '>=16.0.0'} - - '@swc/core-darwin-arm64@1.7.23': - resolution: {integrity: sha512-yyOHPfti6yKlQulfVWMt7BVKst+SyEZYCWuQSGMn1KgmNCH/bYufRWfQXIhkGSj44ZkEepJmsJ8tDyIb4k5WyA==} - engines: {node: '>=10'} - cpu: [arm64] - os: [darwin] - - '@swc/core-darwin-x64@1.7.23': - resolution: {integrity: sha512-GzqHwQ0Y1VyjdI/bBKFX2GKm5HD3PIB6OhuAQtWZMTtEr2yIrlT0YK2T+XKh7oIg31JwxGBeQdBk3KTI7DARmQ==} - engines: {node: '>=10'} - cpu: [x64] - os: [darwin] - - '@swc/core-linux-arm-gnueabihf@1.7.23': - resolution: {integrity: sha512-qwX4gB41OS6/OZkHcpTqLFGsdmvoZyffnJIlgB/kZKwH3lfeJWzv6vx57zXtNpM/t7GoQEe0VZUVdmNjxSxBZw==} - engines: {node: '>=10'} - cpu: [arm] - os: [linux] - - '@swc/core-linux-arm64-gnu@1.7.23': - resolution: {integrity: sha512-TsrbUZdMaUwzI7+g/8rHPLWbntMKYSu5Bn5IBSqVKPeyqaXxNnlIUnWXgXcUcRAc+T+Y8ADfr7EiFz9iz5DuSA==} - engines: {node: '>=10'} - cpu: [arm64] - os: [linux] - - '@swc/core-linux-arm64-musl@1.7.23': - resolution: {integrity: sha512-JEdtwdthazKq4PBz53KSubwwK8MvqODAihGSAzc8u3Unq4ojcvaS8b0CwLBeD+kTQ78HpxOXTt3DsFIxpgaCAA==} - engines: {node: '>=10'} - cpu: [arm64] - os: [linux] - - '@swc/core-linux-x64-gnu@1.7.23': - resolution: {integrity: sha512-V51gFPWaVAHbI1yg9ahsoya3aB4uawye3SZ5uQWgcP7wdCdiv60dw4F5nuPJf5Z1oXD3U/BslXuamv8Oh9vXqQ==} - engines: {node: '>=10'} - cpu: [x64] - os: [linux] - - '@swc/core-linux-x64-musl@1.7.23': - resolution: {integrity: sha512-BBqQi4+UdeRqag3yM4IJjaHG4yc1o3l9ksENHToE0o/u2DT0FY5+K/DiYGZLC1JHbSFzNqRCYsa7DIzRtZ0A1A==} - engines: {node: '>=10'} - cpu: [x64] - os: [linux] - - '@swc/core-win32-arm64-msvc@1.7.23': - resolution: {integrity: sha512-JPk6pvCKncL6bXG7p+NLZf8PWx4FakVvKNdwGeMrYunb+yk1IZf7qf9LJk8+GDGF5QviDXPs8opZrTrfsW80fA==} - engines: {node: '>=10'} - cpu: [arm64] - os: [win32] - - '@swc/core-win32-ia32-msvc@1.7.23': - resolution: {integrity: sha512-2Whxi8d+bLQBzJcQ5qYPHlk02YYVGsMVav0fWk+FnX2z1QRREIu1L1xvrpi7gBpjXp6BIU40ya8GiKeekNT2bg==} - engines: {node: '>=10'} - cpu: [ia32] - os: [win32] - - '@swc/core-win32-x64-msvc@1.7.23': - resolution: {integrity: sha512-82fARk4/yJ40kwWKY/gdKDisPdtgJE9jgpl/vkNG3alyJxrCzuNM7+CtiKoYbXLeqM8GQTS3wlvCaJu9oQ8dag==} - engines: {node: '>=10'} - cpu: [x64] - os: [win32] - - '@swc/core@1.7.23': - resolution: {integrity: sha512-VDNkpDvDlreGh2E3tlDj8B3piiuLhhQA/7rIVZpiLUvG1YpucAa6N7iDXA7Gc/+Hah8spaCg/qvEaBkCmcIYCQ==} - engines: {node: '>=10'} - peerDependencies: - '@swc/helpers': '*' - peerDependenciesMeta: - '@swc/helpers': - optional: true - - '@swc/counter@0.1.3': - resolution: {integrity: sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ==} - - '@swc/types@0.1.12': - resolution: {integrity: sha512-wBJA+SdtkbFhHjTMYH+dEH1y4VpfGdAc2Kw/LK09i9bXd/K6j6PkDcFCEzb6iVfZMkPRrl/q0e3toqTAJdkIVA==} - - '@temporalio/activity@1.11.1': - resolution: {integrity: sha512-3nfkYBvWzHeAd/0a7M4GdfrUcDs6dA6L5bvGRJQYxNZ0h1X7Shd/FXQ5Yvfqz2LAa4NKlgZ4qcjfnUiulJkVyA==} - - '@temporalio/client@1.11.1': - resolution: {integrity: sha512-1cEh6y7gB1bV/dZbSLL8TsEWM9/glj4j46V8sOS1B3ykoK2+5KaBvkUBKeS/SVH8puqo1zaWbaYEWD8kajg6/g==} - - '@temporalio/common@1.11.1': - resolution: {integrity: sha512-8fQwmYHMakSbqMX26FNfiLwBwVwsMNbGmVfGaFOqy72QZuUiXxgmImXQhK6ir1ieeH8bLyY5laAosB4W6VPBAA==} - - '@temporalio/core-bridge@1.11.1': - resolution: {integrity: sha512-zgBHPdzQqOHyetVR12k4L1vrGlrwFFsMZ1iNaSJoaf8M4C5whYlzv13nUvXThZkaN06/xQCop6L02XkmhxkzYA==} - - '@temporalio/proto@1.11.1': - resolution: {integrity: sha512-qTNSsU88IOJbv6lhWyW0qAHAYg4euEW/NNPgAp5ohfPkqaw+/j51QABO9E6m/KQwMZ6x43z2JjQRS37aC6Psiw==} - - '@temporalio/worker@1.11.1': - resolution: {integrity: sha512-PiYbPLUH0UIakVtcZzJ4064Z3bOYrk+pOL8w8YxBlE85IQKQG6tVm9yrqd/XpC9lk0JOJhdw6KLcm64kDRrryw==} - engines: {node: '>= 16.0.0'} - - '@temporalio/workflow@1.11.1': - resolution: {integrity: sha512-aTfoHPc6O/7UioOSe+PHmpzLZveG2qn0VI/snlclPFs8VO8r5hcbXb5jYeuzCi26mm7Z+Dxmf2hCK77tJlPeKg==} - - '@tsconfig/node10@1.0.11': - resolution: {integrity: sha512-DcRjDCujK/kCk/cUe8Xz8ZSpm8mS3mNNpta+jGCA6USEDfktlNvm1+IuZ9eTcDbNk41BHwpHHeW+N1lKCz4zOw==} - - '@tsconfig/node12@1.0.11': - resolution: {integrity: sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==} - - '@tsconfig/node14@1.0.3': - resolution: {integrity: sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==} - - '@tsconfig/node16@1.0.4': - resolution: {integrity: sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==} - - '@tsconfig/node20@20.1.4': - resolution: {integrity: sha512-sqgsT69YFeLWf5NtJ4Xq/xAF8p4ZQHlmGW74Nu2tD4+g5fAsposc4ZfaaPixVu4y01BEiDCWLRDCvDM5JOsRxg==} - - '@types/chai@4.3.19': - resolution: {integrity: sha512-2hHHvQBVE2FiSK4eN0Br6snX9MtolHaTo/batnLjlGRhoQzlCL61iVpxoqO7SfFyOw+P/pwv+0zNHzKoGWz9Cw==} - - '@types/cheerio@0.22.35': - resolution: {integrity: sha512-yD57BchKRvTV+JD53UZ6PD8KWY5g5rvvMLRnZR3EQBCZXiDT/HR+pKpMzFGlWNhFrXlo7VPZXtKvIEwZkAWOIA==} - - '@types/debug@4.1.12': - resolution: {integrity: sha512-vIChWdVG3LG1SMxEvI/AK+FWJthlrqlTu7fbrlywTkkaONwk/UAGaULXRlf8vkzFBLVm0zkMdCquhL5aOjhXPQ==} - - '@types/estree@1.0.5': - resolution: {integrity: sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==} - - '@types/fluent-ffmpeg@2.1.26': - resolution: {integrity: sha512-0JVF3wdQG+pN0ImwWD0bNgJiKF2OHg/7CDBHw5UIbRTvlnkgGHK6V5doE54ltvhud4o31/dEiHm23CAlxFiUQg==} - - '@types/imapflow@1.0.19': - resolution: {integrity: sha512-U48VZXe4XYhS3AFNI+4ougW8OXI4VaKjyF0nGXgVzIN8SPs9lh2LPNRM0HmIM+hUTw60l7MHgbQO8hsf+Q4U5w==} - - '@types/json-schema@7.0.15': - resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==} - - '@types/luxon@3.4.2': - resolution: {integrity: sha512-TifLZlFudklWlMBfhubvgqTXRzLDI5pCbGa4P8a3wPyUQSW+1xQ5eDsreP9DWHX3tjq1ke96uYG/nwundroWcA==} - - '@types/mailparser@3.4.4': - resolution: {integrity: sha512-C6Znp2QVS25JqtuPyxj38Qh+QoFcLycdxsvcc6IZCGekhaMBzbdTXzwGzhGoYb3TfKu8IRCNV0sV1o3Od97cEQ==} - - '@types/mocha@10.0.7': - resolution: {integrity: sha512-GN8yJ1mNTcFcah/wKEFIJckJx9iJLoMSzWcfRRuxz/Jk+U6KQNnml+etbtxFK8lPjzOw3zp4Ha/kjSst9fsHYw==} - - '@types/ms@0.7.34': - resolution: {integrity: sha512-nG96G3Wp6acyAgJqGasjODb+acrI7KltPiRxzHPXnP3NgI28bpQDRv53olbqGXbfcgF5aiiHmO3xpwEpS5Ld9g==} - - '@types/node@18.19.48': - resolution: {integrity: sha512-7WevbG4ekUcRQSZzOwxWgi5dZmTak7FaxXDoW7xVxPBmKx1rTzfmRLkeCgJzcbBnOV2dkhAPc8cCeT6agocpjg==} - - '@types/node@20.16.3': - resolution: {integrity: sha512-/wdGiWRkMOm53gAsSyFMXFZHbVg7C6CbkrzHNpaHoYfsUWPg7m6ZRKtvQjgvQ9i8WT540a3ydRlRQbxjY30XxQ==} - - '@types/node@22.5.2': - resolution: {integrity: sha512-acJsPTEqYqulZS/Yp/S3GgeE6GZ0qYODUR8aVr/DkhHQ8l9nd4j5x1/ZJy9/gHrRlFMqkO6i0I3E27Alu4jjPg==} - - '@types/pg@8.11.8': - resolution: {integrity: sha512-IqpCf8/569txXN/HoP5i1LjXfKZWL76Yr2R77xgeIICUbAYHeoaEZFhYHo2uDftecLWrTJUq63JvQu8q3lnDyA==} - - '@types/retry@0.12.2': - resolution: {integrity: sha512-XISRgDJ2Tc5q4TRqvgJtzsRkFYNJzZrhTdtMoGVBttwzzQJkPnS3WWTFc7kuDRoPtPakl+T+OfdEUjYJj7Jbow==} - - '@types/sinon@17.0.3': - resolution: {integrity: sha512-j3uovdn8ewky9kRBG19bOwaZbexJu/XjtkHyjvUgt4xfPFz18dcORIMqnYh66Fx3Powhcr85NT5+er3+oViapw==} - - '@types/sinonjs__fake-timers@8.1.5': - resolution: {integrity: sha512-mQkU2jY8jJEF7YHjHvsQO8+3ughTL1mcnn96igfhONmR+fUPSKIkefQYpSe8bsly2Ep7oQbn/6VG5/9/0qcArQ==} - - '@webassemblyjs/ast@1.12.1': - resolution: {integrity: sha512-EKfMUOPRRUTy5UII4qJDGPpqfwjOmZ5jeGFwid9mnoqIFK+e0vqoi1qH56JpmZSzEL53jKnNzScdmftJyG5xWg==} - - '@webassemblyjs/floating-point-hex-parser@1.11.6': - resolution: {integrity: sha512-ejAj9hfRJ2XMsNHk/v6Fu2dGS+i4UaXBXGemOfQ/JfQ6mdQg/WXtwleQRLLS4OvfDhv8rYnVwH27YJLMyYsxhw==} - - '@webassemblyjs/helper-api-error@1.11.6': - resolution: {integrity: sha512-o0YkoP4pVu4rN8aTJgAyj9hC2Sv5UlkzCHhxqWj8butaLvnpdc2jOwh4ewE6CX0txSfLn/UYaV/pheS2Txg//Q==} - - '@webassemblyjs/helper-buffer@1.12.1': - resolution: {integrity: sha512-nzJwQw99DNDKr9BVCOZcLuJJUlqkJh+kVzVl6Fmq/tI5ZtEyWT1KZMyOXltXLZJmDtvLCDgwsyrkohEtopTXCw==} - - '@webassemblyjs/helper-numbers@1.11.6': - resolution: {integrity: sha512-vUIhZ8LZoIWHBohiEObxVm6hwP034jwmc9kuq5GdHZH0wiLVLIPcMCdpJzG4C11cHoQ25TFIQj9kaVADVX7N3g==} - - '@webassemblyjs/helper-wasm-bytecode@1.11.6': - resolution: {integrity: sha512-sFFHKwcmBprO9e7Icf0+gddyWYDViL8bpPjJJl0WHxCdETktXdmtWLGVzoHbqUcY4Be1LkNfwTmXOJUFZYSJdA==} - - '@webassemblyjs/helper-wasm-section@1.12.1': - resolution: {integrity: sha512-Jif4vfB6FJlUlSbgEMHUyk1j234GTNG9dBJ4XJdOySoj518Xj0oGsNi59cUQF4RRMS9ouBUxDDdyBVfPTypa5g==} - - '@webassemblyjs/ieee754@1.11.6': - resolution: {integrity: sha512-LM4p2csPNvbij6U1f19v6WR56QZ8JcHg3QIJTlSwzFcmx6WSORicYj6I63f9yU1kEUtrpG+kjkiIAkevHpDXrg==} - - '@webassemblyjs/leb128@1.11.6': - resolution: {integrity: sha512-m7a0FhE67DQXgouf1tbN5XQcdWoNgaAuoULHIfGFIEVKA6tu/edls6XnIlkmS6FrXAquJRPni3ZZKjw6FSPjPQ==} - - '@webassemblyjs/utf8@1.11.6': - resolution: {integrity: sha512-vtXf2wTQ3+up9Zsg8sa2yWiQpzSsMyXj0qViVP6xKGCUT8p8YJ6HqI7l5eCnWx1T/FYdsv07HQs2wTFbbof/RA==} - - '@webassemblyjs/wasm-edit@1.12.1': - resolution: {integrity: sha512-1DuwbVvADvS5mGnXbE+c9NfA8QRcZ6iKquqjjmR10k6o+zzsRVesil54DKexiowcFCPdr/Q0qaMgB01+SQ1u6g==} - - '@webassemblyjs/wasm-gen@1.12.1': - resolution: {integrity: sha512-TDq4Ojh9fcohAw6OIMXqiIcTq5KUXTGRkVxbSo1hQnSy6lAM5GSdfwWeSxpAo0YzgsgF182E/U0mDNhuA0tW7w==} - - '@webassemblyjs/wasm-opt@1.12.1': - resolution: {integrity: sha512-Jg99j/2gG2iaz3hijw857AVYekZe2SAskcqlWIZXjji5WStnOpVoat3gQfT/Q5tb2djnCjBtMocY/Su1GfxPBg==} - - '@webassemblyjs/wasm-parser@1.12.1': - resolution: {integrity: sha512-xikIi7c2FHXysxXe3COrVUPSheuBtpcfhbpFj4gmu7KRLYOzANztwUU0IbsqvMqzuNK2+glRGWCEqZo1WCLyAQ==} - - '@webassemblyjs/wast-printer@1.12.1': - resolution: {integrity: sha512-+X4WAlOisVWQMikjbcvY2e0rwPsKQ9F688lksZhBcPycBBuii3O7m8FACbDMWDojpAqvjIncrG8J0XHKyQfVeA==} - - '@xtuc/ieee754@1.2.0': - resolution: {integrity: sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA==} - - '@xtuc/long@4.2.2': - resolution: {integrity: sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ==} - - abort-controller@3.0.0: - resolution: {integrity: sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==} - engines: {node: '>=6.5'} - - abstract-logging@2.0.1: - resolution: {integrity: sha512-2BjRTZxTPvheOvGbBslFSYOUkr+SjPtOnrLP33f+VIWLzezQpZcqVg7ja3L4dBXmzzgwT+a029jRx5PCi3JuiA==} - - acorn-import-attributes@1.9.5: - resolution: {integrity: sha512-n02Vykv5uA3eHGM/Z2dQrcD56kL8TyDb2p1+0P83PClMnC/nc+anbQRhIOWnSq4Ke/KvDPrY3C9hDtC/A3eHnQ==} - peerDependencies: - acorn: ^8 - - acorn-walk@8.3.3: - resolution: {integrity: sha512-MxXdReSRhGO7VlFe1bRG/oI7/mdLV9B9JJT0N8vZOhF7gFRR5l3M8W9G8JxmKV+JC5mGqJ0QvqfSOLsCPa4nUw==} - engines: {node: '>=0.4.0'} - - acorn@8.12.1: - resolution: {integrity: sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==} - engines: {node: '>=0.4.0'} - hasBin: true - - ajv-formats@2.1.1: - resolution: {integrity: sha512-Wx0Kx52hxE7C18hkMEggYlEifqWZtYaRgouJor+WMdPnQyEK13vgEWyVNup7SoeeoLMsr4kf5h6dOW11I15MUA==} - peerDependencies: - ajv: ^8.0.0 - peerDependenciesMeta: - ajv: - optional: true - - ajv-formats@3.0.1: - resolution: {integrity: sha512-8iUql50EUR+uUcdRQ3HDqa6EVyo3docL8g5WJ3FNcWmu62IbkGUue/pEyLBW8VGKKucTPgqeks4fIU1DA4yowQ==} - peerDependencies: - ajv: ^8.0.0 - peerDependenciesMeta: - ajv: - optional: true - - ajv-keywords@3.5.2: - resolution: {integrity: sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==} - peerDependencies: - ajv: ^6.9.1 - - ajv@6.12.6: - resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==} - - ajv@8.17.1: - resolution: {integrity: sha512-B/gBuNg5SiMTrPkC+A2+cW0RszwxYmn6VYxB/inlBStS5nx6xHIt/ehKRhIMhqusl7a8LjQoZnjCs5vhwxOQ1g==} - - ansi-colors@4.1.3: - resolution: {integrity: sha512-/6w/C21Pm1A7aZitlI5Ni/2J6FFQN8i1Cvz3kHABAAbw93v/NlvKdVOqz7CCWz/3iv/JplRSEEZ83XION15ovw==} - engines: {node: '>=6'} - - ansi-regex@5.0.1: - resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==} - engines: {node: '>=8'} - - ansi-regex@6.0.1: - resolution: {integrity: sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==} - engines: {node: '>=12'} - - ansi-styles@3.2.1: - resolution: {integrity: sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==} - engines: {node: '>=4'} - - ansi-styles@4.3.0: - resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} - engines: {node: '>=8'} - - ansi-styles@6.2.1: - resolution: {integrity: sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==} - engines: {node: '>=12'} - - any-promise@1.3.0: - resolution: {integrity: sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==} - - anymatch@3.1.3: - resolution: {integrity: sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==} - engines: {node: '>= 8'} - - arg@4.1.3: - resolution: {integrity: sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==} - - arg@5.0.2: - resolution: {integrity: sha512-PYjyFOLKQ9y57JvQ6QLo8dAgNqswh8M1RMJYdQduT6xbWSgK36P/Z/v+p888pM69jMMfS8Xd8F6I1kQ/I9HUGg==} - - argparse@2.0.1: - resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==} - - array-union@2.1.0: - resolution: {integrity: sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==} - engines: {node: '>=8'} - - asap@2.0.6: - resolution: {integrity: sha512-BSHWgDSAiKs50o2Re8ppvp3seVHXSRM44cdSsT9FfNEUUZLOGWVCsiWaRPWM1Znn+mqZ1OfVZ3z3DWEzSp7hRA==} - - assertion-error@2.0.1: - resolution: {integrity: sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==} - engines: {node: '>=12'} - - async@0.2.10: - resolution: {integrity: sha512-eAkdoKxU6/LkKDBzLpT+t6Ff5EtfSF4wx1WfJiPEEV7WNLnDaRXk0oVysiEPm262roaachGexwUv94WhSgN5TQ==} - - atomic-sleep@1.0.0: - resolution: {integrity: sha512-kNOjDqAh7px0XWNI+4QbzoiR/nTkHAWNud2uvnJquD1/x5a7EQZMJT0AczqK0Qn67oY/TTQ1LbUKajZpp3I9tQ==} - engines: {node: '>=8.0.0'} - - avvio@8.4.0: - resolution: {integrity: sha512-CDSwaxINFy59iNwhYnkvALBwZiTydGkOecZyPkqBpABYR1KqGEsET0VOOYDwtleZSUIdeY36DC2bSZ24CO1igA==} - - babel-plugin-polyfill-corejs2@0.4.11: - resolution: {integrity: sha512-sMEJ27L0gRHShOh5G54uAAPaiCOygY/5ratXuiyb2G46FmlSpc9eFCzYVyDiPxfNbwzA7mYahmjQc5q+CZQ09Q==} - peerDependencies: - '@babel/core': ^7.4.0 || ^8.0.0-0 <8.0.0 - - babel-plugin-polyfill-corejs3@0.10.6: - resolution: {integrity: sha512-b37+KR2i/khY5sKmWNVQAnitvquQbNdWy6lJdsr0kmquCKEEUgMKK4SboVM3HtfnZilfjr4MMQ7vY58FVWDtIA==} - peerDependencies: - '@babel/core': ^7.4.0 || ^8.0.0-0 <8.0.0 - - babel-plugin-polyfill-regenerator@0.6.2: - resolution: {integrity: sha512-2R25rQZWP63nGwaAswvDazbPXfrM3HwVoBXK6HcqeKrSrL/JqcC/rDcf95l4r7LXLyxDXc8uQDa064GubtCABg==} - peerDependencies: - '@babel/core': ^7.4.0 || ^8.0.0-0 <8.0.0 - - balanced-match@1.0.2: - resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} - - base64-js@1.5.1: - resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==} - - bath-es5@3.0.3: - resolution: {integrity: sha512-PdCioDToH3t84lP40kUFCKWCOCH389Dl1kbC8FGoqOwamxsmqxxnJSXdkTOsPoNHXjem4+sJ+bbNoQm5zeCqxg==} - - binary-extensions@2.3.0: - resolution: {integrity: sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==} - engines: {node: '>=8'} - - boolbase@1.0.0: - resolution: {integrity: sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==} - - bowser@2.11.0: - resolution: {integrity: sha512-AlcaJBi/pqqJBIQ8U9Mcpc9i8Aqxn88Skv5d+xBX006BY5u8N3mGLHa5Lgppa7L/HfwgwLgZ6NYs+Ag6uUmJRA==} - - brace-expansion@1.1.11: - resolution: {integrity: sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==} - - brace-expansion@2.0.1: - resolution: {integrity: sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==} - - braces@3.0.3: - resolution: {integrity: sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==} - engines: {node: '>=8'} - - browser-stdout@1.3.1: - resolution: {integrity: sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==} - - browserslist@4.23.3: - resolution: {integrity: sha512-btwCFJVjI4YWDNfau8RhZ+B1Q/VLoUITrm3RlP6y1tYGWIOa+InuYiRGXUBXo8nA1qKmHMyLB/iVQg5TT4eFoA==} - engines: {node: ^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7} - hasBin: true - - buffer-from@1.1.2: - resolution: {integrity: sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==} - - buffer@5.6.0: - resolution: {integrity: sha512-/gDYp/UtU0eA1ys8bOs9J6a+E/KWIY+DZ+Q2WESNUA0jFRsJOc0SNUO6xJ5SGA1xueg3NL65W6s+NY5l9cunuw==} - - buffer@6.0.3: - resolution: {integrity: sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==} - - bundle-require@5.0.0: - resolution: {integrity: sha512-GuziW3fSSmopcx4KRymQEJVbZUfqlCqcq7dvs6TYwKRZiegK/2buMxQTPs6MGlNv50wms1699qYO54R8XfRX4w==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - peerDependencies: - esbuild: '>=0.18' - - cac@6.7.14: - resolution: {integrity: sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==} - engines: {node: '>=8'} - - call-bind@1.0.7: - resolution: {integrity: sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==} - engines: {node: '>= 0.4'} - - camelcase@6.3.0: - resolution: {integrity: sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==} - engines: {node: '>=10'} - - caniuse-lite@1.0.30001655: - resolution: {integrity: sha512-jRGVy3iSGO5Uutn2owlb5gR6qsGngTw9ZTb4ali9f3glshcNmJ2noam4Mo9zia5P9Dk3jNNydy7vQjuE5dQmfg==} - - cargo-cp-artifact@0.1.9: - resolution: {integrity: sha512-6F+UYzTaGB+awsTXg0uSJA1/b/B3DDJzpKVRu0UmyI7DmNeaAl2RFHuTGIN6fEgpadRxoXGb7gbC1xo4C3IdyA==} - hasBin: true - - chai@5.1.1: - resolution: {integrity: sha512-pT1ZgP8rPNqUgieVaEY+ryQr6Q4HXNg8Ei9UnLUrjN4IA7dvQC5JB+/kxVcPNDHyBcc/26CXPkbNzq3qwrOEKA==} - engines: {node: '>=12'} - - chalk@2.4.2: - resolution: {integrity: sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==} - engines: {node: '>=4'} - - chalk@4.1.2: - resolution: {integrity: sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==} - engines: {node: '>=10'} - - check-error@2.1.1: - resolution: {integrity: sha512-OAlb+T7V4Op9OwdkjmguYRqncdlx5JiofwOAUkmTF+jNdHwzTaTs4sRAGpzLF3oOz5xAyDGrPgeIDFQmDOTiJw==} - engines: {node: '>= 16'} - - cheerio-select@2.1.0: - resolution: {integrity: sha512-9v9kG0LvzrlcungtnJtpGNxY+fzECQKhK4EGJX2vByejiMX84MFNQw4UxPJl3bFbTMw+Dfs37XaIkCwTZfLh4g==} - - cheerio@1.0.0: - resolution: {integrity: sha512-quS9HgjQpdaXOvsZz82Oz7uxtXiy6UIsIQcpBj7HRw2M63Skasm9qlDocAM7jNuaxdhpPU7c4kJN+gA5MCu4ww==} - engines: {node: '>=18.17'} - - chokidar@3.6.0: - resolution: {integrity: sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==} - engines: {node: '>= 8.10.0'} - - chrome-trace-event@1.0.4: - resolution: {integrity: sha512-rNjApaLzuwaOTjCiT8lSDdGN1APCiqkChLMJxJPWLunPAt5fy8xgU9/jNOchV84wfIxrA0lRQB7oCT8jrn/wrQ==} - engines: {node: '>=6.0'} - - cliui@7.0.4: - resolution: {integrity: sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==} - - cliui@8.0.1: - resolution: {integrity: sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==} - engines: {node: '>=12'} - - color-convert@1.9.3: - resolution: {integrity: sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==} - - color-convert@2.0.1: - resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} - engines: {node: '>=7.0.0'} - - color-name@1.1.3: - resolution: {integrity: sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==} - - color-name@1.1.4: - resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} - - color-string@1.9.1: - resolution: {integrity: sha512-shrVawQFojnZv6xM40anx4CkoDP+fZsw/ZerEMsW/pyzsRbElpsL/DBVW7q3ExxwusdNXI3lXpuhEZkzs8p5Eg==} - - color@4.2.3: - resolution: {integrity: sha512-1rXeuUUiGGrykh+CeBdu5Ie7OJwinCgQY0bc7GCRxy5xVHy+moaqkpL/jqQq0MtQOeYcrqEz4abc5f0KtU7W4A==} - engines: {node: '>=12.5.0'} - - commander@12.1.0: - resolution: {integrity: sha512-Vw8qHK3bZM9y/P10u3Vib8o/DdkvA2OtPtZvD871QKjy74Wj1WSKFILMPRPSdUSx5RFK1arlJzEtA4PkFgnbuA==} - engines: {node: '>=18'} - - commander@2.20.3: - resolution: {integrity: sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==} - - commander@4.1.1: - resolution: {integrity: sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==} - engines: {node: '>= 6'} - - concat-map@0.0.1: - resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} - - concurrently@8.2.2: - resolution: {integrity: sha512-1dP4gpXFhei8IOtlXRE/T/4H88ElHgTiUzh71YUmtjTEHMSRS2Z/fgOxHSxxusGHogsRfxNq1vyAwxSC+EVyDg==} - engines: {node: ^14.13.0 || >=16.0.0} - hasBin: true - - consola@3.2.3: - resolution: {integrity: sha512-I5qxpzLv+sJhTVEoLYNcTW+bThDCPsit0vLNKShZx6rLtpilNpmmeTPaeqJb9ZE9dV3DGaeby6Vuhrw38WjeyQ==} - engines: {node: ^14.18.0 || >=16.10.0} - - content-disposition@0.5.4: - resolution: {integrity: sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==} - engines: {node: '>= 0.6'} - - convert-source-map@2.0.0: - resolution: {integrity: sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==} - - cookie@0.5.0: - resolution: {integrity: sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==} - engines: {node: '>= 0.6'} - - cookie@0.6.0: - resolution: {integrity: sha512-U71cyTamuh1CRNCfpGY6to28lxvNwPG4Guz/EVjgf3Jmzv0vlDp1atT9eS5dDjMYHucpHbWns6Lwf3BKz6svdw==} - engines: {node: '>= 0.6'} - - core-js-compat@3.38.1: - resolution: {integrity: sha512-JRH6gfXxGmrzF3tZ57lFx97YARxCXPaMzPo6jELZhv88pBH5VXpQ+y0znKGlFnzuaihqhLbefxSJxWJMPtfDzw==} - - create-require@1.1.1: - resolution: {integrity: sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==} - - cross-spawn@7.0.3: - resolution: {integrity: sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==} - engines: {node: '>= 8'} - - csprng@0.1.2: - resolution: {integrity: sha512-D3WAbvvgUVIqSxUfdvLeGjuotsB32bvfVPd+AaaTWMtyUeC9zgCnw5xs94no89yFLVsafvY9dMZEhTwsY/ZecA==} - engines: {node: '>=0.6.0'} - - css-select@5.1.0: - resolution: {integrity: sha512-nwoRF1rvRRnnCqqY7updORDsuqKzqYJ28+oSMaJMMgOauh3fvwHqMS7EZpIPqK8GL+g9mKxF1vP/ZjSeNjEVHg==} - - css-what@6.1.0: - resolution: {integrity: sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw==} - engines: {node: '>= 6'} - - data-uri-to-buffer@4.0.1: - resolution: {integrity: sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A==} - engines: {node: '>= 12'} - - date-fns@2.30.0: - resolution: {integrity: sha512-fnULvOpxnC5/Vg3NCiWelDsLiUc9bRwAPs/+LfTLNvetFCtCTN+yQz15C/fs4AwX1R9K5GLtLfn8QW+dWisaAw==} - engines: {node: '>=0.11'} - - date-fns@3.6.0: - resolution: {integrity: sha512-fRHTG8g/Gif+kSh50gaGEdToemgfj74aRX3swtiouboip5JDLAyDE9F11nHMIcvOaXeOC6D7SpNhi7uFyB7Uww==} - - debug@4.3.6: - resolution: {integrity: sha512-O/09Bd4Z1fBrU4VzkhFqVgpPzaGbw6Sm9FEkBT1A/YBXQFGuuSxa1dN2nxgxS34JmKXqYx8CZAwEVoJFImUXIg==} - engines: {node: '>=6.0'} - peerDependencies: - supports-color: '*' - peerDependenciesMeta: - supports-color: - optional: true - - decamelize@4.0.0: - resolution: {integrity: sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==} - engines: {node: '>=10'} - - deep-eql@5.0.2: - resolution: {integrity: sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==} - engines: {node: '>=6'} - - deepmerge@4.3.1: - resolution: {integrity: sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==} - engines: {node: '>=0.10.0'} - - define-data-property@1.1.4: - resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==} - engines: {node: '>= 0.4'} - - depd@2.0.0: - resolution: {integrity: sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==} - engines: {node: '>= 0.8'} - - dereference-json-schema@0.2.1: - resolution: {integrity: sha512-uzJsrg225owJyRQ8FNTPHIuBOdSzIZlHhss9u6W8mp7jJldHqGuLv9cULagP/E26QVJDnjtG8U7Dw139mM1ydA==} - - detect-libc@2.0.3: - resolution: {integrity: sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw==} - engines: {node: '>=8'} - - diff@4.0.2: - resolution: {integrity: sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==} - engines: {node: '>=0.3.1'} - - diff@5.2.0: - resolution: {integrity: sha512-uIFDxqpRZGZ6ThOk84hEfqWoHx2devRFvpTZcTHur85vImfaxUbTW9Ryh4CpCuDnToOP1CEtXKIgytHBPVff5A==} - engines: {node: '>=0.3.1'} - - dir-glob@3.0.1: - resolution: {integrity: sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==} - engines: {node: '>=8'} - - dom-serializer@2.0.0: - resolution: {integrity: sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg==} - - domelementtype@2.3.0: - resolution: {integrity: sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==} - - domhandler@5.0.3: - resolution: {integrity: sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==} - engines: {node: '>= 4'} - - domutils@3.1.0: - resolution: {integrity: sha512-H78uMmQtI2AhgDJjWeQmHwJJ2bLPD3GMmO7Zja/ZZh84wkm+4ut+IUnUdRa8uCGX88DiVx1j6FRe1XfxEgjEZA==} - - dotenv@16.4.5: - resolution: {integrity: sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==} - engines: {node: '>=12'} - - eastasianwidth@0.2.0: - resolution: {integrity: sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==} - - electron-to-chromium@1.5.13: - resolution: {integrity: sha512-lbBcvtIJ4J6sS4tb5TLp1b4LyfCdMkwStzXPyAgVgTRAsep4bvrAGaBOP7ZJtQMNJpSQ9SqG4brWOroNaQtm7Q==} - - emoji-regex@8.0.0: - resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==} - - emoji-regex@9.2.2: - resolution: {integrity: sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==} - - encoding-japanese@2.0.0: - resolution: {integrity: sha512-++P0RhebUC8MJAwJOsT93dT+5oc5oPImp1HubZpAuCZ5kTLnhuuBhKHj2jJeO/Gj93idPBWmIuQ9QWMe5rX3pQ==} - engines: {node: '>=8.10.0'} - - encoding-japanese@2.1.0: - resolution: {integrity: sha512-58XySVxUgVlBikBTbQ8WdDxBDHIdXucB16LO5PBHR8t75D54wQrNo4cg+58+R1CtJfKnsVsvt9XlteRaR8xw1w==} - engines: {node: '>=8.10.0'} - - encoding-japanese@2.2.0: - resolution: {integrity: sha512-EuJWwlHPZ1LbADuKTClvHtwbaFn4rOD+dRAbWysqEOXRc2Uui0hJInNJrsdH0c+OhJA4nrCBdSkW4DD5YxAo6A==} - engines: {node: '>=8.10.0'} - - encoding-sniffer@0.2.0: - resolution: {integrity: sha512-ju7Wq1kg04I3HtiYIOrUrdfdDvkyO9s5XM8QAj/bN61Yo/Vb4vgJxy5vi4Yxk01gWHbrofpPtpxM8bKger9jhg==} - - enhanced-resolve@5.17.1: - resolution: {integrity: sha512-LMHl3dXhTcfv8gM4kEzIUeTQ+7fpdA0l2tUf34BddXPkz2A5xJ5L/Pchd5BL6rdccM9QGvu0sWZzK1Z1t4wwyg==} - engines: {node: '>=10.13.0'} - - entities@4.5.0: - resolution: {integrity: sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==} - engines: {node: '>=0.12'} - - es-define-property@1.0.0: - resolution: {integrity: sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==} - engines: {node: '>= 0.4'} - - es-errors@1.3.0: - resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==} - engines: {node: '>= 0.4'} - - es-module-lexer@1.5.4: - resolution: {integrity: sha512-MVNK56NiMrOwitFB7cqDwq0CQutbw+0BvLshJSse0MUNU+y1FC3bUS/AQg7oUng+/wKrrki7JfmwtVHkVfPLlw==} - - esbuild@0.23.1: - resolution: {integrity: sha512-VVNz/9Sa0bs5SELtn3f7qhJCDPCF5oMEl5cO9/SSinpE9hbPVvxbd572HH5AKiP7WD8INO53GgfDDhRjkylHEg==} - engines: {node: '>=18'} - hasBin: true - - escalade@3.2.0: - resolution: {integrity: sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==} - engines: {node: '>=6'} - - escape-html@1.0.3: - resolution: {integrity: sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==} - - escape-string-regexp@1.0.5: - resolution: {integrity: sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==} - engines: {node: '>=0.8.0'} - - escape-string-regexp@4.0.0: - resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} - engines: {node: '>=10'} - - eslint-scope@5.1.1: - resolution: {integrity: sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==} - engines: {node: '>=8.0.0'} - - esmock@2.6.7: - resolution: {integrity: sha512-4DmjZ0qQIG+NQV1njHvWrua/cZEuJq56A3pSELT2BjNuol1aads7BluofCbLErdO41Ic1XCd2UMepVLpjL64YQ==} - engines: {node: '>=14.16.0'} - - esrecurse@4.3.0: - resolution: {integrity: sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==} - engines: {node: '>=4.0'} - - estraverse@4.3.0: - resolution: {integrity: sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==} - engines: {node: '>=4.0'} - - estraverse@5.3.0: - resolution: {integrity: sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==} - engines: {node: '>=4.0'} - - esutils@2.0.3: - resolution: {integrity: sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==} - engines: {node: '>=0.10.0'} - - event-target-shim@5.0.1: - resolution: {integrity: sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==} - engines: {node: '>=6'} - - events@3.3.0: - resolution: {integrity: sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==} - engines: {node: '>=0.8.x'} - - execa@5.1.1: - resolution: {integrity: sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==} - engines: {node: '>=10'} - - execa@8.0.1: - resolution: {integrity: sha512-VyhnebXciFV2DESc+p6B+y0LjSm0krU4OgJN44qFAhBY0TJ+1V61tYD2+wHusZ6F9n5K+vl8k0sTy7PEfV4qpg==} - engines: {node: '>=16.17'} - - fast-content-type-parse@1.1.0: - resolution: {integrity: sha512-fBHHqSTFLVnR61C+gltJuE5GkVQMV0S2nqUO8TJ+5Z3qAKG8vAx4FKai1s5jq/inV1+sREynIWSuQ6HgoSXpDQ==} - - fast-decode-uri-component@1.0.1: - resolution: {integrity: sha512-WKgKWg5eUxvRZGwW8FvfbaH7AXSh2cL+3j5fMGzUMCxWBJ3dV3a7Wz8y2f/uQ0e3B6WmodD3oS54jTQ9HVTIIg==} - - fast-deep-equal@3.1.3: - resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==} - - fast-glob@3.3.2: - resolution: {integrity: sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==} - engines: {node: '>=8.6.0'} - - fast-json-stable-stringify@2.1.0: - resolution: {integrity: sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==} - - fast-json-stringify@5.16.1: - resolution: {integrity: sha512-KAdnLvy1yu/XrRtP+LJnxbBGrhN+xXu+gt3EUvZhYGKCr3lFHq/7UFJHHFgmJKoqlh6B40bZLEv7w46B0mqn1g==} - - fast-querystring@1.1.2: - resolution: {integrity: sha512-g6KuKWmFXc0fID8WWH0jit4g0AGBoJhCkJMb1RmbsSEUNvQ+ZC8D6CUZ+GtF8nMzSPXnhiePyyqqipzNNEnHjg==} - - fast-redact@3.5.0: - resolution: {integrity: sha512-dwsoQlS7h9hMeYUq1W++23NDcBLV4KqONnITDV9DjfS3q1SgDGVrBdvvTLUotWtPSD7asWDV9/CmsZPy8Hf70A==} - engines: {node: '>=6'} - - fast-uri@2.4.0: - resolution: {integrity: sha512-ypuAmmMKInk5q7XcepxlnUWDLWv4GFtaJqAzWKqn62IpQ3pejtr5dTVbt3vwqVaMKmkNR55sTT+CqUKIaT21BA==} - - fast-uri@3.0.1: - resolution: {integrity: sha512-MWipKbbYiYI0UC7cl8m/i/IWTqfC8YXsqjzybjddLsFjStroQzsHXkc73JutMvBiXmOvapk+axIl79ig5t55Bw==} - - fast-xml-parser@4.4.1: - resolution: {integrity: sha512-xkjOecfnKGkSsOwtZ5Pz7Us/T6mrbPQrq0nh+aCO5V9nk5NLWmasAHumTKjiPJPWANe+kAZ84Jc8ooJkzZ88Sw==} - hasBin: true - - fastify-plugin@4.5.1: - resolution: {integrity: sha512-stRHYGeuqpEZTL1Ef0Ovr2ltazUT9g844X5z/zEBFLG8RYlpDiOCIG+ATvYEp+/zmc7sN29mcIMp8gvYplYPIQ==} - - fastify@4.28.1: - resolution: {integrity: sha512-kFWUtpNr4i7t5vY2EJPCN2KgMVpuqfU4NjnJNCgiNB900oiDeYqaNDRcAfeBbOF5hGixixxcKnOU4KN9z6QncQ==} - - fastq@1.17.1: - resolution: {integrity: sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==} - - faye-websocket@0.11.4: - resolution: {integrity: sha512-CzbClwlXAuiRQAlUyfqPgvPoNKTckTPGfwZV4ZdAhVcP2lh9KUxJg2b5GkE7XbjKQ3YJnQ9z6D9ntLAlB+tP8g==} - engines: {node: '>=0.8.0'} - - faye@1.4.0: - resolution: {integrity: sha512-kRrIg4be8VNYhycS2PY//hpBJSzZPr/DBbcy9VWelhZMW3KhyLkQR0HL0k0MNpmVoNFF4EdfMFkNAWjTP65g6w==} - engines: {node: '>=0.8.0'} - - fetch-blob@3.2.0: - resolution: {integrity: sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ==} - engines: {node: ^12.20 || >= 14.13} - - fill-range@7.1.1: - resolution: {integrity: sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==} - engines: {node: '>=8'} - - find-my-way@8.2.0: - resolution: {integrity: sha512-HdWXgFYc6b1BJcOBDBwjqWuHJj1WYiqrxSh25qtU4DabpMFdj/gSunNBQb83t+8Zt67D7CXEzJWTkxaShMTMOA==} - engines: {node: '>=14'} - - find-up@5.0.0: - resolution: {integrity: sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==} - engines: {node: '>=10'} - - flat@5.0.2: - resolution: {integrity: sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==} - hasBin: true - - fluent-ffmpeg@2.1.3: - resolution: {integrity: sha512-Be3narBNt2s6bsaqP6Jzq91heDgOEaDCJAXcE3qcma/EJBSy5FB4cvO31XBInuAuKBx8Kptf8dkhjK0IOru39Q==} - engines: {node: '>=18'} - - foreground-child@3.3.0: - resolution: {integrity: sha512-Ld2g8rrAyMYFXBhEqMz8ZAHBi4J4uS1i/CxGMDnjyFWddMXLVcDp051DZfu+t7+ab7Wv6SMqpWmyFIj5UbfFvg==} - engines: {node: '>=14'} - - formdata-polyfill@4.0.10: - resolution: {integrity: sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g==} - engines: {node: '>=12.20.0'} - - forwarded@0.2.0: - resolution: {integrity: sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==} - engines: {node: '>= 0.6'} - - fs-monkey@1.0.6: - resolution: {integrity: sha512-b1FMfwetIKymC0eioW7mTywihSQE4oLzQn1dB6rZB5fx/3NpNEdAWeCSMB+60/AeT0TCXsxzAlcYVEFCTAksWg==} - - fs.realpath@1.0.0: - resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==} - - fsevents@2.3.3: - resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} - engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} - os: [darwin] - - function-bind@1.1.2: - resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} - - gensync@1.0.0-beta.2: - resolution: {integrity: sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==} - engines: {node: '>=6.9.0'} - - get-caller-file@2.0.5: - resolution: {integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==} - engines: {node: 6.* || 8.* || >= 10.*} - - get-func-name@2.0.2: - resolution: {integrity: sha512-8vXOvuE167CtIc3OyItco7N/dpRtBbYOsPsXCz7X/PMnlGjYjSGuZJgM1Y7mmew7BKf9BqvLX2tnOVy1BBUsxQ==} - - get-intrinsic@1.2.4: - resolution: {integrity: sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==} - engines: {node: '>= 0.4'} - - get-stream@6.0.1: - resolution: {integrity: sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==} - engines: {node: '>=10'} - - get-stream@8.0.1: - resolution: {integrity: sha512-VaUJspBffn/LMCJVoMvSAdmscJyS1auj5Zulnn5UoYcY531UWmdwhRWkcGKnGU93m5HSXP9LP2usOryrBtQowA==} - engines: {node: '>=16'} - - get-tsconfig@4.8.0: - resolution: {integrity: sha512-Pgba6TExTZ0FJAn1qkJAjIeKoDJ3CsI2ChuLohJnZl/tTU8MVrq3b+2t5UOPfRa4RMsorClBjJALkJUMjG1PAw==} - - glob-parent@5.1.2: - resolution: {integrity: sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==} - engines: {node: '>= 6'} - - glob-to-regexp@0.4.1: - resolution: {integrity: sha512-lkX1HJXwyMcprw/5YUZc2s7DrpAiHB21/V+E1rHUrVNokkvB6bqMzT0VfV6/86ZNabt1k14YOIaT7nDvOX3Iiw==} - - glob@10.4.5: - resolution: {integrity: sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg==} - hasBin: true - - glob@7.2.3: - resolution: {integrity: sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==} - deprecated: Glob versions prior to v9 are no longer supported - - glob@8.1.0: - resolution: {integrity: sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==} - engines: {node: '>=12'} - deprecated: Glob versions prior to v9 are no longer supported - - globals@11.12.0: - resolution: {integrity: sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==} - engines: {node: '>=4'} - - globby@11.1.0: - resolution: {integrity: sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==} - engines: {node: '>=10'} - - gopd@1.0.1: - resolution: {integrity: sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==} - - graceful-fs@4.2.11: - resolution: {integrity: sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==} - - has-flag@3.0.0: - resolution: {integrity: sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==} - engines: {node: '>=4'} - - has-flag@4.0.0: - resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} - engines: {node: '>=8'} - - has-property-descriptors@1.0.2: - resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==} - - has-proto@1.0.3: - resolution: {integrity: sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==} - engines: {node: '>= 0.4'} - - has-symbols@1.0.3: - resolution: {integrity: sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==} - engines: {node: '>= 0.4'} - - hasown@2.0.2: - resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} - engines: {node: '>= 0.4'} - - he@1.2.0: - resolution: {integrity: sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==} - hasBin: true - - heap-js@2.5.0: - resolution: {integrity: sha512-kUGoI3p7u6B41z/dp33G6OaL7J4DRqRYwVmeIlwLClx7yaaAy7hoDExnuejTKtuDwfcatGmddHDEOjf6EyIxtQ==} - engines: {node: '>=10.0.0'} - - html-to-text@9.0.5: - resolution: {integrity: sha512-qY60FjREgVZL03vJU6IfMV4GDjGBIoOyvuFdpBDIX9yTlDw0TjxVBQp+P8NvpdIXNJvfWBTNul7fsAQJq2FNpg==} - engines: {node: '>=14'} - - htmlparser2@8.0.2: - resolution: {integrity: sha512-GYdjWKDkbRLkZ5geuHs5NY1puJ+PXwP7+fHPRz06Eirsb9ugf6d8kkXav6ADhcODhFFPMIXyxkxSuMf3D6NCFA==} - - htmlparser2@9.1.0: - resolution: {integrity: sha512-5zfg6mHUoaer/97TxnGpxmbR7zJtPwIYFMZ/H5ucTlPZhKvtum05yiPK3Mgai3a0DyVxv7qYqoweaEd2nrYQzQ==} - - http-errors@2.0.0: - resolution: {integrity: sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==} - engines: {node: '>= 0.8'} - - http-parser-js@0.5.8: - resolution: {integrity: sha512-SGeBX54F94Wgu5RH3X5jsDtf4eHyRogWX1XGT3b4HuW3tQPM4AaBzoUji/4AAJNXCEOWZ5O0DgZmJw1947gD5Q==} - - human-signals@2.1.0: - resolution: {integrity: sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==} - engines: {node: '>=10.17.0'} - - human-signals@5.0.0: - resolution: {integrity: sha512-AXcZb6vzzrFAUE61HnN4mpLqd/cSIwNQjtNWR0euPm6y0iqx3G4gOXaIDdtdDwZmhwe82LA6+zinmW4UBWVePQ==} - engines: {node: '>=16.17.0'} - - hyperdyperid@1.2.0: - resolution: {integrity: sha512-Y93lCzHYgGWdrJ66yIktxiaGULYc6oGiABxhcO5AufBeOyoIdZF7bIfLaOrbM0iGIOXQQgxxRrFEnb+Y6w1n4A==} - engines: {node: '>=10.18'} - - iconv-lite@0.6.3: - resolution: {integrity: sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==} - engines: {node: '>=0.10.0'} - - ieee754@1.2.1: - resolution: {integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==} - - ignore-by-default@1.0.1: - resolution: {integrity: sha512-Ius2VYcGNk7T90CppJqcIkS5ooHUZyIQK+ClZfMfMNFEF9VSE73Fq+906u/CWu92x4gzZMWOwfFYckPObzdEbA==} - - ignore@5.3.2: - resolution: {integrity: sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==} - engines: {node: '>= 4'} - - imapflow@1.0.164: - resolution: {integrity: sha512-+KAmLrpqq2Q0Ts1imMP4svydfhYznlvlLHhgtTb8NiIcccZvdRNfdHVP8/RrGaw0hy0TOaluawsm/6q+TqdLPw==} - - inflight@1.0.6: - resolution: {integrity: sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==} - deprecated: This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful. - - inherits@2.0.4: - resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} - - ip-address@9.0.5: - resolution: {integrity: sha512-zHtQzGojZXTwZTHQqra+ETKd4Sn3vgi7uBmlPoXVWZqYvuKmtI0l/VZTjqGmJY9x88GGOaZ9+G9ES8hC4T4X8g==} - engines: {node: '>= 12'} - - ipaddr.js@1.9.1: - resolution: {integrity: sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==} - engines: {node: '>= 0.10'} - - is-arrayish@0.3.2: - resolution: {integrity: sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==} - - is-binary-path@2.1.0: - resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==} - engines: {node: '>=8'} - - is-core-module@2.15.1: - resolution: {integrity: sha512-z0vtXSwucUJtANQWldhbtbt7BnL0vxiFjIdDLAatwhDYty2bad6s+rijD6Ri4YuYJubLzIJLUidCh09e1djEVQ==} - engines: {node: '>= 0.4'} - - is-extglob@2.1.1: - resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} - engines: {node: '>=0.10.0'} - - is-fullwidth-code-point@3.0.0: - resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==} - engines: {node: '>=8'} - - is-glob@4.0.3: - resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==} - engines: {node: '>=0.10.0'} - - is-network-error@1.1.0: - resolution: {integrity: sha512-tUdRRAnhT+OtCZR/LxZelH/C7QtjtFrTu5tXCA8pl55eTUElUHT+GPYV8MBMBvea/j+NxQqVt3LbWMRir7Gx9g==} - engines: {node: '>=16'} - - is-number@7.0.0: - resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==} - engines: {node: '>=0.12.0'} - - is-plain-obj@2.1.0: - resolution: {integrity: sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==} - engines: {node: '>=8'} - - is-stream@2.0.1: - resolution: {integrity: sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==} - engines: {node: '>=8'} - - is-stream@3.0.0: - resolution: {integrity: sha512-LnQR4bZ9IADDRSkvpqMGvt/tEJWclzklNgSw48V5EAaAeDd6qGvN8ei6k5p0tvxSR171VmGyHuTiAOfxAbr8kA==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - is-unicode-supported@0.1.0: - resolution: {integrity: sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==} - engines: {node: '>=10'} - - isexe@2.0.0: - resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} - - isexe@3.1.1: - resolution: {integrity: sha512-LpB/54B+/2J5hqQ7imZHfdU31OlgQqx7ZicVlkm9kzg9/w8GKLEcFfJl/t7DCEDueOyBAD6zCCwTO6Fzs0NoEQ==} - engines: {node: '>=16'} - - jackspeak@3.4.3: - resolution: {integrity: sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==} - - jest-worker@27.5.1: - resolution: {integrity: sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg==} - engines: {node: '>= 10.13.0'} - - joycon@3.1.1: - resolution: {integrity: sha512-34wB/Y7MW7bzjKRjUKTa46I2Z7eV62Rkhva+KkopW7Qvv/OSWBqvkSY7vusOPrNuZcUG3tApvdVgNB8POj3SPw==} - engines: {node: '>=10'} - - js-tokens@4.0.0: - resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} - - js-yaml@4.1.0: - resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==} - hasBin: true - - jsbn@1.1.0: - resolution: {integrity: sha512-4bYVV3aAMtDTTu4+xsDYa6sy9GyJ69/amsu9sYF2zqjiEoZA5xJi3BrfX3uY+/IekIu7MwdObdbDWpoZdBv3/A==} - - jsesc@0.5.0: - resolution: {integrity: sha512-uZz5UnB7u4T9LvwmFqXii7pZSouaRPorGs5who1Ip7VO0wxanFvBL7GkM6dTHlgX+jhBApRetaWpnDabOeTcnA==} - hasBin: true - - jsesc@2.5.2: - resolution: {integrity: sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==} - engines: {node: '>=4'} - hasBin: true - - json-parse-even-better-errors@2.3.1: - resolution: {integrity: sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==} - - json-schema-ref-resolver@1.0.1: - resolution: {integrity: sha512-EJAj1pgHc1hxF6vo2Z3s69fMjO1INq6eGHXZ8Z6wCQeldCuwxGK9Sxf4/cScGn3FZubCVUehfWtcDM/PLteCQw==} - - json-schema-resolver@2.0.0: - resolution: {integrity: sha512-pJ4XLQP4Q9HTxl6RVDLJ8Cyh1uitSs0CzDBAz1uoJ4sRD/Bk7cFSXL1FUXDW3zJ7YnfliJx6eu8Jn283bpZ4Yg==} - engines: {node: '>=10'} - - json-schema-traverse@0.4.1: - resolution: {integrity: sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==} - - json-schema-traverse@1.0.0: - resolution: {integrity: sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==} - - json5@2.2.3: - resolution: {integrity: sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==} - engines: {node: '>=6'} - hasBin: true - - just-extend@6.2.0: - resolution: {integrity: sha512-cYofQu2Xpom82S6qD778jBDpwvvy39s1l/hrYij2u9AMdQcGRpaBu6kY4mVhuno5kJVi1DAz4aiphA2WI1/OAw==} - - just-performance@4.3.0: - resolution: {integrity: sha512-L7RjvtJsL0QO8xFs5wEoDDzzJwoiowRw6Rn/GnvldlchS2JQr9wFYPiwZcDfrbbujEKqKN0tvENdbjXdYhDp5Q==} - - leac@0.6.0: - resolution: {integrity: sha512-y+SqErxb8h7nE/fiEX07jsbuhrpO9lL8eca7/Y1nuWV2moNlXhyd59iDGcRf6moVyDMbmTNzL40SUyrFU/yDpg==} - - libbase64@1.2.1: - resolution: {integrity: sha512-l+nePcPbIG1fNlqMzrh68MLkX/gTxk/+vdvAb388Ssi7UuUN31MI44w4Yf33mM3Cm4xDfw48mdf3rkdHszLNew==} - - libbase64@1.3.0: - resolution: {integrity: sha512-GgOXd0Eo6phYgh0DJtjQ2tO8dc0IVINtZJeARPeiIJqge+HdsWSuaDTe8ztQ7j/cONByDZ3zeB325AHiv5O0dg==} - - libmime@5.2.0: - resolution: {integrity: sha512-X2U5Wx0YmK0rXFbk67ASMeqYIkZ6E5vY7pNWRKtnNzqjvdYYG8xtPDpCnuUEnPU9vlgNev+JoSrcaKSUaNvfsw==} - - libmime@5.3.5: - resolution: {integrity: sha512-nSlR1yRZ43L3cZCiWEw7ali3jY29Hz9CQQ96Oy+sSspYnIP5N54ucOPHqooBsXzwrX1pwn13VUE05q4WmzfaLg==} - - libqp@2.0.1: - resolution: {integrity: sha512-Ka0eC5LkF3IPNQHJmYBWljJsw0UvM6j+QdKRbWyCdTmYwvIDE6a7bCm0UkTAL/K+3KXK5qXT/ClcInU01OpdLg==} - - libqp@2.1.0: - resolution: {integrity: sha512-O6O6/fsG5jiUVbvdgT7YX3xY3uIadR6wEZ7+vy9u7PKHAlSEB6blvC1o5pHBjgsi95Uo0aiBBdkyFecj6jtb7A==} - - light-my-request@5.13.0: - resolution: {integrity: sha512-9IjUN9ZyCS9pTG+KqTDEQo68Sui2lHsYBrfMyVUTTZ3XhH8PMZq7xO94Kr+eP9dhi/kcKsx4N41p2IXEBil1pQ==} - - lilconfig@3.1.2: - resolution: {integrity: sha512-eop+wDAvpItUys0FWkHIKeC9ybYrTGbU41U5K7+bttZZeohvnY7M9dZ5kB21GNWiFT2q1OoPTvncPCgSOVO5ow==} - engines: {node: '>=14'} - - limiter@2.1.0: - resolution: {integrity: sha512-361TYz6iay6n+9KvUUImqdLuFigK+K79qrUtBsXhJTLdH4rIt/r1y8r1iozwh8KbZNpujbFTSh74mJ7bwbAMOw==} - - lines-and-columns@1.2.4: - resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} - - linkify-it@5.0.0: - resolution: {integrity: sha512-5aHCbzQRADcdP+ATqnDuhhJ/MRIqDkZX5pyjFHRRysS8vZ5AbqGEoFIb6pYHPZ+L/OC2Lc+xT8uHVVR5CAK/wQ==} - - load-tsconfig@0.2.5: - resolution: {integrity: sha512-IXO6OCs9yg8tMKzfPZ1YmheJbZCiEsnBdcB03l0OcfK9prKnJb96siuHCr5Fl37/yo9DnKU+TLpxzTUspw9shg==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - loader-runner@4.3.0: - resolution: {integrity: sha512-3R/1M+yS3j5ou80Me59j7F9IMs4PXs3VqRrm0TU3AbKPxlmpoY1TNscJV/oGJXo8qCatFGTfDbY6W6ipGOYXfg==} - engines: {node: '>=6.11.5'} - - locate-path@6.0.0: - resolution: {integrity: sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==} - engines: {node: '>=10'} - - lodash.camelcase@4.3.0: - resolution: {integrity: sha512-TwuEnCnxbc3rAvhf/LbG7tJUDzhqXyFnv3dtzLOPgCG/hODL7WFnsbwktkD7yUV0RrreP/l1PALq/YSg6VvjlA==} - - lodash.debounce@4.0.8: - resolution: {integrity: sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow==} - - lodash.get@4.4.2: - resolution: {integrity: sha512-z+Uw/vLuy6gQe8cfaFWD7p0wVv8fJl3mbzXh33RS+0oW2wvUqiRXiQ69gLWSLpgB5/6sU+r6BlQR0MBILadqTQ==} - - lodash.merge@4.6.2: - resolution: {integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==} - - lodash.sortby@4.7.0: - resolution: {integrity: sha512-HDWXG8isMntAyRF5vZ7xKuEvOhT4AhlRt/3czTSjvGUxjYCBVRQY48ViDHyfYz9VIoBkW4TMGQNapx+l3RUwdA==} - - lodash@4.17.21: - resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==} - - log-symbols@4.1.0: - resolution: {integrity: sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==} - engines: {node: '>=10'} - - long@5.2.3: - resolution: {integrity: sha512-lcHwpNoggQTObv5apGNCTdJrO69eHOZMi4BNC+rTLER8iHAqGrUVeLh/irVIM7zTw2bOXA8T6uNPeujwOLg/2Q==} - - loupe@3.1.1: - resolution: {integrity: sha512-edNu/8D5MKVfGVFRhFf8aAxiTM6Wumfz5XsaatSxlD3w4R1d/WEKUTydCdPGbl9K7QG/Ca3GnDV2sIKIpXRQcw==} - - lru-cache@10.4.3: - resolution: {integrity: sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==} - - lru-cache@5.1.1: - resolution: {integrity: sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==} - - luxon@1.28.1: - resolution: {integrity: sha512-gYHAa180mKrNIUJCbwpmD0aTu9kV0dREDrwNnuyFAsO1Wt0EVYSZelPnJlbj9HplzXX/YWXHFTL45kvZ53M0pw==} - - mailparser@3.7.1: - resolution: {integrity: sha512-RCnBhy5q8XtB3mXzxcAfT1huNqN93HTYYyL6XawlIKycfxM/rXPg9tXoZ7D46+SgCS1zxKzw+BayDQSvncSTTw==} - - mailsplit@5.4.0: - resolution: {integrity: sha512-wnYxX5D5qymGIPYLwnp6h8n1+6P6vz/MJn5AzGjZ8pwICWssL+CCQjWBIToOVHASmATot4ktvlLo6CyLfOXWYA==} - - make-error@1.3.6: - resolution: {integrity: sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==} - - memfs@4.11.1: - resolution: {integrity: sha512-LZcMTBAgqUUKNXZagcZxvXXfgF1bHX7Y7nQ0QyEiNbRJgE29GhgPd8Yna1VQcLlPiHt/5RFJMWYN9Uv/VPNvjQ==} - engines: {node: '>= 4.0.0'} - - merge-stream@2.0.0: - resolution: {integrity: sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==} - - merge2@1.4.1: - resolution: {integrity: sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==} - engines: {node: '>= 8'} - - micromatch@4.0.8: - resolution: {integrity: sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==} - engines: {node: '>=8.6'} - - mime-db@1.52.0: - resolution: {integrity: sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==} - engines: {node: '>= 0.6'} - - mime-types@2.1.35: - resolution: {integrity: sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==} - engines: {node: '>= 0.6'} - - mime@3.0.0: - resolution: {integrity: sha512-jSCU7/VB1loIWBZe14aEYHU/+1UMEHoaO7qxCOVJOw9GgH72VAWppxNcjU+x9a2k3GSIBXNKxXQFqRvvZ7vr3A==} - engines: {node: '>=10.0.0'} - hasBin: true - - mimic-fn@2.1.0: - resolution: {integrity: sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==} - engines: {node: '>=6'} - - mimic-fn@4.0.0: - resolution: {integrity: sha512-vqiC06CuhBTUdZH+RYl8sFrL096vA45Ok5ISO6sE/Mr1jRbGH4Csnhi8f3wKVl7x8mO4Au7Ir9D3Oyv1VYMFJw==} - engines: {node: '>=12'} - - minimatch@3.1.2: - resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==} - - minimatch@5.1.6: - resolution: {integrity: sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==} - engines: {node: '>=10'} - - minimatch@9.0.5: - resolution: {integrity: sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==} - engines: {node: '>=16 || 14 >=14.17'} - - minipass@7.1.2: - resolution: {integrity: sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==} - engines: {node: '>=16 || 14 >=14.17'} - - mocha@10.7.3: - resolution: {integrity: sha512-uQWxAu44wwiACGqjbPYmjo7Lg8sFrS3dQe7PP2FQI+woptP4vZXSMcfMyFL/e1yFEeEpV4RtyTpZROOKmxis+A==} - engines: {node: '>= 14.0.0'} - hasBin: true - - mock-json-schema@1.1.1: - resolution: {integrity: sha512-YV23vlsLP1EEOy0EviUvZTluXjLR+rhMzeayP2rcDiezj3RW01MhOSQkbQskdtg0K2fnGas5LKbSXgNjAOSX4A==} - - ms@2.1.2: - resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==} - - ms@2.1.3: - resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} - - ms@3.0.0-canary.1: - resolution: {integrity: sha512-kh8ARjh8rMN7Du2igDRO9QJnqCb2xYTJxyQYK7vJJS4TvLLmsbyhiKpSW+t+y26gyOyMd0riphX0GeWKU3ky5g==} - engines: {node: '>=12.13'} - - mz@2.7.0: - resolution: {integrity: sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==} - - neo-async@2.6.2: - resolution: {integrity: sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==} - - nise@6.0.0: - resolution: {integrity: sha512-K8ePqo9BFvN31HXwEtTNGzgrPpmvgciDsFz8aztFjt4LqKO/JeFD8tBOeuDiCMXrIl/m1YvfH8auSpxfaD09wg==} - - node-domexception@1.0.0: - resolution: {integrity: sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==} - engines: {node: '>=10.5.0'} - - node-fetch@3.3.2: - resolution: {integrity: sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - node-releases@2.0.18: - resolution: {integrity: sha512-d9VeXT4SJ7ZeOqGX6R5EM022wpL+eWPooLI+5UpWn2jCT1aosUQEhQP214x33Wkwx3JQMvIm+tIoVOdodFS40g==} - - nodemailer@6.9.13: - resolution: {integrity: sha512-7o38Yogx6krdoBf3jCAqnIN4oSQFx+fMa0I7dK1D+me9kBxx12D+/33wSb+fhOCtIxvYJ+4x4IMEhmhCKfAiOA==} - engines: {node: '>=6.0.0'} - - nodemailer@6.9.14: - resolution: {integrity: sha512-Dobp/ebDKBvz91sbtRKhcznLThrKxKt97GI2FAlAyy+fk19j73Uz3sBXolVtmcXjaorivqsbbbjDY+Jkt4/bQA==} - engines: {node: '>=6.0.0'} - - nodemon@3.1.4: - resolution: {integrity: sha512-wjPBbFhtpJwmIeY2yP7QF+UKzPfltVGtfce1g/bB15/8vCGZj8uxD62b/b9M9/WVgme0NZudpownKN+c0plXlQ==} - engines: {node: '>=10'} - hasBin: true - - normalize-path@3.0.0: - resolution: {integrity: sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==} - engines: {node: '>=0.10.0'} - - npm-run-path@4.0.1: - resolution: {integrity: sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==} - engines: {node: '>=8'} - - npm-run-path@5.3.0: - resolution: {integrity: sha512-ppwTtiJZq0O/ai0z7yfudtBpWIoxM8yE6nHi1X47eFR2EWORqfbu6CnPlNsjeN683eT0qG6H/Pyf9fCcvjnnnQ==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - nth-check@2.1.1: - resolution: {integrity: sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==} - - object-assign@4.1.1: - resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} - engines: {node: '>=0.10.0'} - - object-inspect@1.13.2: - resolution: {integrity: sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==} - engines: {node: '>= 0.4'} - - obuf@1.1.2: - resolution: {integrity: sha512-PX1wu0AmAdPqOL1mWhqmlOd8kOIZQwGZw6rh7uby9fTc5lhaOWFLX3I6R1hrF9k3zUY40e6igsLGkDXK92LJNg==} - - on-exit-leak-free@2.1.2: - resolution: {integrity: sha512-0eJJY6hXLGf1udHwfNftBqH+g73EU4B504nZeKpz1sYRKafAghwxEJunB2O7rDZkL4PGfsMVnTXZ2EjibbqcsA==} - engines: {node: '>=14.0.0'} - - once@1.4.0: - resolution: {integrity: sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==} - - onetime@5.1.2: - resolution: {integrity: sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==} - engines: {node: '>=6'} - - onetime@6.0.0: - resolution: {integrity: sha512-1FlR+gjXK7X+AsAHso35MnyN5KqGwJRi/31ft6x0M194ht7S+rWAvd7PHss9xSKMzE0asv1pyIHaJYq+BbacAQ==} - engines: {node: '>=12'} - - openapi-backend@5.10.6: - resolution: {integrity: sha512-vTjBRys/O4JIHdlRHUKZ7pxS+gwIJreAAU9dvYRFrImtPzQ5qxm5a6B8BTVT9m6I8RGGsShJv35MAc3Tu2/y/A==} - engines: {node: '>=12.0.0'} - - openapi-schema-validator@12.1.3: - resolution: {integrity: sha512-xTHOmxU/VQGUgo7Cm0jhwbklOKobXby+/237EG967+3TQEYJztMgX9Q5UE2taZKwyKPUq0j11dngpGjUuxz1hQ==} - - openapi-types@12.1.3: - resolution: {integrity: sha512-N4YtSYJqghVu4iek2ZUvcN/0aqH1kRDuNqzcycDxhOUpg7GdvLa2F3DgS6yBNhInhv2r/6I0Flkn7CqL8+nIcw==} - - p-limit@3.1.0: - resolution: {integrity: sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==} - engines: {node: '>=10'} - - p-locate@5.0.0: - resolution: {integrity: sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==} - engines: {node: '>=10'} - - p-retry@6.2.0: - resolution: {integrity: sha512-JA6nkq6hKyWLLasXQXUrO4z8BUZGUt/LjlJxx8Gb2+2ntodU/SS63YZ8b0LUTbQ8ZB9iwOfhEPhg4ykKnn2KsA==} - engines: {node: '>=16.17'} - - package-json-from-dist@1.0.0: - resolution: {integrity: sha512-dATvCeZN/8wQsGywez1mzHtTlP22H8OEfPrVMLNr4/eGa+ijtLn/6M5f0dY8UKNrC2O9UCU6SSoG3qRKnt7STw==} - - parse5-htmlparser2-tree-adapter@7.0.0: - resolution: {integrity: sha512-B77tOZrqqfUfnVcOrUvfdLbz4pu4RopLD/4vmu3HUPswwTA8OH0EMW9BlWR2B0RCoiZRAHEUu7IxeP1Pd1UU+g==} - - parse5-parser-stream@7.1.2: - resolution: {integrity: sha512-JyeQc9iwFLn5TbvvqACIF/VXG6abODeB3Fwmv/TGdLk2LfbWkaySGY72at4+Ty7EkPZj854u4CrICqNk2qIbow==} - - parse5@7.1.2: - resolution: {integrity: sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==} - - parseley@0.12.1: - resolution: {integrity: sha512-e6qHKe3a9HWr0oMRVDTRhKce+bRO8VGQR3NyVwcjwrbhMmFCX9KszEV35+rn4AdilFAq9VPxP/Fe1wC9Qjd2lw==} - - path-equal@1.2.5: - resolution: {integrity: sha512-i73IctDr3F2W+bsOWDyyVm/lqsXO47aY9nsFZUjTT/aljSbkxHxxCoyZ9UUrM8jK0JVod+An+rl48RCsvWM+9g==} - - path-exists@4.0.0: - resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==} - engines: {node: '>=8'} - - path-is-absolute@1.0.1: - resolution: {integrity: sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==} - engines: {node: '>=0.10.0'} - - path-key@3.1.1: - resolution: {integrity: sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==} - engines: {node: '>=8'} - - path-key@4.0.0: - resolution: {integrity: sha512-haREypq7xkM7ErfgIyA0z+Bj4AGKlMSdlQE2jvJo6huWD1EdkKYV+G/T4nq0YEF2vgTT8kqMFKo1uHn950r4SQ==} - engines: {node: '>=12'} - - path-parse@1.0.7: - resolution: {integrity: sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==} - - path-scurry@1.11.1: - resolution: {integrity: sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==} - engines: {node: '>=16 || 14 >=14.18'} - - path-to-regexp@6.2.2: - resolution: {integrity: sha512-GQX3SSMokngb36+whdpRXE+3f9V8UzyAorlYvOGx87ufGHehNTn5lCxrKtLyZ4Yl/wEKnNnr98ZzOwwDZV5ogw==} - - path-type@4.0.0: - resolution: {integrity: sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==} - engines: {node: '>=8'} - - pathval@2.0.0: - resolution: {integrity: sha512-vE7JKRyES09KiunauX7nd2Q9/L7lhok4smP9RZTDeD4MVs72Dp2qNFVz39Nz5a0FVEW0BJR6C0DYrq6unoziZA==} - engines: {node: '>= 14.16'} - - peberminta@0.9.0: - resolution: {integrity: sha512-XIxfHpEuSJbITd1H3EeQwpcZbTLHc+VVr8ANI9t5sit565tsI4/xK3KWTUFE2e6QiangUkh3B0jihzmGnNrRsQ==} - - pg-cloudflare@1.1.1: - resolution: {integrity: sha512-xWPagP/4B6BgFO+EKz3JONXv3YDgvkbVrGw2mTo3D6tVDQRh1e7cqVGvyR3BE+eQgAvx1XhW/iEASj4/jCWl3Q==} - - pg-connection-string@2.6.4: - resolution: {integrity: sha512-v+Z7W/0EO707aNMaAEfiGnGL9sxxumwLl2fJvCQtMn9Fxsg+lPpPkdcyBSv/KFgpGdYkMfn+EI1Or2EHjpgLCA==} - - pg-int8@1.0.1: - resolution: {integrity: sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw==} - engines: {node: '>=4.0.0'} - - pg-numeric@1.0.2: - resolution: {integrity: sha512-BM/Thnrw5jm2kKLE5uJkXqqExRUY/toLHda65XgFTBTFYZyopbKjBe29Ii3RbkvlsMoFwD+tHeGaCjjv0gHlyw==} - engines: {node: '>=4'} - - pg-pool@3.6.2: - resolution: {integrity: sha512-Htjbg8BlwXqSBQ9V8Vjtc+vzf/6fVUuak/3/XXKA9oxZprwW3IMDQTGHP+KDmVL7rtd+R1QjbnCFPuTHm3G4hg==} - peerDependencies: - pg: '>=8.0' - - pg-protocol@1.6.1: - resolution: {integrity: sha512-jPIlvgoD63hrEuihvIg+tJhoGjUsLPn6poJY9N5CnlPd91c2T18T/9zBtLxZSb1EhYxBRoZJtzScCaWlYLtktg==} - - pg-types@2.2.0: - resolution: {integrity: sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==} - engines: {node: '>=4'} - - pg-types@4.0.2: - resolution: {integrity: sha512-cRL3JpS3lKMGsKaWndugWQoLOCoP+Cic8oseVcbr0qhPzYD5DWXK+RZ9LY9wxRf7RQia4SCwQlXk0q6FCPrVng==} - engines: {node: '>=10'} - - pg@8.12.0: - resolution: {integrity: sha512-A+LHUSnwnxrnL/tZ+OLfqR1SxLN3c/pgDztZ47Rpbsd4jUytsTtwQo/TLPRzPJMp/1pbhYVhH9cuSZLAajNfjQ==} - engines: {node: '>= 8.0.0'} - peerDependencies: - pg-native: '>=3.0.1' - peerDependenciesMeta: - pg-native: - optional: true - - pgpass@1.0.5: - resolution: {integrity: sha512-FdW9r/jQZhSeohs1Z3sI1yxFQNFvMcnmfuj4WBMUTxOrAyLMaTcE1aAMBiTlbMNaXvBCQuVi0R7hd8udDSP7ug==} - - picocolors@1.1.0: - resolution: {integrity: sha512-TQ92mBOW0l3LeMeyLV6mzy/kWr8lkd/hp3mTg7wYK7zJhuBStmGMBG0BdeDZS/dZx1IukaX6Bk11zcln25o1Aw==} - - picomatch@2.3.1: - resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==} - engines: {node: '>=8.6'} - - pino-abstract-transport@1.2.0: - resolution: {integrity: sha512-Guhh8EZfPCfH+PMXAb6rKOjGQEoy0xlAIn+irODG5kgfYV+BQ0rGYYWTIel3P5mmyXqkYkPmdIkywsn6QKUR1Q==} - - pino-std-serializers@7.0.0: - resolution: {integrity: sha512-e906FRY0+tV27iq4juKzSYPbUj2do2X2JX4EzSca1631EB2QJQUqGbDuERal7LCtOpxl6x3+nvo9NPZcmjkiFA==} - - pino@9.2.0: - resolution: {integrity: sha512-g3/hpwfujK5a4oVbaefoJxezLzsDgLcNJeITvC6yrfwYeT9la+edCK42j5QpEQSQCZgTKapXvnQIdgZwvRaZug==} - hasBin: true - - pino@9.4.0: - resolution: {integrity: sha512-nbkQb5+9YPhQRz/BeQmrWpEknAaqjpAqRK8NwJpmrX/JHu7JuZC5G1CeAwJDJfGes4h+YihC6in3Q2nGb+Y09w==} - hasBin: true - - pirates@4.0.6: - resolution: {integrity: sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==} - engines: {node: '>= 6'} - - postcss-load-config@6.0.1: - resolution: {integrity: sha512-oPtTM4oerL+UXmx+93ytZVN82RrlY/wPUV8IeDxFrzIjXOLF1pN+EmKPLbubvKHT2HC20xXsCAH2Z+CKV6Oz/g==} - engines: {node: '>= 18'} - peerDependencies: - jiti: '>=1.21.0' - postcss: '>=8.0.9' - tsx: ^4.8.1 - yaml: ^2.4.2 - peerDependenciesMeta: - jiti: - optional: true - postcss: - optional: true - tsx: - optional: true - yaml: - optional: true - - postgres-array@2.0.0: - resolution: {integrity: sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA==} - engines: {node: '>=4'} - - postgres-array@3.0.2: - resolution: {integrity: sha512-6faShkdFugNQCLwucjPcY5ARoW1SlbnrZjmGl0IrrqewpvxvhSLHimCVzqeuULCbG0fQv7Dtk1yDbG3xv7Veog==} - engines: {node: '>=12'} - - postgres-bytea@1.0.0: - resolution: {integrity: sha512-xy3pmLuQqRBZBXDULy7KbaitYqLcmxigw14Q5sj8QBVLqEwXfeybIKVWiqAXTlcvdvb0+xkOtDbfQMOf4lST1w==} - engines: {node: '>=0.10.0'} - - postgres-bytea@3.0.0: - resolution: {integrity: sha512-CNd4jim9RFPkObHSjVHlVrxoVQXz7quwNFpz7RY1okNNme49+sVyiTvTRobiLV548Hx/hb1BG+iE7h9493WzFw==} - engines: {node: '>= 6'} - - postgres-date@1.0.7: - resolution: {integrity: sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q==} - engines: {node: '>=0.10.0'} - - postgres-date@2.1.0: - resolution: {integrity: sha512-K7Juri8gtgXVcDfZttFKVmhglp7epKb1K4pgrkLxehjqkrgPhfG6OO8LHLkfaqkbpjNRnra018XwAr1yQFWGcA==} - engines: {node: '>=12'} - - postgres-interval@1.2.0: - resolution: {integrity: sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==} - engines: {node: '>=0.10.0'} - - postgres-interval@3.0.0: - resolution: {integrity: sha512-BSNDnbyZCXSxgA+1f5UU2GmwhoI0aU5yMxRGO8CdFEcY2BQF9xm/7MqKnYoM1nJDk8nONNWDk9WeSmePFhQdlw==} - engines: {node: '>=12'} - - postgres-range@1.1.4: - resolution: {integrity: sha512-i/hbxIE9803Alj/6ytL7UHQxRvZkI9O4Sy+J3HGc4F4oo/2eQAjTSNJ0bfxyse3bH0nuVesCk+3IRLaMtG3H6w==} - - prevvy@8.0.1: - resolution: {integrity: sha512-ztl6JrbdlICy9cBOxN/aVcT8bp8G08SXMUPns/MJvHHMB6cc3goJiTQwTGLL0X2PoMUYr31YZgd6X6k87NU4OQ==} - - process-warning@3.0.0: - resolution: {integrity: sha512-mqn0kFRl0EoqhnL0GQ0veqFHyIN1yig9RHh/InzORTUiZHFRAur+aMtRkELNwGs9aNwKS6tg/An4NYBPGwvtzQ==} - - process-warning@4.0.0: - resolution: {integrity: sha512-/MyYDxttz7DfGMMHiysAsFE4qF+pQYAA8ziO/3NcRVrQ5fSk+Mns4QZA/oRPFzvcqNoVJXQNWNAsdwBXLUkQKw==} - - process@0.11.10: - resolution: {integrity: sha512-cdGef/drWFoydD1JsMzuFf8100nZl+GT+yacc2bEced5f9Rjk4z+WtFUTBu9PhOi9j/jfmBPu0mMEY4wIdAF8A==} - engines: {node: '>= 0.6.0'} - - proto3-json-serializer@2.0.2: - resolution: {integrity: sha512-SAzp/O4Yh02jGdRc+uIrGoe87dkN/XtwxfZ4ZyafJHymd79ozp5VG5nyZ7ygqPM5+cpLDjjGnYFUkngonyDPOQ==} - engines: {node: '>=14.0.0'} - - protobufjs@7.4.0: - resolution: {integrity: sha512-mRUWCc3KUU4w1jU8sGxICXH/gNS94DvI1gxqDvBzhj1JpcsimQkYiOJfwsPUykUI5ZaspFbSgmBLER8IrQ3tqw==} - engines: {node: '>=12.0.0'} - - proxy-addr@2.0.7: - resolution: {integrity: sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==} - engines: {node: '>= 0.10'} - - psl@1.9.0: - resolution: {integrity: sha512-E/ZsdU4HLs/68gYzgGTkMicWTLPdAftJLfJFlLUAAKZGkStNU72sZjT66SnMDVOfOWY/YAoiD7Jxa9iHvngcag==} - - pstree.remy@1.1.8: - resolution: {integrity: sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==} - - punycode.js@2.3.1: - resolution: {integrity: sha512-uxFIHU0YlHYhDQtV4R9J6a52SLx28BCjT+4ieh7IGbgwVJWO+km431c4yRlREUAsAmt/uMjQUyQHNEPf0M39CA==} - engines: {node: '>=6'} - - punycode@2.3.1: - resolution: {integrity: sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==} - engines: {node: '>=6'} - - qs@6.13.0: - resolution: {integrity: sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==} - engines: {node: '>=0.6'} - - querystringify@2.2.0: - resolution: {integrity: sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==} - - queue-microtask@1.2.3: - resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==} - - quick-format-unescaped@4.0.4: - resolution: {integrity: sha512-tYC1Q1hgyRuHgloV/YXs2w15unPVh8qfu/qCTfhTYamaw7fyhumKa2yGpdSo87vY32rIclj+4fWYQXUMs9EHvg==} - - randombytes@2.1.0: - resolution: {integrity: sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==} - - rate-limiter-flexible@5.0.3: - resolution: {integrity: sha512-lWx2y8NBVlTOLPyqs+6y7dxfEpT6YFqKy3MzWbCy95sTTOhOuxufP2QvRyOHpfXpB9OUJPbVLybw3z3AVAS5fA==} - - readable-stream@3.6.2: - resolution: {integrity: sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==} - engines: {node: '>= 6'} - - readable-stream@4.5.2: - resolution: {integrity: sha512-yjavECdqeZ3GLXNgRXgeQEdz9fvDDkNKyHnbHRFtOr7/LcfgBcmct7t/ET+HaCTqfh06OzoAxrkN/IfjJBVe+g==} - engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} - - readdirp@3.6.0: - resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==} - engines: {node: '>=8.10.0'} - - real-require@0.2.0: - resolution: {integrity: sha512-57frrGM/OCTLqLOAh0mhVA9VBMHd+9U7Zb2THMGdBUoZVOtGbJzjxsYGDJ3A9AYYCP4hn6y1TVbaOfzWtm5GFg==} - engines: {node: '>= 12.13.0'} - - regenerate-unicode-properties@10.1.1: - resolution: {integrity: sha512-X007RyZLsCJVVrjgEFVpLUTZwyOZk3oiL75ZcuYjlIWd6rNJtOjkBwQc5AsRrpbKVkxN6sklw/k/9m2jJYOf8Q==} - engines: {node: '>=4'} - - regenerate@1.4.2: - resolution: {integrity: sha512-zrceR/XhGYU/d/opr2EKO7aRHUeiBI8qjtfHqADTwZd6Szfy16la6kqD0MIUs5z5hx6AaKa+PixpPrR289+I0A==} - - regenerator-runtime@0.14.1: - resolution: {integrity: sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==} - - regenerator-transform@0.15.2: - resolution: {integrity: sha512-hfMp2BoF0qOk3uc5V20ALGDS2ddjQaLrdl7xrGXvAIow7qeWRM2VA2HuCHkUKk9slq3VwEwLNK3DFBqDfPGYtg==} - - regexpu-core@5.3.2: - resolution: {integrity: sha512-RAM5FlZz+Lhmo7db9L298p2vHP5ZywrVXmVXpmAD9GuL5MPH6t9ROw1iA/wfHkQ76Qe7AaPF0nGuim96/IrQMQ==} - engines: {node: '>=4'} - - regjsparser@0.9.1: - resolution: {integrity: sha512-dQUtn90WanSNl+7mQKcXAgZxvUe7Z0SqXlgzv0za4LwiUhyzBC58yQO3liFoUgu8GiJVInAhJjkj1N0EtQ5nkQ==} - hasBin: true - - require-directory@2.1.1: - resolution: {integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==} - engines: {node: '>=0.10.0'} - - require-from-string@2.0.2: - resolution: {integrity: sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==} - engines: {node: '>=0.10.0'} - - requires-port@1.0.0: - resolution: {integrity: sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==} - - resolve-from@5.0.0: - resolution: {integrity: sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==} - engines: {node: '>=8'} - - resolve-pkg-maps@1.0.0: - resolution: {integrity: sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==} - - resolve@1.22.8: - resolution: {integrity: sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==} - hasBin: true - - ret@0.4.3: - resolution: {integrity: sha512-0f4Memo5QP7WQyUEAYUO3esD/XjOc3Zjjg5CPsAq1p8sIu0XPeMbHJemKA0BO7tV0X7+A0FoEpbmHXWxPyD3wQ==} - engines: {node: '>=10'} - - retry@0.13.1: - resolution: {integrity: sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg==} - engines: {node: '>= 4'} - - reusify@1.0.4: - resolution: {integrity: sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==} - engines: {iojs: '>=1.0.0', node: '>=0.10.0'} - - rfdc@1.4.1: - resolution: {integrity: sha512-q1b3N5QkRUWUl7iyylaaj3kOpIT0N2i9MqIEQXP73GVsN9cw3fdx8X63cEmWhJGi2PPCF23Ijp7ktmd39rawIA==} - - rollup@4.21.2: - resolution: {integrity: sha512-e3TapAgYf9xjdLvKQCkQTnbTKd4a6jwlpQSJJFokHGaX2IVjoEqkIIhiQfqsi0cdwlOD+tQGuOd5AJkc5RngBw==} - engines: {node: '>=18.0.0', npm: '>=8.0.0'} - hasBin: true - - run-parallel@1.2.0: - resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} - - rxjs@7.8.1: - resolution: {integrity: sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg==} - - safe-buffer@5.2.1: - resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} - - safe-regex2@3.1.0: - resolution: {integrity: sha512-RAAZAGbap2kBfbVhvmnTFv73NWLMvDGOITFYTZBAaY8eR+Ir4ef7Up/e7amo+y1+AH+3PtLkrt9mvcTsG9LXug==} - - safe-stable-stringify@2.5.0: - resolution: {integrity: sha512-b3rppTKm9T+PsVCBEOUR46GWI7fdOs00VKZ1+9c1EWDaDMvjQc6tUwuFyIprgGgTcWoVHSKrU8H31ZHA2e0RHA==} - engines: {node: '>=10'} - - safer-buffer@2.1.2: - resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==} - - schema-utils@3.3.0: - resolution: {integrity: sha512-pN/yOAvcC+5rQ5nERGuwrjLlYvLTbCibnZ1I7B1LaiAz9BRBlE9GMgE/eqV30P7aJQUf7Ddimy/RsbYO/GrVGg==} - engines: {node: '>= 10.13.0'} - - secure-json-parse@2.7.0: - resolution: {integrity: sha512-6aU+Rwsezw7VR8/nyvKTx8QpWH9FrcYiXXlqC4z5d5XQBDRqtbfsRjnwGyqbi3gddNtWHuEk9OANUotL26qKUw==} - - selderee@0.11.0: - resolution: {integrity: sha512-5TF+l7p4+OsnP8BCCvSyZiSPc4x4//p5uPwK8TCnVPJYRmU2aYKMpOXvw8zM5a5JvuuCGN1jmsMwuU2W02ukfA==} - - semver@6.3.1: - resolution: {integrity: sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==} - hasBin: true - - semver@7.6.3: - resolution: {integrity: sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==} - engines: {node: '>=10'} - hasBin: true - - sequin@0.1.1: - resolution: {integrity: sha512-hJWMZRwP75ocoBM+1/YaCsvS0j5MTPeBHJkS2/wruehl9xwtX30HlDF1Gt6UZ8HHHY8SJa2/IL+jo+JJCd59rA==} - engines: {node: '>=0.4.0'} - - serialize-javascript@6.0.2: - resolution: {integrity: sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g==} - - set-cookie-parser@2.7.0: - resolution: {integrity: sha512-lXLOiqpkUumhRdFF3k1osNXCy9akgx/dyPZ5p8qAg9seJzXr5ZrlqZuWIMuY6ejOsVLE6flJ5/h3lsn57fQ/PQ==} - - set-function-length@1.2.2: - resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==} - engines: {node: '>= 0.4'} - - setprototypeof@1.2.0: - resolution: {integrity: sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==} - - sharp@0.33.5: - resolution: {integrity: sha512-haPVm1EkS9pgvHrQ/F3Xy+hgcuMV0Wm9vfIBSiwZ05k+xgb0PkBQpGsAA/oWdDobNaZTH5ppvHtzCFbnSEwHVw==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - - shebang-command@2.0.0: - resolution: {integrity: sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==} - engines: {node: '>=8'} - - shebang-regex@3.0.0: - resolution: {integrity: sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==} - engines: {node: '>=8'} - - shell-quote@1.8.1: - resolution: {integrity: sha512-6j1W9l1iAs/4xYBI1SYOVZyFcCis9b4KCLQ8fgAGG07QvzaRLVVRQvAy85yNmmZSjYjg4MWh4gNvlPujU/5LpA==} - - side-channel@1.0.6: - resolution: {integrity: sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==} - engines: {node: '>= 0.4'} - - signal-exit@3.0.7: - resolution: {integrity: sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==} - - signal-exit@4.1.0: - resolution: {integrity: sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==} - engines: {node: '>=14'} - - simple-swizzle@0.2.2: - resolution: {integrity: sha512-JA//kQgZtbuY83m+xT+tXJkmJncGMTFT+C+g2h2R9uxkYIrE2yy9sgmcLhCnw57/WSD+Eh3J97FPEDFnbXnDUg==} - - simple-update-notifier@2.0.0: - resolution: {integrity: sha512-a2B9Y0KlNXl9u/vsW6sTIu9vGEpfKu2wRV6l1H3XEas/0gUIzGzBoP/IouTcUQbm9JWZLH3COxyn03TYlFax6w==} - engines: {node: '>=10'} - - sinon@18.0.0: - resolution: {integrity: sha512-+dXDXzD1sBO6HlmZDd7mXZCR/y5ECiEiGCBSGuFD/kZ0bDTofPYc6JaeGmPSF+1j1MejGUWkORbYOLDyvqCWpA==} - - slash@3.0.0: - resolution: {integrity: sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==} - engines: {node: '>=8'} - - slugify@1.6.6: - resolution: {integrity: sha512-h+z7HKHYXj6wJU+AnS/+IH8Uh9fdcX1Lrhg1/VMdf9PwoBQXFcXiAdsy2tSK0P6gKwJLXp02r90ahUCqHk9rrw==} - engines: {node: '>=8.0.0'} - - smart-buffer@4.2.0: - resolution: {integrity: sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg==} - engines: {node: '>= 6.0.0', npm: '>= 3.0.0'} - - socks@2.8.3: - resolution: {integrity: sha512-l5x7VUUWbjVFbafGLxPWkYsHIhEvmF85tbIeFZWc8ZPtoMyybuEhL7Jye/ooC4/d48FgOjSJXgsF/AJPYCW8Zw==} - engines: {node: '>= 10.0.0', npm: '>= 3.0.0'} - - sonic-boom@4.1.0: - resolution: {integrity: sha512-NGipjjRicyJJ03rPiZCJYjwlsuP2d1/5QUviozRXC7S3WdVWNK5e3Ojieb9CCyfhq2UC+3+SRd9nG3I2lPRvUw==} - - source-map-js@1.2.0: - resolution: {integrity: sha512-itJW8lvSA0TXEphiRoawsCksnlf8SyvmFzIhltqAHluXd88pkCd+cXJVHTDwdCr0IzwptSm035IHQktUu1QUMg==} - engines: {node: '>=0.10.0'} - - source-map-loader@4.0.2: - resolution: {integrity: sha512-oYwAqCuL0OZhBoSgmdrLa7mv9MjommVMiQIWgcztf+eS4+8BfcUee6nenFnDhKOhzAVnk5gpZdfnz1iiBv+5sg==} - engines: {node: '>= 14.15.0'} - peerDependencies: - webpack: ^5.72.1 - - source-map-support@0.5.21: - resolution: {integrity: sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==} - - source-map@0.6.1: - resolution: {integrity: sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==} - engines: {node: '>=0.10.0'} - - source-map@0.7.4: - resolution: {integrity: sha512-l3BikUxvPOcn5E74dZiq5BGsTb5yEwhaTSzccU6t4sDOH8NWJCstKO5QT2CvtFoK6F0saL7p9xHAqHOlCPJygA==} - engines: {node: '>= 8'} - - source-map@0.8.0-beta.0: - resolution: {integrity: sha512-2ymg6oRBpebeZi9UUNsgQ89bhx01TcTkmNTGnNO88imTmbSgy4nfujrgVEFKWpMTEGA11EDkTt7mqObTPdigIA==} - engines: {node: '>= 8'} - - spawn-command@0.0.2: - resolution: {integrity: sha512-zC8zGoGkmc8J9ndvml8Xksr1Amk9qBujgbF0JAIWO7kXr43w0h/0GJNM/Vustixu+YE8N/MTrQ7N31FvHUACxQ==} - - split2@4.2.0: - resolution: {integrity: sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==} - engines: {node: '>= 10.x'} - - sprintf-js@1.1.3: - resolution: {integrity: sha512-Oo+0REFV59/rz3gfJNKQiBlwfHaSESl1pcGyABQsnnIfWOFt6JNj5gCog2U6MLZ//IGYD+nA8nI+mTShREReaA==} - - statuses@2.0.1: - resolution: {integrity: sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==} - engines: {node: '>= 0.8'} - - stream-browserify@3.0.0: - resolution: {integrity: sha512-H73RAHsVBapbim0tU2JwwOiXUj+fikfiaoYAKHF3VJfA0pe2BCzkhAHBlLG6REzE+2WNZcxOXjK7lkso+9euLA==} - - string-width@4.2.3: - resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} - engines: {node: '>=8'} - - string-width@5.1.2: - resolution: {integrity: sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==} - engines: {node: '>=12'} - - string_decoder@1.3.0: - resolution: {integrity: sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==} - - strip-ansi@6.0.1: - resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} - engines: {node: '>=8'} - - strip-ansi@7.1.0: - resolution: {integrity: sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==} - engines: {node: '>=12'} - - strip-final-newline@2.0.0: - resolution: {integrity: sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==} - engines: {node: '>=6'} - - strip-final-newline@3.0.0: - resolution: {integrity: sha512-dOESqjYr96iWYylGObzd39EuNTa5VJxyvVAEm5Jnh7KGo75V43Hk1odPQkNDyXNmUR6k+gEiDVXnjB8HJ3crXw==} - engines: {node: '>=12'} - - strip-json-comments@3.1.1: - resolution: {integrity: sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==} - engines: {node: '>=8'} - - strnum@1.0.5: - resolution: {integrity: sha512-J8bbNyKKXl5qYcR36TIO8W3mVGVHrmmxsd5PAItGkmyzwJvybiw2IVq5nqd0i4LSNSkB/sx9VHllbfFdr9k1JA==} - - sucrase@3.35.0: - resolution: {integrity: sha512-8EbVDiu9iN/nESwxeSxDKe0dunta1GOlHufmSSXxMD2z2/tMZpDMpvXQGsc+ajGo8y2uYUmixaSRUc/QPoQ0GA==} - engines: {node: '>=16 || 14 >=14.17'} - hasBin: true - - supports-color@5.5.0: - resolution: {integrity: sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==} - engines: {node: '>=4'} - - supports-color@7.2.0: - resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==} - engines: {node: '>=8'} - - supports-color@8.1.1: - resolution: {integrity: sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==} - engines: {node: '>=10'} - - supports-preserve-symlinks-flag@1.0.0: - resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==} - engines: {node: '>= 0.4'} - - swagger-editor-dist@4.13.1: - resolution: {integrity: sha512-rCalVoyq7h1lZa/aJaHux8JTrEAmGHdpHqVzJH/rQeP35MByQzV709RcqGMRjgN/oONFoUVf35qCiVZZITC41A==} - - swagger-ui-dist@5.17.14: - resolution: {integrity: sha512-CVbSfaLpstV65OnSjbXfVd6Sta3q3F7Cj/yYuvHMp1P90LztOLs6PfUnKEVAeiIVQt9u2SaPwv0LiH/OyMjHRw==} - - swc-loader@0.2.6: - resolution: {integrity: sha512-9Zi9UP2YmDpgmQVbyOPJClY0dwf58JDyDMQ7uRc4krmc72twNI2fvlBWHLqVekBpPc7h5NJkGVT1zNDxFrqhvg==} - peerDependencies: - '@swc/core': ^1.2.147 - webpack: '>=2' - - tapable@2.2.1: - resolution: {integrity: sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ==} - engines: {node: '>=6'} - - terser-webpack-plugin@5.3.10: - resolution: {integrity: sha512-BKFPWlPDndPs+NGGCr1U59t0XScL5317Y0UReNrHaw9/FwhPENlq6bfgs+4yPfyP51vqC1bQ4rp1EfXW5ZSH9w==} - engines: {node: '>= 10.13.0'} - peerDependencies: - '@swc/core': '*' - esbuild: '*' - uglify-js: '*' - webpack: ^5.1.0 - peerDependenciesMeta: - '@swc/core': - optional: true - esbuild: - optional: true - uglify-js: - optional: true - - terser@5.31.6: - resolution: {integrity: sha512-PQ4DAriWzKj+qgehQ7LK5bQqCFNMmlhjR2PFFLuqGCpuCAauxemVBWwWOxo3UIwWQx8+Pr61Df++r76wDmkQBg==} - engines: {node: '>=10'} - hasBin: true - - thenify-all@1.6.0: - resolution: {integrity: sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==} - engines: {node: '>=0.8'} - - thenify@3.3.1: - resolution: {integrity: sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==} - - thingies@1.21.0: - resolution: {integrity: sha512-hsqsJsFMsV+aD4s3CWKk85ep/3I9XzYV/IXaSouJMYIoDlgyi11cBhsqYe9/geRfB0YIikBQg6raRaM+nIMP9g==} - engines: {node: '>=10.18'} - peerDependencies: - tslib: ^2 - - thread-stream@3.1.0: - resolution: {integrity: sha512-OqyPZ9u96VohAyMfJykzmivOrY2wfMSf3C5TtFJVgN+Hm6aj+voFhlK+kZEIv2FBh1X6Xp3DlnCOfEQ3B2J86A==} - - tlds@1.252.0: - resolution: {integrity: sha512-GA16+8HXvqtfEnw/DTcwB0UU354QE1n3+wh08oFjr6Znl7ZLAeUgYzCcK+/CCrOyE0vnHR8/pu3XXG3vDijXpQ==} - hasBin: true - - to-fast-properties@2.0.0: - resolution: {integrity: sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==} - engines: {node: '>=4'} - - to-regex-range@5.0.1: - resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} - engines: {node: '>=8.0'} - - toad-cache@3.7.0: - resolution: {integrity: sha512-/m8M+2BJUpoJdgAHoG+baCwBT+tf2VraSfkBgl0Y00qIWt41DJ8R5B8nsEw0I58YwF5IZH6z24/2TobDKnqSWw==} - engines: {node: '>=12'} - - toidentifier@1.0.1: - resolution: {integrity: sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==} - engines: {node: '>=0.6'} - - touch@3.1.1: - resolution: {integrity: sha512-r0eojU4bI8MnHr8c5bNo7lJDdI2qXlWWJk6a9EAFG7vbhTjElYhBVS3/miuE0uOuoLdb8Mc/rVfsmm6eo5o9GA==} - hasBin: true - - tough-cookie@4.1.4: - resolution: {integrity: sha512-Loo5UUvLD9ScZ6jh8beX1T6sO1w2/MpCRpEP7V280GKMVUQ0Jzar2U3UJPsrdbziLEMMhu3Ujnq//rhiFuIeag==} - engines: {node: '>=6'} - - tr46@1.0.1: - resolution: {integrity: sha512-dTpowEjclQ7Kgx5SdBkqRzVhERQXov8/l9Ft9dVM9fmg0W0KQSVaXX9T4i6twCPNtYiZM53lpSSUAwJbFPOHxA==} - - tree-dump@1.0.2: - resolution: {integrity: sha512-dpev9ABuLWdEubk+cIaI9cHwRNNDjkBBLXTwI4UCUFdQ5xXKqNXoK4FEciw/vxf+NQ7Cb7sGUyeUtORvHIdRXQ==} - engines: {node: '>=10.0'} - peerDependencies: - tslib: '2' - - tree-kill@1.2.2: - resolution: {integrity: sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==} - hasBin: true - - ts-interface-checker@0.1.13: - resolution: {integrity: sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==} - - ts-json-schema-generator@2.3.0: - resolution: {integrity: sha512-t4lBQAwZc0sOJq9LJt3NgbznIcslVnm0JeEMFq8qIRklpMRY8jlYD0YmnRWbqBKANxkby91P1XanSSlSOFpUmg==} - engines: {node: '>=18.0.0'} - hasBin: true - - ts-node@10.9.2: - resolution: {integrity: sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==} - hasBin: true - peerDependencies: - '@swc/core': '>=1.2.50' - '@swc/wasm': '>=1.2.50' - '@types/node': '*' - typescript: '>=2.7' - peerDependenciesMeta: - '@swc/core': - optional: true - '@swc/wasm': - optional: true - - tslib@2.7.0: - resolution: {integrity: sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA==} - - tsup@8.2.4: - resolution: {integrity: sha512-akpCPePnBnC/CXgRrcy72ZSntgIEUa1jN0oJbbvpALWKNOz1B7aM+UVDWGRGIO/T/PZugAESWDJUAb5FD48o8Q==} - engines: {node: '>=18'} - hasBin: true - peerDependencies: - '@microsoft/api-extractor': ^7.36.0 - '@swc/core': ^1 - postcss: ^8.4.12 - typescript: '>=4.5.0' - peerDependenciesMeta: - '@microsoft/api-extractor': - optional: true - '@swc/core': - optional: true - postcss: - optional: true - typescript: - optional: true - - tsx@4.19.0: - resolution: {integrity: sha512-bV30kM7bsLZKZIOCHeMNVMJ32/LuJzLVajkQI/qf92J2Qr08ueLQvW00PUZGiuLPP760UINwupgUj8qrSCPUKg==} - engines: {node: '>=18.0.0'} - hasBin: true - - tunnel-agent@0.6.0: - resolution: {integrity: sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==} - - type-detect@4.0.8: - resolution: {integrity: sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==} - engines: {node: '>=4'} - - type-detect@4.1.0: - resolution: {integrity: sha512-Acylog8/luQ8L7il+geoSxhEkazvkslg7PSNKOX59mbB9cOveP5aq9h74Y7YU8yDpJwetzQQrfIwtf4Wp4LKcw==} - engines: {node: '>=4'} - - typescript-json-schema@0.65.1: - resolution: {integrity: sha512-tuGH7ff2jPaUYi6as3lHyHcKpSmXIqN7/mu50x3HlYn0EHzLpmt3nplZ7EuhUkO0eqDRc9GqWNkfjgBPIS9kxg==} - hasBin: true - - typescript@5.5.4: - resolution: {integrity: sha512-Mtq29sKDAEYP7aljRgtPOpTvOfbwRWlS6dPRzwjdE+C0R4brX/GUyhHSecbHMFLNBLcJIPt9nl9yG5TZ1weH+Q==} - engines: {node: '>=14.17'} - hasBin: true - - uc.micro@2.1.0: - resolution: {integrity: sha512-ARDJmphmdvUk6Glw7y9DQ2bFkKBHwQHLi2lsaH6PPmz/Ka9sFOBsBluozhDltWmnv9u/cF6Rt87znRTPV+yp/A==} - - undefsafe@2.0.5: - resolution: {integrity: sha512-WxONCrssBM8TSPRqN5EmsjVrsv4A8X12J4ArBiiayv3DyyG3ZlIg6yysuuSYdZsVz3TKcTg2fd//Ujd4CHV1iA==} - - undici-types@5.26.5: - resolution: {integrity: sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==} - - undici-types@6.19.8: - resolution: {integrity: sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==} - - undici@6.19.8: - resolution: {integrity: sha512-U8uCCl2x9TK3WANvmBavymRzxbfFYG+tAu+fgx3zxQy3qdagQqBLwJVrdyO1TBfUXvfKveMKJZhpvUYoOjM+4g==} - engines: {node: '>=18.17'} - - unicode-canonical-property-names-ecmascript@2.0.0: - resolution: {integrity: sha512-yY5PpDlfVIU5+y/BSCxAJRBIS1Zc2dDG3Ujq+sR0U+JjUevW2JhocOF+soROYDSaAezOzOKuyyixhD6mBknSmQ==} - engines: {node: '>=4'} - - unicode-match-property-ecmascript@2.0.0: - resolution: {integrity: sha512-5kaZCrbp5mmbz5ulBkDkbY0SsPOjKqVS35VpL9ulMPfSl0J0Xsm+9Evphv9CoIZFwre7aJoa94AY6seMKGVN5Q==} - engines: {node: '>=4'} - - unicode-match-property-value-ecmascript@2.1.0: - resolution: {integrity: sha512-qxkjQt6qjg/mYscYMC0XKRn3Rh0wFPlfxB0xkt9CfyTvpX1Ra0+rAmdX2QyAobptSEvuy4RtpPRui6XkV+8wjA==} - engines: {node: '>=4'} - - unicode-property-aliases-ecmascript@2.1.0: - resolution: {integrity: sha512-6t3foTQI9qne+OZoVQB/8x8rk2k1eVy1gRXhV3oFQ5T6R1dqQ1xtin3XqSlx3+ATBkliTaR/hHyJBm+LVPNM8w==} - engines: {node: '>=4'} - - unionfs@4.5.4: - resolution: {integrity: sha512-qI3RvJwwdFcWUdZz1dWgAyLSfGlY2fS2pstvwkZBUTnkxjcnIvzriBLtqJTKz9FtArAvJeiVCqHlxhOw8Syfyw==} - - universalify@0.2.0: - resolution: {integrity: sha512-CJ1QgKmNg3CwvAv/kOFmtnEN05f0D/cn9QntgNOQlQF9dgvVTHj3t+8JPdjqawCHk7V/KA+fbUqzZ9XWhcqPUg==} - engines: {node: '>= 4.0.0'} - - update-browserslist-db@1.1.0: - resolution: {integrity: sha512-EdRAaAyk2cUE1wOf2DkEhzxqOQvFOoRJFNS6NeyJ01Gp2beMRpBAINjM2iDXE3KCuKhwnvHIQCJm6ThL2Z+HzQ==} - hasBin: true - peerDependencies: - browserslist: '>= 4.21.0' - - uri-js@4.4.1: - resolution: {integrity: sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==} - - url-parse@1.5.10: - resolution: {integrity: sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ==} - - util-deprecate@1.0.2: - resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==} - - uuid@9.0.1: - resolution: {integrity: sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==} - hasBin: true - - v8-compile-cache-lib@3.0.1: - resolution: {integrity: sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==} - - watchpack@2.4.2: - resolution: {integrity: sha512-TnbFSbcOCcDgjZ4piURLCbJ3nJhznVh9kw6F6iokjiFPl8ONxe9A6nMDVXDiNbrSfLILs6vB07F7wLBrwPYzJw==} - engines: {node: '>=10.13.0'} - - web-streams-polyfill@3.3.3: - resolution: {integrity: sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw==} - engines: {node: '>= 8'} - - webidl-conversions@4.0.2: - resolution: {integrity: sha512-YQ+BmxuTgd6UXZW3+ICGfyqRyHXVlD5GtQr5+qjiNW7bF0cqrzX500HVXPBOvgXb5YnzDd+h0zqyv61KUD7+Sg==} - - webpack-sources@3.2.3: - resolution: {integrity: sha512-/DyMEOrDgLKKIG0fmvtz+4dUX/3Ghozwgm6iPp8KRhvn+eQf9+Q7GWxVNMk3+uCPWfdXYC4ExGBckIXdFEfH1w==} - engines: {node: '>=10.13.0'} - - webpack@5.94.0: - resolution: {integrity: sha512-KcsGn50VT+06JH/iunZJedYGUJS5FGjow8wb9c0v5n1Om8O1g4L6LjtfxwlXIATopoQu+vOXXa7gYisWxCoPyg==} - engines: {node: '>=10.13.0'} - hasBin: true - peerDependencies: - webpack-cli: '*' - peerDependenciesMeta: - webpack-cli: - optional: true - - websocket-driver@0.7.4: - resolution: {integrity: sha512-b17KeDIQVjvb0ssuSDF2cYXSg2iztliJ4B9WdsuB6J952qCPKmnVq4DyW5motImXHDC1cBT/1UezrJVsKw5zjg==} - engines: {node: '>=0.8.0'} - - websocket-extensions@0.1.4: - resolution: {integrity: sha512-OqedPIGOfsDlo31UNwYbCFMSaO9m9G/0faIHj5/dZFDMFqPTcx6UwqyOy3COEaEOg/9VsGIpdqn62W5KhoKSpg==} - engines: {node: '>=0.8.0'} - - whatwg-encoding@3.1.1: - resolution: {integrity: sha512-6qN4hJdMwfYBtE3YBTTHhoeuUrDBPZmbQaxWAqSALV/MeEnR5z1xd8UKud2RAkFoPkmB+hli1TZSnyi84xz1vQ==} - engines: {node: '>=18'} - - whatwg-mimetype@4.0.0: - resolution: {integrity: sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg==} - engines: {node: '>=18'} - - whatwg-url@7.1.0: - resolution: {integrity: sha512-WUu7Rg1DroM7oQvGWfOiAK21n74Gg+T4elXEQYkOhtyLeWiJFoOGLXPKI/9gzIie9CtwVLm8wtw6YJdKyxSjeg==} - - which@1.3.1: - resolution: {integrity: sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==} - hasBin: true - - which@2.0.2: - resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==} - engines: {node: '>= 8'} - hasBin: true - - which@4.0.0: - resolution: {integrity: sha512-GlaYyEb07DPxYCKhKzplCWBJtvxZcZMrL+4UkrTSJHHPyZU4mYYTv3qaOe77H7EODLSSopAUFAc6W8U4yqvscg==} - engines: {node: ^16.13.0 || >=18.0.0} - hasBin: true - - workerpool@6.5.1: - resolution: {integrity: sha512-Fs4dNYcsdpYSAfVxhnl1L5zTksjvOJxtC5hzMNl+1t9B8hTJTdKDyZ5ju7ztgPy+ft9tBFXoOlDNiOT9WUXZlA==} - - wrap-ansi@7.0.0: - resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} - engines: {node: '>=10'} - - wrap-ansi@8.1.0: - resolution: {integrity: sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==} - engines: {node: '>=12'} - - wrappy@1.0.2: - resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==} - - xpath@0.0.34: - resolution: {integrity: sha512-FxF6+rkr1rNSQrhUNYrAFJpRXNzlDoMxeXN5qI84939ylEv3qqPFKa85Oxr6tDaJKqwW6KKyo2v26TSv3k6LeA==} - engines: {node: '>=0.6.0'} - - xtend@4.0.2: - resolution: {integrity: sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==} - engines: {node: '>=0.4'} - - y18n@5.0.8: - resolution: {integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==} - engines: {node: '>=10'} - - yallist@3.1.1: - resolution: {integrity: sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==} - - yaml@2.5.0: - resolution: {integrity: sha512-2wWLbGbYDiSqqIKoPjar3MPgB94ErzCtrNE1FdqGuaO0pi2JGjmE8aW8TDZwzU7vuxcGRdL/4gPQwQ7hD5AMSw==} - engines: {node: '>= 14'} - hasBin: true - - yargs-parser@20.2.9: - resolution: {integrity: sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==} - engines: {node: '>=10'} - - yargs-parser@21.1.1: - resolution: {integrity: sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==} - engines: {node: '>=12'} - - yargs-unparser@2.0.0: - resolution: {integrity: sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==} - engines: {node: '>=10'} - - yargs@16.2.0: - resolution: {integrity: sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==} - engines: {node: '>=10'} - - yargs@17.7.2: - resolution: {integrity: sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==} - engines: {node: '>=12'} - - yn@3.1.1: - resolution: {integrity: sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==} - engines: {node: '>=6'} - - yocto-queue@0.1.0: - resolution: {integrity: sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==} - engines: {node: '>=10'} - -snapshots: - - '@ampproject/remapping@2.3.0': - dependencies: - '@jridgewell/gen-mapping': 0.3.5 - '@jridgewell/trace-mapping': 0.3.25 - - '@apidevtools/json-schema-ref-parser@11.7.0': - dependencies: - '@jsdevtools/ono': 7.1.3 - '@types/json-schema': 7.0.15 - js-yaml: 4.1.0 - - '@aws-crypto/crc32@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/crc32c@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/sha1-browser@5.2.0': - dependencies: - '@aws-crypto/supports-web-crypto': 5.2.0 - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-locate-window': 3.568.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-crypto/sha256-browser@5.2.0': - dependencies: - '@aws-crypto/sha256-js': 5.2.0 - '@aws-crypto/supports-web-crypto': 5.2.0 - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-locate-window': 3.568.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-crypto/sha256-js@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/supports-web-crypto@5.2.0': - dependencies: - tslib: 2.7.0 - - '@aws-crypto/util@5.2.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-sdk/client-s3@3.637.0': - dependencies: - '@aws-crypto/sha1-browser': 5.2.0 - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-bucket-endpoint': 3.620.0 - '@aws-sdk/middleware-expect-continue': 3.620.0 - '@aws-sdk/middleware-flexible-checksums': 3.620.0 - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-location-constraint': 3.609.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-sdk-s3': 3.635.0 - '@aws-sdk/middleware-ssec': 3.609.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/signature-v4-multi-region': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@aws-sdk/xml-builder': 3.609.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/eventstream-serde-browser': 3.0.6 - '@smithy/eventstream-serde-config-resolver': 3.0.3 - '@smithy/eventstream-serde-node': 3.0.5 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-blob-browser': 3.1.2 - '@smithy/hash-node': 3.0.3 - '@smithy/hash-stream-node': 3.1.2 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/md5-js': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-stream': 3.1.3 - '@smithy/util-utf8': 3.0.0 - '@smithy/util-waiter': 3.1.2 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sso@3.637.0': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sts@3.637.0': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/core@3.635.0': - dependencies: - '@smithy/core': 2.4.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-middleware': 3.0.3 - fast-xml-parser: 4.4.1 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-env@3.620.1': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-http@3.635.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-stream': 3.1.3 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-ini@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/credential-provider-env': 3.620.1 - '@aws-sdk/credential-provider-http': 3.635.0 - '@aws-sdk/credential-provider-process': 3.620.1 - '@aws-sdk/credential-provider-sso': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/credential-provider-web-identity': 3.621.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - aws-crt - - '@aws-sdk/credential-provider-node@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/credential-provider-env': 3.620.1 - '@aws-sdk/credential-provider-http': 3.635.0 - '@aws-sdk/credential-provider-ini': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/credential-provider-process': 3.620.1 - '@aws-sdk/credential-provider-sso': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/credential-provider-web-identity': 3.621.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - '@aws-sdk/client-sts' - - aws-crt - - '@aws-sdk/credential-provider-process@3.620.1': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-sso@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))': - dependencies: - '@aws-sdk/client-sso': 3.637.0 - '@aws-sdk/token-providers': 3.614.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - aws-crt - - '@aws-sdk/credential-provider-web-identity@3.621.0(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/lib-storage@3.637.0(@aws-sdk/client-s3@3.637.0)': - dependencies: - '@aws-sdk/client-s3': 3.637.0 - '@smithy/abort-controller': 3.1.1 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/smithy-client': 3.2.0 - buffer: 5.6.0 - events: 3.3.0 - stream-browserify: 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-bucket-endpoint@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-arn-parser': 3.568.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-expect-continue@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-flexible-checksums@3.620.0': - dependencies: - '@aws-crypto/crc32': 5.2.0 - '@aws-crypto/crc32c': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@smithy/is-array-buffer': 3.0.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-host-header@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-location-constraint@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-logger@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-recursion-detection@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-sdk-s3@3.635.0': - dependencies: - '@aws-sdk/core': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-arn-parser': 3.568.0 - '@smithy/core': 2.4.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-stream': 3.1.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-ssec@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-user-agent@3.637.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/region-config-resolver@3.614.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@aws-sdk/s3-request-presigner@3.637.0': - dependencies: - '@aws-sdk/signature-v4-multi-region': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-format-url': 3.609.0 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/signature-v4-multi-region@3.635.0': - dependencies: - '@aws-sdk/middleware-sdk-s3': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/token-providers@3.614.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))': - dependencies: - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/types@3.609.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/util-arn-parser@3.568.0': - dependencies: - tslib: 2.7.0 - - '@aws-sdk/util-endpoints@3.637.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - '@smithy/util-endpoints': 2.0.5 - tslib: 2.7.0 - - '@aws-sdk/util-format-url@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/querystring-builder': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/util-locate-window@3.568.0': - dependencies: - tslib: 2.7.0 - - '@aws-sdk/util-user-agent-browser@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - bowser: 2.11.0 - tslib: 2.7.0 - - '@aws-sdk/util-user-agent-node@3.614.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/xml-builder@3.609.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@babel/code-frame@7.24.7': - dependencies: - '@babel/highlight': 7.24.7 - picocolors: 1.1.0 - - '@babel/compat-data@7.25.4': {} - - '@babel/core@7.25.2': - dependencies: - '@ampproject/remapping': 2.3.0 - '@babel/code-frame': 7.24.7 - '@babel/generator': 7.25.6 - '@babel/helper-compilation-targets': 7.25.2 - '@babel/helper-module-transforms': 7.25.2(@babel/core@7.25.2) - '@babel/helpers': 7.25.6 - '@babel/parser': 7.25.6 - '@babel/template': 7.25.0 - '@babel/traverse': 7.25.6 - '@babel/types': 7.25.6 - convert-source-map: 2.0.0 - debug: 4.3.6 - gensync: 1.0.0-beta.2 - json5: 2.2.3 - semver: 6.3.1 - transitivePeerDependencies: - - supports-color - - '@babel/generator@7.25.6': - dependencies: - '@babel/types': 7.25.6 - '@jridgewell/gen-mapping': 0.3.5 - '@jridgewell/trace-mapping': 0.3.25 - jsesc: 2.5.2 - - '@babel/helper-annotate-as-pure@7.24.7': - dependencies: - '@babel/types': 7.25.6 - - '@babel/helper-builder-binary-assignment-operator-visitor@7.24.7': - dependencies: - '@babel/traverse': 7.25.6 - '@babel/types': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/helper-compilation-targets@7.25.2': - dependencies: - '@babel/compat-data': 7.25.4 - '@babel/helper-validator-option': 7.24.8 - browserslist: 4.23.3 - lru-cache: 5.1.1 - semver: 6.3.1 - - '@babel/helper-create-class-features-plugin@7.25.4(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-annotate-as-pure': 7.24.7 - '@babel/helper-member-expression-to-functions': 7.24.8 - '@babel/helper-optimise-call-expression': 7.24.7 - '@babel/helper-replace-supers': 7.25.0(@babel/core@7.25.2) - '@babel/helper-skip-transparent-expression-wrappers': 7.24.7 - '@babel/traverse': 7.25.6 - semver: 6.3.1 - transitivePeerDependencies: - - supports-color - - '@babel/helper-create-regexp-features-plugin@7.25.2(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-annotate-as-pure': 7.24.7 - regexpu-core: 5.3.2 - semver: 6.3.1 - - '@babel/helper-define-polyfill-provider@0.6.2(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-compilation-targets': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - debug: 4.3.6 - lodash.debounce: 4.0.8 - resolve: 1.22.8 - transitivePeerDependencies: - - supports-color - - '@babel/helper-member-expression-to-functions@7.24.8': - dependencies: - '@babel/traverse': 7.25.6 - '@babel/types': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/helper-module-imports@7.24.7': - dependencies: - '@babel/traverse': 7.25.6 - '@babel/types': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/helper-module-transforms@7.25.2(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-module-imports': 7.24.7 - '@babel/helper-simple-access': 7.24.7 - '@babel/helper-validator-identifier': 7.24.7 - '@babel/traverse': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/helper-optimise-call-expression@7.24.7': - dependencies: - '@babel/types': 7.25.6 - - '@babel/helper-plugin-utils@7.24.8': {} - - '@babel/helper-remap-async-to-generator@7.25.0(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-annotate-as-pure': 7.24.7 - '@babel/helper-wrap-function': 7.25.0 - '@babel/traverse': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/helper-replace-supers@7.25.0(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-member-expression-to-functions': 7.24.8 - '@babel/helper-optimise-call-expression': 7.24.7 - '@babel/traverse': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/helper-simple-access@7.24.7': - dependencies: - '@babel/traverse': 7.25.6 - '@babel/types': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/helper-skip-transparent-expression-wrappers@7.24.7': - dependencies: - '@babel/traverse': 7.25.6 - '@babel/types': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/helper-string-parser@7.24.8': {} - - '@babel/helper-validator-identifier@7.24.7': {} - - '@babel/helper-validator-option@7.24.8': {} - - '@babel/helper-wrap-function@7.25.0': - dependencies: - '@babel/template': 7.25.0 - '@babel/traverse': 7.25.6 - '@babel/types': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/helpers@7.25.6': - dependencies: - '@babel/template': 7.25.0 - '@babel/types': 7.25.6 - - '@babel/highlight@7.24.7': - dependencies: - '@babel/helper-validator-identifier': 7.24.7 - chalk: 2.4.2 - js-tokens: 4.0.0 - picocolors: 1.1.0 - - '@babel/parser@7.25.6': - dependencies: - '@babel/types': 7.25.6 - - '@babel/plugin-bugfix-firefox-class-in-computed-class-key@7.25.3(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/traverse': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/plugin-bugfix-safari-class-field-initializer-scope@7.25.0(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression@7.25.0(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/helper-skip-transparent-expression-wrappers': 7.24.7 - '@babel/plugin-transform-optional-chaining': 7.24.8(@babel/core@7.25.2) - transitivePeerDependencies: - - supports-color - - '@babel/plugin-bugfix-v8-static-class-fields-redefine-readonly@7.25.0(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/traverse': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/plugin-proposal-private-property-in-object@7.21.0-placeholder-for-preset-env.2(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - - '@babel/plugin-syntax-async-generators@7.8.4(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-class-properties@7.12.13(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-class-static-block@7.14.5(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-dynamic-import@7.8.3(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-export-namespace-from@7.8.3(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-import-assertions@7.25.6(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-import-attributes@7.25.6(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-import-meta@7.10.4(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-json-strings@7.8.3(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-jsx@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-logical-assignment-operators@7.10.4(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-nullish-coalescing-operator@7.8.3(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-numeric-separator@7.10.4(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-object-rest-spread@7.8.3(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-optional-catch-binding@7.8.3(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-optional-chaining@7.8.3(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-private-property-in-object@7.14.5(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-top-level-await@7.14.5(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-typescript@7.25.4(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-syntax-unicode-sets-regex@7.18.6(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-create-regexp-features-plugin': 7.25.2(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-arrow-functions@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-async-generator-functions@7.25.4(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/helper-remap-async-to-generator': 7.25.0(@babel/core@7.25.2) - '@babel/plugin-syntax-async-generators': 7.8.4(@babel/core@7.25.2) - '@babel/traverse': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-async-to-generator@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-module-imports': 7.24.7 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/helper-remap-async-to-generator': 7.25.0(@babel/core@7.25.2) - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-block-scoped-functions@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-block-scoping@7.25.0(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-class-properties@7.25.4(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-create-class-features-plugin': 7.25.4(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-class-static-block@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-create-class-features-plugin': 7.25.4(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - '@babel/plugin-syntax-class-static-block': 7.14.5(@babel/core@7.25.2) - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-classes@7.25.4(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-annotate-as-pure': 7.24.7 - '@babel/helper-compilation-targets': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/helper-replace-supers': 7.25.0(@babel/core@7.25.2) - '@babel/traverse': 7.25.6 - globals: 11.12.0 - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-computed-properties@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/template': 7.25.0 - - '@babel/plugin-transform-destructuring@7.24.8(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-dotall-regex@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-create-regexp-features-plugin': 7.25.2(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-duplicate-keys@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-duplicate-named-capturing-groups-regex@7.25.0(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-create-regexp-features-plugin': 7.25.2(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-dynamic-import@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/plugin-syntax-dynamic-import': 7.8.3(@babel/core@7.25.2) - - '@babel/plugin-transform-exponentiation-operator@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-builder-binary-assignment-operator-visitor': 7.24.7 - '@babel/helper-plugin-utils': 7.24.8 - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-export-namespace-from@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/plugin-syntax-export-namespace-from': 7.8.3(@babel/core@7.25.2) - - '@babel/plugin-transform-for-of@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/helper-skip-transparent-expression-wrappers': 7.24.7 - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-function-name@7.25.1(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-compilation-targets': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/traverse': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-json-strings@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/plugin-syntax-json-strings': 7.8.3(@babel/core@7.25.2) - - '@babel/plugin-transform-literals@7.25.2(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-logical-assignment-operators@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/plugin-syntax-logical-assignment-operators': 7.10.4(@babel/core@7.25.2) - - '@babel/plugin-transform-member-expression-literals@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-modules-amd@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-module-transforms': 7.25.2(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-modules-commonjs@7.24.8(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-module-transforms': 7.25.2(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - '@babel/helper-simple-access': 7.24.7 - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-modules-systemjs@7.25.0(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-module-transforms': 7.25.2(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - '@babel/helper-validator-identifier': 7.24.7 - '@babel/traverse': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-modules-umd@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-module-transforms': 7.25.2(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-named-capturing-groups-regex@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-create-regexp-features-plugin': 7.25.2(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-new-target@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-nullish-coalescing-operator@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/plugin-syntax-nullish-coalescing-operator': 7.8.3(@babel/core@7.25.2) - - '@babel/plugin-transform-numeric-separator@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/plugin-syntax-numeric-separator': 7.10.4(@babel/core@7.25.2) - - '@babel/plugin-transform-object-rest-spread@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-compilation-targets': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/plugin-syntax-object-rest-spread': 7.8.3(@babel/core@7.25.2) - '@babel/plugin-transform-parameters': 7.24.7(@babel/core@7.25.2) - - '@babel/plugin-transform-object-super@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/helper-replace-supers': 7.25.0(@babel/core@7.25.2) - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-optional-catch-binding@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/plugin-syntax-optional-catch-binding': 7.8.3(@babel/core@7.25.2) - - '@babel/plugin-transform-optional-chaining@7.24.8(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/helper-skip-transparent-expression-wrappers': 7.24.7 - '@babel/plugin-syntax-optional-chaining': 7.8.3(@babel/core@7.25.2) - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-parameters@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-private-methods@7.25.4(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-create-class-features-plugin': 7.25.4(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-private-property-in-object@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-annotate-as-pure': 7.24.7 - '@babel/helper-create-class-features-plugin': 7.25.4(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - '@babel/plugin-syntax-private-property-in-object': 7.14.5(@babel/core@7.25.2) - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-property-literals@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-regenerator@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - regenerator-transform: 0.15.2 - - '@babel/plugin-transform-reserved-words@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-shorthand-properties@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-spread@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/helper-skip-transparent-expression-wrappers': 7.24.7 - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-sticky-regex@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-template-literals@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-typeof-symbol@7.24.8(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-typescript@7.25.2(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-annotate-as-pure': 7.24.7 - '@babel/helper-create-class-features-plugin': 7.25.4(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - '@babel/helper-skip-transparent-expression-wrappers': 7.24.7 - '@babel/plugin-syntax-typescript': 7.25.4(@babel/core@7.25.2) - transitivePeerDependencies: - - supports-color - - '@babel/plugin-transform-unicode-escapes@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-unicode-property-regex@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-create-regexp-features-plugin': 7.25.2(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-unicode-regex@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-create-regexp-features-plugin': 7.25.2(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/plugin-transform-unicode-sets-regex@7.25.4(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-create-regexp-features-plugin': 7.25.2(@babel/core@7.25.2) - '@babel/helper-plugin-utils': 7.24.8 - - '@babel/preset-env@7.25.4(@babel/core@7.25.2)': - dependencies: - '@babel/compat-data': 7.25.4 - '@babel/core': 7.25.2 - '@babel/helper-compilation-targets': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/helper-validator-option': 7.24.8 - '@babel/plugin-bugfix-firefox-class-in-computed-class-key': 7.25.3(@babel/core@7.25.2) - '@babel/plugin-bugfix-safari-class-field-initializer-scope': 7.25.0(@babel/core@7.25.2) - '@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression': 7.25.0(@babel/core@7.25.2) - '@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-bugfix-v8-static-class-fields-redefine-readonly': 7.25.0(@babel/core@7.25.2) - '@babel/plugin-proposal-private-property-in-object': 7.21.0-placeholder-for-preset-env.2(@babel/core@7.25.2) - '@babel/plugin-syntax-async-generators': 7.8.4(@babel/core@7.25.2) - '@babel/plugin-syntax-class-properties': 7.12.13(@babel/core@7.25.2) - '@babel/plugin-syntax-class-static-block': 7.14.5(@babel/core@7.25.2) - '@babel/plugin-syntax-dynamic-import': 7.8.3(@babel/core@7.25.2) - '@babel/plugin-syntax-export-namespace-from': 7.8.3(@babel/core@7.25.2) - '@babel/plugin-syntax-import-assertions': 7.25.6(@babel/core@7.25.2) - '@babel/plugin-syntax-import-attributes': 7.25.6(@babel/core@7.25.2) - '@babel/plugin-syntax-import-meta': 7.10.4(@babel/core@7.25.2) - '@babel/plugin-syntax-json-strings': 7.8.3(@babel/core@7.25.2) - '@babel/plugin-syntax-logical-assignment-operators': 7.10.4(@babel/core@7.25.2) - '@babel/plugin-syntax-nullish-coalescing-operator': 7.8.3(@babel/core@7.25.2) - '@babel/plugin-syntax-numeric-separator': 7.10.4(@babel/core@7.25.2) - '@babel/plugin-syntax-object-rest-spread': 7.8.3(@babel/core@7.25.2) - '@babel/plugin-syntax-optional-catch-binding': 7.8.3(@babel/core@7.25.2) - '@babel/plugin-syntax-optional-chaining': 7.8.3(@babel/core@7.25.2) - '@babel/plugin-syntax-private-property-in-object': 7.14.5(@babel/core@7.25.2) - '@babel/plugin-syntax-top-level-await': 7.14.5(@babel/core@7.25.2) - '@babel/plugin-syntax-unicode-sets-regex': 7.18.6(@babel/core@7.25.2) - '@babel/plugin-transform-arrow-functions': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-async-generator-functions': 7.25.4(@babel/core@7.25.2) - '@babel/plugin-transform-async-to-generator': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-block-scoped-functions': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-block-scoping': 7.25.0(@babel/core@7.25.2) - '@babel/plugin-transform-class-properties': 7.25.4(@babel/core@7.25.2) - '@babel/plugin-transform-class-static-block': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-classes': 7.25.4(@babel/core@7.25.2) - '@babel/plugin-transform-computed-properties': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-destructuring': 7.24.8(@babel/core@7.25.2) - '@babel/plugin-transform-dotall-regex': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-duplicate-keys': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-duplicate-named-capturing-groups-regex': 7.25.0(@babel/core@7.25.2) - '@babel/plugin-transform-dynamic-import': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-exponentiation-operator': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-export-namespace-from': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-for-of': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-function-name': 7.25.1(@babel/core@7.25.2) - '@babel/plugin-transform-json-strings': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-literals': 7.25.2(@babel/core@7.25.2) - '@babel/plugin-transform-logical-assignment-operators': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-member-expression-literals': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-modules-amd': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-modules-commonjs': 7.24.8(@babel/core@7.25.2) - '@babel/plugin-transform-modules-systemjs': 7.25.0(@babel/core@7.25.2) - '@babel/plugin-transform-modules-umd': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-named-capturing-groups-regex': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-new-target': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-nullish-coalescing-operator': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-numeric-separator': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-object-rest-spread': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-object-super': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-optional-catch-binding': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-optional-chaining': 7.24.8(@babel/core@7.25.2) - '@babel/plugin-transform-parameters': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-private-methods': 7.25.4(@babel/core@7.25.2) - '@babel/plugin-transform-private-property-in-object': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-property-literals': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-regenerator': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-reserved-words': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-shorthand-properties': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-spread': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-sticky-regex': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-template-literals': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-typeof-symbol': 7.24.8(@babel/core@7.25.2) - '@babel/plugin-transform-unicode-escapes': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-unicode-property-regex': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-unicode-regex': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-unicode-sets-regex': 7.25.4(@babel/core@7.25.2) - '@babel/preset-modules': 0.1.6-no-external-plugins(@babel/core@7.25.2) - babel-plugin-polyfill-corejs2: 0.4.11(@babel/core@7.25.2) - babel-plugin-polyfill-corejs3: 0.10.6(@babel/core@7.25.2) - babel-plugin-polyfill-regenerator: 0.6.2(@babel/core@7.25.2) - core-js-compat: 3.38.1 - semver: 6.3.1 - transitivePeerDependencies: - - supports-color - - '@babel/preset-modules@0.1.6-no-external-plugins(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/types': 7.25.6 - esutils: 2.0.3 - - '@babel/preset-typescript@7.24.7(@babel/core@7.25.2)': - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-plugin-utils': 7.24.8 - '@babel/helper-validator-option': 7.24.8 - '@babel/plugin-syntax-jsx': 7.24.7(@babel/core@7.25.2) - '@babel/plugin-transform-modules-commonjs': 7.24.8(@babel/core@7.25.2) - '@babel/plugin-transform-typescript': 7.25.2(@babel/core@7.25.2) - transitivePeerDependencies: - - supports-color - - '@babel/regjsgen@0.8.0': {} - - '@babel/runtime@7.25.6': - dependencies: - regenerator-runtime: 0.14.1 - - '@babel/template@7.25.0': - dependencies: - '@babel/code-frame': 7.24.7 - '@babel/parser': 7.25.6 - '@babel/types': 7.25.6 - - '@babel/traverse@7.25.6': - dependencies: - '@babel/code-frame': 7.24.7 - '@babel/generator': 7.25.6 - '@babel/parser': 7.25.6 - '@babel/template': 7.25.0 - '@babel/types': 7.25.6 - debug: 4.3.6 - globals: 11.12.0 - transitivePeerDependencies: - - supports-color - - '@babel/types@7.25.6': - dependencies: - '@babel/helper-string-parser': 7.24.8 - '@babel/helper-validator-identifier': 7.24.7 - to-fast-properties: 2.0.0 - - '@cspotcode/source-map-support@0.8.1': - dependencies: - '@jridgewell/trace-mapping': 0.3.9 - - '@emnapi/runtime@1.2.0': - dependencies: - tslib: 2.7.0 - optional: true - - '@esbuild/aix-ppc64@0.23.1': - optional: true - - '@esbuild/android-arm64@0.23.1': - optional: true - - '@esbuild/android-arm@0.23.1': - optional: true - - '@esbuild/android-x64@0.23.1': - optional: true - - '@esbuild/darwin-arm64@0.23.1': - optional: true - - '@esbuild/darwin-x64@0.23.1': - optional: true - - '@esbuild/freebsd-arm64@0.23.1': - optional: true - - '@esbuild/freebsd-x64@0.23.1': - optional: true - - '@esbuild/linux-arm64@0.23.1': - optional: true - - '@esbuild/linux-arm@0.23.1': - optional: true - - '@esbuild/linux-ia32@0.23.1': - optional: true - - '@esbuild/linux-loong64@0.23.1': - optional: true - - '@esbuild/linux-mips64el@0.23.1': - optional: true - - '@esbuild/linux-ppc64@0.23.1': - optional: true - - '@esbuild/linux-riscv64@0.23.1': - optional: true - - '@esbuild/linux-s390x@0.23.1': - optional: true - - '@esbuild/linux-x64@0.23.1': - optional: true - - '@esbuild/netbsd-x64@0.23.1': - optional: true - - '@esbuild/openbsd-arm64@0.23.1': - optional: true - - '@esbuild/openbsd-x64@0.23.1': - optional: true - - '@esbuild/sunos-x64@0.23.1': - optional: true - - '@esbuild/win32-arm64@0.23.1': - optional: true - - '@esbuild/win32-ia32@0.23.1': - optional: true - - '@esbuild/win32-x64@0.23.1': - optional: true - - '@fastify/accept-negotiator@1.1.0': {} - - '@fastify/ajv-compiler@3.6.0': - dependencies: - ajv: 8.17.1 - ajv-formats: 2.1.1(ajv@8.17.1) - fast-uri: 2.4.0 - - '@fastify/error@3.4.1': {} - - '@fastify/fast-json-stringify-compiler@4.3.0': - dependencies: - fast-json-stringify: 5.16.1 - - '@fastify/merge-json-schemas@0.1.1': - dependencies: - fast-deep-equal: 3.1.3 - - '@fastify/send@2.1.0': - dependencies: - '@lukeed/ms': 2.0.2 - escape-html: 1.0.3 - fast-decode-uri-component: 1.0.1 - http-errors: 2.0.0 - mime: 3.0.0 - - '@fastify/static@7.0.4': - dependencies: - '@fastify/accept-negotiator': 1.1.0 - '@fastify/send': 2.1.0 - content-disposition: 0.5.4 - fastify-plugin: 4.5.1 - fastq: 1.17.1 - glob: 10.4.5 - - '@fastify/swagger-ui@4.1.0': - dependencies: - '@fastify/static': 7.0.4 - fastify-plugin: 4.5.1 - openapi-types: 12.1.3 - rfdc: 1.4.1 - yaml: 2.5.0 - - '@fastify/swagger@8.15.0': - dependencies: - fastify-plugin: 4.5.1 - json-schema-resolver: 2.0.0 - openapi-types: 12.1.3 - rfdc: 1.4.1 - yaml: 2.5.0 - transitivePeerDependencies: - - supports-color - - '@grpc/grpc-js@1.11.1': - dependencies: - '@grpc/proto-loader': 0.7.13 - '@js-sdsl/ordered-map': 4.4.2 - - '@grpc/proto-loader@0.7.13': - dependencies: - lodash.camelcase: 4.3.0 - long: 5.2.3 - protobufjs: 7.4.0 - yargs: 17.7.2 - - '@img/sharp-darwin-arm64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-darwin-arm64': 1.0.4 - optional: true - - '@img/sharp-darwin-x64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-darwin-x64': 1.0.4 - optional: true - - '@img/sharp-libvips-darwin-arm64@1.0.4': - optional: true - - '@img/sharp-libvips-darwin-x64@1.0.4': - optional: true - - '@img/sharp-libvips-linux-arm64@1.0.4': - optional: true - - '@img/sharp-libvips-linux-arm@1.0.5': - optional: true - - '@img/sharp-libvips-linux-s390x@1.0.4': - optional: true - - '@img/sharp-libvips-linux-x64@1.0.4': - optional: true - - '@img/sharp-libvips-linuxmusl-arm64@1.0.4': - optional: true - - '@img/sharp-libvips-linuxmusl-x64@1.0.4': - optional: true - - '@img/sharp-linux-arm64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linux-arm64': 1.0.4 - optional: true - - '@img/sharp-linux-arm@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linux-arm': 1.0.5 - optional: true - - '@img/sharp-linux-s390x@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linux-s390x': 1.0.4 - optional: true - - '@img/sharp-linux-x64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linux-x64': 1.0.4 - optional: true - - '@img/sharp-linuxmusl-arm64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linuxmusl-arm64': 1.0.4 - optional: true - - '@img/sharp-linuxmusl-x64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linuxmusl-x64': 1.0.4 - optional: true - - '@img/sharp-wasm32@0.33.5': - dependencies: - '@emnapi/runtime': 1.2.0 - optional: true - - '@img/sharp-win32-ia32@0.33.5': - optional: true - - '@img/sharp-win32-x64@0.33.5': - optional: true - - '@isaacs/cliui@8.0.2': - dependencies: - string-width: 5.1.2 - string-width-cjs: string-width@4.2.3 - strip-ansi: 7.1.0 - strip-ansi-cjs: strip-ansi@6.0.1 - wrap-ansi: 8.1.0 - wrap-ansi-cjs: wrap-ansi@7.0.0 - - '@jridgewell/gen-mapping@0.3.5': - dependencies: - '@jridgewell/set-array': 1.2.1 - '@jridgewell/sourcemap-codec': 1.5.0 - '@jridgewell/trace-mapping': 0.3.25 - - '@jridgewell/resolve-uri@3.1.2': {} - - '@jridgewell/set-array@1.2.1': {} - - '@jridgewell/source-map@0.3.6': - dependencies: - '@jridgewell/gen-mapping': 0.3.5 - '@jridgewell/trace-mapping': 0.3.25 - - '@jridgewell/sourcemap-codec@1.5.0': {} - - '@jridgewell/trace-mapping@0.3.25': - dependencies: - '@jridgewell/resolve-uri': 3.1.2 - '@jridgewell/sourcemap-codec': 1.5.0 - - '@jridgewell/trace-mapping@0.3.9': - dependencies: - '@jridgewell/resolve-uri': 3.1.2 - '@jridgewell/sourcemap-codec': 1.5.0 - - '@js-sdsl/ordered-map@4.4.2': {} - - '@jsdevtools/ono@7.1.3': {} - - '@jsonjoy.com/base64@1.1.2(tslib@2.7.0)': - dependencies: - tslib: 2.7.0 - - '@jsonjoy.com/json-pack@1.1.0(tslib@2.7.0)': - dependencies: - '@jsonjoy.com/base64': 1.1.2(tslib@2.7.0) - '@jsonjoy.com/util': 1.3.0(tslib@2.7.0) - hyperdyperid: 1.2.0 - thingies: 1.21.0(tslib@2.7.0) - tslib: 2.7.0 - - '@jsonjoy.com/util@1.3.0(tslib@2.7.0)': - dependencies: - tslib: 2.7.0 - - '@lukeed/ms@2.0.2': {} - - '@noble/hashes@1.5.0': {} - - '@nodelib/fs.scandir@2.1.5': - dependencies: - '@nodelib/fs.stat': 2.0.5 - run-parallel: 1.2.0 - - '@nodelib/fs.stat@2.0.5': {} - - '@nodelib/fs.walk@1.2.8': - dependencies: - '@nodelib/fs.scandir': 2.1.5 - fastq: 1.17.1 - - '@paralleldrive/cuid2@2.2.2': - dependencies: - '@noble/hashes': 1.5.0 - - '@pkgjs/parseargs@0.11.0': - optional: true - - '@protobufjs/aspromise@1.1.2': {} - - '@protobufjs/base64@1.1.2': {} - - '@protobufjs/codegen@2.0.4': {} - - '@protobufjs/eventemitter@1.1.0': {} - - '@protobufjs/fetch@1.1.0': - dependencies: - '@protobufjs/aspromise': 1.1.2 - '@protobufjs/inquire': 1.1.0 - - '@protobufjs/float@1.0.2': {} - - '@protobufjs/inquire@1.1.0': {} - - '@protobufjs/path@1.1.2': {} - - '@protobufjs/pool@1.1.0': {} - - '@protobufjs/utf8@1.1.0': {} - - '@rollup/rollup-android-arm-eabi@4.21.2': - optional: true - - '@rollup/rollup-android-arm64@4.21.2': - optional: true - - '@rollup/rollup-darwin-arm64@4.21.2': - optional: true - - '@rollup/rollup-darwin-x64@4.21.2': - optional: true - - '@rollup/rollup-linux-arm-gnueabihf@4.21.2': - optional: true - - '@rollup/rollup-linux-arm-musleabihf@4.21.2': - optional: true - - '@rollup/rollup-linux-arm64-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-arm64-musl@4.21.2': - optional: true - - '@rollup/rollup-linux-powerpc64le-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-riscv64-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-s390x-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-x64-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-x64-musl@4.21.2': - optional: true - - '@rollup/rollup-win32-arm64-msvc@4.21.2': - optional: true - - '@rollup/rollup-win32-ia32-msvc@4.21.2': - optional: true - - '@rollup/rollup-win32-x64-msvc@4.21.2': - optional: true - - '@selderee/plugin-htmlparser2@0.11.0': - dependencies: - domhandler: 5.0.3 - selderee: 0.11.0 - - '@sinonjs/commons@2.0.0': - dependencies: - type-detect: 4.0.8 - - '@sinonjs/commons@3.0.1': - dependencies: - type-detect: 4.0.8 - - '@sinonjs/fake-timers@11.3.1': - dependencies: - '@sinonjs/commons': 3.0.1 - - '@sinonjs/samsam@8.0.0': - dependencies: - '@sinonjs/commons': 2.0.0 - lodash.get: 4.4.2 - type-detect: 4.1.0 - - '@sinonjs/text-encoding@0.7.3': {} - - '@smithy/abort-controller@3.1.1': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/chunked-blob-reader-native@3.0.0': - dependencies: - '@smithy/util-base64': 3.0.0 - tslib: 2.7.0 - - '@smithy/chunked-blob-reader@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/config-resolver@3.0.5': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@smithy/core@2.4.0': - dependencies: - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/credential-provider-imds@3.2.0': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - tslib: 2.7.0 - - '@smithy/eventstream-codec@3.1.2': - dependencies: - '@aws-crypto/crc32': 5.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-hex-encoding': 3.0.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-browser@3.0.6': - dependencies: - '@smithy/eventstream-serde-universal': 3.0.5 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-config-resolver@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-node@3.0.5': - dependencies: - '@smithy/eventstream-serde-universal': 3.0.5 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-universal@3.0.5': - dependencies: - '@smithy/eventstream-codec': 3.1.2 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/fetch-http-handler@3.2.4': - dependencies: - '@smithy/protocol-http': 4.1.0 - '@smithy/querystring-builder': 3.0.3 - '@smithy/types': 3.3.0 - '@smithy/util-base64': 3.0.0 - tslib: 2.7.0 - - '@smithy/hash-blob-browser@3.1.2': - dependencies: - '@smithy/chunked-blob-reader': 3.0.0 - '@smithy/chunked-blob-reader-native': 3.0.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/hash-node@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/hash-stream-node@3.1.2': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/invalid-dependency@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/is-array-buffer@2.2.0': - dependencies: - tslib: 2.7.0 - - '@smithy/is-array-buffer@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/md5-js@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/middleware-content-length@3.0.5': - dependencies: - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/middleware-endpoint@3.1.0': - dependencies: - '@smithy/middleware-serde': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@smithy/middleware-retry@3.0.15': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/service-error-classification': 3.0.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - tslib: 2.7.0 - uuid: 9.0.1 - - '@smithy/middleware-serde@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/middleware-stack@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/node-config-provider@3.1.4': - dependencies: - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/node-http-handler@3.1.4': - dependencies: - '@smithy/abort-controller': 3.1.1 - '@smithy/protocol-http': 4.1.0 - '@smithy/querystring-builder': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/property-provider@3.1.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/protocol-http@4.1.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/querystring-builder@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-uri-escape': 3.0.0 - tslib: 2.7.0 - - '@smithy/querystring-parser@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/service-error-classification@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - - '@smithy/shared-ini-file-loader@3.1.4': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/signature-v4@4.1.0': - dependencies: - '@smithy/is-array-buffer': 3.0.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-hex-encoding': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-uri-escape': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/smithy-client@3.2.0': - dependencies: - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-stack': 3.0.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-stream': 3.1.3 - tslib: 2.7.0 - - '@smithy/types@3.3.0': - dependencies: - tslib: 2.7.0 - - '@smithy/url-parser@3.0.3': - dependencies: - '@smithy/querystring-parser': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-base64@3.0.0': - dependencies: - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-body-length-browser@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-body-length-node@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-buffer-from@2.2.0': - dependencies: - '@smithy/is-array-buffer': 2.2.0 - tslib: 2.7.0 - - '@smithy/util-buffer-from@3.0.0': - dependencies: - '@smithy/is-array-buffer': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-config-provider@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-defaults-mode-browser@3.0.15': - dependencies: - '@smithy/property-provider': 3.1.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - bowser: 2.11.0 - tslib: 2.7.0 - - '@smithy/util-defaults-mode-node@3.0.15': - dependencies: - '@smithy/config-resolver': 3.0.5 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-endpoints@2.0.5': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-hex-encoding@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-middleware@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-retry@3.0.3': - dependencies: - '@smithy/service-error-classification': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-stream@3.1.3': - dependencies: - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-base64': 3.0.0 - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-hex-encoding': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-uri-escape@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-utf8@2.3.0': - dependencies: - '@smithy/util-buffer-from': 2.2.0 - tslib: 2.7.0 - - '@smithy/util-utf8@3.0.0': - dependencies: - '@smithy/util-buffer-from': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-waiter@3.1.2': - dependencies: - '@smithy/abort-controller': 3.1.1 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@swc/core-darwin-arm64@1.7.23': - optional: true - - '@swc/core-darwin-x64@1.7.23': - optional: true - - '@swc/core-linux-arm-gnueabihf@1.7.23': - optional: true - - '@swc/core-linux-arm64-gnu@1.7.23': - optional: true - - '@swc/core-linux-arm64-musl@1.7.23': - optional: true - - '@swc/core-linux-x64-gnu@1.7.23': - optional: true - - '@swc/core-linux-x64-musl@1.7.23': - optional: true - - '@swc/core-win32-arm64-msvc@1.7.23': - optional: true - - '@swc/core-win32-ia32-msvc@1.7.23': - optional: true - - '@swc/core-win32-x64-msvc@1.7.23': - optional: true - - '@swc/core@1.7.23': - dependencies: - '@swc/counter': 0.1.3 - '@swc/types': 0.1.12 - optionalDependencies: - '@swc/core-darwin-arm64': 1.7.23 - '@swc/core-darwin-x64': 1.7.23 - '@swc/core-linux-arm-gnueabihf': 1.7.23 - '@swc/core-linux-arm64-gnu': 1.7.23 - '@swc/core-linux-arm64-musl': 1.7.23 - '@swc/core-linux-x64-gnu': 1.7.23 - '@swc/core-linux-x64-musl': 1.7.23 - '@swc/core-win32-arm64-msvc': 1.7.23 - '@swc/core-win32-ia32-msvc': 1.7.23 - '@swc/core-win32-x64-msvc': 1.7.23 - - '@swc/counter@0.1.3': {} - - '@swc/types@0.1.12': - dependencies: - '@swc/counter': 0.1.3 - - '@temporalio/activity@1.11.1': - dependencies: - '@temporalio/common': 1.11.1 - abort-controller: 3.0.0 - - '@temporalio/client@1.11.1': - dependencies: - '@grpc/grpc-js': 1.11.1 - '@temporalio/common': 1.11.1 - '@temporalio/proto': 1.11.1 - abort-controller: 3.0.0 - long: 5.2.3 - uuid: 9.0.1 - - '@temporalio/common@1.11.1': - dependencies: - '@temporalio/proto': 1.11.1 - long: 5.2.3 - ms: 3.0.0-canary.1 - proto3-json-serializer: 2.0.2 - - '@temporalio/core-bridge@1.11.1': - dependencies: - '@temporalio/common': 1.11.1 - arg: 5.0.2 - cargo-cp-artifact: 0.1.9 - which: 4.0.0 - - '@temporalio/proto@1.11.1': - dependencies: - long: 5.2.3 - protobufjs: 7.4.0 - - '@temporalio/worker@1.11.1(esbuild@0.23.1)': - dependencies: - '@swc/core': 1.7.23 - '@temporalio/activity': 1.11.1 - '@temporalio/client': 1.11.1 - '@temporalio/common': 1.11.1 - '@temporalio/core-bridge': 1.11.1 - '@temporalio/proto': 1.11.1 - '@temporalio/workflow': 1.11.1 - abort-controller: 3.0.0 - heap-js: 2.5.0 - memfs: 4.11.1 - rxjs: 7.8.1 - source-map: 0.7.4 - source-map-loader: 4.0.2(webpack@5.94.0(@swc/core@1.7.23)(esbuild@0.23.1)) - supports-color: 8.1.1 - swc-loader: 0.2.6(@swc/core@1.7.23)(webpack@5.94.0(@swc/core@1.7.23)(esbuild@0.23.1)) - unionfs: 4.5.4 - webpack: 5.94.0(@swc/core@1.7.23)(esbuild@0.23.1) - transitivePeerDependencies: - - '@swc/helpers' - - esbuild - - uglify-js - - webpack-cli - - '@temporalio/workflow@1.11.1': - dependencies: - '@temporalio/common': 1.11.1 - '@temporalio/proto': 1.11.1 - - '@tsconfig/node10@1.0.11': {} - - '@tsconfig/node12@1.0.11': {} - - '@tsconfig/node14@1.0.3': {} - - '@tsconfig/node16@1.0.4': {} - - '@tsconfig/node20@20.1.4': {} - - '@types/chai@4.3.19': {} - - '@types/cheerio@0.22.35': - dependencies: - '@types/node': 22.5.2 - - '@types/debug@4.1.12': - dependencies: - '@types/ms': 0.7.34 - - '@types/estree@1.0.5': {} - - '@types/fluent-ffmpeg@2.1.26': - dependencies: - '@types/node': 22.5.2 - - '@types/imapflow@1.0.19': - dependencies: - '@types/node': 22.5.2 - - '@types/json-schema@7.0.15': {} - - '@types/luxon@3.4.2': {} - - '@types/mailparser@3.4.4': - dependencies: - '@types/node': 22.5.2 - iconv-lite: 0.6.3 - - '@types/mocha@10.0.7': {} - - '@types/ms@0.7.34': {} - - '@types/node@18.19.48': - dependencies: - undici-types: 5.26.5 - - '@types/node@20.16.3': - dependencies: - undici-types: 6.19.8 - - '@types/node@22.5.2': - dependencies: - undici-types: 6.19.8 - - '@types/pg@8.11.8': - dependencies: - '@types/node': 22.5.2 - pg-protocol: 1.6.1 - pg-types: 4.0.2 - - '@types/retry@0.12.2': {} - - '@types/sinon@17.0.3': - dependencies: - '@types/sinonjs__fake-timers': 8.1.5 - - '@types/sinonjs__fake-timers@8.1.5': {} - - '@webassemblyjs/ast@1.12.1': - dependencies: - '@webassemblyjs/helper-numbers': 1.11.6 - '@webassemblyjs/helper-wasm-bytecode': 1.11.6 - - '@webassemblyjs/floating-point-hex-parser@1.11.6': {} - - '@webassemblyjs/helper-api-error@1.11.6': {} - - '@webassemblyjs/helper-buffer@1.12.1': {} - - '@webassemblyjs/helper-numbers@1.11.6': - dependencies: - '@webassemblyjs/floating-point-hex-parser': 1.11.6 - '@webassemblyjs/helper-api-error': 1.11.6 - '@xtuc/long': 4.2.2 - - '@webassemblyjs/helper-wasm-bytecode@1.11.6': {} - - '@webassemblyjs/helper-wasm-section@1.12.1': - dependencies: - '@webassemblyjs/ast': 1.12.1 - '@webassemblyjs/helper-buffer': 1.12.1 - '@webassemblyjs/helper-wasm-bytecode': 1.11.6 - '@webassemblyjs/wasm-gen': 1.12.1 - - '@webassemblyjs/ieee754@1.11.6': - dependencies: - '@xtuc/ieee754': 1.2.0 - - '@webassemblyjs/leb128@1.11.6': - dependencies: - '@xtuc/long': 4.2.2 - - '@webassemblyjs/utf8@1.11.6': {} - - '@webassemblyjs/wasm-edit@1.12.1': - dependencies: - '@webassemblyjs/ast': 1.12.1 - '@webassemblyjs/helper-buffer': 1.12.1 - '@webassemblyjs/helper-wasm-bytecode': 1.11.6 - '@webassemblyjs/helper-wasm-section': 1.12.1 - '@webassemblyjs/wasm-gen': 1.12.1 - '@webassemblyjs/wasm-opt': 1.12.1 - '@webassemblyjs/wasm-parser': 1.12.1 - '@webassemblyjs/wast-printer': 1.12.1 - - '@webassemblyjs/wasm-gen@1.12.1': - dependencies: - '@webassemblyjs/ast': 1.12.1 - '@webassemblyjs/helper-wasm-bytecode': 1.11.6 - '@webassemblyjs/ieee754': 1.11.6 - '@webassemblyjs/leb128': 1.11.6 - '@webassemblyjs/utf8': 1.11.6 - - '@webassemblyjs/wasm-opt@1.12.1': - dependencies: - '@webassemblyjs/ast': 1.12.1 - '@webassemblyjs/helper-buffer': 1.12.1 - '@webassemblyjs/wasm-gen': 1.12.1 - '@webassemblyjs/wasm-parser': 1.12.1 - - '@webassemblyjs/wasm-parser@1.12.1': - dependencies: - '@webassemblyjs/ast': 1.12.1 - '@webassemblyjs/helper-api-error': 1.11.6 - '@webassemblyjs/helper-wasm-bytecode': 1.11.6 - '@webassemblyjs/ieee754': 1.11.6 - '@webassemblyjs/leb128': 1.11.6 - '@webassemblyjs/utf8': 1.11.6 - - '@webassemblyjs/wast-printer@1.12.1': - dependencies: - '@webassemblyjs/ast': 1.12.1 - '@xtuc/long': 4.2.2 - - '@xtuc/ieee754@1.2.0': {} - - '@xtuc/long@4.2.2': {} - - abort-controller@3.0.0: - dependencies: - event-target-shim: 5.0.1 - - abstract-logging@2.0.1: {} - - acorn-import-attributes@1.9.5(acorn@8.12.1): - dependencies: - acorn: 8.12.1 - - acorn-walk@8.3.3: - dependencies: - acorn: 8.12.1 - - acorn@8.12.1: {} - - ajv-formats@2.1.1(ajv@8.17.1): - optionalDependencies: - ajv: 8.17.1 - - ajv-formats@3.0.1(ajv@8.17.1): - optionalDependencies: - ajv: 8.17.1 - - ajv-keywords@3.5.2(ajv@6.12.6): - dependencies: - ajv: 6.12.6 - - ajv@6.12.6: - dependencies: - fast-deep-equal: 3.1.3 - fast-json-stable-stringify: 2.1.0 - json-schema-traverse: 0.4.1 - uri-js: 4.4.1 - - ajv@8.17.1: - dependencies: - fast-deep-equal: 3.1.3 - fast-uri: 3.0.1 - json-schema-traverse: 1.0.0 - require-from-string: 2.0.2 - - ansi-colors@4.1.3: {} - - ansi-regex@5.0.1: {} - - ansi-regex@6.0.1: {} - - ansi-styles@3.2.1: - dependencies: - color-convert: 1.9.3 - - ansi-styles@4.3.0: - dependencies: - color-convert: 2.0.1 - - ansi-styles@6.2.1: {} - - any-promise@1.3.0: {} - - anymatch@3.1.3: - dependencies: - normalize-path: 3.0.0 - picomatch: 2.3.1 - - arg@4.1.3: {} - - arg@5.0.2: {} - - argparse@2.0.1: {} - - array-union@2.1.0: {} - - asap@2.0.6: {} - - assertion-error@2.0.1: {} - - async@0.2.10: {} - - atomic-sleep@1.0.0: {} - - avvio@8.4.0: - dependencies: - '@fastify/error': 3.4.1 - fastq: 1.17.1 - - babel-plugin-polyfill-corejs2@0.4.11(@babel/core@7.25.2): - dependencies: - '@babel/compat-data': 7.25.4 - '@babel/core': 7.25.2 - '@babel/helper-define-polyfill-provider': 0.6.2(@babel/core@7.25.2) - semver: 6.3.1 - transitivePeerDependencies: - - supports-color - - babel-plugin-polyfill-corejs3@0.10.6(@babel/core@7.25.2): - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-define-polyfill-provider': 0.6.2(@babel/core@7.25.2) - core-js-compat: 3.38.1 - transitivePeerDependencies: - - supports-color - - babel-plugin-polyfill-regenerator@0.6.2(@babel/core@7.25.2): - dependencies: - '@babel/core': 7.25.2 - '@babel/helper-define-polyfill-provider': 0.6.2(@babel/core@7.25.2) - transitivePeerDependencies: - - supports-color - - balanced-match@1.0.2: {} - - base64-js@1.5.1: {} - - bath-es5@3.0.3: {} - - binary-extensions@2.3.0: {} - - boolbase@1.0.0: {} - - bowser@2.11.0: {} - - brace-expansion@1.1.11: - dependencies: - balanced-match: 1.0.2 - concat-map: 0.0.1 - - brace-expansion@2.0.1: - dependencies: - balanced-match: 1.0.2 - - braces@3.0.3: - dependencies: - fill-range: 7.1.1 - - browser-stdout@1.3.1: {} - - browserslist@4.23.3: - dependencies: - caniuse-lite: 1.0.30001655 - electron-to-chromium: 1.5.13 - node-releases: 2.0.18 - update-browserslist-db: 1.1.0(browserslist@4.23.3) - - buffer-from@1.1.2: {} - - buffer@5.6.0: - dependencies: - base64-js: 1.5.1 - ieee754: 1.2.1 - - buffer@6.0.3: - dependencies: - base64-js: 1.5.1 - ieee754: 1.2.1 - - bundle-require@5.0.0(esbuild@0.23.1): - dependencies: - esbuild: 0.23.1 - load-tsconfig: 0.2.5 - - cac@6.7.14: {} - - call-bind@1.0.7: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - set-function-length: 1.2.2 - - camelcase@6.3.0: {} - - caniuse-lite@1.0.30001655: {} - - cargo-cp-artifact@0.1.9: {} - - chai@5.1.1: - dependencies: - assertion-error: 2.0.1 - check-error: 2.1.1 - deep-eql: 5.0.2 - loupe: 3.1.1 - pathval: 2.0.0 - - chalk@2.4.2: - dependencies: - ansi-styles: 3.2.1 - escape-string-regexp: 1.0.5 - supports-color: 5.5.0 - - chalk@4.1.2: - dependencies: - ansi-styles: 4.3.0 - supports-color: 7.2.0 - - check-error@2.1.1: {} - - cheerio-select@2.1.0: - dependencies: - boolbase: 1.0.0 - css-select: 5.1.0 - css-what: 6.1.0 - domelementtype: 2.3.0 - domhandler: 5.0.3 - domutils: 3.1.0 - - cheerio@1.0.0: - dependencies: - cheerio-select: 2.1.0 - dom-serializer: 2.0.0 - domhandler: 5.0.3 - domutils: 3.1.0 - encoding-sniffer: 0.2.0 - htmlparser2: 9.1.0 - parse5: 7.1.2 - parse5-htmlparser2-tree-adapter: 7.0.0 - parse5-parser-stream: 7.1.2 - undici: 6.19.8 - whatwg-mimetype: 4.0.0 - - chokidar@3.6.0: - dependencies: - anymatch: 3.1.3 - braces: 3.0.3 - glob-parent: 5.1.2 - is-binary-path: 2.1.0 - is-glob: 4.0.3 - normalize-path: 3.0.0 - readdirp: 3.6.0 - optionalDependencies: - fsevents: 2.3.3 - - chrome-trace-event@1.0.4: {} - - cliui@7.0.4: - dependencies: - string-width: 4.2.3 - strip-ansi: 6.0.1 - wrap-ansi: 7.0.0 - - cliui@8.0.1: - dependencies: - string-width: 4.2.3 - strip-ansi: 6.0.1 - wrap-ansi: 7.0.0 - - color-convert@1.9.3: - dependencies: - color-name: 1.1.3 - - color-convert@2.0.1: - dependencies: - color-name: 1.1.4 - - color-name@1.1.3: {} - - color-name@1.1.4: {} - - color-string@1.9.1: - dependencies: - color-name: 1.1.4 - simple-swizzle: 0.2.2 - - color@4.2.3: - dependencies: - color-convert: 2.0.1 - color-string: 1.9.1 - - commander@12.1.0: {} - - commander@2.20.3: {} - - commander@4.1.1: {} - - concat-map@0.0.1: {} - - concurrently@8.2.2: - dependencies: - chalk: 4.1.2 - date-fns: 2.30.0 - lodash: 4.17.21 - rxjs: 7.8.1 - shell-quote: 1.8.1 - spawn-command: 0.0.2 - supports-color: 8.1.1 - tree-kill: 1.2.2 - yargs: 17.7.2 - - consola@3.2.3: {} - - content-disposition@0.5.4: - dependencies: - safe-buffer: 5.2.1 - - convert-source-map@2.0.0: {} - - cookie@0.5.0: {} - - cookie@0.6.0: {} - - core-js-compat@3.38.1: - dependencies: - browserslist: 4.23.3 - - create-require@1.1.1: {} - - cross-spawn@7.0.3: - dependencies: - path-key: 3.1.1 - shebang-command: 2.0.0 - which: 2.0.2 - - csprng@0.1.2: - dependencies: - sequin: 0.1.1 - - css-select@5.1.0: - dependencies: - boolbase: 1.0.0 - css-what: 6.1.0 - domhandler: 5.0.3 - domutils: 3.1.0 - nth-check: 2.1.1 - - css-what@6.1.0: {} - - data-uri-to-buffer@4.0.1: {} - - date-fns@2.30.0: - dependencies: - '@babel/runtime': 7.25.6 - - date-fns@3.6.0: {} - - debug@4.3.6: - dependencies: - ms: 2.1.2 - - debug@4.3.6(supports-color@5.5.0): - dependencies: - ms: 2.1.2 - optionalDependencies: - supports-color: 5.5.0 - - debug@4.3.6(supports-color@8.1.1): - dependencies: - ms: 2.1.2 - optionalDependencies: - supports-color: 8.1.1 - - decamelize@4.0.0: {} - - deep-eql@5.0.2: {} - - deepmerge@4.3.1: {} - - define-data-property@1.1.4: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - gopd: 1.0.1 - - depd@2.0.0: {} - - dereference-json-schema@0.2.1: {} - - detect-libc@2.0.3: {} - - diff@4.0.2: {} - - diff@5.2.0: {} - - dir-glob@3.0.1: - dependencies: - path-type: 4.0.0 - - dom-serializer@2.0.0: - dependencies: - domelementtype: 2.3.0 - domhandler: 5.0.3 - entities: 4.5.0 - - domelementtype@2.3.0: {} - - domhandler@5.0.3: - dependencies: - domelementtype: 2.3.0 - - domutils@3.1.0: - dependencies: - dom-serializer: 2.0.0 - domelementtype: 2.3.0 - domhandler: 5.0.3 - - dotenv@16.4.5: {} - - eastasianwidth@0.2.0: {} - - electron-to-chromium@1.5.13: {} - - emoji-regex@8.0.0: {} - - emoji-regex@9.2.2: {} - - encoding-japanese@2.0.0: {} - - encoding-japanese@2.1.0: {} - - encoding-japanese@2.2.0: {} - - encoding-sniffer@0.2.0: - dependencies: - iconv-lite: 0.6.3 - whatwg-encoding: 3.1.1 - - enhanced-resolve@5.17.1: - dependencies: - graceful-fs: 4.2.11 - tapable: 2.2.1 - - entities@4.5.0: {} - - es-define-property@1.0.0: - dependencies: - get-intrinsic: 1.2.4 - - es-errors@1.3.0: {} - - es-module-lexer@1.5.4: {} - - esbuild@0.23.1: - optionalDependencies: - '@esbuild/aix-ppc64': 0.23.1 - '@esbuild/android-arm': 0.23.1 - '@esbuild/android-arm64': 0.23.1 - '@esbuild/android-x64': 0.23.1 - '@esbuild/darwin-arm64': 0.23.1 - '@esbuild/darwin-x64': 0.23.1 - '@esbuild/freebsd-arm64': 0.23.1 - '@esbuild/freebsd-x64': 0.23.1 - '@esbuild/linux-arm': 0.23.1 - '@esbuild/linux-arm64': 0.23.1 - '@esbuild/linux-ia32': 0.23.1 - '@esbuild/linux-loong64': 0.23.1 - '@esbuild/linux-mips64el': 0.23.1 - '@esbuild/linux-ppc64': 0.23.1 - '@esbuild/linux-riscv64': 0.23.1 - '@esbuild/linux-s390x': 0.23.1 - '@esbuild/linux-x64': 0.23.1 - '@esbuild/netbsd-x64': 0.23.1 - '@esbuild/openbsd-arm64': 0.23.1 - '@esbuild/openbsd-x64': 0.23.1 - '@esbuild/sunos-x64': 0.23.1 - '@esbuild/win32-arm64': 0.23.1 - '@esbuild/win32-ia32': 0.23.1 - '@esbuild/win32-x64': 0.23.1 - - escalade@3.2.0: {} - - escape-html@1.0.3: {} - - escape-string-regexp@1.0.5: {} - - escape-string-regexp@4.0.0: {} - - eslint-scope@5.1.1: - dependencies: - esrecurse: 4.3.0 - estraverse: 4.3.0 - - esmock@2.6.7: {} - - esrecurse@4.3.0: - dependencies: - estraverse: 5.3.0 - - estraverse@4.3.0: {} - - estraverse@5.3.0: {} - - esutils@2.0.3: {} - - event-target-shim@5.0.1: {} - - events@3.3.0: {} - - execa@5.1.1: - dependencies: - cross-spawn: 7.0.3 - get-stream: 6.0.1 - human-signals: 2.1.0 - is-stream: 2.0.1 - merge-stream: 2.0.0 - npm-run-path: 4.0.1 - onetime: 5.1.2 - signal-exit: 3.0.7 - strip-final-newline: 2.0.0 - - execa@8.0.1: - dependencies: - cross-spawn: 7.0.3 - get-stream: 8.0.1 - human-signals: 5.0.0 - is-stream: 3.0.0 - merge-stream: 2.0.0 - npm-run-path: 5.3.0 - onetime: 6.0.0 - signal-exit: 4.1.0 - strip-final-newline: 3.0.0 - - fast-content-type-parse@1.1.0: {} - - fast-decode-uri-component@1.0.1: {} - - fast-deep-equal@3.1.3: {} - - fast-glob@3.3.2: - dependencies: - '@nodelib/fs.stat': 2.0.5 - '@nodelib/fs.walk': 1.2.8 - glob-parent: 5.1.2 - merge2: 1.4.1 - micromatch: 4.0.8 - - fast-json-stable-stringify@2.1.0: {} - - fast-json-stringify@5.16.1: - dependencies: - '@fastify/merge-json-schemas': 0.1.1 - ajv: 8.17.1 - ajv-formats: 3.0.1(ajv@8.17.1) - fast-deep-equal: 3.1.3 - fast-uri: 2.4.0 - json-schema-ref-resolver: 1.0.1 - rfdc: 1.4.1 - - fast-querystring@1.1.2: - dependencies: - fast-decode-uri-component: 1.0.1 - - fast-redact@3.5.0: {} - - fast-uri@2.4.0: {} - - fast-uri@3.0.1: {} - - fast-xml-parser@4.4.1: - dependencies: - strnum: 1.0.5 - - fastify-plugin@4.5.1: {} - - fastify@4.28.1: - dependencies: - '@fastify/ajv-compiler': 3.6.0 - '@fastify/error': 3.4.1 - '@fastify/fast-json-stringify-compiler': 4.3.0 - abstract-logging: 2.0.1 - avvio: 8.4.0 - fast-content-type-parse: 1.1.0 - fast-json-stringify: 5.16.1 - find-my-way: 8.2.0 - light-my-request: 5.13.0 - pino: 9.4.0 - process-warning: 3.0.0 - proxy-addr: 2.0.7 - rfdc: 1.4.1 - secure-json-parse: 2.7.0 - semver: 7.6.3 - toad-cache: 3.7.0 - - fastq@1.17.1: - dependencies: - reusify: 1.0.4 - - faye-websocket@0.11.4: - dependencies: - websocket-driver: 0.7.4 - - faye@1.4.0: - dependencies: - asap: 2.0.6 - csprng: 0.1.2 - faye-websocket: 0.11.4 - safe-buffer: 5.2.1 - tough-cookie: 4.1.4 - tunnel-agent: 0.6.0 - - fetch-blob@3.2.0: - dependencies: - node-domexception: 1.0.0 - web-streams-polyfill: 3.3.3 - - fill-range@7.1.1: - dependencies: - to-regex-range: 5.0.1 - - find-my-way@8.2.0: - dependencies: - fast-deep-equal: 3.1.3 - fast-querystring: 1.1.2 - safe-regex2: 3.1.0 - - find-up@5.0.0: - dependencies: - locate-path: 6.0.0 - path-exists: 4.0.0 - - flat@5.0.2: {} - - fluent-ffmpeg@2.1.3: - dependencies: - async: 0.2.10 - which: 1.3.1 - - foreground-child@3.3.0: - dependencies: - cross-spawn: 7.0.3 - signal-exit: 4.1.0 - - formdata-polyfill@4.0.10: - dependencies: - fetch-blob: 3.2.0 - - forwarded@0.2.0: {} - - fs-monkey@1.0.6: {} - - fs.realpath@1.0.0: {} - - fsevents@2.3.3: - optional: true - - function-bind@1.1.2: {} - - gensync@1.0.0-beta.2: {} - - get-caller-file@2.0.5: {} - - get-func-name@2.0.2: {} - - get-intrinsic@1.2.4: - dependencies: - es-errors: 1.3.0 - function-bind: 1.1.2 - has-proto: 1.0.3 - has-symbols: 1.0.3 - hasown: 2.0.2 - - get-stream@6.0.1: {} - - get-stream@8.0.1: {} - - get-tsconfig@4.8.0: - dependencies: - resolve-pkg-maps: 1.0.0 - - glob-parent@5.1.2: - dependencies: - is-glob: 4.0.3 - - glob-to-regexp@0.4.1: {} - - glob@10.4.5: - dependencies: - foreground-child: 3.3.0 - jackspeak: 3.4.3 - minimatch: 9.0.5 - minipass: 7.1.2 - package-json-from-dist: 1.0.0 - path-scurry: 1.11.1 - - glob@7.2.3: - dependencies: - fs.realpath: 1.0.0 - inflight: 1.0.6 - inherits: 2.0.4 - minimatch: 3.1.2 - once: 1.4.0 - path-is-absolute: 1.0.1 - - glob@8.1.0: - dependencies: - fs.realpath: 1.0.0 - inflight: 1.0.6 - inherits: 2.0.4 - minimatch: 5.1.6 - once: 1.4.0 - - globals@11.12.0: {} - - globby@11.1.0: - dependencies: - array-union: 2.1.0 - dir-glob: 3.0.1 - fast-glob: 3.3.2 - ignore: 5.3.2 - merge2: 1.4.1 - slash: 3.0.0 - - gopd@1.0.1: - dependencies: - get-intrinsic: 1.2.4 - - graceful-fs@4.2.11: {} - - has-flag@3.0.0: {} - - has-flag@4.0.0: {} - - has-property-descriptors@1.0.2: - dependencies: - es-define-property: 1.0.0 - - has-proto@1.0.3: {} - - has-symbols@1.0.3: {} - - hasown@2.0.2: - dependencies: - function-bind: 1.1.2 - - he@1.2.0: {} - - heap-js@2.5.0: {} - - html-to-text@9.0.5: - dependencies: - '@selderee/plugin-htmlparser2': 0.11.0 - deepmerge: 4.3.1 - dom-serializer: 2.0.0 - htmlparser2: 8.0.2 - selderee: 0.11.0 - - htmlparser2@8.0.2: - dependencies: - domelementtype: 2.3.0 - domhandler: 5.0.3 - domutils: 3.1.0 - entities: 4.5.0 - - htmlparser2@9.1.0: - dependencies: - domelementtype: 2.3.0 - domhandler: 5.0.3 - domutils: 3.1.0 - entities: 4.5.0 - - http-errors@2.0.0: - dependencies: - depd: 2.0.0 - inherits: 2.0.4 - setprototypeof: 1.2.0 - statuses: 2.0.1 - toidentifier: 1.0.1 - - http-parser-js@0.5.8: {} - - human-signals@2.1.0: {} - - human-signals@5.0.0: {} - - hyperdyperid@1.2.0: {} - - iconv-lite@0.6.3: - dependencies: - safer-buffer: 2.1.2 - - ieee754@1.2.1: {} - - ignore-by-default@1.0.1: {} - - ignore@5.3.2: {} - - imapflow@1.0.164: - dependencies: - encoding-japanese: 2.2.0 - iconv-lite: 0.6.3 - libbase64: 1.3.0 - libmime: 5.3.5 - libqp: 2.1.0 - mailsplit: 5.4.0 - nodemailer: 6.9.14 - pino: 9.2.0 - socks: 2.8.3 - - inflight@1.0.6: - dependencies: - once: 1.4.0 - wrappy: 1.0.2 - - inherits@2.0.4: {} - - ip-address@9.0.5: - dependencies: - jsbn: 1.1.0 - sprintf-js: 1.1.3 - - ipaddr.js@1.9.1: {} - - is-arrayish@0.3.2: {} - - is-binary-path@2.1.0: - dependencies: - binary-extensions: 2.3.0 - - is-core-module@2.15.1: - dependencies: - hasown: 2.0.2 - - is-extglob@2.1.1: {} - - is-fullwidth-code-point@3.0.0: {} - - is-glob@4.0.3: - dependencies: - is-extglob: 2.1.1 - - is-network-error@1.1.0: {} - - is-number@7.0.0: {} - - is-plain-obj@2.1.0: {} - - is-stream@2.0.1: {} - - is-stream@3.0.0: {} - - is-unicode-supported@0.1.0: {} - - isexe@2.0.0: {} - - isexe@3.1.1: {} - - jackspeak@3.4.3: - dependencies: - '@isaacs/cliui': 8.0.2 - optionalDependencies: - '@pkgjs/parseargs': 0.11.0 - - jest-worker@27.5.1: - dependencies: - '@types/node': 22.5.2 - merge-stream: 2.0.0 - supports-color: 8.1.1 - - joycon@3.1.1: {} - - js-tokens@4.0.0: {} - - js-yaml@4.1.0: - dependencies: - argparse: 2.0.1 - - jsbn@1.1.0: {} - - jsesc@0.5.0: {} - - jsesc@2.5.2: {} - - json-parse-even-better-errors@2.3.1: {} - - json-schema-ref-resolver@1.0.1: - dependencies: - fast-deep-equal: 3.1.3 - - json-schema-resolver@2.0.0: - dependencies: - debug: 4.3.6 - rfdc: 1.4.1 - uri-js: 4.4.1 - transitivePeerDependencies: - - supports-color - - json-schema-traverse@0.4.1: {} - - json-schema-traverse@1.0.0: {} - - json5@2.2.3: {} - - just-extend@6.2.0: {} - - just-performance@4.3.0: {} - - leac@0.6.0: {} - - libbase64@1.2.1: {} - - libbase64@1.3.0: {} - - libmime@5.2.0: - dependencies: - encoding-japanese: 2.0.0 - iconv-lite: 0.6.3 - libbase64: 1.2.1 - libqp: 2.0.1 - - libmime@5.3.5: - dependencies: - encoding-japanese: 2.1.0 - iconv-lite: 0.6.3 - libbase64: 1.3.0 - libqp: 2.1.0 - - libqp@2.0.1: {} - - libqp@2.1.0: {} - - light-my-request@5.13.0: - dependencies: - cookie: 0.6.0 - process-warning: 3.0.0 - set-cookie-parser: 2.7.0 - - lilconfig@3.1.2: {} - - limiter@2.1.0: - dependencies: - just-performance: 4.3.0 - - lines-and-columns@1.2.4: {} - - linkify-it@5.0.0: - dependencies: - uc.micro: 2.1.0 - - load-tsconfig@0.2.5: {} - - loader-runner@4.3.0: {} - - locate-path@6.0.0: - dependencies: - p-locate: 5.0.0 - - lodash.camelcase@4.3.0: {} - - lodash.debounce@4.0.8: {} - - lodash.get@4.4.2: {} - - lodash.merge@4.6.2: {} - - lodash.sortby@4.7.0: {} - - lodash@4.17.21: {} - - log-symbols@4.1.0: - dependencies: - chalk: 4.1.2 - is-unicode-supported: 0.1.0 - - long@5.2.3: {} - - loupe@3.1.1: - dependencies: - get-func-name: 2.0.2 - - lru-cache@10.4.3: {} - - lru-cache@5.1.1: - dependencies: - yallist: 3.1.1 - - luxon@1.28.1: {} - - mailparser@3.7.1: - dependencies: - encoding-japanese: 2.1.0 - he: 1.2.0 - html-to-text: 9.0.5 - iconv-lite: 0.6.3 - libmime: 5.3.5 - linkify-it: 5.0.0 - mailsplit: 5.4.0 - nodemailer: 6.9.13 - punycode.js: 2.3.1 - tlds: 1.252.0 - - mailsplit@5.4.0: - dependencies: - libbase64: 1.2.1 - libmime: 5.2.0 - libqp: 2.0.1 - - make-error@1.3.6: {} - - memfs@4.11.1: - dependencies: - '@jsonjoy.com/json-pack': 1.1.0(tslib@2.7.0) - '@jsonjoy.com/util': 1.3.0(tslib@2.7.0) - tree-dump: 1.0.2(tslib@2.7.0) - tslib: 2.7.0 - - merge-stream@2.0.0: {} - - merge2@1.4.1: {} - - micromatch@4.0.8: - dependencies: - braces: 3.0.3 - picomatch: 2.3.1 - - mime-db@1.52.0: {} - - mime-types@2.1.35: - dependencies: - mime-db: 1.52.0 - - mime@3.0.0: {} - - mimic-fn@2.1.0: {} - - mimic-fn@4.0.0: {} - - minimatch@3.1.2: - dependencies: - brace-expansion: 1.1.11 - - minimatch@5.1.6: - dependencies: - brace-expansion: 2.0.1 - - minimatch@9.0.5: - dependencies: - brace-expansion: 2.0.1 - - minipass@7.1.2: {} - - mocha@10.7.3: - dependencies: - ansi-colors: 4.1.3 - browser-stdout: 1.3.1 - chokidar: 3.6.0 - debug: 4.3.6(supports-color@8.1.1) - diff: 5.2.0 - escape-string-regexp: 4.0.0 - find-up: 5.0.0 - glob: 8.1.0 - he: 1.2.0 - js-yaml: 4.1.0 - log-symbols: 4.1.0 - minimatch: 5.1.6 - ms: 2.1.3 - serialize-javascript: 6.0.2 - strip-json-comments: 3.1.1 - supports-color: 8.1.1 - workerpool: 6.5.1 - yargs: 16.2.0 - yargs-parser: 20.2.9 - yargs-unparser: 2.0.0 - - mock-json-schema@1.1.1: - dependencies: - lodash: 4.17.21 - - ms@2.1.2: {} - - ms@2.1.3: {} - - ms@3.0.0-canary.1: {} - - mz@2.7.0: - dependencies: - any-promise: 1.3.0 - object-assign: 4.1.1 - thenify-all: 1.6.0 - - neo-async@2.6.2: {} - - nise@6.0.0: - dependencies: - '@sinonjs/commons': 3.0.1 - '@sinonjs/fake-timers': 11.3.1 - '@sinonjs/text-encoding': 0.7.3 - just-extend: 6.2.0 - path-to-regexp: 6.2.2 - - node-domexception@1.0.0: {} - - node-fetch@3.3.2: - dependencies: - data-uri-to-buffer: 4.0.1 - fetch-blob: 3.2.0 - formdata-polyfill: 4.0.10 - - node-releases@2.0.18: {} - - nodemailer@6.9.13: {} - - nodemailer@6.9.14: {} - - nodemon@3.1.4: - dependencies: - chokidar: 3.6.0 - debug: 4.3.6(supports-color@5.5.0) - ignore-by-default: 1.0.1 - minimatch: 3.1.2 - pstree.remy: 1.1.8 - semver: 7.6.3 - simple-update-notifier: 2.0.0 - supports-color: 5.5.0 - touch: 3.1.1 - undefsafe: 2.0.5 - - normalize-path@3.0.0: {} - - npm-run-path@4.0.1: - dependencies: - path-key: 3.1.1 - - npm-run-path@5.3.0: - dependencies: - path-key: 4.0.0 - - nth-check@2.1.1: - dependencies: - boolbase: 1.0.0 - - object-assign@4.1.1: {} - - object-inspect@1.13.2: {} - - obuf@1.1.2: {} - - on-exit-leak-free@2.1.2: {} - - once@1.4.0: - dependencies: - wrappy: 1.0.2 - - onetime@5.1.2: - dependencies: - mimic-fn: 2.1.0 - - onetime@6.0.0: - dependencies: - mimic-fn: 4.0.0 - - openapi-backend@5.10.6: - dependencies: - '@apidevtools/json-schema-ref-parser': 11.7.0 - ajv: 8.17.1 - bath-es5: 3.0.3 - cookie: 0.5.0 - dereference-json-schema: 0.2.1 - lodash: 4.17.21 - mock-json-schema: 1.1.1 - openapi-schema-validator: 12.1.3 - openapi-types: 12.1.3 - qs: 6.13.0 - - openapi-schema-validator@12.1.3: - dependencies: - ajv: 8.17.1 - ajv-formats: 2.1.1(ajv@8.17.1) - lodash.merge: 4.6.2 - openapi-types: 12.1.3 - - openapi-types@12.1.3: {} - - p-limit@3.1.0: - dependencies: - yocto-queue: 0.1.0 - - p-locate@5.0.0: - dependencies: - p-limit: 3.1.0 - - p-retry@6.2.0: - dependencies: - '@types/retry': 0.12.2 - is-network-error: 1.1.0 - retry: 0.13.1 - - package-json-from-dist@1.0.0: {} - - parse5-htmlparser2-tree-adapter@7.0.0: - dependencies: - domhandler: 5.0.3 - parse5: 7.1.2 - - parse5-parser-stream@7.1.2: - dependencies: - parse5: 7.1.2 - - parse5@7.1.2: - dependencies: - entities: 4.5.0 - - parseley@0.12.1: - dependencies: - leac: 0.6.0 - peberminta: 0.9.0 - - path-equal@1.2.5: {} - - path-exists@4.0.0: {} - - path-is-absolute@1.0.1: {} - - path-key@3.1.1: {} - - path-key@4.0.0: {} - - path-parse@1.0.7: {} - - path-scurry@1.11.1: - dependencies: - lru-cache: 10.4.3 - minipass: 7.1.2 - - path-to-regexp@6.2.2: {} - - path-type@4.0.0: {} - - pathval@2.0.0: {} - - peberminta@0.9.0: {} - - pg-cloudflare@1.1.1: - optional: true - - pg-connection-string@2.6.4: {} - - pg-int8@1.0.1: {} - - pg-numeric@1.0.2: {} - - pg-pool@3.6.2(pg@8.12.0): - dependencies: - pg: 8.12.0 - - pg-protocol@1.6.1: {} - - pg-types@2.2.0: - dependencies: - pg-int8: 1.0.1 - postgres-array: 2.0.0 - postgres-bytea: 1.0.0 - postgres-date: 1.0.7 - postgres-interval: 1.2.0 - - pg-types@4.0.2: - dependencies: - pg-int8: 1.0.1 - pg-numeric: 1.0.2 - postgres-array: 3.0.2 - postgres-bytea: 3.0.0 - postgres-date: 2.1.0 - postgres-interval: 3.0.0 - postgres-range: 1.1.4 - - pg@8.12.0: - dependencies: - pg-connection-string: 2.6.4 - pg-pool: 3.6.2(pg@8.12.0) - pg-protocol: 1.6.1 - pg-types: 2.2.0 - pgpass: 1.0.5 - optionalDependencies: - pg-cloudflare: 1.1.1 - - pgpass@1.0.5: - dependencies: - split2: 4.2.0 - - picocolors@1.1.0: {} - - picomatch@2.3.1: {} - - pino-abstract-transport@1.2.0: - dependencies: - readable-stream: 4.5.2 - split2: 4.2.0 - - pino-std-serializers@7.0.0: {} - - pino@9.2.0: - dependencies: - atomic-sleep: 1.0.0 - fast-redact: 3.5.0 - on-exit-leak-free: 2.1.2 - pino-abstract-transport: 1.2.0 - pino-std-serializers: 7.0.0 - process-warning: 3.0.0 - quick-format-unescaped: 4.0.4 - real-require: 0.2.0 - safe-stable-stringify: 2.5.0 - sonic-boom: 4.1.0 - thread-stream: 3.1.0 - - pino@9.4.0: - dependencies: - atomic-sleep: 1.0.0 - fast-redact: 3.5.0 - on-exit-leak-free: 2.1.2 - pino-abstract-transport: 1.2.0 - pino-std-serializers: 7.0.0 - process-warning: 4.0.0 - quick-format-unescaped: 4.0.4 - real-require: 0.2.0 - safe-stable-stringify: 2.5.0 - sonic-boom: 4.1.0 - thread-stream: 3.1.0 - - pirates@4.0.6: {} - - postcss-load-config@6.0.1(tsx@4.19.0)(yaml@2.5.0): - dependencies: - lilconfig: 3.1.2 - optionalDependencies: - tsx: 4.19.0 - yaml: 2.5.0 - - postgres-array@2.0.0: {} - - postgres-array@3.0.2: {} - - postgres-bytea@1.0.0: {} - - postgres-bytea@3.0.0: - dependencies: - obuf: 1.1.2 - - postgres-date@1.0.7: {} - - postgres-date@2.1.0: {} - - postgres-interval@1.2.0: - dependencies: - xtend: 4.0.2 - - postgres-interval@3.0.0: {} - - postgres-range@1.1.4: {} - - prevvy@8.0.1: - dependencies: - '@types/debug': 4.1.12 - '@types/fluent-ffmpeg': 2.1.26 - '@types/luxon': 3.4.2 - '@types/node': 20.16.3 - debug: 4.3.6 - execa: 8.0.1 - fluent-ffmpeg: 2.1.3 - luxon: 1.28.1 - typescript: 5.5.4 - transitivePeerDependencies: - - supports-color - - process-warning@3.0.0: {} - - process-warning@4.0.0: {} - - process@0.11.10: {} - - proto3-json-serializer@2.0.2: - dependencies: - protobufjs: 7.4.0 - - protobufjs@7.4.0: - dependencies: - '@protobufjs/aspromise': 1.1.2 - '@protobufjs/base64': 1.1.2 - '@protobufjs/codegen': 2.0.4 - '@protobufjs/eventemitter': 1.1.0 - '@protobufjs/fetch': 1.1.0 - '@protobufjs/float': 1.0.2 - '@protobufjs/inquire': 1.1.0 - '@protobufjs/path': 1.1.2 - '@protobufjs/pool': 1.1.0 - '@protobufjs/utf8': 1.1.0 - '@types/node': 22.5.2 - long: 5.2.3 - - proxy-addr@2.0.7: - dependencies: - forwarded: 0.2.0 - ipaddr.js: 1.9.1 - - psl@1.9.0: {} - - pstree.remy@1.1.8: {} - - punycode.js@2.3.1: {} - - punycode@2.3.1: {} - - qs@6.13.0: - dependencies: - side-channel: 1.0.6 - - querystringify@2.2.0: {} - - queue-microtask@1.2.3: {} - - quick-format-unescaped@4.0.4: {} - - randombytes@2.1.0: - dependencies: - safe-buffer: 5.2.1 - - rate-limiter-flexible@5.0.3: {} - - readable-stream@3.6.2: - dependencies: - inherits: 2.0.4 - string_decoder: 1.3.0 - util-deprecate: 1.0.2 - - readable-stream@4.5.2: - dependencies: - abort-controller: 3.0.0 - buffer: 6.0.3 - events: 3.3.0 - process: 0.11.10 - string_decoder: 1.3.0 - - readdirp@3.6.0: - dependencies: - picomatch: 2.3.1 - - real-require@0.2.0: {} - - regenerate-unicode-properties@10.1.1: - dependencies: - regenerate: 1.4.2 - - regenerate@1.4.2: {} - - regenerator-runtime@0.14.1: {} - - regenerator-transform@0.15.2: - dependencies: - '@babel/runtime': 7.25.6 - - regexpu-core@5.3.2: - dependencies: - '@babel/regjsgen': 0.8.0 - regenerate: 1.4.2 - regenerate-unicode-properties: 10.1.1 - regjsparser: 0.9.1 - unicode-match-property-ecmascript: 2.0.0 - unicode-match-property-value-ecmascript: 2.1.0 - - regjsparser@0.9.1: - dependencies: - jsesc: 0.5.0 - - require-directory@2.1.1: {} - - require-from-string@2.0.2: {} - - requires-port@1.0.0: {} - - resolve-from@5.0.0: {} - - resolve-pkg-maps@1.0.0: {} - - resolve@1.22.8: - dependencies: - is-core-module: 2.15.1 - path-parse: 1.0.7 - supports-preserve-symlinks-flag: 1.0.0 - - ret@0.4.3: {} - - retry@0.13.1: {} - - reusify@1.0.4: {} - - rfdc@1.4.1: {} - - rollup@4.21.2: - dependencies: - '@types/estree': 1.0.5 - optionalDependencies: - '@rollup/rollup-android-arm-eabi': 4.21.2 - '@rollup/rollup-android-arm64': 4.21.2 - '@rollup/rollup-darwin-arm64': 4.21.2 - '@rollup/rollup-darwin-x64': 4.21.2 - '@rollup/rollup-linux-arm-gnueabihf': 4.21.2 - '@rollup/rollup-linux-arm-musleabihf': 4.21.2 - '@rollup/rollup-linux-arm64-gnu': 4.21.2 - '@rollup/rollup-linux-arm64-musl': 4.21.2 - '@rollup/rollup-linux-powerpc64le-gnu': 4.21.2 - '@rollup/rollup-linux-riscv64-gnu': 4.21.2 - '@rollup/rollup-linux-s390x-gnu': 4.21.2 - '@rollup/rollup-linux-x64-gnu': 4.21.2 - '@rollup/rollup-linux-x64-musl': 4.21.2 - '@rollup/rollup-win32-arm64-msvc': 4.21.2 - '@rollup/rollup-win32-ia32-msvc': 4.21.2 - '@rollup/rollup-win32-x64-msvc': 4.21.2 - fsevents: 2.3.3 - - run-parallel@1.2.0: - dependencies: - queue-microtask: 1.2.3 - - rxjs@7.8.1: - dependencies: - tslib: 2.7.0 - - safe-buffer@5.2.1: {} - - safe-regex2@3.1.0: - dependencies: - ret: 0.4.3 - - safe-stable-stringify@2.5.0: {} - - safer-buffer@2.1.2: {} - - schema-utils@3.3.0: - dependencies: - '@types/json-schema': 7.0.15 - ajv: 6.12.6 - ajv-keywords: 3.5.2(ajv@6.12.6) - - secure-json-parse@2.7.0: {} - - selderee@0.11.0: - dependencies: - parseley: 0.12.1 - - semver@6.3.1: {} - - semver@7.6.3: {} - - sequin@0.1.1: {} - - serialize-javascript@6.0.2: - dependencies: - randombytes: 2.1.0 - - set-cookie-parser@2.7.0: {} - - set-function-length@1.2.2: - dependencies: - define-data-property: 1.1.4 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - gopd: 1.0.1 - has-property-descriptors: 1.0.2 - - setprototypeof@1.2.0: {} - - sharp@0.33.5: - dependencies: - color: 4.2.3 - detect-libc: 2.0.3 - semver: 7.6.3 - optionalDependencies: - '@img/sharp-darwin-arm64': 0.33.5 - '@img/sharp-darwin-x64': 0.33.5 - '@img/sharp-libvips-darwin-arm64': 1.0.4 - '@img/sharp-libvips-darwin-x64': 1.0.4 - '@img/sharp-libvips-linux-arm': 1.0.5 - '@img/sharp-libvips-linux-arm64': 1.0.4 - '@img/sharp-libvips-linux-s390x': 1.0.4 - '@img/sharp-libvips-linux-x64': 1.0.4 - '@img/sharp-libvips-linuxmusl-arm64': 1.0.4 - '@img/sharp-libvips-linuxmusl-x64': 1.0.4 - '@img/sharp-linux-arm': 0.33.5 - '@img/sharp-linux-arm64': 0.33.5 - '@img/sharp-linux-s390x': 0.33.5 - '@img/sharp-linux-x64': 0.33.5 - '@img/sharp-linuxmusl-arm64': 0.33.5 - '@img/sharp-linuxmusl-x64': 0.33.5 - '@img/sharp-wasm32': 0.33.5 - '@img/sharp-win32-ia32': 0.33.5 - '@img/sharp-win32-x64': 0.33.5 - - shebang-command@2.0.0: - dependencies: - shebang-regex: 3.0.0 - - shebang-regex@3.0.0: {} - - shell-quote@1.8.1: {} - - side-channel@1.0.6: - dependencies: - call-bind: 1.0.7 - es-errors: 1.3.0 - get-intrinsic: 1.2.4 - object-inspect: 1.13.2 - - signal-exit@3.0.7: {} - - signal-exit@4.1.0: {} - - simple-swizzle@0.2.2: - dependencies: - is-arrayish: 0.3.2 - - simple-update-notifier@2.0.0: - dependencies: - semver: 7.6.3 - - sinon@18.0.0: - dependencies: - '@sinonjs/commons': 3.0.1 - '@sinonjs/fake-timers': 11.3.1 - '@sinonjs/samsam': 8.0.0 - diff: 5.2.0 - nise: 6.0.0 - supports-color: 7.2.0 - - slash@3.0.0: {} - - slugify@1.6.6: {} - - smart-buffer@4.2.0: {} - - socks@2.8.3: - dependencies: - ip-address: 9.0.5 - smart-buffer: 4.2.0 - - sonic-boom@4.1.0: - dependencies: - atomic-sleep: 1.0.0 - - source-map-js@1.2.0: {} - - source-map-loader@4.0.2(webpack@5.94.0(@swc/core@1.7.23)(esbuild@0.23.1)): - dependencies: - iconv-lite: 0.6.3 - source-map-js: 1.2.0 - webpack: 5.94.0(@swc/core@1.7.23)(esbuild@0.23.1) - - source-map-support@0.5.21: - dependencies: - buffer-from: 1.1.2 - source-map: 0.6.1 - - source-map@0.6.1: {} - - source-map@0.7.4: {} - - source-map@0.8.0-beta.0: - dependencies: - whatwg-url: 7.1.0 - - spawn-command@0.0.2: {} - - split2@4.2.0: {} - - sprintf-js@1.1.3: {} - - statuses@2.0.1: {} - - stream-browserify@3.0.0: - dependencies: - inherits: 2.0.4 - readable-stream: 3.6.2 - - string-width@4.2.3: - dependencies: - emoji-regex: 8.0.0 - is-fullwidth-code-point: 3.0.0 - strip-ansi: 6.0.1 - - string-width@5.1.2: - dependencies: - eastasianwidth: 0.2.0 - emoji-regex: 9.2.2 - strip-ansi: 7.1.0 - - string_decoder@1.3.0: - dependencies: - safe-buffer: 5.2.1 - - strip-ansi@6.0.1: - dependencies: - ansi-regex: 5.0.1 - - strip-ansi@7.1.0: - dependencies: - ansi-regex: 6.0.1 - - strip-final-newline@2.0.0: {} - - strip-final-newline@3.0.0: {} - - strip-json-comments@3.1.1: {} - - strnum@1.0.5: {} - - sucrase@3.35.0: - dependencies: - '@jridgewell/gen-mapping': 0.3.5 - commander: 4.1.1 - glob: 10.4.5 - lines-and-columns: 1.2.4 - mz: 2.7.0 - pirates: 4.0.6 - ts-interface-checker: 0.1.13 - - supports-color@5.5.0: - dependencies: - has-flag: 3.0.0 - - supports-color@7.2.0: - dependencies: - has-flag: 4.0.0 - - supports-color@8.1.1: - dependencies: - has-flag: 4.0.0 - - supports-preserve-symlinks-flag@1.0.0: {} - - swagger-editor-dist@4.13.1: {} - - swagger-ui-dist@5.17.14: {} - - swc-loader@0.2.6(@swc/core@1.7.23)(webpack@5.94.0(@swc/core@1.7.23)(esbuild@0.23.1)): - dependencies: - '@swc/core': 1.7.23 - '@swc/counter': 0.1.3 - webpack: 5.94.0(@swc/core@1.7.23)(esbuild@0.23.1) - - tapable@2.2.1: {} - - terser-webpack-plugin@5.3.10(@swc/core@1.7.23)(esbuild@0.23.1)(webpack@5.94.0(@swc/core@1.7.23)(esbuild@0.23.1)): - dependencies: - '@jridgewell/trace-mapping': 0.3.25 - jest-worker: 27.5.1 - schema-utils: 3.3.0 - serialize-javascript: 6.0.2 - terser: 5.31.6 - webpack: 5.94.0(@swc/core@1.7.23)(esbuild@0.23.1) - optionalDependencies: - '@swc/core': 1.7.23 - esbuild: 0.23.1 - - terser@5.31.6: - dependencies: - '@jridgewell/source-map': 0.3.6 - acorn: 8.12.1 - commander: 2.20.3 - source-map-support: 0.5.21 - - thenify-all@1.6.0: - dependencies: - thenify: 3.3.1 - - thenify@3.3.1: - dependencies: - any-promise: 1.3.0 - - thingies@1.21.0(tslib@2.7.0): - dependencies: - tslib: 2.7.0 - - thread-stream@3.1.0: - dependencies: - real-require: 0.2.0 - - tlds@1.252.0: {} - - to-fast-properties@2.0.0: {} - - to-regex-range@5.0.1: - dependencies: - is-number: 7.0.0 - - toad-cache@3.7.0: {} - - toidentifier@1.0.1: {} - - touch@3.1.1: {} - - tough-cookie@4.1.4: - dependencies: - psl: 1.9.0 - punycode: 2.3.1 - universalify: 0.2.0 - url-parse: 1.5.10 - - tr46@1.0.1: - dependencies: - punycode: 2.3.1 - - tree-dump@1.0.2(tslib@2.7.0): - dependencies: - tslib: 2.7.0 - - tree-kill@1.2.2: {} - - ts-interface-checker@0.1.13: {} - - ts-json-schema-generator@2.3.0: - dependencies: - '@types/json-schema': 7.0.15 - commander: 12.1.0 - glob: 10.4.5 - json5: 2.2.3 - normalize-path: 3.0.0 - safe-stable-stringify: 2.5.0 - tslib: 2.7.0 - typescript: 5.5.4 - - ts-node@10.9.2(@swc/core@1.7.23)(@types/node@18.19.48)(typescript@5.5.4): - dependencies: - '@cspotcode/source-map-support': 0.8.1 - '@tsconfig/node10': 1.0.11 - '@tsconfig/node12': 1.0.11 - '@tsconfig/node14': 1.0.3 - '@tsconfig/node16': 1.0.4 - '@types/node': 18.19.48 - acorn: 8.12.1 - acorn-walk: 8.3.3 - arg: 4.1.3 - create-require: 1.1.1 - diff: 4.0.2 - make-error: 1.3.6 - typescript: 5.5.4 - v8-compile-cache-lib: 3.0.1 - yn: 3.1.1 - optionalDependencies: - '@swc/core': 1.7.23 - - ts-node@10.9.2(@swc/core@1.7.23)(@types/node@22.5.2)(typescript@5.5.4): - dependencies: - '@cspotcode/source-map-support': 0.8.1 - '@tsconfig/node10': 1.0.11 - '@tsconfig/node12': 1.0.11 - '@tsconfig/node14': 1.0.3 - '@tsconfig/node16': 1.0.4 - '@types/node': 22.5.2 - acorn: 8.12.1 - acorn-walk: 8.3.3 - arg: 4.1.3 - create-require: 1.1.1 - diff: 4.0.2 - make-error: 1.3.6 - typescript: 5.5.4 - v8-compile-cache-lib: 3.0.1 - yn: 3.1.1 - optionalDependencies: - '@swc/core': 1.7.23 - - tslib@2.7.0: {} - - tsup@8.2.4(@swc/core@1.7.23)(tsx@4.19.0)(typescript@5.5.4)(yaml@2.5.0): - dependencies: - bundle-require: 5.0.0(esbuild@0.23.1) - cac: 6.7.14 - chokidar: 3.6.0 - consola: 3.2.3 - debug: 4.3.6 - esbuild: 0.23.1 - execa: 5.1.1 - globby: 11.1.0 - joycon: 3.1.1 - picocolors: 1.1.0 - postcss-load-config: 6.0.1(tsx@4.19.0)(yaml@2.5.0) - resolve-from: 5.0.0 - rollup: 4.21.2 - source-map: 0.8.0-beta.0 - sucrase: 3.35.0 - tree-kill: 1.2.2 - optionalDependencies: - '@swc/core': 1.7.23 - typescript: 5.5.4 - transitivePeerDependencies: - - jiti - - supports-color - - tsx - - yaml - - tsx@4.19.0: - dependencies: - esbuild: 0.23.1 - get-tsconfig: 4.8.0 - optionalDependencies: - fsevents: 2.3.3 - - tunnel-agent@0.6.0: - dependencies: - safe-buffer: 5.2.1 - - type-detect@4.0.8: {} - - type-detect@4.1.0: {} - - typescript-json-schema@0.65.1(@swc/core@1.7.23): - dependencies: - '@types/json-schema': 7.0.15 - '@types/node': 18.19.48 - glob: 7.2.3 - path-equal: 1.2.5 - safe-stable-stringify: 2.5.0 - ts-node: 10.9.2(@swc/core@1.7.23)(@types/node@18.19.48)(typescript@5.5.4) - typescript: 5.5.4 - yargs: 17.7.2 - transitivePeerDependencies: - - '@swc/core' - - '@swc/wasm' - - typescript@5.5.4: {} - - uc.micro@2.1.0: {} - - undefsafe@2.0.5: {} - - undici-types@5.26.5: {} - - undici-types@6.19.8: {} - - undici@6.19.8: {} - - unicode-canonical-property-names-ecmascript@2.0.0: {} - - unicode-match-property-ecmascript@2.0.0: - dependencies: - unicode-canonical-property-names-ecmascript: 2.0.0 - unicode-property-aliases-ecmascript: 2.1.0 - - unicode-match-property-value-ecmascript@2.1.0: {} - - unicode-property-aliases-ecmascript@2.1.0: {} - - unionfs@4.5.4: - dependencies: - fs-monkey: 1.0.6 - - universalify@0.2.0: {} - - update-browserslist-db@1.1.0(browserslist@4.23.3): - dependencies: - browserslist: 4.23.3 - escalade: 3.2.0 - picocolors: 1.1.0 - - uri-js@4.4.1: - dependencies: - punycode: 2.3.1 - - url-parse@1.5.10: - dependencies: - querystringify: 2.2.0 - requires-port: 1.0.0 - - util-deprecate@1.0.2: {} - - uuid@9.0.1: {} - - v8-compile-cache-lib@3.0.1: {} - - watchpack@2.4.2: - dependencies: - glob-to-regexp: 0.4.1 - graceful-fs: 4.2.11 - - web-streams-polyfill@3.3.3: {} - - webidl-conversions@4.0.2: {} - - webpack-sources@3.2.3: {} - - webpack@5.94.0(@swc/core@1.7.23)(esbuild@0.23.1): - dependencies: - '@types/estree': 1.0.5 - '@webassemblyjs/ast': 1.12.1 - '@webassemblyjs/wasm-edit': 1.12.1 - '@webassemblyjs/wasm-parser': 1.12.1 - acorn: 8.12.1 - acorn-import-attributes: 1.9.5(acorn@8.12.1) - browserslist: 4.23.3 - chrome-trace-event: 1.0.4 - enhanced-resolve: 5.17.1 - es-module-lexer: 1.5.4 - eslint-scope: 5.1.1 - events: 3.3.0 - glob-to-regexp: 0.4.1 - graceful-fs: 4.2.11 - json-parse-even-better-errors: 2.3.1 - loader-runner: 4.3.0 - mime-types: 2.1.35 - neo-async: 2.6.2 - schema-utils: 3.3.0 - tapable: 2.2.1 - terser-webpack-plugin: 5.3.10(@swc/core@1.7.23)(esbuild@0.23.1)(webpack@5.94.0(@swc/core@1.7.23)(esbuild@0.23.1)) - watchpack: 2.4.2 - webpack-sources: 3.2.3 - transitivePeerDependencies: - - '@swc/core' - - esbuild - - uglify-js - - websocket-driver@0.7.4: - dependencies: - http-parser-js: 0.5.8 - safe-buffer: 5.2.1 - websocket-extensions: 0.1.4 - - websocket-extensions@0.1.4: {} - - whatwg-encoding@3.1.1: - dependencies: - iconv-lite: 0.6.3 - - whatwg-mimetype@4.0.0: {} - - whatwg-url@7.1.0: - dependencies: - lodash.sortby: 4.7.0 - tr46: 1.0.1 - webidl-conversions: 4.0.2 - - which@1.3.1: - dependencies: - isexe: 2.0.0 - - which@2.0.2: - dependencies: - isexe: 2.0.0 - - which@4.0.0: - dependencies: - isexe: 3.1.1 - - workerpool@6.5.1: {} - - wrap-ansi@7.0.0: - dependencies: - ansi-styles: 4.3.0 - string-width: 4.2.3 - strip-ansi: 6.0.1 - - wrap-ansi@8.1.0: - dependencies: - ansi-styles: 6.2.1 - string-width: 5.1.2 - strip-ansi: 7.1.0 - - wrappy@1.0.2: {} - - xpath@0.0.34: {} - - xtend@4.0.2: {} - - y18n@5.0.8: {} - - yallist@3.1.1: {} - - yaml@2.5.0: {} - - yargs-parser@20.2.9: {} - - yargs-parser@21.1.1: {} - - yargs-unparser@2.0.0: - dependencies: - camelcase: 6.3.0 - decamelize: 4.0.0 - flat: 5.0.2 - is-plain-obj: 2.1.0 - - yargs@16.2.0: - dependencies: - cliui: 7.0.4 - escalade: 3.2.0 - get-caller-file: 2.0.5 - require-directory: 2.1.1 - string-width: 4.2.3 - y18n: 5.0.8 - yargs-parser: 20.2.9 - - yargs@17.7.2: - dependencies: - cliui: 8.0.1 - escalade: 3.2.0 - get-caller-file: 2.0.5 - require-directory: 2.1.1 - string-width: 4.2.3 - y18n: 5.0.8 - yargs-parser: 21.1.1 - - yn@3.1.1: {} - - yocto-queue@0.1.0: {} diff --git a/services/scout/src/api.yaml b/services/scout/src/api.yaml deleted file mode 100644 index 1221577..0000000 --- a/services/scout/src/api.yaml +++ /dev/null @@ -1,55 +0,0 @@ -openapi: '3.0.1' -info: - title: "@futureporn/scout" - version: '4.0.1' - description: REST API for acquiring vtuber data - -paths: - - '/pets': - get: - operationId: getPets - summary: List pets - responses: - 200: - $ref: '#/components/responses/PetListWithExample' - - '/pets/{id}': - get: - operationId: getPetById - summary: Get pet by its id - parameters: - - name: id - in: path - required: true - schema: - type: integer - minimum: 1 - responses: - 200: - $ref: '#/components/responses/PetResponseWithSchema' - -components: - responses: - PetListWithExample: - description: List of pets - content: - 'application/json': - example: - - id: 1 - name: Garfield - - id: 2 - name: Odie - PetResponseWithSchema: - description: A single pet - content: - 'application/json': - schema: - type: object - properties: - id: - type: integer - minimum: 1 - name: - type: string - example: Garfield diff --git a/services/scout/src/cb.spec.ts b/services/scout/src/cb.spec.ts deleted file mode 100644 index ac04298..0000000 --- a/services/scout/src/cb.spec.ts +++ /dev/null @@ -1,45 +0,0 @@ -import { describe } from 'mocha' -import { expect } from 'chai'; -import { getBroadcasterDisplayName, getInitialRoomDossier, getRandomRoom } from './cb.js' -import { dirname, join } from 'path'; -import { fileURLToPath } from 'url'; -import fs from 'node:fs/promises' -const __dirname = dirname(fileURLToPath(import.meta.url)); -const cbRoomFixturePath = join(__dirname, './fixtures/cb.projektmelody.fixture.html') - -describe('cb', function () { - let fixture: string - beforeEach(async function () { - const fileContent = await fs.readFile(cbRoomFixturePath, { encoding: 'utf-8' }) - fixture = fileContent - }) - describe('unit', function () { - describe('getInitialRoomDossier', function () { - it('should return a Dossier object', function () { - const dossier = getInitialRoomDossier(fixture) - expect(dossier).to.have.property('broadcaster_username', 'projektmelody') - expect(dossier).to.have.property('broadcaster_gender', 'female') - expect(dossier).to.have.property('room_uid', 'G0TWFS5') - }) - }) - describe('getBroadcasterDisplayName', function () { - it('should get a formatted display name', function () { - expect(fixture).to.match(/Chaturbate/) - const displayName = getBroadcasterDisplayName(fixture) - expect(displayName).to.equal('Projektmelody') - }) - }) - }) - describe('integration', function () { - describe('getRandomRoom', function () { - it('should return a Room object of an online room', async function () { - this.timeout(1000*16) - const room = await getRandomRoom() - expect(room).to.have.property('url') - expect(room).to.have.property('name') - expect(room.name).to.match(/[a-z_]/) - expect(room.url).to.match(/https:\/\//) - }) - }) - }) -}) \ No newline at end of file diff --git a/services/scout/src/cb.ts b/services/scout/src/cb.ts deleted file mode 100644 index c27aa07..0000000 --- a/services/scout/src/cb.ts +++ /dev/null @@ -1,233 +0,0 @@ -import * as cheerio from 'cheerio' -import fetch from 'node-fetch' -import scrapingFetch from './scrapingFetch.ts'; - -interface Dossier { - viewer_uid: null | string; - is_age_verified: boolean; - age: number; - room_status: 'offline' | 'online'; - num_viewers: number; - wschat_host: string; - viewer_username: string; - viewer_gender: 'm' | 'f'; - allow_anonymous_tipping: boolean; - chat_username: string; - chat_password: string; - broadcaster_username: string; - room_pass: string; - last_pass: string; - chat_rules: string; - room_title: string; - room_uid: string; - broadcaster_uid: string; - broadcaster_gender: 'female' | 'male'; - apps_running: string; - hls_source: string; - dismissible_messages: string[]; - edge_auth: string; - is_widescreen: boolean; - allow_private_shows: boolean; - private_show_price: number; - private_min_minutes: number; - allow_show_recordings: boolean; - spy_private_show_price: number; - private_show_id: string; - low_satisfaction_score: boolean; - hidden_message: string; - following: boolean; - follow_notification_frequency: string; - is_moderator: boolean; - chat_settings: { - font_size: string; - show_emoticons: boolean; - emoticon_autocomplete_delay: number; - sort_users_key: string; - room_entry_for: 'org' | 'user'; - room_leave_for: 'org' | 'user'; - c2c_notify_limit: number; - silence_broadcasters: string; - allowed_chat: 'all' | 'whisper'; - collapse_notices: boolean; - highest_token_color: string; - mod_expire: number; - max_pm_age: number; - font_family: string; - font_color: string; - tip_volume: number; - ignored_users: string; - }; - broadcaster_on_new_chat: boolean; - token_balance: number; - is_supporter: boolean; - needs_supporter_to_pm: boolean; - server_name: string; - num_followed: number; - num_followed_online: number; - has_studio: boolean; - is_mobile: boolean; - ignored_emoticons: string[]; - tfa_enabled: boolean; - satisfaction_score: { - percent: number; - up_votes: number; - down_votes: number; - max: number; - }; - hide_satisfaction_score: boolean; - tips_in_past_24_hours: number; - last_vote_in_past_24_hours: string | null; - last_vote_in_past_90_days_down: boolean; - show_mobile_site_banner_link: boolean; - exploring_hashtag: string; - source_name: 'un' | 'o'; - performer_has_fanclub: boolean; - opt_out: boolean; - fan_club_is_member: boolean; - asp_auth_url: string; - browser_id: string; - edge_region: string; - userlist_color: 'g' | 'b' | 'r'; - active_password: boolean; - fan_club_paid_with_tokens: boolean; - quality: { - quality: string; - rate: number; - stopped: boolean; - }; -} - - -export interface ChaturbateModel { - gender: string; - location: string; - current_show: 'public' | 'private'; - username: string; - room_subject: string; - tags: string[]; - is_new: boolean; - num_users: number; - num_followers: number; - country: string; - spoken_languages: string; - display_name: string; - birthday: string; - is_hd: boolean; - age: number; - seconds_online: number; - image_url: string; - image_url_360x270: string; - chat_room_url_revshare: string; - iframe_embed_revshare: string; - chat_room_url: string; - iframe_embed: string; - slug: string; -} - -export interface ChaturbateOnlineModelsResponse { - results: ChaturbateModel[], - count: number -} - -export interface Room { - name: string; - url: string; -} - - - // display_name is tricky. - // there is no capitaliziation in dossier.broadcaster_username. - // there is a `Real Name` k/v field in CB bios, but it's not foolproof as the broadcaster can enter anything, append notes, etc. - // so for now I'm just going to use the lowercase dossier.broadcaster_username - // actually, we can get username from - -export async function fetchHtml(url: string): Promise { - const fetchOptions = { - headers: { - 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/108.0.0.0 Safari/537.36', - } - } - const res = await scrapingFetch(url, fetchOptions); - const body = await res.text() - return body -} - -export function getBroadcasterDisplayName(html: string): string|null { - if (!html) throw new Error('getBroadcasterDisplayName requires html as argument'); - const $ = cheerio.load(html); - const ogTitle = $('meta[property="og:title"]')?.attr('content')?.split(' ')?.at(1); - return (!!ogTitle) ? ogTitle : null -} - - -/** - * getInitialRoomDossier - * - * Gets data about the model from their CB page. - * - * @param {String} roomUrl example: https://chaturbate.com/projektmelody - * @returns {Object} initialRoomDossier - * - * @todo please get a fixture of when a room is in password mode - */ -export function getInitialRoomDossier(html: string): Dossier|null { - if (!html) throw new Error('getInitialRoomDossier requires html as argument'); - - - try { - const $ = cheerio.load(html); - let rawScript = $('script:contains(window.initialRoomDossier)').html(); - if (!rawScript) { - throw new Error('window.initialRoomDossier is null. This could mean the channel is in password mode'); - } - let rawDossier = rawScript.slice(rawScript.indexOf('"'), rawScript.lastIndexOf('"') + 1); - let dossier: Dossier = JSON.parse(JSON.parse(rawDossier)); - return dossier; - } catch (error) { - if (error instanceof Error) { - // Handle the error gracefully - console.error(`Error fetching initial room dossier: ${error.message}`); - } else { - console.error('caught an exotic error, uh-oh') - console.error(error) - } - return null; - } -} - - - -export async function getRandomRoom(): Promise { - try { - const res = await scrapingFetch('https://chaturbate.com/api/public/affiliates/onlinerooms/?wm=DiPkB&client_ip=request_ip', { - headers: { - accept: 'application/json' - } - }); - const data = await res.json() as ChaturbateOnlineModelsResponse; - - if (!data || !Array.isArray(data.results) || data.results.length === 0) { - throw new Error('No results found'); - } - - const results = data.results; - const randomIndex = Math.floor(Math.random() * results.length); - - if (!results[randomIndex]) { - throw new Error('No result found at random index'); - } - - const username = results[randomIndex].username; - return { - url: `https://chaturbate.com/${username}`, - name: username - } - } catch (error) { - if (error instanceof Error) { - console.error(`Error in getRandomRoom: ${error.message}`); - } else { - console.error('An unexpected error occurred'); - } - throw error; // Re-throw the error to propagate it further - } -} \ No newline at end of file diff --git a/services/scout/src/config.ts b/services/scout/src/config.ts deleted file mode 100644 index 19daa34..0000000 --- a/services/scout/src/config.ts +++ /dev/null @@ -1,35 +0,0 @@ -import 'dotenv/config' - -const requiredEnvVars = [ - 'HTTP_PROXY', - 'POSTGREST_URL', - 'NODE_ENV', - 'WORKER_CONNECTION_STRING', - 'PORT', -] as const; - -const getEnvVar = (key: typeof requiredEnvVars[number]): string => { - const value = process.env[key]; - if (!value) { - throw new Error(`Missing ${key} env var`); - } - return value; -}; - -export interface Config { - postgrestUrl: string; - httpProxy: string; - nodeEnv: string; - workerConnectionString: string; - port: number; -} - - -export const configs: Config = { - httpProxy: getEnvVar('HTTP_PROXY'), - postgrestUrl: getEnvVar('POSTGREST_URL'), - nodeEnv: getEnvVar('NODE_ENV'), - workerConnectionString: getEnvVar('WORKER_CONNECTION_STRING'), - port: parseInt(getEnvVar('PORT')), -} - diff --git a/services/scout/src/css/SwaggerDark.css b/services/scout/src/css/SwaggerDark.css deleted file mode 100644 index c7348f0..0000000 --- a/services/scout/src/css/SwaggerDark.css +++ /dev/null @@ -1,851 +0,0 @@ -/* @see https://github.com/Amoenus/SwaggerDark/tree/master */ - -@media only screen and (prefers-color-scheme: dark) { - - a { color: #8c8cfa; } - - ::-webkit-scrollbar-track-piece { background-color: rgba(255, 255, 255, .2) !important; } - - ::-webkit-scrollbar-track { background-color: rgba(255, 255, 255, .3) !important; } - - ::-webkit-scrollbar-thumb { background-color: rgba(255, 255, 255, .5) !important; } - - embed[type="application/pdf"] { filter: invert(90%); } - - html { - background: #1f1f1f !important; - box-sizing: border-box; - filter: contrast(100%) brightness(100%) saturate(100%); - overflow-y: scroll; - } - - body { - background: #1f1f1f; - background-color: #1f1f1f; - background-image: none !important; - } - - button, input, select, textarea { - background-color: #1f1f1f; - color: #bfbfbf; - } - - font, html { color: #bfbfbf; } - - .swagger-ui, .swagger-ui section h3 { color: #b5bac9; } - - .swagger-ui a { background-color: transparent; } - - .swagger-ui mark { - background-color: #664b00; - color: #bfbfbf; - } - - .swagger-ui legend { color: inherit; } - - .swagger-ui .debug * { outline: #e6da99 solid 1px; } - - .swagger-ui .debug-white * { outline: #fff solid 1px; } - - .swagger-ui .debug-black * { outline: #bfbfbf solid 1px; } - - .swagger-ui .debug-grid { background: url() 0 0; } - - .swagger-ui .debug-grid-16 { background: url() 0 0; } - - .swagger-ui .debug-grid-8-solid { background: url() 0 0 #1c1c21; } - - .swagger-ui .debug-grid-16-solid { background: url() 0 0 #1c1c21; } - - .swagger-ui .b--black { border-color: #000; } - - .swagger-ui .b--near-black { border-color: #121212; } - - .swagger-ui .b--dark-gray { border-color: #333; } - - .swagger-ui .b--mid-gray { border-color: #545454; } - - .swagger-ui .b--gray { border-color: #787878; } - - .swagger-ui .b--silver { border-color: #999; } - - .swagger-ui .b--light-silver { border-color: #6e6e6e; } - - .swagger-ui .b--moon-gray { border-color: #4d4d4d; } - - .swagger-ui .b--light-gray { border-color: #2b2b2b; } - - .swagger-ui .b--near-white { border-color: #242424; } - - .swagger-ui .b--white { border-color: #1c1c21; } - - .swagger-ui .b--white-90 { border-color: rgba(28, 28, 33, .9); } - - .swagger-ui .b--white-80 { border-color: rgba(28, 28, 33, .8); } - - .swagger-ui .b--white-70 { border-color: rgba(28, 28, 33, .7); } - - .swagger-ui .b--white-60 { border-color: rgba(28, 28, 33, .6); } - - .swagger-ui .b--white-50 { border-color: rgba(28, 28, 33, .5); } - - .swagger-ui .b--white-40 { border-color: rgba(28, 28, 33, .4); } - - .swagger-ui .b--white-30 { border-color: rgba(28, 28, 33, .3); } - - .swagger-ui .b--white-20 { border-color: rgba(28, 28, 33, .2); } - - .swagger-ui .b--white-10 { border-color: rgba(28, 28, 33, .1); } - - .swagger-ui .b--white-05 { border-color: rgba(28, 28, 33, .05); } - - .swagger-ui .b--white-025 { border-color: rgba(28, 28, 33, .024); } - - .swagger-ui .b--white-0125 { border-color: rgba(28, 28, 33, .01); } - - .swagger-ui .b--black-90 { border-color: rgba(0, 0, 0, .9); } - - .swagger-ui .b--black-80 { border-color: rgba(0, 0, 0, .8); } - - .swagger-ui .b--black-70 { border-color: rgba(0, 0, 0, .7); } - - .swagger-ui .b--black-60 { border-color: rgba(0, 0, 0, .6); } - - .swagger-ui .b--black-50 { border-color: rgba(0, 0, 0, .5); } - - .swagger-ui .b--black-40 { border-color: rgba(0, 0, 0, .4); } - - .swagger-ui .b--black-30 { border-color: rgba(0, 0, 0, .3); } - - .swagger-ui .b--black-20 { border-color: rgba(0, 0, 0, .2); } - - .swagger-ui .b--black-10 { border-color: rgba(0, 0, 0, .1); } - - .swagger-ui .b--black-05 { border-color: rgba(0, 0, 0, .05); } - - .swagger-ui .b--black-025 { border-color: rgba(0, 0, 0, .024); } - - .swagger-ui .b--black-0125 { border-color: rgba(0, 0, 0, .01); } - - .swagger-ui .b--dark-red { border-color: #bc2f36; } - - .swagger-ui .b--red { border-color: #c83932; } - - .swagger-ui .b--light-red { border-color: #ab3c2b; } - - .swagger-ui .b--orange { border-color: #cc6e33; } - - .swagger-ui .b--purple { border-color: #5e2ca5; } - - .swagger-ui .b--light-purple { border-color: #672caf; } - - .swagger-ui .b--dark-pink { border-color: #ab2b81; } - - .swagger-ui .b--hot-pink { border-color: #c03086; } - - .swagger-ui .b--pink { border-color: #8f2464; } - - .swagger-ui .b--light-pink { border-color: #721d4d; } - - .swagger-ui .b--dark-green { border-color: #1c6e50; } - - .swagger-ui .b--green { border-color: #279b70; } - - .swagger-ui .b--light-green { border-color: #228762; } - - .swagger-ui .b--navy { border-color: #0d1d35; } - - .swagger-ui .b--dark-blue { border-color: #20497e; } - - .swagger-ui .b--blue { border-color: #4380d0; } - - .swagger-ui .b--light-blue { border-color: #20517e; } - - .swagger-ui .b--lightest-blue { border-color: #143a52; } - - .swagger-ui .b--washed-blue { border-color: #0c312d; } - - .swagger-ui .b--washed-green { border-color: #0f3d2c; } - - .swagger-ui .b--washed-red { border-color: #411010; } - - .swagger-ui .b--transparent { border-color: transparent; } - - .swagger-ui .b--gold, .swagger-ui .b--light-yellow, .swagger-ui .b--washed-yellow, .swagger-ui .b--yellow { border-color: #664b00; } - - .swagger-ui .shadow-1 { box-shadow: rgba(0, 0, 0, .2) 0 0 4px 2px; } - - .swagger-ui .shadow-2 { box-shadow: rgba(0, 0, 0, .2) 0 0 8px 2px; } - - .swagger-ui .shadow-3 { box-shadow: rgba(0, 0, 0, .2) 2px 2px 4px 2px; } - - .swagger-ui .shadow-4 { box-shadow: rgba(0, 0, 0, .2) 2px 2px 8px 0; } - - .swagger-ui .shadow-5 { box-shadow: rgba(0, 0, 0, .2) 4px 4px 8px 0; } - - @media screen and (min-width: 30em) { - .swagger-ui .shadow-1-ns { box-shadow: rgba(0, 0, 0, .2) 0 0 4px 2px; } - - .swagger-ui .shadow-2-ns { box-shadow: rgba(0, 0, 0, .2) 0 0 8px 2px; } - - .swagger-ui .shadow-3-ns { box-shadow: rgba(0, 0, 0, .2) 2px 2px 4px 2px; } - - .swagger-ui .shadow-4-ns { box-shadow: rgba(0, 0, 0, .2) 2px 2px 8px 0; } - - .swagger-ui .shadow-5-ns { box-shadow: rgba(0, 0, 0, .2) 4px 4px 8px 0; } - } - - @media screen and (max-width: 60em) and (min-width: 30em) { - .swagger-ui .shadow-1-m { box-shadow: rgba(0, 0, 0, .2) 0 0 4px 2px; } - - .swagger-ui .shadow-2-m { box-shadow: rgba(0, 0, 0, .2) 0 0 8px 2px; } - - .swagger-ui .shadow-3-m { box-shadow: rgba(0, 0, 0, .2) 2px 2px 4px 2px; } - - .swagger-ui .shadow-4-m { box-shadow: rgba(0, 0, 0, .2) 2px 2px 8px 0; } - - .swagger-ui .shadow-5-m { box-shadow: rgba(0, 0, 0, .2) 4px 4px 8px 0; } - } - - @media screen and (min-width: 60em) { - .swagger-ui .shadow-1-l { box-shadow: rgba(0, 0, 0, .2) 0 0 4px 2px; } - - .swagger-ui .shadow-2-l { box-shadow: rgba(0, 0, 0, .2) 0 0 8px 2px; } - - .swagger-ui .shadow-3-l { box-shadow: rgba(0, 0, 0, .2) 2px 2px 4px 2px; } - - .swagger-ui .shadow-4-l { box-shadow: rgba(0, 0, 0, .2) 2px 2px 8px 0; } - - .swagger-ui .shadow-5-l { box-shadow: rgba(0, 0, 0, .2) 4px 4px 8px 0; } - } - - .swagger-ui .black-05 { color: rgba(191, 191, 191, .05); } - - .swagger-ui .bg-black-05 { background-color: rgba(0, 0, 0, .05); } - - .swagger-ui .black-90, .swagger-ui .hover-black-90:focus, .swagger-ui .hover-black-90:hover { color: rgba(191, 191, 191, .9); } - - .swagger-ui .black-80, .swagger-ui .hover-black-80:focus, .swagger-ui .hover-black-80:hover { color: rgba(191, 191, 191, .8); } - - .swagger-ui .black-70, .swagger-ui .hover-black-70:focus, .swagger-ui .hover-black-70:hover { color: rgba(191, 191, 191, .7); } - - .swagger-ui .black-60, .swagger-ui .hover-black-60:focus, .swagger-ui .hover-black-60:hover { color: rgba(191, 191, 191, .6); } - - .swagger-ui .black-50, .swagger-ui .hover-black-50:focus, .swagger-ui .hover-black-50:hover { color: rgba(191, 191, 191, .5); } - - .swagger-ui .black-40, .swagger-ui .hover-black-40:focus, .swagger-ui .hover-black-40:hover { color: rgba(191, 191, 191, .4); } - - .swagger-ui .black-30, .swagger-ui .hover-black-30:focus, .swagger-ui .hover-black-30:hover { color: rgba(191, 191, 191, .3); } - - .swagger-ui .black-20, .swagger-ui .hover-black-20:focus, .swagger-ui .hover-black-20:hover { color: rgba(191, 191, 191, .2); } - - .swagger-ui .black-10, .swagger-ui .hover-black-10:focus, .swagger-ui .hover-black-10:hover { color: rgba(191, 191, 191, .1); } - - .swagger-ui .hover-white-90:focus, .swagger-ui .hover-white-90:hover, .swagger-ui .white-90 { color: rgba(255, 255, 255, .9); } - - .swagger-ui .hover-white-80:focus, .swagger-ui .hover-white-80:hover, .swagger-ui .white-80 { color: rgba(255, 255, 255, .8); } - - .swagger-ui .hover-white-70:focus, .swagger-ui .hover-white-70:hover, .swagger-ui .white-70 { color: rgba(255, 255, 255, .7); } - - .swagger-ui .hover-white-60:focus, .swagger-ui .hover-white-60:hover, .swagger-ui .white-60 { color: rgba(255, 255, 255, .6); } - - .swagger-ui .hover-white-50:focus, .swagger-ui .hover-white-50:hover, .swagger-ui .white-50 { color: rgba(255, 255, 255, .5); } - - .swagger-ui .hover-white-40:focus, .swagger-ui .hover-white-40:hover, .swagger-ui .white-40 { color: rgba(255, 255, 255, .4); } - - .swagger-ui .hover-white-30:focus, .swagger-ui .hover-white-30:hover, .swagger-ui .white-30 { color: rgba(255, 255, 255, .3); } - - .swagger-ui .hover-white-20:focus, .swagger-ui .hover-white-20:hover, .swagger-ui .white-20 { color: rgba(255, 255, 255, .2); } - - .swagger-ui .hover-white-10:focus, .swagger-ui .hover-white-10:hover, .swagger-ui .white-10 { color: rgba(255, 255, 255, .1); } - - .swagger-ui .hover-moon-gray:focus, .swagger-ui .hover-moon-gray:hover, .swagger-ui .moon-gray { color: #ccc; } - - .swagger-ui .hover-light-gray:focus, .swagger-ui .hover-light-gray:hover, .swagger-ui .light-gray { color: #ededed; } - - .swagger-ui .hover-near-white:focus, .swagger-ui .hover-near-white:hover, .swagger-ui .near-white { color: #f5f5f5; } - - .swagger-ui .dark-red, .swagger-ui .hover-dark-red:focus, .swagger-ui .hover-dark-red:hover { color: #e6999d; } - - .swagger-ui .hover-red:focus, .swagger-ui .hover-red:hover, .swagger-ui .red { color: #e69d99; } - - .swagger-ui .hover-light-red:focus, .swagger-ui .hover-light-red:hover, .swagger-ui .light-red { color: #e6a399; } - - .swagger-ui .hover-orange:focus, .swagger-ui .hover-orange:hover, .swagger-ui .orange { color: #e6b699; } - - .swagger-ui .gold, .swagger-ui .hover-gold:focus, .swagger-ui .hover-gold:hover { color: #e6d099; } - - .swagger-ui .hover-yellow:focus, .swagger-ui .hover-yellow:hover, .swagger-ui .yellow { color: #e6da99; } - - .swagger-ui .hover-light-yellow:focus, .swagger-ui .hover-light-yellow:hover, .swagger-ui .light-yellow { color: #ede6b6; } - - .swagger-ui .hover-purple:focus, .swagger-ui .hover-purple:hover, .swagger-ui .purple { color: #b99ae4; } - - .swagger-ui .hover-light-purple:focus, .swagger-ui .hover-light-purple:hover, .swagger-ui .light-purple { color: #bb99e6; } - - .swagger-ui .dark-pink, .swagger-ui .hover-dark-pink:focus, .swagger-ui .hover-dark-pink:hover { color: #e699cc; } - - .swagger-ui .hot-pink, .swagger-ui .hover-hot-pink:focus, .swagger-ui .hover-hot-pink:hover, .swagger-ui .hover-pink:focus, .swagger-ui .hover-pink:hover, .swagger-ui .pink { color: #e699c7; } - - .swagger-ui .hover-light-pink:focus, .swagger-ui .hover-light-pink:hover, .swagger-ui .light-pink { color: #edb6d5; } - - .swagger-ui .dark-green, .swagger-ui .green, .swagger-ui .hover-dark-green:focus, .swagger-ui .hover-dark-green:hover, .swagger-ui .hover-green:focus, .swagger-ui .hover-green:hover { color: #99e6c9; } - - .swagger-ui .hover-light-green:focus, .swagger-ui .hover-light-green:hover, .swagger-ui .light-green { color: #a1e8ce; } - - .swagger-ui .hover-navy:focus, .swagger-ui .hover-navy:hover, .swagger-ui .navy { color: #99b8e6; } - - .swagger-ui .blue, .swagger-ui .dark-blue, .swagger-ui .hover-blue:focus, .swagger-ui .hover-blue:hover, .swagger-ui .hover-dark-blue:focus, .swagger-ui .hover-dark-blue:hover { color: #99bae6; } - - .swagger-ui .hover-light-blue:focus, .swagger-ui .hover-light-blue:hover, .swagger-ui .light-blue { color: #a9cbea; } - - .swagger-ui .hover-lightest-blue:focus, .swagger-ui .hover-lightest-blue:hover, .swagger-ui .lightest-blue { color: #d6e9f5; } - - .swagger-ui .hover-washed-blue:focus, .swagger-ui .hover-washed-blue:hover, .swagger-ui .washed-blue { color: #f7fdfc; } - - .swagger-ui .hover-washed-green:focus, .swagger-ui .hover-washed-green:hover, .swagger-ui .washed-green { color: #ebfaf4; } - - .swagger-ui .hover-washed-yellow:focus, .swagger-ui .hover-washed-yellow:hover, .swagger-ui .washed-yellow { color: #fbf9ef; } - - .swagger-ui .hover-washed-red:focus, .swagger-ui .hover-washed-red:hover, .swagger-ui .washed-red { color: #f9e7e7; } - - .swagger-ui .color-inherit, .swagger-ui .hover-inherit:focus, .swagger-ui .hover-inherit:hover { color: inherit; } - - .swagger-ui .bg-black-90, .swagger-ui .hover-bg-black-90:focus, .swagger-ui .hover-bg-black-90:hover { background-color: rgba(0, 0, 0, .9); } - - .swagger-ui .bg-black-80, .swagger-ui .hover-bg-black-80:focus, .swagger-ui .hover-bg-black-80:hover { background-color: rgba(0, 0, 0, .8); } - - .swagger-ui .bg-black-70, .swagger-ui .hover-bg-black-70:focus, .swagger-ui .hover-bg-black-70:hover { background-color: rgba(0, 0, 0, .7); } - - .swagger-ui .bg-black-60, .swagger-ui .hover-bg-black-60:focus, .swagger-ui .hover-bg-black-60:hover { background-color: rgba(0, 0, 0, .6); } - - .swagger-ui .bg-black-50, .swagger-ui .hover-bg-black-50:focus, .swagger-ui .hover-bg-black-50:hover { background-color: rgba(0, 0, 0, .5); } - - .swagger-ui .bg-black-40, .swagger-ui .hover-bg-black-40:focus, .swagger-ui .hover-bg-black-40:hover { background-color: rgba(0, 0, 0, .4); } - - .swagger-ui .bg-black-30, .swagger-ui .hover-bg-black-30:focus, .swagger-ui .hover-bg-black-30:hover { background-color: rgba(0, 0, 0, .3); } - - .swagger-ui .bg-black-20, .swagger-ui .hover-bg-black-20:focus, .swagger-ui .hover-bg-black-20:hover { background-color: rgba(0, 0, 0, .2); } - - .swagger-ui .bg-white-90, .swagger-ui .hover-bg-white-90:focus, .swagger-ui .hover-bg-white-90:hover { background-color: rgba(28, 28, 33, .9); } - - .swagger-ui .bg-white-80, .swagger-ui .hover-bg-white-80:focus, .swagger-ui .hover-bg-white-80:hover { background-color: rgba(28, 28, 33, .8); } - - .swagger-ui .bg-white-70, .swagger-ui .hover-bg-white-70:focus, .swagger-ui .hover-bg-white-70:hover { background-color: rgba(28, 28, 33, .7); } - - .swagger-ui .bg-white-60, .swagger-ui .hover-bg-white-60:focus, .swagger-ui .hover-bg-white-60:hover { background-color: rgba(28, 28, 33, .6); } - - .swagger-ui .bg-white-50, .swagger-ui .hover-bg-white-50:focus, .swagger-ui .hover-bg-white-50:hover { background-color: rgba(28, 28, 33, .5); } - - .swagger-ui .bg-white-40, .swagger-ui .hover-bg-white-40:focus, .swagger-ui .hover-bg-white-40:hover { background-color: rgba(28, 28, 33, .4); } - - .swagger-ui .bg-white-30, .swagger-ui .hover-bg-white-30:focus, .swagger-ui .hover-bg-white-30:hover { background-color: rgba(28, 28, 33, .3); } - - .swagger-ui .bg-white-20, .swagger-ui .hover-bg-white-20:focus, .swagger-ui .hover-bg-white-20:hover { background-color: rgba(28, 28, 33, .2); } - - .swagger-ui .bg-black, .swagger-ui .hover-bg-black:focus, .swagger-ui .hover-bg-black:hover { background-color: #000; } - - .swagger-ui .bg-near-black, .swagger-ui .hover-bg-near-black:focus, .swagger-ui .hover-bg-near-black:hover { background-color: #121212; } - - .swagger-ui .bg-dark-gray, .swagger-ui .hover-bg-dark-gray:focus, .swagger-ui .hover-bg-dark-gray:hover { background-color: #333; } - - .swagger-ui .bg-mid-gray, .swagger-ui .hover-bg-mid-gray:focus, .swagger-ui .hover-bg-mid-gray:hover { background-color: #545454; } - - .swagger-ui .bg-gray, .swagger-ui .hover-bg-gray:focus, .swagger-ui .hover-bg-gray:hover { background-color: #787878; } - - .swagger-ui .bg-silver, .swagger-ui .hover-bg-silver:focus, .swagger-ui .hover-bg-silver:hover { background-color: #999; } - - .swagger-ui .bg-white, .swagger-ui .hover-bg-white:focus, .swagger-ui .hover-bg-white:hover { background-color: #1c1c21; } - - .swagger-ui .bg-transparent, .swagger-ui .hover-bg-transparent:focus, .swagger-ui .hover-bg-transparent:hover { background-color: transparent; } - - .swagger-ui .bg-dark-red, .swagger-ui .hover-bg-dark-red:focus, .swagger-ui .hover-bg-dark-red:hover { background-color: #bc2f36; } - - .swagger-ui .bg-red, .swagger-ui .hover-bg-red:focus, .swagger-ui .hover-bg-red:hover { background-color: #c83932; } - - .swagger-ui .bg-light-red, .swagger-ui .hover-bg-light-red:focus, .swagger-ui .hover-bg-light-red:hover { background-color: #ab3c2b; } - - .swagger-ui .bg-orange, .swagger-ui .hover-bg-orange:focus, .swagger-ui .hover-bg-orange:hover { background-color: #cc6e33; } - - .swagger-ui .bg-gold, .swagger-ui .bg-light-yellow, .swagger-ui .bg-washed-yellow, .swagger-ui .bg-yellow, .swagger-ui .hover-bg-gold:focus, .swagger-ui .hover-bg-gold:hover, .swagger-ui .hover-bg-light-yellow:focus, .swagger-ui .hover-bg-light-yellow:hover, .swagger-ui .hover-bg-washed-yellow:focus, .swagger-ui .hover-bg-washed-yellow:hover, .swagger-ui .hover-bg-yellow:focus, .swagger-ui .hover-bg-yellow:hover { background-color: #664b00; } - - .swagger-ui .bg-purple, .swagger-ui .hover-bg-purple:focus, .swagger-ui .hover-bg-purple:hover { background-color: #5e2ca5; } - - .swagger-ui .bg-light-purple, .swagger-ui .hover-bg-light-purple:focus, .swagger-ui .hover-bg-light-purple:hover { background-color: #672caf; } - - .swagger-ui .bg-dark-pink, .swagger-ui .hover-bg-dark-pink:focus, .swagger-ui .hover-bg-dark-pink:hover { background-color: #ab2b81; } - - .swagger-ui .bg-hot-pink, .swagger-ui .hover-bg-hot-pink:focus, .swagger-ui .hover-bg-hot-pink:hover { background-color: #c03086; } - - .swagger-ui .bg-pink, .swagger-ui .hover-bg-pink:focus, .swagger-ui .hover-bg-pink:hover { background-color: #8f2464; } - - .swagger-ui .bg-light-pink, .swagger-ui .hover-bg-light-pink:focus, .swagger-ui .hover-bg-light-pink:hover { background-color: #721d4d; } - - .swagger-ui .bg-dark-green, .swagger-ui .hover-bg-dark-green:focus, .swagger-ui .hover-bg-dark-green:hover { background-color: #1c6e50; } - - .swagger-ui .bg-green, .swagger-ui .hover-bg-green:focus, .swagger-ui .hover-bg-green:hover { background-color: #279b70; } - - .swagger-ui .bg-light-green, .swagger-ui .hover-bg-light-green:focus, .swagger-ui .hover-bg-light-green:hover { background-color: #228762; } - - .swagger-ui .bg-navy, .swagger-ui .hover-bg-navy:focus, .swagger-ui .hover-bg-navy:hover { background-color: #0d1d35; } - - .swagger-ui .bg-dark-blue, .swagger-ui .hover-bg-dark-blue:focus, .swagger-ui .hover-bg-dark-blue:hover { background-color: #20497e; } - - .swagger-ui .bg-blue, .swagger-ui .hover-bg-blue:focus, .swagger-ui .hover-bg-blue:hover { background-color: #4380d0; } - - .swagger-ui .bg-light-blue, .swagger-ui .hover-bg-light-blue:focus, .swagger-ui .hover-bg-light-blue:hover { background-color: #20517e; } - - .swagger-ui .bg-lightest-blue, .swagger-ui .hover-bg-lightest-blue:focus, .swagger-ui .hover-bg-lightest-blue:hover { background-color: #143a52; } - - .swagger-ui .bg-washed-blue, .swagger-ui .hover-bg-washed-blue:focus, .swagger-ui .hover-bg-washed-blue:hover { background-color: #0c312d; } - - .swagger-ui .bg-washed-green, .swagger-ui .hover-bg-washed-green:focus, .swagger-ui .hover-bg-washed-green:hover { background-color: #0f3d2c; } - - .swagger-ui .bg-washed-red, .swagger-ui .hover-bg-washed-red:focus, .swagger-ui .hover-bg-washed-red:hover { background-color: #411010; } - - .swagger-ui .bg-inherit, .swagger-ui .hover-bg-inherit:focus, .swagger-ui .hover-bg-inherit:hover { background-color: inherit; } - - .swagger-ui .shadow-hover { transition: all .5s cubic-bezier(.165, .84, .44, 1) 0s; } - - .swagger-ui .shadow-hover::after { - border-radius: inherit; - box-shadow: rgba(0, 0, 0, .2) 0 0 16px 2px; - content: ""; - height: 100%; - left: 0; - opacity: 0; - position: absolute; - top: 0; - transition: opacity .5s cubic-bezier(.165, .84, .44, 1) 0s; - width: 100%; - z-index: -1; - } - - .swagger-ui .bg-animate, .swagger-ui .bg-animate:focus, .swagger-ui .bg-animate:hover { transition: background-color .15s ease-in-out 0s; } - - .swagger-ui .nested-links a { - color: #99bae6; - transition: color .15s ease-in 0s; - } - - .swagger-ui .nested-links a:focus, .swagger-ui .nested-links a:hover { - color: #a9cbea; - transition: color .15s ease-in 0s; - } - - .swagger-ui .opblock-tag { - border-bottom: 1px solid rgba(58, 64, 80, .3); - color: #b5bac9; - transition: all .2s ease 0s; - } - - .swagger-ui .opblock-tag svg, .swagger-ui section.models h4 svg { transition: all .4s ease 0s; } - - .swagger-ui .opblock { - border: 1px solid #000; - border-radius: 4px; - box-shadow: rgba(0, 0, 0, .19) 0 0 3px; - margin: 0 0 15px; - } - - .swagger-ui .opblock .tab-header .tab-item.active h4 span::after { background: gray; } - - .swagger-ui .opblock.is-open .opblock-summary { border-bottom: 1px solid #000; } - - .swagger-ui .opblock .opblock-section-header { - background: rgba(28, 28, 33, .8); - box-shadow: rgba(0, 0, 0, .1) 0 1px 2px; - } - - .swagger-ui .opblock .opblock-section-header > label > span { padding: 0 10px 0 0; } - - .swagger-ui .opblock .opblock-summary-method { - background: #000; - color: #fff; - text-shadow: rgba(0, 0, 0, .1) 0 1px 0; - } - - .swagger-ui .opblock.opblock-post { - background: rgba(72, 203, 144, .1); - border-color: #48cb90; - } - - .swagger-ui .opblock.opblock-post .opblock-summary-method, .swagger-ui .opblock.opblock-post .tab-header .tab-item.active h4 span::after { background: #48cb90; } - - .swagger-ui .opblock.opblock-post .opblock-summary { border-color: #48cb90; } - - .swagger-ui .opblock.opblock-put { - background: rgba(213, 157, 88, .1); - border-color: #d59d58; - } - - .swagger-ui .opblock.opblock-put .opblock-summary-method, .swagger-ui .opblock.opblock-put .tab-header .tab-item.active h4 span::after { background: #d59d58; } - - .swagger-ui .opblock.opblock-put .opblock-summary { border-color: #d59d58; } - - .swagger-ui .opblock.opblock-delete { - background: rgba(200, 50, 50, .1); - border-color: #c83232; - } - - .swagger-ui .opblock.opblock-delete .opblock-summary-method, .swagger-ui .opblock.opblock-delete .tab-header .tab-item.active h4 span::after { background: #c83232; } - - .swagger-ui .opblock.opblock-delete .opblock-summary { border-color: #c83232; } - - .swagger-ui .opblock.opblock-get { - background: rgba(42, 105, 167, .1); - border-color: #2a69a7; - } - - .swagger-ui .opblock.opblock-get .opblock-summary-method, .swagger-ui .opblock.opblock-get .tab-header .tab-item.active h4 span::after { background: #2a69a7; } - - .swagger-ui .opblock.opblock-get .opblock-summary { border-color: #2a69a7; } - - .swagger-ui .opblock.opblock-patch { - background: rgba(92, 214, 188, .1); - border-color: #5cd6bc; - } - - .swagger-ui .opblock.opblock-patch .opblock-summary-method, .swagger-ui .opblock.opblock-patch .tab-header .tab-item.active h4 span::after { background: #5cd6bc; } - - .swagger-ui .opblock.opblock-patch .opblock-summary { border-color: #5cd6bc; } - - .swagger-ui .opblock.opblock-head { - background: rgba(140, 63, 207, .1); - border-color: #8c3fcf; - } - - .swagger-ui .opblock.opblock-head .opblock-summary-method, .swagger-ui .opblock.opblock-head .tab-header .tab-item.active h4 span::after { background: #8c3fcf; } - - .swagger-ui .opblock.opblock-head .opblock-summary { border-color: #8c3fcf; } - - .swagger-ui .opblock.opblock-options { - background: rgba(36, 89, 143, .1); - border-color: #24598f; - } - - .swagger-ui .opblock.opblock-options .opblock-summary-method, .swagger-ui .opblock.opblock-options .tab-header .tab-item.active h4 span::after { background: #24598f; } - - .swagger-ui .opblock.opblock-options .opblock-summary { border-color: #24598f; } - - .swagger-ui .opblock.opblock-deprecated { - background: rgba(46, 46, 46, .1); - border-color: #2e2e2e; - opacity: .6; - } - - .swagger-ui .opblock.opblock-deprecated .opblock-summary-method, .swagger-ui .opblock.opblock-deprecated .tab-header .tab-item.active h4 span::after { background: #2e2e2e; } - - .swagger-ui .opblock.opblock-deprecated .opblock-summary { border-color: #2e2e2e; } - - .swagger-ui .filter .operation-filter-input { border: 2px solid #2b3446; } - - .swagger-ui .tab li:first-of-type::after { background: rgba(0, 0, 0, .2); } - - .swagger-ui .download-contents { - background: #7c8192; - color: #fff; - } - - .swagger-ui .scheme-container { - background: #1c1c21; - box-shadow: rgba(0, 0, 0, .15) 0 1px 2px 0; - } - - .swagger-ui .loading-container .loading::before { - animation: 1s linear 0s infinite normal none running rotation, .5s ease 0s 1 normal none running opacity; - border-color: rgba(0, 0, 0, .6) rgba(84, 84, 84, .1) rgba(84, 84, 84, .1); - } - - .swagger-ui .response-control-media-type--accept-controller select { border-color: #196619; } - - .swagger-ui .response-control-media-type__accept-message { color: #99e699; } - - .swagger-ui .version-pragma__message code { background-color: #3b3b3b; } - - .swagger-ui .btn { - background: 0 0; - border: 2px solid gray; - box-shadow: rgba(0, 0, 0, .1) 0 1px 2px; - color: #b5bac9; - } - - .swagger-ui .btn:hover { box-shadow: rgba(0, 0, 0, .3) 0 0 5px; } - - .swagger-ui .btn.authorize, .swagger-ui .btn.cancel { - background-color: transparent; - border-color: #a72a2a; - color: #e69999; - } - - .swagger-ui .btn.cancel:hover { - background-color: #a72a2a; - color: #fff; - } - - .swagger-ui .btn.authorize { - border-color: #48cb90; - color: #9ce3c3; - } - - .swagger-ui .btn.authorize svg { fill: #9ce3c3; } - - .btn.authorize.unlocked:hover { - background-color: #48cb90; - color: #fff; - } - - .btn.authorize.unlocked:hover svg { - fill: #fbfbfb; - } - - .swagger-ui .btn.execute { - background-color: #5892d5; - border-color: #5892d5; - color: #fff; - } - - .swagger-ui .copy-to-clipboard { background: #7c8192; } - - .swagger-ui .copy-to-clipboard button { background: url("data:image/svg+xml;charset=utf-8,") 50% center no-repeat; } - - .swagger-ui select { - background: url("data:image/svg+xml;charset=utf-8,") right 10px center/20px no-repeat #212121; - background: url() right 10px center/20px no-repeat #1c1c21; - border: 2px solid #41444e; - } - - .swagger-ui select[multiple] { background: #212121; } - - .swagger-ui button.invalid, .swagger-ui input[type=email].invalid, .swagger-ui input[type=file].invalid, .swagger-ui input[type=password].invalid, .swagger-ui input[type=search].invalid, .swagger-ui input[type=text].invalid, .swagger-ui select.invalid, .swagger-ui textarea.invalid { - background: #390e0e; - border-color: #c83232; - } - - .swagger-ui input[type=email], .swagger-ui input[type=file], .swagger-ui input[type=password], .swagger-ui input[type=search], .swagger-ui input[type=text], .swagger-ui textarea { - background: #1c1c21; - border: 1px solid #404040; - } - - .swagger-ui textarea { - background: rgba(28, 28, 33, .8); - color: #b5bac9; - } - - .swagger-ui input[disabled], .swagger-ui select[disabled] { - background-color: #1f1f1f; - color: #bfbfbf; - } - - .swagger-ui textarea[disabled] { - background-color: #41444e; - color: #fff; - } - - .swagger-ui select[disabled] { border-color: #878787; } - - .swagger-ui textarea:focus { border: 2px solid #2a69a7; } - - .swagger-ui .checkbox input[type=checkbox] + label > .item { - background: #303030; - box-shadow: #303030 0 0 0 2px; - } - - .swagger-ui .checkbox input[type=checkbox]:checked + label > .item { background: url("data:image/svg+xml;charset=utf-8,") 50% center no-repeat #303030; } - - .swagger-ui .dialog-ux .backdrop-ux { background: rgba(0, 0, 0, .8); } - - .swagger-ui .dialog-ux .modal-ux { - background: #1c1c21; - border: 1px solid #2e2e2e; - box-shadow: rgba(0, 0, 0, .2) 0 10px 30px 0; - } - - .swagger-ui .dialog-ux .modal-ux-header .close-modal { background: 0 0; } - - .swagger-ui .model .deprecated span, .swagger-ui .model .deprecated td { color: #bfbfbf !important; } - - .swagger-ui .model-toggle::after { background: url("data:image/svg+xml;charset=utf-8,") 50% center/100% no-repeat; } - - .swagger-ui .model-hint { - background: rgba(0, 0, 0, .7); - color: #ebebeb; - } - - .swagger-ui section.models { border: 1px solid rgba(58, 64, 80, .3); } - - .swagger-ui section.models.is-open h4 { border-bottom: 1px solid rgba(58, 64, 80, .3); } - - .swagger-ui section.models .model-container { background: rgba(0, 0, 0, .05); } - - .swagger-ui section.models .model-container:hover { background: rgba(0, 0, 0, .07); } - - .swagger-ui .model-box { background: rgba(0, 0, 0, .1); } - - .swagger-ui .prop-type { color: #aaaad4; } - - .swagger-ui table thead tr td, .swagger-ui table thead tr th { - border-bottom: 1px solid rgba(58, 64, 80, .2); - color: #b5bac9; - } - - .swagger-ui .parameter__name.required::after { color: rgba(230, 153, 153, .6); } - - .swagger-ui .topbar .download-url-wrapper .select-label { color: #f0f0f0; } - - .swagger-ui .topbar .download-url-wrapper .download-url-button { - background: #63a040; - color: #fff; - } - - .swagger-ui .info .title small { background: #7c8492; } - - .swagger-ui .info .title small.version-stamp { background-color: #7a9b27; } - - .swagger-ui .auth-container .errors { - background-color: #350d0d; - color: #b5bac9; - } - - .swagger-ui .errors-wrapper { - background: rgba(200, 50, 50, .1); - border: 2px solid #c83232; - } - - .swagger-ui .markdown code, .swagger-ui .renderedmarkdown code { - background: rgba(0, 0, 0, .05); - color: #c299e6; - } - - .swagger-ui .model-toggle:after { background: url() 50% no-repeat; } - - /* arrows for each operation and request are now white */ - .arrow, #large-arrow-up { fill: #fff; } - - #unlocked { fill: #fff; } - - ::-webkit-scrollbar-track { background-color: #646464 !important; } - - ::-webkit-scrollbar-thumb { - background-color: #242424 !important; - border: 2px solid #3e4346 !important; - } - - ::-webkit-scrollbar-button:vertical:start:decrement { - background: linear-gradient(130deg, #696969 40%, rgba(255, 0, 0, 0) 41%), linear-gradient(230deg, #696969 40%, transparent 41%), linear-gradient(0deg, #696969 40%, transparent 31%); - background-color: #b6b6b6; - } - - ::-webkit-scrollbar-button:vertical:end:increment { - background: linear-gradient(310deg, #696969 40%, transparent 41%), linear-gradient(50deg, #696969 40%, transparent 41%), linear-gradient(180deg, #696969 40%, transparent 31%); - background-color: #b6b6b6; - } - - ::-webkit-scrollbar-button:horizontal:end:increment { - background: linear-gradient(210deg, #696969 40%, transparent 41%), linear-gradient(330deg, #696969 40%, transparent 41%), linear-gradient(90deg, #696969 30%, transparent 31%); - background-color: #b6b6b6; - } - - ::-webkit-scrollbar-button:horizontal:start:decrement { - background: linear-gradient(30deg, #696969 40%, transparent 41%), linear-gradient(150deg, #696969 40%, transparent 41%), linear-gradient(270deg, #696969 30%, transparent 31%); - background-color: #b6b6b6; - } - - ::-webkit-scrollbar-button, ::-webkit-scrollbar-track-piece { background-color: #3e4346 !important; } - - .swagger-ui .black, .swagger-ui .checkbox, .swagger-ui .dark-gray, .swagger-ui .download-url-wrapper .loading, .swagger-ui .errors-wrapper .errors small, .swagger-ui .fallback, .swagger-ui .filter .loading, .swagger-ui .gray, .swagger-ui .hover-black:focus, .swagger-ui .hover-black:hover, .swagger-ui .hover-dark-gray:focus, .swagger-ui .hover-dark-gray:hover, .swagger-ui .hover-gray:focus, .swagger-ui .hover-gray:hover, .swagger-ui .hover-light-silver:focus, .swagger-ui .hover-light-silver:hover, .swagger-ui .hover-mid-gray:focus, .swagger-ui .hover-mid-gray:hover, .swagger-ui .hover-near-black:focus, .swagger-ui .hover-near-black:hover, .swagger-ui .hover-silver:focus, .swagger-ui .hover-silver:hover, .swagger-ui .light-silver, .swagger-ui .markdown pre, .swagger-ui .mid-gray, .swagger-ui .model .property, .swagger-ui .model .property.primitive, .swagger-ui .model-title, .swagger-ui .near-black, .swagger-ui .parameter__extension, .swagger-ui .parameter__in, .swagger-ui .prop-format, .swagger-ui .renderedmarkdown pre, .swagger-ui .response-col_links .response-undocumented, .swagger-ui .response-col_status .response-undocumented, .swagger-ui .silver, .swagger-ui section.models h4, .swagger-ui section.models h5, .swagger-ui span.token-not-formatted, .swagger-ui span.token-string, .swagger-ui table.headers .header-example, .swagger-ui table.model tr.description, .swagger-ui table.model tr.extension { color: #bfbfbf; } - - .swagger-ui .hover-white:focus, .swagger-ui .hover-white:hover, .swagger-ui .info .title small pre, .swagger-ui .topbar a, .swagger-ui .white { color: #fff; } - - .swagger-ui .bg-black-10, .swagger-ui .hover-bg-black-10:focus, .swagger-ui .hover-bg-black-10:hover, .swagger-ui .stripe-dark:nth-child(2n + 1) { background-color: rgba(0, 0, 0, .1); } - - .swagger-ui .bg-white-10, .swagger-ui .hover-bg-white-10:focus, .swagger-ui .hover-bg-white-10:hover, .swagger-ui .stripe-light:nth-child(2n + 1) { background-color: rgba(28, 28, 33, .1); } - - .swagger-ui .bg-light-silver, .swagger-ui .hover-bg-light-silver:focus, .swagger-ui .hover-bg-light-silver:hover, .swagger-ui .striped--light-silver:nth-child(2n + 1) { background-color: #6e6e6e; } - - .swagger-ui .bg-moon-gray, .swagger-ui .hover-bg-moon-gray:focus, .swagger-ui .hover-bg-moon-gray:hover, .swagger-ui .striped--moon-gray:nth-child(2n + 1) { background-color: #4d4d4d; } - - .swagger-ui .bg-light-gray, .swagger-ui .hover-bg-light-gray:focus, .swagger-ui .hover-bg-light-gray:hover, .swagger-ui .striped--light-gray:nth-child(2n + 1) { background-color: #2b2b2b; } - - .swagger-ui .bg-near-white, .swagger-ui .hover-bg-near-white:focus, .swagger-ui .hover-bg-near-white:hover, .swagger-ui .striped--near-white:nth-child(2n + 1) { background-color: #242424; } - - .swagger-ui .opblock-tag:hover, .swagger-ui section.models h4:hover { background: rgba(0, 0, 0, .02); } - - .swagger-ui .checkbox p, .swagger-ui .dialog-ux .modal-ux-content h4, .swagger-ui .dialog-ux .modal-ux-content p, .swagger-ui .dialog-ux .modal-ux-header h3, .swagger-ui .errors-wrapper .errors h4, .swagger-ui .errors-wrapper hgroup h4, .swagger-ui .info .base-url, .swagger-ui .info .title, .swagger-ui .info h1, .swagger-ui .info h2, .swagger-ui .info h3, .swagger-ui .info h4, .swagger-ui .info h5, .swagger-ui .info li, .swagger-ui .info p, .swagger-ui .info table, .swagger-ui .loading-container .loading::after, .swagger-ui .model, .swagger-ui .opblock .opblock-section-header h4, .swagger-ui .opblock .opblock-section-header > label, .swagger-ui .opblock .opblock-summary-description, .swagger-ui .opblock .opblock-summary-operation-id, .swagger-ui .opblock .opblock-summary-path, .swagger-ui .opblock .opblock-summary-path__deprecated, .swagger-ui .opblock-description-wrapper, .swagger-ui .opblock-description-wrapper h4, .swagger-ui .opblock-description-wrapper p, .swagger-ui .opblock-external-docs-wrapper, .swagger-ui .opblock-external-docs-wrapper h4, .swagger-ui .opblock-external-docs-wrapper p, .swagger-ui .opblock-tag small, .swagger-ui .opblock-title_normal, .swagger-ui .opblock-title_normal h4, .swagger-ui .opblock-title_normal p, .swagger-ui .parameter__name, .swagger-ui .parameter__type, .swagger-ui .response-col_links, .swagger-ui .response-col_status, .swagger-ui .responses-inner h4, .swagger-ui .responses-inner h5, .swagger-ui .scheme-container .schemes > label, .swagger-ui .scopes h2, .swagger-ui .servers > label, .swagger-ui .tab li, .swagger-ui label, .swagger-ui select, .swagger-ui table.headers td { color: #b5bac9; } - - .swagger-ui .download-url-wrapper .failed, .swagger-ui .filter .failed, .swagger-ui .model-deprecated-warning, .swagger-ui .parameter__deprecated, .swagger-ui .parameter__name.required span, .swagger-ui table.model tr.property-row .star { color: #e69999; } - - .swagger-ui .opblock-body pre.microlight, .swagger-ui textarea.curl { - background: #41444e; - border-radius: 4px; - color: #fff; - } - - .swagger-ui .expand-methods svg, .swagger-ui .expand-methods:hover svg { fill: #bfbfbf; } - - .swagger-ui .auth-container, .swagger-ui .dialog-ux .modal-ux-header { border-bottom: 1px solid #2e2e2e; } - - .swagger-ui .topbar .download-url-wrapper .select-label select, .swagger-ui .topbar .download-url-wrapper input[type=text] { border: 2px solid #63a040; } - - .swagger-ui .info a, .swagger-ui .info a:hover, .swagger-ui .scopes h2 a { color: #99bde6; } - - /* Dark Scrollbar */ - ::-webkit-scrollbar { - width: 14px; - height: 14px; - } - - ::-webkit-scrollbar-button { - background-color: #3e4346 !important; - } - - ::-webkit-scrollbar-track { - background-color: #646464 !important; - } - - ::-webkit-scrollbar-track-piece { - background-color: #3e4346 !important; - } - - ::-webkit-scrollbar-thumb { - height: 50px; - background-color: #242424 !important; - border: 2px solid #3e4346 !important; - } - - ::-webkit-scrollbar-corner {} - - ::-webkit-resizer {} - - ::-webkit-scrollbar-button:vertical:start:decrement { - background: - linear-gradient(130deg, #696969 40%, rgba(255, 0, 0, 0) 41%), - linear-gradient(230deg, #696969 40%, rgba(0, 0, 0, 0) 41%), - linear-gradient(0deg, #696969 40%, rgba(0, 0, 0, 0) 31%); - background-color: #b6b6b6; - } - - ::-webkit-scrollbar-button:vertical:end:increment { - background: - linear-gradient(310deg, #696969 40%, rgba(0, 0, 0, 0) 41%), - linear-gradient(50deg, #696969 40%, rgba(0, 0, 0, 0) 41%), - linear-gradient(180deg, #696969 40%, rgba(0, 0, 0, 0) 31%); - background-color: #b6b6b6; - } - - ::-webkit-scrollbar-button:horizontal:end:increment { - background: - linear-gradient(210deg, #696969 40%, rgba(0, 0, 0, 0) 41%), - linear-gradient(330deg, #696969 40%, rgba(0, 0, 0, 0) 41%), - linear-gradient(90deg, #696969 30%, rgba(0, 0, 0, 0) 31%); - background-color: #b6b6b6; - } - - ::-webkit-scrollbar-button:horizontal:start:decrement { - background: - linear-gradient(30deg, #696969 40%, rgba(0, 0, 0, 0) 41%), - linear-gradient(150deg, #696969 40%, rgba(0, 0, 0, 0) 41%), - linear-gradient(270deg, #696969 30%, rgba(0, 0, 0, 0) 31%); - background-color: #b6b6b6; - } -} \ No newline at end of file diff --git a/services/scout/src/fansly.spec.ts b/services/scout/src/fansly.spec.ts deleted file mode 100644 index ce555a5..0000000 --- a/services/scout/src/fansly.spec.ts +++ /dev/null @@ -1,47 +0,0 @@ -import { expect } from 'chai' -import { usernameRegex, normalize, urlFromUsername, getAccountData } from './fansly.ts' -import { describe } from 'mocha' - -describe('fansly', function () { - describe('integration', function () { - describe('getAccountData', function () { - it('should get username, id, and location', async function () { - const data = await getAccountData('projektmelody') - // console.log(data) - expect(data).to.have.property('username', 'ProjektMelody') - expect(data).to.have.property('id', '284824898138812416') - expect(data).to.have.property('location', 'Casting Couch, JP') - }) - }) - }) - describe('unit', function () { - describe('regex', function () { - describe('username', function () { - it('should get the username of the channel', function () { - expect(usernameRegex.exec('https://fansly.com/18Plus/posts')?.at(1)).to.equal('18Plus') - expect(usernameRegex.exec('https://fansly.com/projektmelody/posts')?.at(1)).to.equal('projektmelody') - expect(usernameRegex.exec('https://fansly.com/GoodKittenVR')?.at(1)).to.equal('GoodKittenVR') - expect(usernameRegex.exec('https://fansly.com/live/MzLewdieB')?.at(1)).to.equal('MzLewdieB') - expect(usernameRegex.exec('https://fansly.com/live/340602399334871040')?.at(1)).to.equal('340602399334871040') - }) - }) - }) - describe('url', function () { - describe('fromUsername', function () { - it('should accept a channel name and give us a valid channel URL', function () { - expect(urlFromUsername('projektmelody')).to.equal('https://fansly.com/projektmelody') - expect(urlFromUsername('GoodKittenVR')).to.equal('https://fansly.com/GoodKittenVR') - expect(urlFromUsername('MzLewdieB')).to.equal('https://fansly.com/MzLewdieB') - expect(urlFromUsername('340602399334871040')).to.equal('https://fansly.com/340602399334871040') - }) - }) - describe('normalize', function () { - it('should accept a live URL and return a normal channel url.', function () { - expect(normalize('https://fansly.com/live/projektmelody')).to.equal('https://fansly.com/projektmelody') - expect(normalize('https://fansly.com/live/340602399334871040')).to.equal('https://fansly.com/340602399334871040') - expect(normalize('https://fansly.com/live/GoodKittenVR')).to.equal('https://fansly.com/GoodKittenVR') - }) - }) - }) - }) -}) \ No newline at end of file diff --git a/services/scout/src/fansly.ts b/services/scout/src/fansly.ts deleted file mode 100644 index fec8e83..0000000 --- a/services/scout/src/fansly.ts +++ /dev/null @@ -1,171 +0,0 @@ -import { download, getTmpFile } from '@futureporn/utils/file.ts'; -import type { VtuberRecord } from '@futureporn/types'; -import { ua0 } from './ua.ts'; -import scrapingFetch from './scrapingFetch.ts'; - -export interface FanslyProfile { - id: string; - username: string; - displayName?: string | null; - flags: number; - version: number; - followCount: number; - subscriberCount: number; - permissions: { - accountPermissionFlags: { - flags: number; - }; - }; - timelineStats: { - accountId: string; - imageCount: number; - videoCount: number; - bundleCount: number; - bundleImageCount: number; - bundleVideoCount: number; - fetchedAt: number; - }; - profileAccessFlags: number; - profileFlags: number; - about: string; - location: string; - profileSocials: any[]; - pinnedPosts: { - postId: string; - accountId: string; - pos: number; - createdAt: number; - }[]; - statusId: number; - lastSeenAt: number; - postLikes: number; - accountMediaLikes: number; - avatar: { - id: string; - type: number; - status: number; - accountId: string; - mimetype: string; - flags: number; - location: string; - width: number; - height: number; - metadata: string; - updatedAt: number; - createdAt: number; - variants: any[]; - variantHash: {}; - locations: any[]; - }; - banner: { - id: string; - type: number; - status: number; - accountId: string; - mimetype: string; - flags: number; - location: string; - width: number; - height: number; - metadata: string; - updatedAt: number; - createdAt: number; - variants: any[]; - variantHash: {}; - locations: any[]; - }; - mediaStoryState: { - accountId: string; - status: number; - storyCount: number; - version: number; - createdAt: number; - updatedAt: number; - hasActiveStories: boolean; - }; - subscriptionTiers: { - id: string; - accountId: string; - name: string; - color: string; - pos: number; - price: number; - maxSubscribers: number; - subscriptionBenefits?: any[]; - includedTierIds: string[]; - plans: any[]; - }[]; - streaming: { - accountId: string; - channel: { - id: string; - accountId: string; - playbackUrl: string; - chatRoomId: string; - status: number; - version: number; - createdAt: number; - updatedAt?: null | number; - stream: any; - arn: null; - ingestEndpoint: null; - }; - enabled: boolean; - }; - walls: { - id: string; - accountId: string; - pos: null; - name: string; - description: ''; - metadata: ''; - }[]; - profileAccess: boolean; -} - -export const usernameRegex = new RegExp(/^https:\/\/fansly\.com\/(?:live\/)?([^\/]+)/) -export const urlFromUsername = (username: string) => `https://fansly.com/${username}` -export const normalize = (url: string) => { - if (!url) throw new Error('normalized received a null or undefined url.'); - const username = usernameRegex.exec(url)?.at(1) - if (!username) throw new Error('failed to get username from url'); - return urlFromUsername(username) -} - - -export const imageUrlFromUsername = async function image(fanslyUserId: string) { - if (!fanslyUserId) throw new Error(`first arg passed to fansly.data.image must be a {string} fanslyUserId`); - const url = `https://api.fansly.com/api/v1/account/${fanslyUserId}/avatar` - const filePath = getTmpFile('avatar.jpg') - return download({ filePath, url }) -} - -export async function getProfileImage(username: string): Promise { - const data = await getAccountData(username) - return data.avatar.location -} - - -export async function getAccountData(username: string): Promise { - const fetchUrl = `https://apiv3.fansly.com/api/v1/account?usernames=${username}&ngsw-bypass=true` - const fetchOptions = { - method: 'GET', - headers: { - 'User-Agent': ua0, - 'Accept': 'application/json' - } - } - const res = await scrapingFetch(fetchUrl, fetchOptions) - if (!res.ok) { - const body = await res.text() - const msg = `failed to fetch getAccountData res.status=${res.status}, res.statusText=${res.statusText}, body=${body}` - console.error(msg) - throw new Error(msg) - } - const data = await res.json() as any - - if (!data?.success) throw new Error('data.success was expected to exist but it is absent.'); - if (data.success !== true) throw new Error('data.success was not true'); - if (!data?.response) throw new Error('data.response was expected to exist but it is absent.'); - return data.response[0] -} diff --git a/services/scout/src/fastify.ts b/services/scout/src/fastify.ts deleted file mode 100644 index 2ae1e58..0000000 --- a/services/scout/src/fastify.ts +++ /dev/null @@ -1,165 +0,0 @@ - - -import Fastify, { FastifyPluginCallback, FastifyReply, FastifyRequest, FastifyReplyContext, FastifyPluginAsync, FastifyServerOptions } from 'fastify' -import fastifySwagger from '@fastify/swagger' -import fastifySwaggerUi from '@fastify/swagger-ui' -import { readFileSync } from 'node:fs' -import { fileURLToPath } from 'url' -import { dirname, join } from 'node:path' -import { Config } from './config' -import { VtuberRecord, VtuberResponse, VtuberDataScrape } from './schemas.ts' -import scrapeVtuberData from './scrapeVtuberData.ts' -import { getPlaylistUrl } from './ytdlp.ts' -import { getRandomRoom } from './cb.ts' -import { getPackageVersion } from '@futureporn/utils/file.ts' -import { type GenericApiResponse } from '@futureporn/types' - -type VtuberDataRequest = FastifyRequest<{ - Querystring: { url: string } -}> - - -const __dirname = dirname(fileURLToPath(import.meta.url)); -const swaggerDarkCss = readFileSync(join(__dirname, './css/SwaggerDark.css'), { encoding: 'utf-8' }) -const version = getPackageVersion(join(__dirname, '../package.json')) - - -async function fastifySetup(configs: Config) { - - const fastify = Fastify({ - logger: true - }) - - - await fastify.register(fastifySwagger, { - openapi: { - info: { - title: '@futureporn/scout', - description: 'Vtuber data acquisition API', - version - }, - } - }) - await fastify.register(fastifySwaggerUi, { - theme: { - title: '@fp/scout', - css: [ - { filename: 'SwaggerDark.css', content: swaggerDarkCss } - ] - }, - routePrefix: '/', - uiConfig: { - docExpansion: 'list', - deepLinking: true - }, - uiHooks: { - onRequest: function (request: FastifyRequest, reply: FastifyReply, next: any) { next() }, - preHandler: function (request: FastifyRequest, reply: FastifyReply, next: any) { next() } - }, - staticCSP: true, - transformStaticCSP: (header: any) => header, - transformSpecification: (swaggerObject: any, request: FastifyRequest, reply: FastifyReply) => { return swaggerObject }, - transformSpecificationClone: true -}) - -fastify.addSchema(VtuberResponse) -fastify.addSchema(VtuberRecord) -fastify.addSchema(VtuberDataScrape) - - - -fastify.get('/chaturbate/random-room', { - schema: { - response: { - '2xx': { - type: 'object' - } - }, - tags: ['chaturbate'] - } -}, async (req, reply) => { - const room = await getRandomRoom() - console.log(room) - reply.type('application/json').send(JSON.stringify(room)) -}) - - - -fastify.get('/ytdlp/playlist-url', { - schema: { - querystring: { - type: 'object', - properties: { - playlistUrl: { - type: 'string' - } - } - }, - response: { - '2xx': { - error: { type: 'boolean' }, - message: { type: 'string' }, - data: { type: 'object', properties: { - playlistUrl: { type: 'string' } - }} - } - }, - tags: ['yt-dlp'] - } -}, async (req: VtuberDataRequest, reply) => { - try { - const { data, error, message } = await getPlaylistUrl(req.query.url) - console.log(`playlistUrl=${data.playlistUrl}`) - reply.type('application/json').send(JSON.stringify({ - data: { - playlistUrl: data.playlistUrl - }, - error: error, - message: message - })) - } catch (e) { - reply.type('application/json').send(JSON.stringify({ data: null, error: e })) - } -}) - - - -fastify.get('/vtuber/data', { - schema: { - querystring: { - type: 'object', - properties: { - url: { - type: 'string', - description: 'URL of a vtuber profile on Chaturbate or Fansly. ex: https://chaturbate.com/projektmelody' - } - } - }, - response: { - '2xx': { $ref: 'VtuberDataScrape' } - }, - tags: [ - 'vtuber' - ] - } -}, async (req: VtuberDataRequest, reply) => { - console.log(`we received a request with url=${req.query.url}`) - const data = await scrapeVtuberData(req.query.url) - reply.type('application/json').send(data) -}) - - - -fastify.listen({ host: '0.0.0.0', port: configs.port }, function (err, address) { - console.log(`@futureporn/scout listening on ${address}`) - if (err) { - fastify.log.error(err) - process.exit(1) - } -}) - - -} - - -export default fastifySetup diff --git a/services/scout/src/fixtures/.gitignore b/services/scout/src/fixtures/.gitignore deleted file mode 100644 index 314f02b..0000000 --- a/services/scout/src/fixtures/.gitignore +++ /dev/null @@ -1 +0,0 @@ -*.txt \ No newline at end of file diff --git a/services/scout/src/fixtures/cb.projektmelody.fixture.html b/services/scout/src/fixtures/cb.projektmelody.fixture.html deleted file mode 100644 index 80ee9b1..0000000 --- a/services/scout/src/fixtures/cb.projektmelody.fixture.html +++ /dev/null @@ -1,718 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - -Chaturbate - 100% Free Chat & Webcams - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/services/scout/src/hello.ts b/services/scout/src/hello.ts deleted file mode 100644 index 6e158df..0000000 --- a/services/scout/src/hello.ts +++ /dev/null @@ -1,9 +0,0 @@ -import icons from './icons.ts' - -export default function hello(thing: string) { - if (thing === 'world') { - return icons.world - } else { - return 'hi' - } -} \ No newline at end of file diff --git a/services/scout/src/icons.ts b/services/scout/src/icons.ts deleted file mode 100644 index 6a37fe8..0000000 --- a/services/scout/src/icons.ts +++ /dev/null @@ -1,4 +0,0 @@ -export default ({ - world: '🌏', - sun: '☀️' -}) \ No newline at end of file diff --git a/services/scout/src/index.browser.js b/services/scout/src/index.browser.js deleted file mode 100644 index df35654..0000000 --- a/services/scout/src/index.browser.js +++ /dev/null @@ -1,174 +0,0 @@ -// watches chaturbate for go live - -import puppeteer from 'puppeteer-extra'; -import { dirname } from 'path'; -import { fileURLToPath } from 'url'; -import StealthPlugin from 'puppeteer-extra-plugin-stealth' -import delay from 'delay' -import ReplPlugin from 'puppeteer-extra-plugin-repl' -import fsp from 'fs/promises'; -import repl from 'puppeteer-extra-plugin-repl'; - -const __dirname = dirname(fileURLToPath(import.meta.url)); -const browserDataDir = __dirname+'/futureporn-scout-datadir'; - - -const searchInputSelector = 'body > app-root > div > div.site-wrapper.nav-bar-visible.nav-bar-top-visible > div > app-explore-route > div > app-account-explore-route > div > input' -const followButtonSelector = "xpath///app-account-follow-button/div/xd-localization-string[contains(., 'Follow')]" -const onlineIndicator = 'a.online-indicator.is-live' - -// @todo Get this value from Strapi -const channels = [ - 'avabrooks' -] - - -const scrollDown = async (page) => { - console.log('scrolling down') - await page.keyboard.press('PageDown') - await page.evaluate(async () => { - window.scrollBy(0, 500); - }); -} - -const handleOnlineChannel = async (page) => { - const href = await page.evaluate(async () => { - const onlineIndicator = 'a.online-indicator.is-live' - const href = document.querySelector(onlineIndicator).href - return href - }) - console.log(`${href} is online!`) - await fsp.appendFile('./data.csv', `${new Date().toISOString()},${href}\n`) -} - -/** - * Open up a tab to the channel. - * Intercept websockets events which tell us when the channel is online - */ -const monitor = async (browser, channel) => { - if (!browser) throw new Error('monitor requires page arg'); - if (!channel) throw new Error('monitor requires channel arg'); - console.log(`monitoring ${channel}`) - const page = await browser.newPage(); - - await page.setRequestInterception(true); - - const url = await new Promise((resolve) => { - page.on('request', interceptRequest => { - const url = interceptRequest.url(); - console.log(url) - if (url.startsWith('wss://realtime')) { - console.log(`request! ${interceptRequest.url()} ${JSON.stringify(interceptRequest.headers(), null, 2)}`); - resolve(interceptRequest.url()); - } - interceptRequest.continue(); - }); - - page.goto(`https://chaturbate.com/${channel}`); - }) - - await page.repl() -} - -const dash = async (page, channel) => { - console.log('check') - await page.goto(`https://chaturbate.com/${channel}`) - - - // look for is-live indicator - try { - console.log('waiting for home page') - await page.waitForSelector('div.stories-scroll-container', { timeout: 15000 }) - console.log('waiting for online indicators') - await page.waitForSelector(onlineIndicator, { timeout: 10000 }) - console.log(`FOUND online channel! ${JSON.stringify(onlineIndicator, null, 2)}`); - handleOnlineChannel(page) - delay(1000) - } catch (e) { - console.error(e) - console.log('error on the dash. lets move on.') - } -} - -const discover = async (page) => { - - // Navigate the page to a URL - await page.goto('https://fansly.com/explore/discover'); - - - console.log('wait for search input') - await page.waitForSelector(searchInputSelector) - - // console.log('click search input') - // await page.click(searchInputSelector) - - console.log('type a random letter') - const letter = (() => { - const letters = 'abcdefghijklmnopqrstuvwxyz' - const randomIndex = Math.floor(Math.random() * letters.length); - return letters[randomIndex]; - })() - await page.type(searchInputSelector, letter) - - - - // // Start an interactive REPL here with the `page` instance. - // await page.repl() - // // Afterwards start REPL with the `browser` instance. - // await browser.repl() - - let go = true - setTimeout(() => { - go = false - }, 1000*30*1) // ~2 minutes of following - - while (go) { - try { - console.log('Wait for Follow/Unfollow buttons') - try { - await page.waitForSelector(followButtonSelector, { timeout: 1000 }) - await page.click(followButtonSelector, { timeout: 1000 }) - } catch {} - - // look for is-live indicator - try { - await page.waitForSelector(onlineIndicator, { timeout: 100 }) - console.log(`FOUND online channel! ${JSON.stringify(onlineIndicator, null, 2)}`); - handleOnlineChannel(page) - } catch (e) { - // console.log('online channel not found') - } - - scrollDown(page) - await page.keyboard.press('PageDown') - } catch (e) { - console.error('Error while waiting for follow button') - console.error(e) - } - await delay(15000) - } -} - -(async () => { - // Launch the browser and open a new blank page - puppeteer.use(StealthPlugin()) - puppeteer.use(ReplPlugin()) - const browser = await puppeteer.launch({ - headless: false, - args: [ - `--user-data-dir=${browserDataDir}` - ] - }); - - - for (const ch of channels) { - await monitor(browser, ch) - // await dash(page) - // await discover(page) // discover & follow - // await dash(page) // dashboard & see who's live - } - - - - -})(); \ No newline at end of file diff --git a/services/scout/src/index.ts b/services/scout/src/index.ts deleted file mode 100644 index 479c565..0000000 --- a/services/scout/src/index.ts +++ /dev/null @@ -1,6 +0,0 @@ - -import fastify from './fastify.js' -import { configs } from './config.js' - -fastify(configs) - diff --git a/services/scout/src/loader.js b/services/scout/src/loader.js deleted file mode 100644 index 4d2438f..0000000 --- a/services/scout/src/loader.js +++ /dev/null @@ -1,13 +0,0 @@ -import { readFileSync } from 'node:fs' -import yaml from 'js-yaml' -import { join, dirname } from 'node:path' -import { fileURLToPath } from 'url'; - -const __dirname = dirname(fileURLToPath(import.meta.url)); - - -export function loadOpenApiSpec () { - const rawFile = readFileSync(join(__dirname, './api.yaml'), { encoding: 'utf-8' }) - const doc = yaml.load(rawFile) - return doc -} diff --git a/services/scout/src/schemas.ts b/services/scout/src/schemas.ts deleted file mode 100644 index af03515..0000000 --- a/services/scout/src/schemas.ts +++ /dev/null @@ -1,28 +0,0 @@ -/** - * schemas.ts - * - * uses ts-json-schema-generator to generate json schema from typescript declarations. - * - */ -import tsj, { Config } from 'ts-json-schema-generator' -import { createRequire } from 'node:module' - - - -const require = createRequire(import.meta.url); -const typesIndex = require.resolve('@futureporn/types/src/index.ts'); -const typesTsConfig = require.resolve('@futureporn/types/tsconfig.json'); - -const config: Config = { - path: typesIndex, - tsconfig: typesTsConfig, - type: '*', - skipTypeCheck: true, - encodeRefs: true, - topRef: false, -} - -export const VtuberResponse = tsj.createGenerator(Object.assign(config, { schemaId: 'VtuberResponse' })).createSchema('VtuberResponse') -export const VtuberRecord = tsj.createGenerator(Object.assign(config, { schemaId: 'VtuberRecord' })).createSchema('VtuberRecord') -export const VtuberDataScrape = tsj.createGenerator(Object.assign(config, { schemaId: 'VtuberDataScrape' })).createSchema('VtuberDataScrape') -// export const VtuberDataScrape = tsj.createGenerator(Object.assign(config, { schemaId: 'VtuberDataScrape' })).createSchema('VtuberDataScrape') diff --git a/services/scout/src/scrapeVtuberData.spec.ts b/services/scout/src/scrapeVtuberData.spec.ts deleted file mode 100644 index 43d269b..0000000 --- a/services/scout/src/scrapeVtuberData.spec.ts +++ /dev/null @@ -1,14 +0,0 @@ -import scrapeVtuberData from "./scrapeVtuberData.ts" -import { expect } from "chai" - -describe('integration', function () { - describe('scrapeVtuberData', function () { - it('should get displayname, username, and id from chaturbate', async function () { - const vtuber = await scrapeVtuberData('https://chaturbate.com/projektmelody') - expect(vtuber).to.have.property('display_name', 'Projektmelody') - expect(vtuber).to.have.property('chaturbate_id', 'G0TWFS5') - expect(vtuber).to.have.property('slug', 'projektmelody') - expect(vtuber).to.have.property('theme_color', '#5C23C0') - }) - }) -}) \ No newline at end of file diff --git a/services/scout/src/scrapeVtuberData.ts b/services/scout/src/scrapeVtuberData.ts deleted file mode 100644 index 706cf17..0000000 --- a/services/scout/src/scrapeVtuberData.ts +++ /dev/null @@ -1,51 +0,0 @@ -import type { VtuberDataScrape } from "@futureporn/types" -import { fetchHtml, getBroadcasterDisplayName, getInitialRoomDossier } from './cb.ts' -import { getAccountData, usernameRegex } from "./fansly.ts" -import { fpSlugify } from "@futureporn/utils/name.ts" - - -/** - * - * scrapeVtuberData - * - * The purpose of this function is to retrieve enough data about a vtuber in order for us to create a database record about them. - */ -export default async function scrapeVtuberData(url: string): Promise { - - let display_name, chaturbate_id, slug, fansly_id, chaturbate, fansly - if (url.match(/chaturbate/)) { - console.log('url matches chaturbate') - let html = await fetchHtml(url) - // console.log(html) - let dossier = getInitialRoomDossier(html) - if (!dossier) throw new Error('failed to fetch initial room dossier.'); - slug = dossier.broadcaster_username - chaturbate_id = dossier.broadcaster_uid - display_name = getBroadcasterDisplayName(html) - chaturbate = url - if (!display_name) throw new Error('failed to get broadcaster display name from chaturbate'); - // console.log(dossier) - } else if (url.match(/fansly/)) { - console.log('url matches fansly') - const username = usernameRegex.exec(url)?.at(1) - if (!username) throw new Error(`failed to parse fansly username from url=${url}`); - const data = await getAccountData(username) - slug = fpSlugify(data.username) - display_name = data?.username - fansly_id = data.id - fansly = url - } else { - throw new Error('The URL does not match a known platform'); - } - - - return { - display_name, - chaturbate_id, - chaturbate, - fansly_id, - fansly, - slug, - } - -} \ No newline at end of file diff --git a/services/scout/src/scrapingFetch.spec.ts b/services/scout/src/scrapingFetch.spec.ts deleted file mode 100644 index bb77480..0000000 --- a/services/scout/src/scrapingFetch.spec.ts +++ /dev/null @@ -1,31 +0,0 @@ -import scrapingFetch from "./scrapingFetch.ts" -import { expect } from "chai" - -describe('scrapingFetch', function () { - describe('unit', function () { - xit('idk', async function () { - - }) - }) - describe('integration', function () { - it('should be rate limited while fetching a url', async function () { - this.timeout(1000*30) - // @see https://dog.ceo/dog-api/ - const dogBreedApiUrl = 'https://dog.ceo/api/breeds/image/random' - const res = await scrapingFetch(dogBreedApiUrl) - const json = await res.json() - expect(json).to.have.property('message') - expect(json).to.have.property('status') - - const res1 = await scrapingFetch(dogBreedApiUrl) - const json1 = await res1.json() - expect(json1).to.have.property('message') - expect(json1).to.have.property('status') - - const res2 = await scrapingFetch(dogBreedApiUrl) - const json2 = await res2.json() - expect(json2).to.have.property('message') - expect(json2).to.have.property('status') - }) - }) -}) \ No newline at end of file diff --git a/services/scout/src/scrapingFetch.ts b/services/scout/src/scrapingFetch.ts deleted file mode 100644 index fc37cba..0000000 --- a/services/scout/src/scrapingFetch.ts +++ /dev/null @@ -1,75 +0,0 @@ -/** - * scrapingFetch - * - * Like a regular fetch(), but this one is rate limited so as not to be a nuisance. - * - * rate limiting provided by https://github.com/animir/node-rate-limiter-flexible - * - * the token count is measured at the database level, so many methods cluster-wide can use scrapingFetch and there - * is a guarantee that our fetches stay within the limits. - */ -import { RateLimiterMemory, RateLimiterPostgres, RateLimiterQueue, type IRateLimiterPostgresOptions } from 'rate-limiter-flexible'; -import pRetry from 'p-retry' -import { configs } from './config.ts' -import pg from 'pg' -const { Pool } = pg - -const rateLimiterOptions = { - points: 1, - duration: 5, -} - -console.log(`connectionString=${configs.workerConnectionString}`) -const storeClient = new Pool({ - connectionString: configs.workerConnectionString, -}) -storeClient.on('error', (err) => { - console.error('A pool error occurred', err.stack) -}) -storeClient.on('connect', () => { - // console.log(`postgres storeClient Pool connedcted!!!!!`) - storeClient.on('error', (err) => { - console.error('A pool error occurred within the connect callbakc', err.stack) - }) -}); - - -const postgresOptions: IRateLimiterPostgresOptions = { - storeClient, - schemaName: 'public', - tableName: 'limiter', - keyPrefix: 'scraping_fetch', - tableCreated: true, -} -console.log(`process.env.NODE_ENV=${configs.nodeEnv}`) -const limiterFlexible = (process.env.NODE_ENV === 'development') - ? new RateLimiterMemory(rateLimiterOptions) - : new RateLimiterPostgres(Object.assign(rateLimiterOptions, postgresOptions)) - -const limiterQueue = new RateLimiterQueue(limiterFlexible, { - maxQueueSize: 64, -}); - - -export default async function scrapingFetch(url: string, options?: RequestInit): Promise { - - const run = async () => { - console.log(`${performance.now()} scrapingFetch() spending 1 token and fetching ${url}`) - await limiterQueue.removeTokens(1) - const res = await fetch(url, options) - return res - } - - return pRetry(run, { - forever: false, - maxTimeout: 8000, - retries: 3, - onFailedAttempt: (e) => { - console.error(e); - console.error(`There was an error DURING scrapingFetch. We will now retry up to ${e.retriesLeft} times. cause=${e.cause}, message=${e.message}, attemptNumber=${e.attemptNumber}, name=${e.name}`) - } - }) - - - -} \ No newline at end of file diff --git a/services/scout/src/signals.js b/services/scout/src/signals.js deleted file mode 100644 index 2fb00e4..0000000 --- a/services/scout/src/signals.js +++ /dev/null @@ -1,297 +0,0 @@ - -import 'dotenv/config' -import qs from 'qs' -import { subMinutes, addMinutes } from 'date-fns' -import { fpSlugify, download } from './utils.js' -import { getProminentColor } from './image.js' -import { getImage } from './vtuber.js' -import fansly from './fansly.js' - -// alternative js libraries for postgres notify/wait -// * https://github.com/imqueue/pg-pubsub -// * https://github.com/voxpelli/node-pg-pubsub -// * https://github.com/andywer/pg-listen - -if (!process.env.SCOUT_STRAPI_API_KEY) throw new Error('SCOUT_STRAPI_API_KEY is missing from env'); -if (!process.env.STRAPI_URL) throw new Error('STRAPI_URL is missing from env'); -if (!process.env.SCOUT_RECENTS_TOKEN) throw new Error('SCOUT_RECENTS_TOKEN is undefined in env'); -if (!process.env.CDN_BUCKET_URL) throw new Error('CDN_BUCKET_URL is undefined in env'); - - - - -/** - * Create a database record which shows this stream exists - * - * It's kind of complicated, but we do it this way so we don't need a backend batch processor. - * Instead, Scout takes as much responsibility as possible and does the work that a human would do if they were creating the records. - * - * In Strapi, we are finding or updating or creating the following content-types. - * * vtuber - * * platform-notification - * * stream - * - * It's a 3 step process, with each step outlined in the function body. - */ -export async function createStreamInDb ({ source, platform, channel, date, url, userId }) { - throw new Error('createStreamInDb is deprecated.'); - - let vtuberId, streamId - - console.log('>> # Step 1') - // # Step 1. - // First we find or create the vtuber - // The vtuber may already be in the db, so we look for that record. All we need is the Vtuber ID. - // If the vtuber is not in the db, we create the vtuber record. - - // GET /api/:pluralApiId?filters[field][operator]=value - const findVtubersFilters = (() => { - if (platform === 'chaturbate') { - return { chaturbate: { $eq: url } } - } else if (platform === 'fansly') { - if (!userId) throw new Error('Fansly userId was undefined, but it is required.') - return { fanslyId: { $eq: userId } } - } - })() - console.log('>>>>> the following is findVtubersFilters.') - console.log(findVtubersFilters) - - const findVtubersQueryString = qs.stringify({ - filters: findVtubersFilters - }, { encode: false }) - console.log(`>>>>> platform=${platform}, url=${url}, userId=${userId}`) - - console.log('>> findVtuber') - const findVtuberRes = await fetch(`${process.env.STRAPI_URL}/api/vtubers?${findVtubersQueryString}`, { - method: 'GET', - headers: { - 'content-type': 'application/json' - } - }) - const findVtuberJson = await findVtuberRes.json() - console.log('>> here is the vtuber json') - console.log(findVtuberJson) - if (findVtuberJson?.data && findVtuberJson.data.length > 0) { - console.log('>>a vtuber was FOUND') - if (findVtuberJson.data.length > 1) throw new Error('There was more than one vtuber match. There must only be one.') - vtuberId = findVtuberJson.data[0].id - console.log('here is the findVtuberJson (as follows)') - console.log(findVtuberJson) - console.log(`the matching vtuber has ID=${vtuberId} (${findVtuberJson.data[0].attributes.displayName})`) - } - - if (!vtuberId) { - console.log('>> !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! (DEPRECATED FUNCTION) vtuberId was not found so we create') - - /** - * We are creating a vtuber record. - * We need a few things. - * * image URL - * * themeColor - * - * To get an image, we have to do a few things. - * * [x] download image from platform - * * [x] get themeColor from image - * * [x] upload image to b2 - * * [x] get B2 cdn link to image - * - * To get themeColor, we need the image locally where we can then run - */ - - // download image from platform - // vtuber.getImage expects a vtuber object, which we don't have yet, so we create a dummy one - const dummyVtuber = { - attributes: { - slug: fpSlugify(channel), - fansly: fansly.url.fromUsername(channel) - } - } - const platformImageUrl = await getImage(dummyVtuber) - const imageFile = await download({ url: platformImageUrl }) - - // get themeColor from image - const themeColor = await getProminentColor(imageFile) - - // upload image to b2 - const b2FileData = await s3.uploadFile(imageFile) - - // get b2 cdn link to image - const imageCdnLink = `${process.env.CDN_BUCKET_URL}/${b2FileData.Key}` - - - const createVtuberRes = await fetch(`${process.env.STRAPI_URL}/api/vtubers`, { - method: 'POST', - headers: { - 'authorization': `Bearer ${process.env.SCOUT_STRAPI_API_KEY}`, - 'content-type': 'application/json' - }, - body: JSON.stringify({ - data: { - displayName: channel, - fansly: (platform === 'fansly') ? url : null, - fanslyId: (platform === 'fansly') ? userId : null, - chaturbate: (platform === 'chaturbate') ? url : null, - slug: fpSlugify(channel), - description1: ' ', - image: imageCdnLink, - themeColor: themeColor || '#dde1ec' - } - }) - }) - const createVtuberJson = await createVtuberRes.json() - console.log('>> createVtuberJson as follows') - console.log(JSON.stringify(createVtuberJson, null, 2)) - if (createVtuberJson.data) { - vtuberId = createVtuberJson.data.id - console.log(`>>> vtuber created with id=${vtuberId}`) - } - } - - if (!vtuberId) throw new Error(`>> we weren't able to find or create a vtuberId so we are panicking. (this should not happen under normal circumstances. Bug desu ka?)`) - console.log(`>># Step 2. vtuberId=${vtuberId}`) - // # Step 2. - // Next we create the platform-notification record. - // This probably doesn't already exist, so we don't check for a pre-existing platform-notification. - const pNotifPayload = { - data: { - source: source, - date: date, - date2: date, - platform: platform, - vtuber: vtuberId, - } - } - console.log('pNotifPayload as follows') - console.log(pNotifPayload) - - const pNotifCreateRes = await fetch(`${process.env.STRAPI_URL}/api/platform-notifications`, { - method: 'POST', - headers: { - 'authorization': `Bearer ${process.env.SCOUT_STRAPI_API_KEY}`, - 'Content-Type': 'application/json' - }, - body: JSON.stringify(pNotifPayload) - }) - const pNotifData = await pNotifCreateRes.json() - if (pNotifData.error) { - console.error('>> we failed to create platform-notification, there was an error in the response') - console.error(pNotifData.error) - throw new Error(pNotifData.error) - } - console.log(`>> pNotifData (json response) is as follows`) - console.log(pNotifData) - if (!pNotifData.data?.id) throw new Error('failed to created pNotifData! The response was missing an id'); - - // # Step 3. - // Finally we find or create the stream record - // The stream may already be in the db (the streamer is multi-platform streaming), so we look for that record. - // This gets a bit tricky. How do we determine one stream from another? - // For now, the rule is 30 minutes of separation. - // Anything <=30m is interpreted as the same stream. Anything >30m is interpreted as a different stream. - // If the stream is not in the db, we create the stream record - const dateSinceRange = subMinutes(new Date(date), 30) - const dateUntilRange = addMinutes(new Date(date), 30) - console.log(`Find a stream within + or - 30 mins of the notif date=${new Date(date).toISOString()}. dateSinceRange=${dateSinceRange.toISOString()}, dateUntilRange=${dateUntilRange.toISOString()}`) - const findStreamQueryString = qs.stringify({ - populate: 'platform_notifications', - filters: { - date: { - $gte: dateSinceRange, - $lte: dateUntilRange - }, - vtuber: { - id: { - '$eq': vtuberId - } - } - } - }, { encode: false }) - - console.log('>> findStream') - const findStreamRes = await fetch(`${process.env.STRAPI_URL}/api/streams?${findStreamQueryString}`, { - method: 'GET', - headers: { - 'authorization': `Bearer ${process.env.SCOUT_STRAPI_API_KEY}`, - 'Content-Type': 'application/json' - } - }) - const findStreamData = await findStreamRes.json() - if (findStreamData?.data && findStreamData.data.length > 0) { - console.log('>> we found a findStreamData json. (there is an existing stream for this e-mail/notification)') - console.log(findStreamData) - streamId = findStreamData.data?.id - - // Before we're done here, we need to do something extra. We need to populate isChaturbateStream and/or isFanslyStream. - // We know which of these booleans to set based on the stream's related platform_notifications - // We go through each pNotif and look at it's platform - let isFanslyStream, isChaturbateStream - for (const pn of findStreamData.platform_notifications) { - if (pn.platform === 'fansly') { - isFanslyStream = true - } else if (pn.platform === 'chaturbate') { - isChaturbateStream = true - } - } - - console.log(`>>> updating stream ${streamId}. isFanslyStream=${isFanslyStream}, isChaturbateStream=${isChaturbateStream}`) - const updateStreamRes = await fetch(`${process.env.STRAPI_URL}/api/stream/${streamId}`, { - method: 'PUT', - headers: { - 'authorization': `Bearer ${process.env.SCOUT_STRAPI_API_KEY}`, - 'content-type': 'application/json' - }, - body: JSON.stringify({ - data: { - isFanslyStream: isFanslyStream, - isChaturbateStream: isChaturbateStream - } - }) - }) - const updateStreamJson = await updateStreamRes.json() - if (updateStreamJson?.error) throw new Error(updateStreamJson); - console.log(`>> assuming a successful update to the stream record. response as follows.`) - console.log(JSON.stringify(updateStreamJson, null, 2)) - } - - if (!streamId) { - console.log('>> did not find a streamId, so we go ahead and create a stream record in the db.') - const createStreamPayload = { - data: { - isFanslyStream: (platform === 'fansly') ? true : false, - isChaturbateStream: (platform === 'chaturbate') ? true : false, - archiveStatus: 'missing', - date: date, - date2: date, - date_str: date, - vtuber: vtuberId, - platform_notifications: [ - pNotifData.data.id - ] - } - } - console.log('>> createStreamPayload as follows') - console.log(createStreamPayload) - const createStreamRes = await fetch(`${process.env.STRAPI_URL}/api/streams`, { - method: 'POST', - headers: { - 'authorization': `Bearer ${process.env.SCOUT_STRAPI_API_KEY}`, - "Content-Type": "application/json" - }, - body: JSON.stringify(createStreamPayload) - }) - const createStreamJson = await createStreamRes.json() - console.log('>> we got the createStreamJson') - console.log(createStreamJson) - if (createStreamJson.error) { - console.error(JSON.stringify(createStreamJson.error, null, 2)) - throw new Error('Failed to create stream in DB due to an error. (see above)') - } - - } - - - -} - - - diff --git a/services/scout/src/spawnWrapper.spec.ts b/services/scout/src/spawnWrapper.spec.ts deleted file mode 100644 index c3e98f2..0000000 --- a/services/scout/src/spawnWrapper.spec.ts +++ /dev/null @@ -1,30 +0,0 @@ -import { expect } from "chai" -import spawnWrapper from "./spawnWrapper.ts" -import { getRandomRoom } from "./cb.ts" - -describe('spawnWrapper', function () { - describe('integration', function () { - this.timeout(1000*8) - let roomUrl: string - this.beforeAll(async function () { - roomUrl = (await getRandomRoom()).url - }) - it('should get a playlistUrl of an active stream', async function () { - // the system under test is the network integration - const {code, output} = await spawnWrapper('yt-dlp', ['-g', roomUrl]) - expect(code).to.equal(0) - expect(output).to.match(/https:\/\/.*\.m3u8/) - }) - // these tests are flaky because the rooms used will not always be in the same state - xit('should handle when the room is offline', async function () { - const {code, output} = await spawnWrapper('yt-dlp', ['-g', 'chaturbate.com/48507961285']) - expect(code).to.equal(1) - expect(output).to.match(/Room is currently offline/) - }) - xit('should handle when the room is passworded', async function () { - const {code, output} = await spawnWrapper('yt-dlp', ['-g', 'chaturbate.com/projektmelody']) - expect(code).to.equal(1) - expect(output).to.match(/Unable to find stream URL/) - }) - }) -}) \ No newline at end of file diff --git a/services/scout/src/spawnWrapper.ts b/services/scout/src/spawnWrapper.ts deleted file mode 100644 index 03c2974..0000000 --- a/services/scout/src/spawnWrapper.ts +++ /dev/null @@ -1,34 +0,0 @@ -import child_process from 'node:child_process' - -export interface SpawnOutput { - code: number; - output: string; -} - - -/** - * we have this child_process.spawn wrapper to make testing easier. - * this function is meant to be mocked during unit tests so the function logic can be tested - * without making a network request. - */ -export default async function spawnWrapper (command: string, args: string[]): Promise { - console.log(`spawnWrapper command=${command}, with the following args ${args.join(' ')}`) - return new Promise((resolve, reject) => { - let output = ''; - const process = child_process.spawn(command, args) - process.on('exit', function (code) { - if (code === undefined || code === null) throw new Error('process exited without an exit code'); - resolve({ code, output }) - }) - process.stdout.on('data', (data) => { - output += data - }) - process.stderr.on('data', (data) => { - output += data - }) - process.on('error', function (e) { - reject(e) - }) - }) -} - diff --git a/services/scout/src/twitter.js b/services/scout/src/twitter.js deleted file mode 100644 index ea900bc..0000000 --- a/services/scout/src/twitter.js +++ /dev/null @@ -1,65 +0,0 @@ - -import * as htmlparser2 from "htmlparser2"; -import { load } from 'cheerio' -import { download } from '@futureporn/utils/file.ts'; -import pRetry, { AbortError } from 'p-retry'; - -if (!process.env.SCOUT_NITTER_ACCESS_KEY) throw new Error('SCOUT_NITTER_ACCESS_KEY was undefined in env'); -if (!process.env.SCOUT_NITTER_URL) throw new Error('SCOUT_NITTER_URL was undefined in env'); - - - -const regex = { - username: new RegExp(/https:\/\/(?:twitter\.com|x\.com)\/([a-zA-Z0-9_]+)/i) -} - -const normalize = (url) => { - if (!url) throw new Error('normalized received a null or undefined url.'); - return fromUsername(twitter.regex.username.exec(url).at(1)) -} - - -const image = async function image (twitterUsername) { - if (!twitterUsername) throw new Error('first arg to twitter.data.image must be a twitterUsername. It was undefined.'); - const requestDataFromNitter = async () => { - const url = `${process.env.SCOUT_NITTER_URL}/${twitterUsername}/rss?key=${process.env.SCOUT_NITTER_ACCESS_KEY}` - // console.log(`fetching from url=${url}`) - const response = await fetch(url); - // Abort retrying if the resource doesn't exist - if (response.status === 404) { - throw new AbortError(response.statusText); - } - return response.text(); - } - const body = await pRetry(requestDataFromNitter, { retries: 5 }); - try { - const dom = htmlparser2.parseDocument(body); - const $ = load(dom, { _useHtmlParser2: true }) - const urls = $('url:contains("profile_images")').first() - const downloadedImageFile = await download({ url: urls.text() }) - return downloadedImageFile - } catch (e) { - console.error(`while fetching rss from nitter, the following error was encountered.`) - console.error(e) - } -} - -const fromUsername = (username) => `https://x.com/${username}` - -const url = { - normalize, - fromUsername -} - -const data = { - image -} - -const twitter = { - regex, - url, - data, -} - - -export default twitter \ No newline at end of file diff --git a/services/scout/src/twitter.spec.js b/services/scout/src/twitter.spec.js deleted file mode 100644 index e2f2403..0000000 --- a/services/scout/src/twitter.spec.js +++ /dev/null @@ -1,28 +0,0 @@ -import { expect } from 'chai' -import twitter from './twitter.js' -import { describe } from 'mocha' -import { tmpFileRegex } from './utils.js' - -describe('twitter', function () { - describe('regex', function () { - describe('username', function () { - it('should get the username of the channel', function () { - expect(twitter.regex.username.exec('https://twitter.com/18Plus').at(1)).to.equal('18Plus') - expect(twitter.regex.username.exec('https://twitter.com/projektmelody').at(1)).to.equal('projektmelody') - expect(twitter.regex.username.exec('https://twitter.com/GoodKittenVR').at(1)).to.equal('GoodKittenVR') - expect(twitter.regex.username.exec('https://x.com/projektmelody').at(1)).to.equal('projektmelody') - expect(twitter.regex.username.exec('https://x.com/18Plus').at(1)).to.equal('18Plus') - expect(twitter.regex.username.exec('https://x.com/GoodKittenVR').at(1)).to.equal('GoodKittenVR') - }) - }) - }) - describe('data', function () { - this.timeout(1000*30) - describe('image', function () { - it("should download the twitter users's avatar and save it to disk", async function () { - const imgFile = await twitter.data.image('projektmelody') - expect(imgFile).to.match(tmpFileRegex) - }) - }) - }) -}) \ No newline at end of file diff --git a/services/scout/src/ua.ts b/services/scout/src/ua.ts deleted file mode 100644 index 7d643f8..0000000 --- a/services/scout/src/ua.ts +++ /dev/null @@ -1 +0,0 @@ -export const ua0 = 'Mozilla/5.0 (X11; Linux x86_64; rv:105.0) Gecko/20100101 Firefox/105.0' \ No newline at end of file diff --git a/services/scout/src/vtuber.spec.js b/services/scout/src/vtuber.spec.js deleted file mode 100644 index b628368..0000000 --- a/services/scout/src/vtuber.spec.js +++ /dev/null @@ -1,36 +0,0 @@ - -import { expect } from 'chai' -import { describe } from 'mocha' -import { getImage } from './vtuber.js' -import { tmpFileRegex } from './utils.js' - -const vtuberFixture0 = { - id: 0, - attributes: { - slug: 'projektmelody', - twitter: 'https://x.com/projektmelody', - fansly: 'https://fansly.com/projektmelody' - } -} -const vtuberFixture1 = { - id: 0, - attributes: { - slug: 'projektmelody', - twitter: undefined, - fanslyId: '284824898138812416' - } -} - -describe('vtuber', function () { - this.timeout(1000*60) - describe('getImage', function () { - it('should download an avatar image from twitter', async function () { - const file = await getImage(vtuberFixture0) - expect(file).to.match(tmpFileRegex) - }) - it('should download an avatar image from fansly', async function () { - const file = await getImage(vtuberFixture1) - expect(file).to.match(tmpFileRegex) - }) - }) -}) \ No newline at end of file diff --git a/services/scout/src/vtuber.ts b/services/scout/src/vtuber.ts deleted file mode 100644 index 31164ad..0000000 --- a/services/scout/src/vtuber.ts +++ /dev/null @@ -1,44 +0,0 @@ -import dotenv from 'dotenv'; -dotenv.config({ - path: '../../.env', -}); - -import { type IVtuber } from '@futureporn/types'; -import twitter from './twitter.js'; -import { getProfileImage } from './fansly.ts'; - -/** - * Acquire a vtuber image from the www - * - * Sources preference - * 1. Twitter - * 2. Fansly - * - * Our task is to download an avatar image of the vtuber. - * A slug is good for pulling a record from the database. From there, we can see any social medias such as Twitter or Fansly. - * Twitter is preferred. - * - * We depend on one of these social media URLs. If there is neither Twitter nor Fansly listed, we throw an error. - * - * @param {IVtuber} vtuber -- vtuber instance from Strapi - * @returns {string} filePath -- path on disk where the image was saved - */ -export async function getImage(vtuber: IVtuber): Promise { - if (!vtuber) throw new Error('first arg must be vtuber instance'); - - const { twitter: twitterUrl, fanslyId: fanslyId } = vtuber.attributes; - const twitterUsername = twitterUrl && twitterUrl.match(/@(\w+)/)?.[1]; - - let img: string|undefined; - if (twitterUrl) { - img = await twitter.data.image(twitterUsername); - } else if (fanslyId) { - img = await getProfileImage(fanslyId); - } else { - const msg = `while attempting to get vtuber image, there was neither twitterUrl nor fanslyId listed. One of these must exist for us to download an image.\nvtuber=${JSON.stringify(vtuber, null, 2)}`; - console.error(msg); - throw new Error(msg); - } - if (!img) throw new Error('failed to acquire vtuber image'); - return img; -} \ No newline at end of file diff --git a/services/scout/src/ytdlp.spec.ts b/services/scout/src/ytdlp.spec.ts deleted file mode 100644 index d550f89..0000000 --- a/services/scout/src/ytdlp.spec.ts +++ /dev/null @@ -1,53 +0,0 @@ -import { getRandomRoom } from './cb.ts' -import { expect } from 'chai' -import esmock from 'esmock' -import { mock } from 'node:test' - -describe('esmock integration', function () { - // sanity test to ensure esmock functionality doesn't break - // here we are overriding the hello.ts module's functionality - // normally it would return an Earth emoji with Asia visible. - it('should return a planet Earth emoji with Americas visible', async function () { - const hello = await esmock('./hello.ts', { - './icons': { world: '🌎' } - }) - expect(hello('world')).to.equal('🌎') - expect(hello()).to.equal('hi') - }) -}) - -describe('ytdlp', function () { - - describe('integration', function () { - let roomUrl: string; - this.beforeAll(async function () { - roomUrl = (await getRandomRoom()).url - }) - }) - describe('unit', function () { - it('should handle 403s by using a proxy', async function () { - this.timeout(2000) - const ytdlpErrorRequestForbidden = "ERROR: [Chaturbate] projektmelody: Unable to download webpage: HTTP Error 403: Forbidden (caused by ); please report this issue on https://github.com/yt-dlp/yt-dlp/issues?q= , filling out the appropriate issue template. Confirm you are on the latest version using yt-dlp -U" - const requestSimulator = mock.fn(() => { - if (requestSimulator.mock.calls.length === 0) { - return { - code: 1, - output: ytdlpErrorRequestForbidden - } - } else { - return { - code: 0, - output: 'https://example.com/playlist.m3u8' - } - } - }) - - const ytdlp = await esmock('./ytdlp.ts', { - // simulate a yt-dlp request getting blocked by Cloudflare - './spawnWrapper.ts': requestSimulator - }) - const url = await ytdlp.getPlaylistUrl('chaturbate.com/projektmelody') - expect(url).to.match(/https:\/\/.*\.m3u8/) - }) - }) -}) \ No newline at end of file diff --git a/services/scout/src/ytdlp.ts b/services/scout/src/ytdlp.ts deleted file mode 100644 index dfb046b..0000000 --- a/services/scout/src/ytdlp.ts +++ /dev/null @@ -1,43 +0,0 @@ -import spawnWrapper from './spawnWrapper.ts' -import 'dotenv/config' -import { configs } from './config.ts' -import { type GenericApiResponse } from '@futureporn/types' -import { ExhaustedRetriesError, RoomOfflineError } from '@futureporn/utils/error.ts' - -const maxRetries = 3 - - - -export async function getPlaylistUrl (roomUrl: string, proxy = false, retries = 0): Promise { - console.log(`getPlaylistUrl roomUrl=${roomUrl} proxy=${false} retries=${retries}`) - let args = ['-4', '-g', roomUrl] - if (proxy) { - console.log(`proxy=${proxy}, HTTP_PROXY=${configs.httpProxy}`) - args = args.concat(['--proxy', configs.httpProxy]) - } - const { code, output } = await spawnWrapper('yt-dlp', args) - if (output.match(/HTTP Error 403/)) { - // we were likely blocked by Cloudflare - // we make the request a second time, this time via proxy - if (retries < maxRetries) return getPlaylistUrl(roomUrl, true, retries+=1); - else throw new ExhaustedRetriesError(); - } else if (output.match(/Unable to find stream URL/)) { - // sometimes this happens. a retry is in order. - if (retries < maxRetries) return getPlaylistUrl(roomUrl, proxy, retries+=1); - else throw new ExhaustedRetriesError() - } else if (code === 0 && output.match(/https:\/\/.*\.m3u8/)) { - // this must be an OK result with a playlist - return { data: { playlistUrl: output.trim() }, error: null, message: 'we got a playlistUrl' } - } else if (code === 1 && output.match(/Room is currently offline/)) { - throw new RoomOfflineError() - } else { - console.error('exotic scenario') - const msg = `We encountered an exotic scenario where code=${code} and output=${output}. Admin: please patch the code to handle this scenario.` - console.error(msg) - throw new Error(msg) - } -} - -export default { - getPlaylistUrl -} \ No newline at end of file diff --git a/services/scout/tsconfig.json b/services/scout/tsconfig.json deleted file mode 100644 index faacd28..0000000 --- a/services/scout/tsconfig.json +++ /dev/null @@ -1,33 +0,0 @@ -{ - "compilerOptions": { - // Base Options recommended for all projects - "allowImportingTsExtensions": true, - "noEmit": true, - "esModuleInterop": true, - "skipLibCheck": true, - "target": "ESNext", - "allowJs": true, - "moduleResolution": "Bundler", - "resolveJsonModule": true, - "moduleDetection": "force", - "isolatedModules": true, - // Enable strict type checking so you can catch bugs early - "strict": true, - "noUncheckedIndexedAccess": true, - "noImplicitOverride": true, - // Transpile our TypeScript code to JavaScript - "module": "ESNext", - "outDir": "dist", - "lib": [ - "ESNext", - "dom" - ] - }, - // Include the necessary files for your project - "include": [ - "src/**/*.ts" - ], - "exclude": [ - "node_modules" - ] -} \ No newline at end of file diff --git a/services/scout/tsup.config.ts b/services/scout/tsup.config.ts deleted file mode 100644 index 1ce909d..0000000 --- a/services/scout/tsup.config.ts +++ /dev/null @@ -1,18 +0,0 @@ -import { defineConfig } from 'tsup' -import { exec } from 'node:child_process' -import { promisify } from 'node:util' - -export default defineConfig({ - entry: [ - 'src/index.ts' - ], - shims: true, - clean: true, - loader: { - '.yml': 'file' - }, - onSuccess: async () => { - const execP = promisify(exec) - // execP('cp ./src/api.yaml ./dist/api.yaml') - }, -}) diff --git a/services/strapi/.editorconfig b/services/strapi/.editorconfig deleted file mode 100644 index 473e451..0000000 --- a/services/strapi/.editorconfig +++ /dev/null @@ -1,16 +0,0 @@ -root = true - -[*] -indent_style = space -indent_size = 2 -end_of_line = lf -charset = utf-8 -trim_trailing_whitespace = true -insert_final_newline = true - -[{package.json,*.yml}] -indent_style = space -indent_size = 2 - -[*.md] -trim_trailing_whitespace = false diff --git a/services/strapi/.eslintignore b/services/strapi/.eslintignore deleted file mode 100644 index 90759a5..0000000 --- a/services/strapi/.eslintignore +++ /dev/null @@ -1,3 +0,0 @@ -.cache -build -**/node_modules/** diff --git a/services/strapi/.eslintrc b/services/strapi/.eslintrc deleted file mode 100644 index b2ca93b..0000000 --- a/services/strapi/.eslintrc +++ /dev/null @@ -1,27 +0,0 @@ -{ - "parser": "babel-eslint", - "extends": "eslint:recommended", - "env": { - "commonjs": true, - "es6": true, - "node": true, - "browser": false - }, - "parserOptions": { - "ecmaFeatures": { - "experimentalObjectRestSpread": true, - "jsx": false - }, - "sourceType": "module" - }, - "globals": { - "strapi": true - }, - "rules": { - "indent": ["error", 2, { "SwitchCase": 1 }], - "linebreak-style": ["error", "unix"], - "no-console": 0, - "quotes": ["error", "single"], - "semi": ["error", "always"] - } -} diff --git a/services/strapi/.gitignore b/services/strapi/.gitignore deleted file mode 100644 index 91e0eb7..0000000 --- a/services/strapi/.gitignore +++ /dev/null @@ -1,115 +0,0 @@ -############################ -# OS X -############################ - -.DS_Store -.AppleDouble -.LSOverride -Icon -.Spotlight-V100 -.Trashes -._* - - -############################ -# Linux -############################ - -*~ - - -############################ -# Windows -############################ - -Thumbs.db -ehthumbs.db -Desktop.ini -$RECYCLE.BIN/ -*.cab -*.msi -*.msm -*.msp - - -############################ -# Packages -############################ - -*.7z -*.csv -*.dat -*.dmg -*.gz -*.iso -*.jar -*.rar -*.tar -*.zip -*.com -*.class -*.dll -*.exe -*.o -*.seed -*.so -*.swo -*.swp -*.swn -*.swm -*.out -*.pid - - -############################ -# Logs and databases -############################ - -.tmp -*.log -*.sql -*.sqlite -*.sqlite3 - - -############################ -# Misc. -############################ - -*# -ssl -.idea -nbproject -public/uploads/* -!public/uploads/.gitkeep - -############################ -# Node.js -############################ - -lib-cov -lcov.info -pids -logs -results -node_modules -.node_history - -############################ -# Tests -############################ - -coverage - -############################ -# Strapi -############################ - -.env -license.txt -exports -.strapi -dist -build -.strapi-updater.json -.strapi-cloud.json diff --git a/services/strapi/README.md b/services/strapi/README.md deleted file mode 100644 index 879d89b..0000000 --- a/services/strapi/README.md +++ /dev/null @@ -1,57 +0,0 @@ -# 🚀 Getting started with Strapi - -Strapi comes with a full featured [Command Line Interface](https://docs.strapi.io/dev-docs/cli) (CLI) which lets you scaffold and manage your project in seconds. - -### `develop` - -Start your Strapi application with autoReload enabled. [Learn more](https://docs.strapi.io/dev-docs/cli#strapi-develop) - -``` -npm run develop -# or -yarn develop -``` - -### `start` - -Start your Strapi application with autoReload disabled. [Learn more](https://docs.strapi.io/dev-docs/cli#strapi-start) - -``` -npm run start -# or -yarn start -``` - -### `build` - -Build your admin panel. [Learn more](https://docs.strapi.io/dev-docs/cli#strapi-build) - -``` -npm run build -# or -yarn build -``` - -## ⚙️ Deployment - -Strapi gives you many possible deployment options for your project including [Strapi Cloud](https://cloud.strapi.io). Browse the [deployment section of the documentation](https://docs.strapi.io/dev-docs/deployment) to find the best solution for your use case. - -## 📚 Learn more - -- [Resource center](https://strapi.io/resource-center) - Strapi resource center. -- [Strapi documentation](https://docs.strapi.io) - Official Strapi documentation. -- [Strapi tutorials](https://strapi.io/tutorials) - List of tutorials made by the core team and the community. -- [Strapi blog](https://strapi.io/blog) - Official Strapi blog containing articles made by the Strapi team and the community. -- [Changelog](https://strapi.io/changelog) - Find out about the Strapi product updates, new features and general improvements. - -Feel free to check out the [Strapi GitHub repository](https://github.com/strapi/strapi). Your feedback and contributions are welcome! - -## ✨ Community - -- [Discord](https://discord.strapi.io) - Come chat with the Strapi community including the core team. -- [Forum](https://forum.strapi.io/) - Place to discuss, ask questions and find answers, show your Strapi project and get feedback or just talk with other Community members. -- [Awesome Strapi](https://github.com/strapi/awesome-strapi) - A curated list of awesome things related to Strapi. - ---- - -🤫 Psst! [Strapi is hiring](https://strapi.io/careers). diff --git a/services/strapi/config/admin.js b/services/strapi/config/admin.js deleted file mode 100644 index fa18851..0000000 --- a/services/strapi/config/admin.js +++ /dev/null @@ -1,17 +0,0 @@ -module.exports = ({ env }) => ({ - auth: { - secret: env('ADMIN_JWT_SECRET'), - }, - apiToken: { - salt: env('API_TOKEN_SALT'), - }, - transfer: { - token: { - salt: env('TRANSFER_TOKEN_SALT'), - }, - }, - flags: { - nps: env.bool('FLAG_NPS', true), - promoteEE: env.bool('FLAG_PROMOTE_EE', true), - }, -}); diff --git a/services/strapi/config/api.js b/services/strapi/config/api.js deleted file mode 100644 index 62f8b65..0000000 --- a/services/strapi/config/api.js +++ /dev/null @@ -1,7 +0,0 @@ -module.exports = { - rest: { - defaultLimit: 25, - maxLimit: 100, - withCount: true, - }, -}; diff --git a/services/strapi/config/database.js b/services/strapi/config/database.js deleted file mode 100644 index 63a17c3..0000000 --- a/services/strapi/config/database.js +++ /dev/null @@ -1,92 +0,0 @@ -const path = require('path'); - -module.exports = ({ env }) => { - const client = env('DATABASE_CLIENT', 'sqlite'); - - const connections = { - mysql: { - connection: { - connectionString: env('DATABASE_URL'), - host: env('DATABASE_HOST', 'localhost'), - port: env.int('DATABASE_PORT', 3306), - database: env('DATABASE_NAME', 'strapi'), - user: env('DATABASE_USERNAME', 'strapi'), - password: env('DATABASE_PASSWORD', 'strapi'), - ssl: env.bool('DATABASE_SSL', false) && { - key: env('DATABASE_SSL_KEY', undefined), - cert: env('DATABASE_SSL_CERT', undefined), - ca: env('DATABASE_SSL_CA', undefined), - capath: env('DATABASE_SSL_CAPATH', undefined), - cipher: env('DATABASE_SSL_CIPHER', undefined), - rejectUnauthorized: env.bool( - 'DATABASE_SSL_REJECT_UNAUTHORIZED', - true - ), - }, - }, - pool: { min: env.int('DATABASE_POOL_MIN', 2), max: env.int('DATABASE_POOL_MAX', 10) }, - }, - mysql2: { - connection: { - host: env('DATABASE_HOST', 'localhost'), - port: env.int('DATABASE_PORT', 3306), - database: env('DATABASE_NAME', 'strapi'), - user: env('DATABASE_USERNAME', 'strapi'), - password: env('DATABASE_PASSWORD', 'strapi'), - ssl: env.bool('DATABASE_SSL', false) && { - key: env('DATABASE_SSL_KEY', undefined), - cert: env('DATABASE_SSL_CERT', undefined), - ca: env('DATABASE_SSL_CA', undefined), - capath: env('DATABASE_SSL_CAPATH', undefined), - cipher: env('DATABASE_SSL_CIPHER', undefined), - rejectUnauthorized: env.bool( - 'DATABASE_SSL_REJECT_UNAUTHORIZED', - true - ), - }, - }, - pool: { min: env.int('DATABASE_POOL_MIN', 2), max: env.int('DATABASE_POOL_MAX', 10) }, - }, - postgres: { - connection: { - connectionString: env('DATABASE_URL'), - host: env('DATABASE_HOST', 'localhost'), - port: env.int('DATABASE_PORT', 5432), - database: env('DATABASE_NAME', 'strapi'), - user: env('DATABASE_USERNAME', 'strapi'), - password: env('DATABASE_PASSWORD', 'strapi'), - ssl: env.bool('DATABASE_SSL', false) && { - key: env('DATABASE_SSL_KEY', undefined), - cert: env('DATABASE_SSL_CERT', undefined), - ca: env('DATABASE_SSL_CA', undefined), - capath: env('DATABASE_SSL_CAPATH', undefined), - cipher: env('DATABASE_SSL_CIPHER', undefined), - rejectUnauthorized: env.bool( - 'DATABASE_SSL_REJECT_UNAUTHORIZED', - true - ), - }, - schema: env('DATABASE_SCHEMA', 'public'), - }, - pool: { min: env.int('DATABASE_POOL_MIN', 2), max: env.int('DATABASE_POOL_MAX', 10) }, - }, - sqlite: { - connection: { - filename: path.join( - __dirname, - '..', - env('DATABASE_FILENAME', '.tmp/data.db') - ), - }, - useNullAsDefault: true, - }, - }; - - return { - connection: { - client, - ...connections[client], - acquireConnectionTimeout: env.int('DATABASE_CONNECTION_TIMEOUT', 60000), - }, - }; -}; diff --git a/services/strapi/config/middlewares.js b/services/strapi/config/middlewares.js deleted file mode 100644 index 6eaf586..0000000 --- a/services/strapi/config/middlewares.js +++ /dev/null @@ -1,12 +0,0 @@ -module.exports = [ - 'strapi::logger', - 'strapi::errors', - 'strapi::security', - 'strapi::cors', - 'strapi::poweredBy', - 'strapi::query', - 'strapi::body', - 'strapi::session', - 'strapi::favicon', - 'strapi::public', -]; diff --git a/services/strapi/config/plugins.js b/services/strapi/config/plugins.js deleted file mode 100644 index 5b3cfd1..0000000 --- a/services/strapi/config/plugins.js +++ /dev/null @@ -1,45 +0,0 @@ -module.exports = () => ({ - 'fuzzy-search': { - enabled: true, - config: { - contentTypes: [{ - uid: 'api::tag.tag', - modelName: 'tag', - transliterate: false, - queryConstraints: { - where: { - '$and': [ - { - publishedAt: { - '$notNull': true - } - }, - ] - } - }, - fuzzysortOptions: { - characterLimit: 32, - threshold: -600, - limit: 10, - keys: [{ - name: 'name', - weight: 100 - }] - } - }] - } - }, - "users-permissions": { - config: { - register: { - allowedFields: [ - "isNamePublic", - "isLinkPublic", - "avatar", - "vanityLink", - "patreonBenefits" - ] - } - } - } -}); diff --git a/services/strapi/config/server.js b/services/strapi/config/server.js deleted file mode 100644 index 15e5966..0000000 --- a/services/strapi/config/server.js +++ /dev/null @@ -1,12 +0,0 @@ -module.exports = ({ env }) => ({ - host: env('HOST', '0.0.0.0'), - port: env.int('PORT', 1337), - proxy: true, - app: { - keys: env.array('APP_KEYS'), - }, - webhooks: { - populateRelations: env.bool('WEBHOOKS_POPULATE_RELATIONS', false), - }, - url: env('STRAPI_URL', 'http://localhost:1337'), -}); diff --git a/services/strapi/database/daily-backup.sh b/services/strapi/database/daily-backup.sh deleted file mode 100644 index 1ee0aeb..0000000 --- a/services/strapi/database/daily-backup.sh +++ /dev/null @@ -1,13 +0,0 @@ -#!/bin/bash - -# daily-backup.sh -# useful for the dokku server -# dokku's backup feature is broken atm https://github.com/dokku/dokku-postgres/issues/274 -# backups are exported from dokku:postgres plugin before being sent to b2 - - -filename="$(date +'%Y-%m-%d_%H-%M-%S').psql" - -dokku postgres:export futureporn-db > "${filename}" -b2-linux upload-file futureporn-db-backup "./${filename}" "${filename}" - diff --git a/services/strapi/database/devDb.sh b/services/strapi/database/devDb.sh deleted file mode 100755 index d33e117..0000000 --- a/services/strapi/database/devDb.sh +++ /dev/null @@ -1,21 +0,0 @@ -#!/bin/bash - - -# Check if the containers already exist -pgadmin_exists=$(docker ps -a --filter "name=pgadmin" --format '{{.Names}}') -strapi_postgres_exists=$(docker ps -a --filter "name=strapi-postgres" --format '{{.Names}}') - -# Run strapi-postgres container if it doesn't exist or is not running -if [ -z "$strapi_postgres_exists" ]; then - docker run -d --name strapi-postgres -p 5432:5432 -e POSTGRES_PASSWORD=$POSTGRES_PASSWORD postgres:14.7 - echo "strapi-postgres container created and started." -else - container_status=$(docker inspect -f '{{.State.Status}}' strapi-postgres) - - if [ "$container_status" != "running" ]; then - docker start strapi-postgres - echo "strapi-postgres container started." - else - echo "strapi-postgres container already exists and is running. Skipping creation." - fi -fi diff --git a/services/strapi/database/migrations/.gitkeep b/services/strapi/database/migrations/.gitkeep deleted file mode 100644 index e69de29..0000000 diff --git a/services/strapi/database/migrations/2023-08-01-relate-vods-to-vtubers-part2.js b/services/strapi/database/migrations/2023-08-01-relate-vods-to-vtubers-part2.js deleted file mode 100644 index 795adca..0000000 --- a/services/strapi/database/migrations/2023-08-01-relate-vods-to-vtubers-part2.js +++ /dev/null @@ -1,25 +0,0 @@ -module.exports = { - async up(knex) { - // ... (Create vods_vtuber_links table if not already created) - - // Get vtuber ID for ProjektMelody (assuming it's 1) - const vtuberId = 1; - - // Get all VODs from the database - const vods = await knex.select('*').from('vods'); - - // For each VOD, associate it with the vtuber (vtuber with ID 1) if not already associated - for (const [index, vod] of vods.entries()) { - const existingAssociation = await knex('vods_vtuber_links') - .where({ vtuber_id: vtuberId, vod_id: vod.id }) - .first(); - if (!existingAssociation) { - await knex('vods_vtuber_links').insert({ - vtuber_id: vtuberId, - vod_id: vod.id, - vod_order: index + 1, // Auto-increment the vod_order number - }); - } - } - } -}; diff --git a/services/strapi/database/migrations/2023-08-17-reformat-cdnUrl.js b/services/strapi/database/migrations/2023-08-17-reformat-cdnUrl.js deleted file mode 100644 index c090e12..0000000 --- a/services/strapi/database/migrations/2023-08-17-reformat-cdnUrl.js +++ /dev/null @@ -1,18 +0,0 @@ -module.exports = { - async up(knex) { - - // Get all B2 Files from the database - const files = await knex.select('*').from('b2_files'); - - // For each B2 File, update cdnUrl - // we do this to change - // erroneous https://futureporn-b2.b-cdn.net/futureporn/:key - // to https://futureporn-b2.b-cdn.net/:key - for (const [index, file] of files.entries()) { - const key = file.key; - const cdnUrl = `https://futureporn-b2.b-cdn.net/${key}`; - await knex('b2_files').update({ cdn_url: cdnUrl }).where({ id: file.id }); - } - }, - }; - \ No newline at end of file diff --git a/services/strapi/database/migrations/2023-08-20-strip-query-string-from-cid.js b/services/strapi/database/migrations/2023-08-20-strip-query-string-from-cid.js deleted file mode 100644 index 0a254f2..0000000 --- a/services/strapi/database/migrations/2023-08-20-strip-query-string-from-cid.js +++ /dev/null @@ -1,23 +0,0 @@ -const stripQueryString = function (text) { - if (!text) return ''; - return text.split(/[?#]/)[0]; -} - -module.exports = { - async up(knex) { - - // Get all vods - const vods = await knex.select('*').from('vods'); - - // For each vod, update videoSrcHash and video240Hash - // we remove any existing ?filename(...) qs from the cid - for (const [index, vod] of vods.entries()) { - const strippedVideoSrcHash = stripQueryString(vod.video_src_hash) - const strippedVideo240Hash = stripQueryString(vod.video_240_hash) - await knex('vods').update({ video_src_hash: strippedVideoSrcHash }).where({ id: vod.id }); - await knex('vods').update({ video_240_hash: strippedVideo240Hash }).where({ id: vod.id }); - } - - }, - }; - \ No newline at end of file diff --git a/services/strapi/database/migrations/2023-08-30-remove-cloudinary.js b/services/strapi/database/migrations/2023-08-30-remove-cloudinary.js deleted file mode 100644 index 6f1f5a7..0000000 --- a/services/strapi/database/migrations/2023-08-30-remove-cloudinary.js +++ /dev/null @@ -1,13 +0,0 @@ -module.exports = { - async up(knex) { - - const toys = await knex.select('*').from('toys'); - for (const [index, toy] of toys.entries()) { - if (toy.image_2) { - const existingImageFilename = new URL(toy.image_2).pathname.split('/').at(-1) - await knex('toys').update({ image_2: `https://futureporn-b2.b-cdn.net/${existingImageFilename}` }).where({ id: toy.id }); - } - } - }, - }; - \ No newline at end of file diff --git a/services/strapi/database/migrations/2023-08-30-toy-image-field-simplify.js b/services/strapi/database/migrations/2023-08-30-toy-image-field-simplify.js deleted file mode 100644 index 66ab1de..0000000 --- a/services/strapi/database/migrations/2023-08-30-toy-image-field-simplify.js +++ /dev/null @@ -1,33 +0,0 @@ -module.exports = { - async up(knex) { - // Add the `image2` field (column) as a short text field - await knex.schema.table('toys', (table) => { - table.string('image_2', 512); - }); - - // Get all toys - const toys = await knex.select('*').from('toys'); - - // Update the image2 field with the previous image URLs - for (const toy of toys) { - // lookup the file morph which maps toy to (image) file - const imageFileId = (await knex.select('file_id').from('files_related_morphs').where({ related_id: toy.id }))[0].file_id - - // get the image data from the file - const imageUrl = (await knex.select('url').from('files').where({ id: imageFileId }))[0].url - - if (!imageUrl) continue; - - // Copy the values from image to image2 - await knex('toys').update({ image_2: imageUrl }).where({ id: toy.id }); - } - - const hasImageColumn = await knex.schema.hasColumn('toys', 'image'); - if (hasImageColumn) { - // Drop the `image` column - table.dropColumn('image'); - } - - - }, -}; diff --git a/services/strapi/database/migrations/2023-09-08-change-date-to-string.js b/services/strapi/database/migrations/2023-09-08-change-date-to-string.js deleted file mode 100644 index 3b03cc0..0000000 --- a/services/strapi/database/migrations/2023-09-08-change-date-to-string.js +++ /dev/null @@ -1,23 +0,0 @@ -module.exports = { - async up(knex) { - // Check if the 'date_2' column exists in the 'vods' table - const hasDate2Column = await knex.schema.hasColumn('vods', 'date_2'); - - if (!hasDate2Column) { - // Add the new 'date_2' column as a string if it doesn't exist - await knex.schema.table('vods', (table) => { - table.string('date_2'); - }); - - // Fetch all existing rows from the 'vods' table - const existingVods = await knex.select('id', 'date').from('vods'); - - // Loop through each row and update 'date_2' with the date value - for (const vod of existingVods) { - await knex('vods') - .where({ id: vod.id }) - .update({ date_2: vod.date.toISOString() }); - } - } - }, -}; diff --git a/services/strapi/database/migrations/2023-09-08-drop-toys-image.js b/services/strapi/database/migrations/2023-09-08-drop-toys-image.js deleted file mode 100644 index 2119714..0000000 --- a/services/strapi/database/migrations/2023-09-08-drop-toys-image.js +++ /dev/null @@ -1,11 +0,0 @@ -module.exports = { - async up(knex) { - const hasColumn = await knex.schema.hasColumn('toys', 'image'); - - if (hasColumn) { - await knex.schema.table('toys', (table) => { - table.dropColumn('image'); - }); - } - } -}; diff --git a/services/strapi/database/migrations/2023-09-08-drop-vod-videosrc.js b/services/strapi/database/migrations/2023-09-08-drop-vod-videosrc.js deleted file mode 100644 index 8d1f878..0000000 --- a/services/strapi/database/migrations/2023-09-08-drop-vod-videosrc.js +++ /dev/null @@ -1,11 +0,0 @@ -module.exports = { - async up(knex) { - const hasColumn = await knex.schema.hasColumn('vods', 'video_src'); - - if (hasColumn) { - await knex.schema.table('vods', (table) => { - table.dropColumn('video_src'); - }); - } - } -}; diff --git a/services/strapi/database/migrations/2023-12-24-add-cuid-to-vods.js b/services/strapi/database/migrations/2023-12-24-add-cuid-to-vods.js deleted file mode 100644 index 5590b99..0000000 --- a/services/strapi/database/migrations/2023-12-24-add-cuid-to-vods.js +++ /dev/null @@ -1,31 +0,0 @@ - -const generateCuid = require('../../misc/generateCuid'); - -module.exports = { - async up(knex) { - - console.log(`MIGRATION-- 2023-12-24-add-cuid-to-vods.js`); - - // Check if the 'cuid' column already exists in the 'vods' table - const hasCuidColumn = await knex.schema.hasColumn('vods', 'cuid'); - - if (!hasCuidColumn) { - // Add the 'cuid' column to the 'vods' table - await knex.schema.table('vods', (table) => { - table.string('cuid'); - }); - } - - // Get all vods from the database - const vods = await knex.select('*').from('vods'); - - // For each vod, populate cuid if it's null or undefined - for (const [index, vod] of vods.entries()) { - if (!vod.cuid) { - await knex('vods').update({ cuid: generateCuid() }).where({ id: vod.id }); - } - } - - }, -}; - diff --git a/services/strapi/database/migrations/2023-12-26-add-cuid-to-streams.js b/services/strapi/database/migrations/2023-12-26-add-cuid-to-streams.js deleted file mode 100644 index 942854b..0000000 --- a/services/strapi/database/migrations/2023-12-26-add-cuid-to-streams.js +++ /dev/null @@ -1,33 +0,0 @@ - -const { init } = require('@paralleldrive/cuid2'); - -module.exports = { - async up(knex) { - - console.log(`MIGRATION-- 2023-12-26-add-cuid-to-streams.js`); - - // Check if the 'cuid' column already exists in the 'streams' table - const hasCuidColumn = await knex.schema.hasColumn('streams', 'cuid'); - - if (!hasCuidColumn) { - // Add the 'cuid' column to the 'streams' table - await knex.schema.table('streams', (table) => { - table.string('cuid'); - }); - } - - // Get all streams from the database - const streams = await knex.select('*').from('streams'); - - // For each stream, populate cuid if it's null or undefined - for (const [index, stream] of streams.entries()) { - if (!stream.cuid) { - const length = 10; - const genCuid = init({ length }); - await knex('streams').update({ cuid: genCuid() }).where({ id: stream.id }); - } - } - - }, -}; - diff --git a/services/strapi/database/migrations/2023-12-27-relate-vods-to-streams.js b/services/strapi/database/migrations/2023-12-27-relate-vods-to-streams.js deleted file mode 100644 index 703275d..0000000 --- a/services/strapi/database/migrations/2023-12-27-relate-vods-to-streams.js +++ /dev/null @@ -1,35 +0,0 @@ - -const { sub, add } = require('date-fns'); - - -module.exports = { - async up(knex) { - console.log(`MIGRATION-- 2023-12-27-relate-vods-to-streams.js`); - - // Get all VODs from the database - const vods = await knex.select('*').from('vods'); - - // For each VOD, associate it with the stream with the nearest date (if not already associated) - for (const [index, vod] of vods.entries()) { - const existingAssociation = await knex('vods_stream_links') - .where({ vod_id: vod.id }) - .first(); - - if (!existingAssociation) { - // get nearest stream within +/- 3 hours - const date2 = new Date(vod.date_2); - const startDate = sub(date2, { hours: 3 }) - const endDate = add(date2, { hours: 3 }); - console.log(`vod.id=${vod.id}, vod.date_2=${vod.date_2}, date2=${date2}, startDate=${startDate}, endDate=${endDate}`) - const stream = await knex('streams') - .whereBetween('date', [startDate, endDate]) - - await knex('vods_stream_links').insert({ - stream_id: stream.id, - vod_id: vod.id, - vod_order: 1, - }); - } - } - }, -}; diff --git a/services/strapi/database/migrations/2023.05.09-video-src-sanity.js.noexec b/services/strapi/database/migrations/2023.05.09-video-src-sanity.js.noexec deleted file mode 100644 index cda7f9d..0000000 --- a/services/strapi/database/migrations/2023.05.09-video-src-sanity.js.noexec +++ /dev/null @@ -1,26 +0,0 @@ - -const fetch = require('node-fetch') - -let problemUrls = [] - -async function checkUrl(url) { - const res = await fetch(url); - if (!res.ok || !res?.headers?.get('x-bz-file-name') || !res?.headers?.get('x-bz-file-id')) problemUrls.push(url) -} - - - -module.exports = { - async up(knex) { - - // Get all VODs from the database - const vods = await knex.select('*').from('vods'); - - // sanity check every B2 URL - for (const vod of vods) { - await checkUrl(vod.video_src) - } - - process.exit(5923423) - }, -}; \ No newline at end of file diff --git a/services/strapi/database/migrations/2023.05.11T12.32.00.convert-to-video-src-b2.js.noexec b/services/strapi/database/migrations/2023.05.11T12.32.00.convert-to-video-src-b2.js.noexec deleted file mode 100644 index 0ccaeb3..0000000 --- a/services/strapi/database/migrations/2023.05.11T12.32.00.convert-to-video-src-b2.js.noexec +++ /dev/null @@ -1,98 +0,0 @@ - -const fetch = require('node-fetch') - -// greets chatgpt -async function getFileDetailsFromUrl(url) { - const controller = new AbortController(); - const signal = controller.signal; - - const options = { - signal, - }; - - let retries = 10; - - while (retries) { - console.log(`fetching~~ ${url}`); - const timeoutId = setTimeout(() => { - console.log('fetch timed out, aborting...'); - controller.abort(); - }, 5000); - - try { - const res = await fetch(url, options); - - clearTimeout(timeoutId); - - console.log('finished fetch'); - if (!res.ok) throw new Error(`problem while getting file from url with url ${url}`); - if (!res?.headers?.get('x-bz-file-name')) throw new Error(`${url} did not have a x-bz-file-name in the response headers`); - if (!res?.headers?.get('x-bz-file-id')) throw new Error(`${url} did not have a x-bz-file-id in the response headers`); - - return { - key: res.headers.get('x-bz-file-name'), - url: url, - uploadId: res.headers.get('x-bz-file-id'), - }; - } catch (err) { - clearTimeout(timeoutId); - retries--; - - if (retries === 0) { - console.error(`Could not fetch file details from URL: ${url}.`); - throw err; - } - - console.warn(`Retrying fetch (${retries} attempts left)`); - } - } -} - - - - -module.exports = { - async up(knex) { - // You have full access to the Knex.js API with an already initialized connection to the database - - // Get all VODs from the database - const vods = await knex.select('*').from('vods'); - - - // Process each VOD - for (const vod of vods) { - - // courtesy timer - await new Promise((resolve) => setTimeout(resolve, 1000)) - - console.log(vod) - // Get the file details from the VOD's video source URL - if (vod?.video_src) { - try { - const fileDetails = await getFileDetailsFromUrl(vod.video_src); - - // Insert the B2 file into the database - const [file] = await knex('b2_files').insert({ - url: fileDetails.url, - key: fileDetails.key, - upload_id: fileDetails.uploadId, - }).returning('id'); - - console.log(file) - console.log(`attempting to insert vod_id:${vod.id}, b_2_file_id:${file.id} for videoSrcB2`) - - // Link the B2 file to the VOD - await knex('vods_video_src_b_2_links').insert({ - vod_id: vod.id, - b_2_file_id: file.id, - }); - } catch (e) { - console.error(e) - console.log(`there was an error so we are skipping vod ${vod.id}`) - } - } else { - console.log(`${vod.id} has no video_src. skipping.`) - } - } - }, -}; diff --git a/services/strapi/database/migrations/2023.05.14T00.42.00.000Z.migrate-tags-to-tag-vod-relations.js b/services/strapi/database/migrations/2023.05.14T00.42.00.000Z.migrate-tags-to-tag-vod-relations.js deleted file mode 100644 index 7c3e24a..0000000 --- a/services/strapi/database/migrations/2023.05.14T00.42.00.000Z.migrate-tags-to-tag-vod-relations.js +++ /dev/null @@ -1,43 +0,0 @@ - -// up until now, tags have been attached directly to each vod object. -// now, tags are not attached to vods. -// instead, tag-vod-relations are used to associate a tag with a vod - -// what we need to do in this migration is -// * create a new tag-vod-relation for each tag in each vod -// * delete tags field in vods - -module.exports = { - async up(knex) { - - console.log('2023.05.14 - migrate tags to tag_vod_relations') - - // get all tags_vods_links - // for each, create a tag-vod-relation - const tagsVodsLinks = await knex.select('*').from('tags_vods_links') - - for (const tvl of tagsVodsLinks) { - // Create a tag-vod-relation entry for each tag - const tvr = await knex('tag_vod_relations') - .insert({ - created_at: new Date(), - updated_at: new Date(), - creator_id: 1 - }) - .returning( - ['id'] - ) - - await knex('tag_vod_relations_tag_links').insert({ - tag_vod_relation_id: tvr[0].id, - tag_id: tvl.tag_id - }) - - await knex('tag_vod_relations_vod_links').insert({ - tag_vod_relation_id: tvr[0].id, - vod_id: tvl.vod_id - }) - } - - }, -}; \ No newline at end of file diff --git a/services/strapi/database/migrations/2023.05.15T02.44.00.000Z.drop-vod-tags.js b/services/strapi/database/migrations/2023.05.15T02.44.00.000Z.drop-vod-tags.js deleted file mode 100644 index 15fc208..0000000 --- a/services/strapi/database/migrations/2023.05.15T02.44.00.000Z.drop-vod-tags.js +++ /dev/null @@ -1,12 +0,0 @@ - -// previously, we tagged vods directly on the vod content-type -// now, we use tag-vod-relation to relate tags to vods. -// thus, we want to get rid of vod.tags -// and also tag.vods - -module.exports = { - async up(knex) { - console.log('2023.05.15 - drop tags_vods_links') - await knex.schema.dropTable('tags_vods_links') - } -} \ No newline at end of file diff --git a/services/strapi/database/migrations/2023.05.25-gimme-the-tags.js.noexec b/services/strapi/database/migrations/2023.05.25-gimme-the-tags.js.noexec deleted file mode 100644 index 65af409..0000000 --- a/services/strapi/database/migrations/2023.05.25-gimme-the-tags.js.noexec +++ /dev/null @@ -1,110 +0,0 @@ - -// const fetch = require('node-fetch') - -// // greets chatgpt -// async function getFileDetailsFromUrl(url) { -// const controller = new AbortController(); -// const signal = controller.signal; - -// const options = { -// signal, -// }; - -// let retries = 10; - -// while (retries) { -// console.log(`fetching ${url}`); -// const timeoutId = setTimeout(() => { -// console.log('fetch timed out, aborting...'); -// controller.abort(); -// }, 5000); - -// try { -// const res = await fetch(url, options); - -// clearTimeout(timeoutId); - -// console.log('finished fetch'); -// if (!res.ok) throw new Error(`problem while getting file from url with url ${url}`); -// if (!res?.headers?.get('x-bz-file-name')) throw new Error(`${url} did not have a x-bz-file-name in the response headers`); -// if (!res?.headers?.get('x-bz-file-id')) throw new Error(`${url} did not have a x-bz-file-id in the response headers`); - -// return { -// key: res.headers.get('x-bz-file-name'), -// url: url, -// uploadId: res.headers.get('x-bz-file-id'), -// }; -// } catch (err) { -// clearTimeout(timeoutId); -// retries--; - -// if (retries === 0) { -// console.error(`Could not fetch file details from URL: ${url}.`); -// throw err; -// } - -// console.warn(`Retrying fetch (${retries} attempts left)`); -// } -// } -// } - - - - -module.exports = { - async up(knex) { - // You have full access to the Knex.js API with an already initialized connection to the database - - - // we iterate through the local, non-strapi backup db first. - // get list of all tags - // for each tag - // * get list of related vods - // * create relation in Strapi - // * - - - - // Get all VODs from the database - const vods = await knex.select('*').from('vods'); - - - // Process each VOD - for (const vod of vods) { - - // courtesy timer - await new Promise((resolve) => setTimeout(resolve, 10)) - - // @todo - - console.log(vod) - // Get the file details from the VOD's video source URL - if (vod?.video_src) { - try { - const fileDetails = await getFileDetailsFromUrl(vod.video_src); - - // Insert the B2 file into the database - const [file] = await knex('b2_files').insert({ - url: fileDetails.url, - key: fileDetails.key, - upload_id: fileDetails.uploadId, - }).returning('id'); - - console.log(file) - console.log(`attempting to insert vod_id:${vod.id}, b_2_file_id:${file.id} for videoSrcB2`) - - // Link the B2 file to the VOD - await knex('vods_video_src_b_2_links').insert({ - vod_id: vod.id, - b_2_file_id: file.id, - }); - } catch (e) { - console.error(e) - console.log(`there was an error so we are skipping vod ${vod.id}`) - } - } else { - console.log(`${vod.id} has no video_src. skipping.`) - } - } - }, -}; \ No newline at end of file diff --git a/services/strapi/database/migrations/2023.05.25T20.44.00.000Z.get-the-og-tags.js b/services/strapi/database/migrations/2023.05.25T20.44.00.000Z.get-the-og-tags.js deleted file mode 100644 index a211ebb..0000000 --- a/services/strapi/database/migrations/2023.05.25T20.44.00.000Z.get-the-og-tags.js +++ /dev/null @@ -1,124 +0,0 @@ -'use strict' - - -require('dotenv').config() - -const { Client } = require('pg') -const fetch = require('node-fetch') -const _ = require('lodash'); -const ogVods = require('../og-tags.json') - - -// const slugify = require('slugify') - - -// function slugifyString (str) { -// return slugify(str, { -// replacement: '-', // replace spaces with replacement character, defaults to `-` -// remove: undefined, // remove characters that match regex, defaults to `undefined` -// lower: true, // convert to lower case, defaults to `false` -// strict: true, // strip special characters except replacement, defaults to `false` -// locale: 'en', // language code of the locale to use -// trim: true // trim leading and trailing replacement chars, defaults to `true` -// }) -// } - - -async function associateTagWithVodsInStrapi (tagId, vodsIds) { - const res = await fetch(`${process.env.STRAPI_URL}/api/tags/${tagId}`, { - method: 'PUT', - headers: { - 'authorization': `Bearer ${process.env.STRAPI_API_KEY}` - }, - data: { - vods: [vodsIds] - } - }) - const json = await res.json() - - - if (!res.ok) throw new Error(JSON.stringify(json)) -} - - - -async function associateVodWithTagsInStrapi (knex, vodId, tagsIds) { - console.log(`updating vodId:${vodId} with tagsIds:${tagsIds}`) - for (const tagId of tagsIds) { - // see if it exists already - const rows = await knex.select('*').from('tags_vods_links').where({ - 'vod_id': vodId, - 'tag_id': tagId - }) - if (rows.length === 0) { - await knex('tags_vods_links').insert({ - vod_id: vodId, - tag_id: tagId - }); - } - } -} - -async function getStrapiVodByAnnounceUrl (knex, announceUrl) { - const rows = await knex.select('*').from('vods').where('announce_url', announceUrl) - return (rows[0]) -} - - - - -async function getStrapiTagByName (knex, tag) { - const rows = await knex.select('*').from('tags').where({ 'name': tag }) - return rows[0] -} - - - - - -module.exports = { - async up(knex) { - // You have full access to the Knex.js API with an already initialized connection to the database - - for (const vod of ogVods) { - // get matching vod in strapi - console.log(vod) - if (vod.announceUrl) { - const strapiVod = await getStrapiVodByAnnounceUrl(knex, vod.announceUrl) - - if (strapiVod) { - // we've got a matching vod - - if (vod.tags) { - console.log(`source vod has tags: ${vod.tags}`) - - let strapiTagsIds = [] - - // for each tag, get the matching strapi tag ID - for (const tag of vod.tags) { - // lookup the strapi tag id - const strapiTag = await getStrapiTagByName(knex, tag) - if (!!strapiTag) { - strapiTagsIds.push(strapiTag.id) - } - } - - console.log(`we are adding the following strapiTagsIds to vod ID ${strapiVod.id}: ${strapiTagsIds}`) - - // create relations between matching vod and the tags - await associateVodWithTagsInStrapi(knex, strapiVod.id, strapiTagsIds) - - } - } - } - } - - // Get all VODs from the database - const vods = await knex.select('*').from('vods'); - - // Process each VOD - for (const vod of vods) { - - } - } -} diff --git a/services/strapi/database/migrations/2023.07.17.relate-vods-to-vtubers.js b/services/strapi/database/migrations/2023.07.17.relate-vods-to-vtubers.js deleted file mode 100644 index 5f035a1..0000000 --- a/services/strapi/database/migrations/2023.07.17.relate-vods-to-vtubers.js +++ /dev/null @@ -1,70 +0,0 @@ -module.exports = { - async up(knex) { - - console.log('Create vtubers table') - await knex.schema.createTable('vtubers', (table) => { - table.increments('id').primary(); - table.string('displayName').notNullable(); - table.string('chaturbate'); - table.string('twitter'); - table.string('patreon'); - table.string('twitch'); - table.string('tiktok'); - table.string('onlyfans'); - table.string('youtube'); - table.string('linktree'); - table.string('carrd'); - table.string('fansly'); - table.string('pornhub'); - table.string('discord'); - table.string('reddit'); - table.string('throne'); - table.string('instagram'); - table.string('facebook'); - table.string('merch'); - table.string('slug').notNullable(); - table.text('description1').notNullable(); - table.text('description2'); - table.string('image').notNullable(); - }); - - console.log('Create vods_vtuber_links table') - await knex.schema.createTable('vods_vtuber_links', (table) => { - table.increments('id').primary(); - table.integer('vod_id').unsigned().references('vods.id'); - table.integer('vtuber_id').unsigned().references('vtubers.id'); - table.integer('vod_order').notNullable(); - }); - - - console.log('Create a vtuber entry for ProjektMelody') - const projektMelody = { - displayName: 'ProjektMelody', - slug: 'projektmelody', // You can customize the slug based on your preference - description1: 'Description for ProjektMelody', // Add your vtuber's description here - image: 'http://futureporn-b2.b-cdn.net/futureporn/projekt-melody.jpg', // Replace with the image filename for ProjektMelody - }; - - console.log('Get all VODs from the database') - const vods = await knex.select('*').from('vods'); - - console.log('get projektmelody id') - // const [projektMelodyId] = await knex('vtubers').insert(projektMelody); - const projektMelodyId = 1 - - console.log(`projektmelodyId is : ${projektMelodyId}`) - - console.log(`For each VOD, associate ProjektMelody vtuber.`) - for (const [index, vod] of vods.entries()) { - console.log(`Check if vtuber_id exists in the vtubers table`) - const vtuber = await knex('vtubers').where('id', projektMelodyId).first(); - if (vtuber) { - await knex('vods_vtuber_links').insert({ - vtuber_id: projektMelodyId, - vod_id: vod.id, - vod_order: index + 1, // Auto-increment the vod_order number - }); - } - } - }, -}; diff --git a/services/strapi/database/migrations/2023.07.31.add-b2-file-cdnUrl.js b/services/strapi/database/migrations/2023.07.31.add-b2-file-cdnUrl.js deleted file mode 100644 index 308f159..0000000 --- a/services/strapi/database/migrations/2023.07.31.add-b2-file-cdnUrl.js +++ /dev/null @@ -1,18 +0,0 @@ -module.exports = { - async up(knex) { - // Add the 'cdn_url' column to the 'b2_files' table - await knex.schema.table('b2_files', (table) => { - table.string('cdn_url'); // Change the data type if needed (e.g., text, varchar, etc.) - }); - - // Get all B2 Files from the database - const files = await knex.select('*').from('b2_files'); - - // For each B2 File, create cdnUrl - for (const [index, file] of files.entries()) { - const key = file.key; - const cdnUrl = `https://futureporn-b2.b-cdn.net/futureporn/${key}`; - await knex('b2_files').update({ cdn_url: cdnUrl }).where({ id: file.id }); - } - }, -}; diff --git a/services/strapi/database/migrations/2024-01-08-add-streams.js.noexec b/services/strapi/database/migrations/2024-01-08-add-streams.js.noexec deleted file mode 100644 index d354107..0000000 --- a/services/strapi/database/migrations/2024-01-08-add-streams.js.noexec +++ /dev/null @@ -1,30 +0,0 @@ -module.exports = { - async up(knex) { - - await knex.schema.createTable('streams', (table) => { - table.increments('id').primary(); - table.string('date_str').notNullable(); - table.string('vods'); - table.string('vtuber'); - table.string('tweet'); - table.string('date'); - table.string('cuid'); - }); - - - // Add the 'cdn_url' column to the 'b2_files' table - await knex.schema.table('b2_files', (table) => { - table.string('cdn_url'); // Change the data type if needed (e.g., text, varchar, etc.) - }); - - // Get all B2 Files from the database - const files = await knex.select('*').from('b2_files'); - - // For each B2 File, create cdnUrl - for (const [index, file] of files.entries()) { - const key = file.key; - const cdnUrl = `https://futureporn-b2.b-cdn.net/futureporn/${key}`; - await knex('b2_files').update({ cdn_url: cdnUrl }).where({ id: file.id }); - } - }, -}; diff --git a/services/strapi/database/migrations/2024-01-14-add-date2-to-streams.js b/services/strapi/database/migrations/2024-01-14-add-date2-to-streams.js deleted file mode 100644 index 5cbd5cf..0000000 --- a/services/strapi/database/migrations/2024-01-14-add-date2-to-streams.js +++ /dev/null @@ -1,29 +0,0 @@ - -module.exports = { - async up(knex) { - - console.log(`MIGRATION-- 2024-01-14-add-date2-to-streams.js`); - - // Check if the 'date_2' column already exists in the 'streams' table - const hasColumn = await knex.schema.hasColumn('streams', 'date_2'); - - if (!hasColumn) { - console.log(`Adding the 'date_2' column to the 'streams' table`); - await knex.schema.table('streams', (table) => { - table.string('date_2'); - }); - } - - // Get all streams from the database - const streams = await knex.select('*').from('streams'); - - // For each stream, populate date_2 if it's null or undefined - for (const [index, stream] of streams.entries()) { - if (stream.date_2 === null && stream.date_str !== null) { - const result = await knex('streams').update({ date_2: stream.date_str }).where({ id: stream.id }); - } - } - - }, -}; - diff --git a/services/strapi/database/migrations/2024-01-15-add-platform-to-streams.js b/services/strapi/database/migrations/2024-01-15-add-platform-to-streams.js deleted file mode 100644 index a1632e9..0000000 --- a/services/strapi/database/migrations/2024-01-15-add-platform-to-streams.js +++ /dev/null @@ -1,49 +0,0 @@ - -module.exports = { - async up(knex) { - - console.log(`MIGRATION-- 2024-01-15-add-platform-streams.js`); - - // Check if the 'platform' column already exists in the 'streams' table - const hasColumn = await knex.schema.hasColumn('streams', 'platform'); - - if (!hasColumn) { - console.log(`Adding the 'platform' column to the 'streams' table`); - await knex.schema.table('streams', (table) => { - table.string('platform'); - }); - } - - // Get all streams from the database - const streams = await knex.select('*').from('streams'); - - // For each stream, populate platform based on the related tweet data - for (const [index, stream] of streams.entries()) { - - const tweetLink = await knex('streams_tweet_links') - .where({ stream_id: stream.id }) - .first(); - - if (tweetLink) { - console.log(tweetLink); - - const tweet = await knex('tweets') - .where({ id: tweetLink.tweet_id }) - .first(); - - console.log(tweet); - - if (!!tweet) { - console.log(`stream ${stream.id} tweet tweet.is_chaturbate_invite=${tweet.is_chaturbate_invite}, tweet.is_fansly_invite=${tweet.is_fansly_invite}`); - await knex('streams').update({ - is_chaturbate_stream: !!tweet.is_chaturbate_invite, - is_fansly_stream: !!tweet.is_fansly_invite - }).where({ id: stream.id }); - } - } - - } - - }, -}; - diff --git a/services/strapi/database/migrations/2024-07-04-add-fansly-id-to-vtuber.js b/services/strapi/database/migrations/2024-07-04-add-fansly-id-to-vtuber.js deleted file mode 100644 index 3314815..0000000 --- a/services/strapi/database/migrations/2024-07-04-add-fansly-id-to-vtuber.js +++ /dev/null @@ -1,28 +0,0 @@ -module.exports = { - async up(knex) { - - console.log(`MIGRATION-- 2024-07-04-add-fansly-id-to-vtuber.js`); - - // Check if the 'fansly_id' column already exists in the 'vtubers' table - const hasColumn = await knex.schema.hasColumn('vtubers', 'fansly_id'); - - if (!hasColumn) { - console.log(`Adding the 'fansly_id' column to the 'vtubers' table`); - await knex.schema.table('vtubers', (table) => { - table.string('fansly_id'); - }); - } - - // Get all vtubers from the database - const vtubers = await knex.select('*').from('vtubers'); - - // For each vtuber, populate fansly_id if it's null or undefined - for (const [_, vtuber] of vtubers.entries()) { - if (!vtuber.fansly_id) { - await knex('vtubers').update({ fansly_id: '' }).where({ id: vtuber.id }); - } - } - - }, -}; - diff --git a/services/strapi/database/og-tags.json b/services/strapi/database/og-tags.json deleted file mode 100644 index 7d838af..0000000 --- a/services/strapi/database/og-tags.json +++ /dev/null @@ -1,1009 +0,0 @@ -[{ - "tags": null, - "date": "2020-02-09T18:14:25.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1226570058073362438" -}, { - "tags": null, - "date": "2020-02-10T02:15:55.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1226691233659183104" -}, { - "tags": null, - "date": "2020-02-12T01:11:48.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1227399871734468610" -}, { - "tags": null, - "date": "2020-02-12T17:14:12.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1227642067733618691" -}, { - "tags": null, - "date": "2020-02-13T02:09:11.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1227776703415693312" -}, { - "tags": null, - "date": "2020-02-15T02:13:56.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1228502672246005760" -}, { - "tags": null, - "date": "2020-02-18T21:17:28.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1229877618067025923" -}, { - "tags": null, - "date": "2020-02-20T01:06:47.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1230297714719674368" -}, { - "tags": null, - "date": "2020-02-22T02:13:38.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1231039313300901889" -}, { - "tags": null, - "date": "2023-02-19T00:15:09.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1627099434508075008" -}, { - "tags": null, - "date": "2020-02-09T01:40:10.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1226319848411717632" -}, { - "tags": null, - "date": "2020-02-21T16:25:47.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1230891377422192641" -}, { - "tags": null, - "date": "2020-03-03T21:06:17.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1234948231311876096" -}, { - "tags": null, - "date": "2020-03-05T01:18:07.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1235373996906078208" -}, { - "tags": null, - "date": "2020-03-07T02:40:52.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1236119594995245057" -}, { - "tags": null, - "date": "2020-03-12T00:01:06.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1237891327922016258" -}, { - "tags": null, - "date": "2020-03-15T20:22:40.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1239285911201988609" -}, { - "tags": null, - "date": "2020-03-17T19:55:46.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1240003917355921412" -}, { - "tags": null, - "date": "2020-03-22T00:11:12.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1241517748276154371" -}, { - "tags": null, - "date": "2020-03-26T00:14:56.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1242968240894234627" -}, { - "tags": null, - "date": "2020-03-01T21:06:17.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1234223457723961344" -}, { - "tags": ["anal", "deep throat", "vore", "stuck in wall", "choking", "puddle", "squirting"], - "date": "2020-02-08T16:12:42.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1226177041109786625" -}, { - "tags": null, - "date": "2020-02-28T16:38:43.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1233431346112061441" -}, { - "tags": null, - "date": "2020-04-14T19:57:03.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1250151101174816769" -}, { - "tags": null, - "date": "2020-04-24T00:55:16.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1253487639430410240" -}, { - "tags": null, - "date": "2020-04-25T23:05:42.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1254184839684726786" -}, { - "tags": null, - "date": "2020-04-26T22:54:50.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1254544493849739264" -}, { - "tags": null, - "date": "2020-05-09T21:59:16.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1259241552016216065" -}, { - "tags": null, - "date": "2020-05-12T19:01:53.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1260284077011238912" -}, { - "tags": null, - "date": "2020-06-27T00:06:39.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1276668227767144452" -}, { - "tags": null, - "date": "2020-02-17T01:05:40.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1229210268447715328" -}, { - "tags": null, - "date": "2020-04-05T22:55:16.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1246934456927215617" -}, { - "tags": null, - "date": "2020-05-05T19:00:56.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1257747123682119680" -}, { - "tags": null, - "date": "2020-05-28T19:57:30.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1266096281002356736" -}, { - "tags": null, - "date": "2020-05-29T23:02:07.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1266505128426815488" -}, { - "tags": null, - "date": "2020-06-01T00:13:53.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1267247965464297473" -}, { - "tags": null, - "date": "2020-06-06T00:02:11.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1269056957295583240" -}, { - "tags": null, - "date": "2020-06-16T18:55:04.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1272965936685953024" -}, { - "tags": null, - "date": "2020-06-18T23:37:57.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1273761903664336897" -}, { - "tags": null, - "date": "2020-06-21T00:09:09.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1274494531312984064" -}, { - "tags": null, - "date": "2020-06-23T20:00:12.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1275519042955153409" -}, { - "tags": null, - "date": "2020-02-27T01:10:55.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1232835470712803328" -}, { - "tags": null, - "date": "2020-05-23T00:02:55.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1263983714893279233" -}, { - "tags": null, - "date": "2020-07-15T00:25:49.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1283196031648358400" -}, { - "tags": null, - "date": "2020-07-18T01:14:51.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1284295535751499776" -}, { - "tags": null, - "date": "2020-07-25T00:00:05.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1286813436119220225" -}, { - "tags": null, - "date": "2020-07-29T20:03:40.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1288565878502699008" -}, { - "tags": null, - "date": "2020-08-01T00:00:00.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1289350128436838400" -}, { - "tags": null, - "date": "2020-08-05T19:52:40.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1291099825396748292" -}, { - "tags": null, - "date": "2020-08-14T22:21:59.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1294398894466629634" -}, { - "tags": null, - "date": "2020-08-19T21:00:11.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1296190247957659649" -}, { - "tags": null, - "date": "2020-03-14T01:38:59.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1238640737668997120" -}, { - "tags": null, - "date": "2020-07-10T23:59:34.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1281739875319009280" -}, { - "tags": null, - "date": "2020-09-11T18:02:31.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1304480454242861056" -}, { - "tags": null, - "date": "2020-09-13T23:00:49.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1305280300130406402" -}, { - "tags": null, - "date": "2020-09-18T22:55:13.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1307090831011205120" -}, { - "tags": null, - "date": "2020-09-28T20:14:36.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1310674290744995843" -}, { - "tags": null, - "date": "2020-10-02T23:48:02.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1312177554824990720" -}, { - "tags": null, - "date": "2020-10-06T22:59:13.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1313614818713231360" -}, { - "tags": null, - "date": "2020-10-09T18:01:05.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1314626953803161601" -}, { - "tags": null, - "date": "2020-04-02T00:11:39.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1245504130753404931" -}, { - "tags": null, - "date": "2020-08-23T22:59:21.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1297669785883484160" -}, { - "tags": null, - "date": "2020-09-20T20:00:12.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1307771561651441665" -}, { - "tags": null, - "date": "2020-10-31T23:03:57.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1322675709379334146" -}, { - "tags": null, - "date": "2020-11-18T21:28:43.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1329174724790915075" -}, { - "tags": null, - "date": "2020-11-23T23:56:04.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1331023742693232646" -}, { - "tags": null, - "date": "2020-11-26T22:03:19.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1332082533526360066" -}, { - "tags": null, - "date": "2020-12-04T21:18:42.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1334970406025965571" -}, { - "tags": null, - "date": "2020-12-07T23:56:41.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1336097330483916800" -}, { - "tags": null, - "date": "2020-04-09T00:06:24.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1248039525215080449" -}, { - "tags": null, - "date": "2020-10-18T18:59:38.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1317903180148166657" -}, { - "tags": null, - "date": "2020-10-29T19:58:10.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1321904176637595648" -}, { - "tags": null, - "date": "2021-01-05T23:07:36.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1346594227233427459" -}, { - "tags": null, - "date": "2021-01-08T23:03:45.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1347680421920862209" -}, { - "tags": null, - "date": "2021-01-12T23:03:22.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1349129873068068873" -}, { - "tags": null, - "date": "2021-01-16T01:01:22.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1350246732391706624" -}, { - "tags": null, - "date": "2021-01-22T23:02:24.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1352753509045776384" -}, { - "tags": null, - "date": "2021-01-26T23:02:21.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1354203048491544581" -}, { - "tags": null, - "date": "2021-02-03T23:00:44.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1357101745961328645" -}, { - "tags": null, - "date": "2020-12-21T21:01:24.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1341126649329410050" -}, { - "tags": null, - "date": "2020-12-28T23:20:13.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1343698295563153408" -}, { - "tags": null, - "date": "2021-02-14T21:13:33.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1361061038620364812" -}, { - "tags": null, - "date": "2021-02-18T00:01:18.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1362190417631076352" -}, { - "tags": null, - "date": "2021-02-20T00:59:31.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1362929843759943681" -}, { - "tags": null, - "date": "2021-02-24T23:01:00.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1364711958105194498" -}, { - "tags": null, - "date": "2021-03-02T23:07:27.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1366887909241012233" -}, { - "tags": null, - "date": "2021-03-12T22:53:59.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1370508398081540098" -}, { - "tags": null, - "date": "2021-03-13T20:58:44.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1370841782506901516" -}, { - "tags": null, - "date": "2021-03-20T21:54:09.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1373392441257103368" -}, { - "tags": null, - "date": "2021-03-22T22:05:15.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1374120011854397442" -}, { - "tags": null, - "date": "2021-02-10T23:26:55.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1359645050792206337" -}, { - "tags": null, - "date": "2021-02-25T00:00:00.000Z", - "announceUrl": null -}, { - "tags": null, - "date": "2021-04-03T01:26:21.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1378156887884005378" -}, { - "tags": null, - "date": "2021-04-03T19:55:31.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1378436020081950726" -}, { - "tags": null, - "date": "2021-04-12T23:11:23.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1381746799506030593" -}, { - "tags": null, - "date": "2021-04-24T17:31:05.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1386009816443805699" -}, { - "tags": null, - "date": "2021-05-01T16:59:01.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1388538459816595463" -}, { - "tags": null, - "date": "2021-05-14T15:53:36.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1393233038298001411" -}, { - "tags": null, - "date": "2021-05-21T22:59:03.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1395876824865988611" -}, { - "tags": null, - "date": "2021-05-25T22:04:21.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1397312608454258690" -}, { - "tags": null, - "date": "2021-06-05T22:02:00.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1401298283499298817" -}, { - "tags": null, - "date": "2021-06-12T22:56:31.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1403848719695482888" -}, { - "tags": null, - "date": "2022-01-09T23:32:49.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1480321695315640328" -}, { - "tags": null, - "date": "2021-06-26T20:08:24.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1408879841315860486" -}, { - "tags": null, - "date": "2021-07-03T22:04:50.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1411445858244169728" -}, { - "tags": null, - "date": "2021-07-07T19:04:50.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1412850109608566786" -}, { - "tags": null, - "date": "2021-07-13T19:52:08.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1415036340580884482" -}, { - "tags": null, - "date": "2021-07-17T20:05:10.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1416489172311887876" -}, { - "tags": null, - "date": "2021-08-04T23:11:38.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1423059078381719564" -}, { - "tags": null, - "date": "2021-08-08T20:26:41.000Z", - "announceUrl": "http://twitter.com/ProjektMelody/status/1424467119006261249" -}, { - "tags": null, - "date": "2021-08-12T19:57:20.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1425909284940955648" -}, { - "tags": null, - "date": "2021-08-17T23:16:50.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1427771428833738755" -}, { - "tags": null, - "date": "2021-08-21T20:13:36.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1429174868935856133" -}, { - "tags": null, - "date": "2022-05-13T19:19:42.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1525194090299670528" -}, { - "tags": null, - "date": "2021-09-09T23:01:37.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1436102522201116673" -}, { - "tags": null, - "date": "2021-09-12T22:03:47.000Z", - "announceUrl": "http://twitter.com/ProjektMelody/status/1437175131223302146" -}, { - "tags": null, - "date": "2021-09-19T19:47:34.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1439677566739259395" -}, { - "tags": null, - "date": "2021-09-21T21:56:01.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1440434669305012228" -}, { - "tags": null, - "date": "2021-09-29T18:55:41.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1443288390007508995" -}, { - "tags": null, - "date": "2021-10-09T20:09:55.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1446930950332506117" -}, { - "tags": null, - "date": "2021-10-11T22:06:20.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1447685023059128320" -}, { - "tags": null, - "date": "2021-10-23T20:18:46.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1452006607626412038" -}, { - "tags": null, - "date": "2021-11-04T22:03:43.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1456381673235492878" -}, { - "tags": null, - "date": "2023-04-04T23:43:09.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1643398832485482497" -}, { - "tags": null, - "date": "2022-09-07T23:13:00.000Z", - "announceUrl": null -}, { - "tags": null, - "date": "2021-11-12T00:08:21.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1458949749986443268" -}, { - "tags": null, - "date": "2021-11-19T20:01:27.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1461786722337972235" -}, { - "tags": null, - "date": "2021-12-06T20:10:31.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1467949594252398599" -}, { - "tags": null, - "date": "2021-12-12T23:12:01.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1470169597546479617" -}, { - "tags": null, - "date": "2021-12-15T23:04:14.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1471254803451887625" -}, { - "tags": null, - "date": "2021-12-17T23:05:54.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1471979997426589699" -}, { - "tags": null, - "date": "2021-12-25T22:34:35.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1474871219610603526" -}, { - "tags": null, - "date": "2021-12-30T23:17:45.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1476694022408261640" -}, { - "tags": null, - "date": "2022-01-03T23:10:33.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1478141764741611527" -}, { - "tags": null, - "date": "2022-01-13T00:03:21.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1481416541493153795" -}, { - "tags": null, - "date": "2022-01-22T22:59:59.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1485024472105164805" -}, { - "tags": null, - "date": "2022-01-26T01:57:25.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1486156290585882626" -}, { - "tags": null, - "date": "2022-02-02T21:48:11.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1488992672022945793" -}, { - "tags": null, - "date": "2022-02-16T23:00:32.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1494084309958987782" -}, { - "tags": null, - "date": "2022-02-18T20:01:16.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1494763968535179274" -}, { - "tags": null, - "date": "2022-03-10T23:02:20.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1502057294246424600" -}, { - "tags": null, - "date": "2022-03-18T01:35:11.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1504632475804446721" -}, { - "tags": null, - "date": "2022-04-06T00:02:17.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1511494467097870340" -}, { - "tags": null, - "date": "2022-04-10T00:12:41.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1512946635655065602" -}, { - "tags": null, - "date": "2022-04-16T23:09:42.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1515467499642531845" -}, { - "tags": null, - "date": "2022-04-22T23:14:35.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1517643056228487168" -}, { - "tags": null, - "date": "2022-05-06T22:59:23.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1522712662349656064" -}, { - "tags": null, - "date": "2022-05-08T22:51:40.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1523435492657664000" -}, { - "tags": null, - "date": "2022-05-19T00:44:58.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1527087885693878275" -}, { - "tags": null, - "date": "2022-05-25T02:06:07.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1529282633762492417" -}, { - "tags": null, - "date": "2022-06-05T23:30:37.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1533592158179139585" -}, { - "tags": null, - "date": "2022-06-09T21:10:03.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1535006331119222784" -}, { - "tags": null, - "date": "2022-06-22T23:46:31.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1539756753147117568" -}, { - "tags": null, - "date": "2022-07-19T03:48:05.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1549239628418490368" -}, { - "tags": null, - "date": "2022-08-03T23:32:24.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1554973488871530501" -}, { - "tags": null, - "date": "2022-08-05T23:15:33.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1555694026376060966" -}, { - "tags": null, - "date": "2022-08-11T20:31:00.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1557827065101127680#m" -}, { - "tags": null, - "date": "2022-08-23T23:28:21.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1562220295683014657" -}, { - "tags": null, - "date": "2022-09-10T21:03:36.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1568706781806821377" -}, { - "tags": null, - "date": "2022-06-18T23:07:00.000Z", - "announceUrl": null -}, { - "tags": null, - "date": "2022-09-13T23:26:31.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1569829911057031170" -}, { - "tags": null, - "date": "2022-09-21T23:38:26.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1572732011529506817" -}, { - "tags": null, - "date": "2022-10-02T02:05:22.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1576392868201365504" -}, { - "tags": null, - "date": "2022-10-04T23:15:33.000Z", - "announceUrl": "http://twitter.com/ProjektMelody/status/1577437242716741632" -}, { - "tags": null, - "date": "2022-10-13T02:22:45.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1580383510908399616" -}, { - "tags": null, - "date": "2022-10-15T00:31:27.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1581080275546537984" -}, { - "tags": ["squirting", "toy", "lovense-lush", "fantasy-for-her-suction", "crave-vesper-necklace", "womanizer-duo", "twitching", "licking", "unboxing", "suction", "review", "orgasm", "tongue", "vibrator", "cum-eating", "pussy-milking", "torture", "rope", "restraint", "bondage", "chains", "dildo", "deep-breathing"], - "date": "2022-10-20T23:21:15.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1583236936445353985" -}, { - "tags": ["jumpsuit", "3d", "new-outfit", "pole-dancing", "lovense-lush", "lovense-hyfy", "blowjob", "bottomless", "ass", "dancing", "pool", "precum", "pussy"], - "date": "2022-11-04T20:29:53.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1588629627907473409" -}, { - "tags": ["video-game", "hentai", "eroge", "daraku-gear", "mobile-game", "buffering", "sponsored", "lovense-lush", "storytime", "hentai", "games", "gacha", "waifu", "womb-tattoo", "bondage", "mind-break", "squirting", "nipple-clamps", "handcuffs", "suction-cups", "prisoner", "slave", "big-brother", "master", "hitachi-magic-wand", "double-penetration", "confusion", "cock-block", "crying", "edging", "cum", "pasties", "robocock", "milking", "hose", "self-censorship", "threesome", "blowjob", "spanking", "cucking", "feet", "sex-training"], - "date": "2022-12-14T00:01:45.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1602816075304935425" -}, { - "tags": ["japan", "monokini", "outfit", "game-of-thrones", "gundam", "growling", "grinding", "flooring", "pool", "lovense-lush", "heavy-breathing", "moaning", "naked", "edging", "womb-tattoo", "pelvic-thrusting", "missionary", "fingering", "cum-drunk", "laughing", "c2c", "self-censorship", "sex-ed", "sex-stories", "glory-hole", "texting", "ass", "2-cooms", "orgasm", "selfie-cam", "non-euclidian", "silly-dancing"], - "date": "2022-12-16T00:30:34.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1603548101561696261" -}, { - "tags": ["silly-music", "guest-dj", "lovense-lush", "3d", "original-outfit", "step-brother", "buffering", "screen-recording", "coffee", "anime-discussion", "monday", "topless", "pasties", "just-chatting", "moaning", "countdown", "orgasm", "hands-free-cum", "yandere", "all-fours", "malfunctioning-hand", "bed", "high-heels", "boobs", "twerking", "factoid", "cowgirl", "unzip", "belly", "femdom", "gamer-chair", "selfie-camera", "pussy-licking", "pussy-closeup", "fingering", "aftershocks", "dakimakura-discussion", "2000s-music", "ghost-in-the-shell", "lgbtq", "spread-legs", "feet", "footjob", "tit-belt"], - "date": "2020-05-17T00:07:30.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1261810539442946049" -}, { - "tags": null, - "date": "2020-05-20T01:16:56.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1262915175037927425" -}, { - "tags": null, - "date": "2020-07-05T00:02:06.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1279566184690659330" -}, { - "tags": ["just-chatting", "chastity", "recovery", "live-2d", "fetish-research", "house-party", "video-game"], - "date": "2023-01-05T20:58:29.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1611104872484556805" -}, { - "tags": ["singing", "christmas", "karaoke", "just-chatting", "holiday", "lovense-lush", "tomboy", "live2d", "debut", "pegging", "asmr", "strapon-dildo", "erotic-roleplay", "edging"], - "date": "2023-01-13T18:21:13.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1613964398506147853" -}, { - "tags": ["just-chatting", "twintails", "g-string", "swimming-pool", "lovense-lush", "3d", "leap-motion", "orgasm-denial", "moaning", "bad-audio", "toilet", "bidet", "succubus-lingerie", "womb-tattoo", "dressup", "tv-discussion", "states-discussion", "naked", "experimental-lighting", "handjob", "asmr-discussion", "tease"], - "date": "2023-01-20T00:37:06.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1616233319439515648" -}, { - "tags": null, - "date": "2020-07-06T20:03:01.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1280230795085578247" -}, { - "tags": null, - "date": "2020-07-07T20:59:26.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1280607381605449729" -}, { - "tags": null, - "date": "2020-08-22T00:03:48.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1296961229475512323" -}, { - "tags": ["asmr", "3dio", "dominatrix", "submission", "breeding", "ai-dungeon", "tentacles", "anal", "fantasy-writing", "moaning", "cum-ban", "heavy-breathing", "orgasm", "denial", "cum-drunk"], - "date": "2023-01-21T20:36:06.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1616897448890273807" -}, { - "tags": null, - "date": "2020-09-05T23:08:12.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1302383058247614465" -}, { - "tags": null, - "date": "2021-07-04T18:50:39.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1411759377816690696" -}, { - "tags": ["bepis", "pasties", "recovery", "hentai-discussion", "topless", "boobs", "live2d", "handjob", "lovense-lush", "guilty-fap", "train-molestation", "sex-dungeon", "bdsm", "santa", "moaning", "mel-noises", "rickroll", "cucked", "cbat", "punishment2-cum", "orgasm", "milking-device", "cumdrunk", "panting", "heat", "feral", "deep-breathing", "monster-girl", "roleplay", "fucking", "gratitude", "snack", "meltys-quest", "eroge", "fat-bastard", "voiceover", "girl-on-girl", "thirsty", "slut", "good-audio", "love"], - "date": "2023-02-17T02:10:04.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1626403575642353665" -}, { - "tags": ["social-media", "aspirations", "first-time", "lovense-lush", "bad-audio", "ahegao", "3d", "lonely", "friends", "dab", "tag-discussion", "deep-breathing", "jerk-off-gesture", "gratitude", "cat-girls", "plexstorm", "hentai-game", "cute-pussy", "origin", "moaning", "twerking", "dancing", "shaking", "strip-tease", "anime-discussion", "video-game-discussion", "ass", "pasties", "high-heels", "harness", "thong", "leggings", "technical-difficulties", "panic-attack", "meditation", "just-chatting", "masturbation", "nipples"], - "date": "2020-02-07T23:21:48.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1225922638687752192" -}, { - "tags": null, - "date": "2020-10-16T23:32:50.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1317247159478136832" -}, { - "tags": ["bad-audio", "new-outfit", "anniversary", "bunny-outfit", "3d", "sexmachine", "raul", "recovery", "bunny-suit", "sit-on-face", "naked", "jacket", "mel-noises", "rickroll", "frickenator", "standing-cum", "legwarmers", "bunny-ears", "ass", "leg-shaking", "glitch", "doggy-style", "breeding", "dirty-talk", "cum-drunk", "trouble-walking", "slut", "food", "masturbation", "dick-sucking", "t1m", "lovense-lush", "pool", "flooring"], - "date": "2023-02-12T23:07:09.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1624907994671489026" -}, { - "tags": ["mel-noises", "crying", "good-audio", "live2d", "moaning", "dildo", "masturbation", "glass-dildo", "teaser", "sex-stories", "horny", "script-writing", "joi", "topless", "pasties", "naked", "asmr", "just-chatting", "self-care", "deep-breathing", "guided-meditation", "tantric-sex", "slut", "orgasm", "hentai-game", "caring"], - "date": "2023-02-09T00:25:47.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1623478232036122624" -}, { - "tags": ["just-chatting", "community", "anxiety", "dork", "live2d", "nut-between-worlds", "pro-social-behavior", "topless", "pasties", "harness", "boobs", "moaning", "blowjob"], - "date": "2023-02-13T03:07:00.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1624907994671489026" -}, { - "tags": null, - "date": "2020-03-01T01:14:02.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1233923415964340225" -}, { - "tags": null, - "date": "2020-03-28T01:42:21.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1243715015829585925" -}, { - "tags": null, - "date": "2020-04-01T00:13:26.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1245142190952796167" -}, { - "tags": null, - "date": "2020-05-02T22:01:33.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1256705414663725058" -}, { - "tags": null, - "date": "2020-06-10T18:58:26.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1270792457941368832" -}, { - "tags": null, - "date": "2020-06-28T00:10:07.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1277031489205678083" -}, { - "tags": null, - "date": "2020-02-25T21:31:41.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1232417908678168576" -}, { - "tags": null, - "date": "2020-03-10T22:00:51.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1237498680690855939" -}, { - "tags": null, - "date": "2020-03-20T15:41:36.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1241027117284044801" -}, { - "tags": null, - "date": "2020-08-28T23:03:16.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1299482712869306368" -}, { - "tags": null, - "date": "2020-09-24T23:25:58.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1309272896456527872" -}, { - "tags": null, - "date": "2020-10-13T23:03:14.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1316152545732157448" -}, { - "tags": null, - "date": "2020-10-23T18:55:31.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1319714083462107139" -}, { - "tags": ["just-chatting", "lovense-lush", "mel-noises", "pro-social-behavior", "topless", "g-string", "thong", "high-heels", "selfie-camera", "ass", "porn-discussion", "panties", "no-nut-november", "moaning", "feet", "feet-licking", "vtuber-discussion", "gamer-chair", "naked", "stretching", "flexibility", "orgasm", "masturbation"], - "date": "2020-11-09T00:55:59.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1325603004695842816" -}, { - "tags": null, - "date": "2020-12-14T21:46:27.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1338601272354758656" -}, { - "tags": null, - "date": "2020-12-24T23:02:13.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1342244216731275265" -}, { - "tags": null, - "date": "2020-08-11T23:01:22.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1293321639728492544" -}, { - "tags": null, - "date": "2020-10-11T22:05:30.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1315413239556190211" -}, { - "tags": null, - "date": "2021-03-03T23:03:48.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1367249376675041285" -}, { - "tags": null, - "date": "2021-04-18T00:30:03.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1383578535340634120" -}, { - "tags": null, - "date": "2021-04-23T23:01:52.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1385730672661303296" -}, { - "tags": null, - "date": "2021-06-06T20:55:39.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1401643974922838019" -}, { - "tags": null, - "date": "2021-06-15T22:57:44.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1404936189946236930" -}, { - "tags": null, - "date": "2021-07-27T23:04:15.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1420158120370872320" -}, { - "tags": null, - "date": "2021-08-30T23:08:30.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1432480377302552587" -}, { - "tags": null, - "date": "2021-09-03T21:59:09.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1433912474542264321" -}, { - "tags": null, - "date": "2021-02-05T22:57:21.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1357825669024120841" -}, { - "tags": null, - "date": "2021-10-04T22:20:50.000Z", - "announceUrl": "http://twitter.com/ProjektMelody/status/1445151953764458501" -}, { - "tags": null, - "date": "2021-11-16T23:07:06.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1460746277717782528" -}, { - "tags": null, - "date": "2021-12-22T23:58:43.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1473805228537438209" -}, { - "tags": null, - "date": "2022-01-15T20:53:35.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1482455948459450369" -}, { - "tags": null, - "date": "2022-01-28T22:13:21.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1487187063048921093" -}, { - "tags": null, - "date": "2022-02-13T21:17:23.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1492971187047374849" -}, { - "tags": null, - "date": "2022-04-13T20:10:50.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1514335322527649798" -}, { - "tags": null, - "date": "2022-05-14T23:02:18.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1525612497935507457" -}, { - "tags": null, - "date": "2022-05-26T18:31:28.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1529892992818876417" -}, { - "tags": null, - "date": "2022-08-01T19:28:31.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1554187338804338696" -}, { - "tags": null, - "date": "2022-09-02T20:06:16.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1565793017960828931" -}, { - "tags": null, - "date": "2022-09-23T23:38:18.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1573456755236954114" -}, { - "tags": ["memes", "anime", "bdsm", "orgasm", "moaning", "laughing", "cum-drunk", "lovense-lush"], - "date": "2022-10-27T23:20:05.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1585773359379677184" -}, { - "tags": ["lovense-hyphy", "lovense-lush", "whispering", "asmr", "sportscar", "pee", "sick", "coughing", "shaking", "cum", "orgasm", "brainfrog", "oral", "sucking", "sex-toy", "cum-drunk", "moaning", "twitching", "thrusting", "grinding", "objectification", "flooring", "keyboard-clicking", "mocap-failure", "muted", "charades", "posing", "riding", "dry-humping", "ass", "high-heels", "legs", "hip-sway", "plug-suit", "chest-harness", "naked", "womb-tattoo", "sexy-dance", "slav-squat", "tail", "pole-dance", "clipping", "licking", "glowing", "dab"], - "date": "2022-11-12T00:13:40.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1591222662487236609" -}, { - "tags": ["eroge", "video-game", "orc", "moaning", "cyber", "cum-drunk", "orgasm", "towel", "petting", "massage", "accupressure", "lingerie", "ass", "thong", "pasties", "adhesive-bandage", "monster-girls", "texting", "lovense-lush", "lovense-hyphy", "broken-toy", "tail", "live2d", "food-porn", "self-care", "aromatherapy", "monster-dick", "hot-sluts", "monster", "fantasy", "cock-sleeve", "feet", "footjob", "fingering", "elf", "cum-whore", "panties", "cum"], - "date": "2022-11-19T22:35:57.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1594097172689166337" -}, { - "tags": ["ass", "tail", "lovense-lush", "conversation", "grinding", "flooring", "going-away", "humping", "laughing", "eating", "harness", "boobs", "toes", "cum-drunk", "aftershocks", "dancing", "pole-dancing", "moaning", "feet", "stepping", "cock-abuse", "extreme-closeup", "butthole", "panties", "shaking", "pussy-licking", "singing", "hentai", "carnival", "fingering", "wedgie", "fetishwear", "watch-along", "pasties", "creampie", "rude-sex", "cumming", "g-string", "idol", "voyeurism", "condom", "gay", "spitting", "pervert", "edging", "mutual-masturbation", "sniffing"], - "date": "2022-11-22T23:04:43.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1595191577068044289" -}, { - "tags": ["hentai", "eroge", "dohna-dohna", "orgasm-denial", "chastity", "video-game", "live2d", "ovulation", "feet", "horny", "dream", "vampire", "just-chatting", "numi", "game-of-thrones", "recovery", "bad-audio", "yandere", "storytime", "spontaneous-song", "crushes"], - "date": "2023-01-01T22:03:49.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1609671764514164737" -}, { - "tags": null, - "date": "2021-09-07T22:01:24.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1435362590893527040" -}, { - "tags": null, - "date": "2021-09-16T20:00:39.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1438593697445302274" -}, { - "tags": null, - "date": "2022-02-07T23:27:14.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1490829536316985348" -}, { - "tags": ["lawnmower", "motorbunny", "bad-audio", "3d", "lovense-lush", "pussy-to-mouth", "screaming", "swimming-pool", "moaning", "deep-breathing", "cock-riding", "mel-noises", "mind-break", "technical-difficulties", "yelling", "echo", "reverse-cowgirl", "muffled-screams", "orgasm", "no-face-tracking", "swearing", "chaturbate-compliance", "spit", "messy", "cum-drinking", "clit-milking-device", "multiple-orgasms", "4-cum", "squirting", "loud-orgasm", "cum-chalice", "schlorp", "sksksksk"], - "date": "2023-01-25T00:19:39.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1618040869587947521" -}, { - "tags": ["lovense-lush", "lovense-hush", "tail", "butt-plug", "double-penetration", "moaning", "blushing", "manual-masturbation", "embarrasment", "orgasm", "just-chatting", "shorts", "stripping", "jiggle-physics", "3d", "boobs", "pasties", "dancing", "swearing", "vtuber-discussion", "t-pose", "ass", "anime-discussion", "selfie-camera", "edging", "jojo-posing", "anal-fingering", "fingering"], - "date": "2020-11-18T01:59:20.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1328880436551049217" -}, { - "tags": null, - "date": "2021-01-03T23:01:53.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1345868009483735042" -}, { - "tags": null, - "date": "2021-01-10T19:57:37.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1348358353676873729" -}, { - "tags": null, - "date": "2021-02-07T23:04:03.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1358552132295798787" -}, { - "tags": ["boobs", "porn-game", "video-game", "bondage", "mobile-game", "android", "moaning", "bikini", "blowjob", "doggy-style", "3d", "sponsored"], - "date": "2022-11-18T01:03:45.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1593409592477638656" -}, { - "tags": ["lovense-lush", "spanking", "ass", "stripping", "bottom-bitch", "permission", "cum-slut", "blowjob", "grinding", "yoga", "cyber", "kegel", "conversation", "bend-over", "squirming", "horny", "dildo-choking", "deep-throating", "finger-stimulation", "begging", "hummer", "orgasm", "cum-drunk", "laughing", "gratitude", "teasing", "crying", "bullying", "moaning", "feet-licking", "womb-tattoo", "harness", "boobs", "vagina", "drugs", "interpretive-dance", "sexy-dancing", "pole-dancing", "flooring", "jig", "silly-dancing", "tail"], - "date": "2022-11-22T01:15:55.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1594862203546501121" -}, { - "tags": ["love-drunk", "noises", "x-mas", "technical-difficulties", "horny", "chat-analytics", "chat-roast", "lovense-lush", "moaning", "self-deprecation-humor", "edging", "just-chatting", "orgasm", "2-cum", "cum-drunk", "multiple-orgasms", "hands-free-cum", "handcuffs", "butt-plug", "anal", "chastity-device", "bdsm", "ball-gag", "collar", "heavy-breathing", "big-o", "snacking", "fat-bastard", "airplane", "mile-high-club", "dirty-talk", "video-game", "malady", "hime-hajime", "big-bang-studios", "a-nut-between-worlds"], - "date": "2022-12-17T22:14:20.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1604238592389238784" -}, { - "tags": ["video-game", "recovery", "bad-audio", "technical-difficulties", "just-chatting", "purino-party", "zooted", "medication", null], - "date": "2023-01-05T02:07:18.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1610820355895037955" -}, { - "tags": ["cum-ban", "soaking", "semantics", "lovense-lush", "dildo", "pussy-noises", "relaxation", "meditation", "games", "meltys-quest", "wet", "asmr", "messy", "moaning", "horny", "blowjob", "doctor-defiance", "cum-brain", "mind-break", "giggles", "mel-noises", "hentai-game", "prank", "ear-rape", "live2d", "voice-acting", "goblin-sex", "drake__selfsuck"], - "date": "2023-01-09T23:07:18.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1612586844298018817" -}, { - "tags": ["bondage", "digital-art", "lovense-lush", "mel-noises", "moaning", "hucow", "pussy-milking", "browser-wars", "edging", "daddy-play", "orgasm", "feet", "live2d", "rapping", "singing", "vibrator", "good-audio", "vibrator-asmr", "spreader-bars", "precum", "puddle", "bdsm", "restraint", "slut", "cum-drunk", "loud-orgasm", "multiple-orgasms", "screaming", "begging", "cum-inside", "mind-break", "aftershocks", "labia-spreader"], - "date": "2023-01-29T22:06:21.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1619819262125944832" -}, { - "tags": ["pasties", "topless", "live2d", "lovense-lush", "lovense-osci", "orgasm", "cum-drunk", "mind-break", "moaning", "mel-noises", "multiple-orgasms", "hentai-watch-along", "dirty-talk", "blowjob", "horny", "tentacles", "double-penetration", "fantasy"], - "date": "2023-02-01T01:38:32.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1620597436988411907" -}, { - "tags": ["cooking", "onigiri", "seiso", "irl", "bad-audio", "rickroll", "dirty-slut", "crepe", "jokes", "just-chatting", "sex-toy-discussion", "booli", "femdom", "topless", "apron", "boobs", "nipples", "bacon", "innuendo", "dirty-talk", "parenting", "stories"], - "date": "2023-02-24T00:17:38.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1628911998787362817" -}, { - "tags": ["3d", "lovense-lush", "lovense-sex-machine", "pool", "dancing", "sway", "ass", "shorts", "leggings", "middrift", "mel-noises", "pig-latin", "moaning", "topless", "bra", "bottomless", "good-audio", "kneeling", "legs-spread", "t1m", "pussy", "grinding", "shaking", "missionary-style", "first-time", "lovense-hyphy", "cum", "orgasm", "multiple-orgasms", "begging", "prone-bone", "daddy", "dirty-talk", "shibari", "closeup", "pov", "6-cums"], - "date": "2023-03-04T02:28:14.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1631843966843863040" -}, { - "tags": ["anime-girl", "horny", "baka", "lovense-lush", "boobs", "harness", "pasties", "cum", "orgasm", "edging", "gunrun", "hormones", "subathon", "fantasy", "sake", "ntrpg", "game-night", "toilet", "ntr", "elderly", "fatherboard", "sneeze", "dream-daddy", "futa-fix-dick-dine-and-dash", "futanari", "asmr", "lovense-hush", "anal", "begging"], - "date": "2023-03-03T02:04:05.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1631475499753979908" -}, { - "tags": ["lovense-partnership", "big-bang-a-nut-between-worlds", "selfcest", "cuck", "video-game", "sponsored-stream", "finger-blasting", "blowjob", "handjob"], - "date": "2023-03-24T21:13:29.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1639374902246404102" -}, { - "tags": ["bikini", "womb-tattoo", "pool", "lovense-lush", "japan", "adhd", "edging", "permission", "cum-slut", "mel-noises", "moaning", "begging", "yelling", "mindbreak", "cum-drunk", "dirty-talk", "bbc", "dildo", "death", "lovense-sex-machine", "bakery", "orgasms", "multiple-orgasms", "ejaculate", "blowjob", "just-chatting", "choking", "joi", "slave", "master", "deepthroat", "sloppy", "3d", "goblin", "anal-breeding", "rapping", "dj", "dancing", "hime-hajime", "bad-audio", "voice-acting", "alto"], - "date": "2023-03-23T21:13:04.000Z", - "announceUrl": "https://twitter.com/ProjektMelody/status/1639012409175072769" -}] \ No newline at end of file diff --git a/services/strapi/favicon.png b/services/strapi/favicon.png deleted file mode 100644 index df668a881b58175effdf6e6d667cd0a29ce98075..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 497 zcmeAS@N?(olHy`uVBq!ia0vp^4j|0I3?%1nZ+ru!Vgr0aT!FNQ%YPz~r|W+YAaeN! zM4m4Hfm|S%I{E+RP5<}p{hyZnKR5G#ea-*Q_WyV8{15c~egC1S$L^sn@(SnS%da63LSXaQ`%;7gOVpveiyr7nQL9IZ= z-A`pVKgIF>S>*T2kpBVWANdax|AfbLJG@L@l;9x3(=mx{2S>{97;cxH>_3zyyzXAU zjjdxQ^WxPEN$-xlJne8sF0xqg#Mjh={AZcjd`yxKaLoVx;=zr@liB5BU%YW{PhSw- uckQM0zm%HSf9!WG&HHyg{qyDQlnaA`#FRum$ErX}4pUXO@geCwXjqE-E diff --git a/services/strapi/jsconfig.json b/services/strapi/jsconfig.json deleted file mode 100644 index e030430..0000000 --- a/services/strapi/jsconfig.json +++ /dev/null @@ -1,9 +0,0 @@ -{ - "compilerOptions": { - "module": "NodeNext", - "moduleResolution": "NodeNext", - "target": "ES2021", - "checkJs": true, - "allowJs": true - } -} diff --git a/services/strapi/misc/generateCuid.js b/services/strapi/misc/generateCuid.js deleted file mode 100644 index c049924..0000000 --- a/services/strapi/misc/generateCuid.js +++ /dev/null @@ -1,7 +0,0 @@ -const { init } = require('@paralleldrive/cuid2'); - -module.exports = function() { - const length = 10; - const genCuid = init({ length }); - return genCuid(); -} \ No newline at end of file diff --git a/services/strapi/package.json b/services/strapi/package.json deleted file mode 100644 index 5a5f7b3..0000000 --- a/services/strapi/package.json +++ /dev/null @@ -1,51 +0,0 @@ -{ - "name": "@futureporn/strapi", - "private": true, - "version": "0.1.0", - "description": "A Strapi application", - "scripts": { - "dev": "pnpm run develop", - "develop": "strapi develop", - "start": "strapi start", - "build": "strapi build", - "strapi": "strapi", - "deploy": "strapi deploy", - "clean": "rm -rf dist", - "superclean": "rm -rf node_modules && rm -rf pnpm-lock.yaml && rm -rf dist" - }, - "dependencies": { - "@11ty/eleventy-fetch": "^4.0.1", - "@aws-sdk/client-s3": "^3.637.0", - "@mux/mux-node": "^8.8.0", - "@paralleldrive/cuid2": "^2.2.2", - "@strapi/plugin-cloud": "4.25.9", - "@strapi/plugin-i18n": "4.25.9", - "@strapi/plugin-users-permissions": "4.25.9", - "@strapi/strapi": "4.25.9", - "@strapi/utils": "^4.25.9", - "canvas": "^2.11.2", - "date-fns": "^3.6.0", - "dotenv": "^16.4.5", - "lodash": "^4.17.21", - "node-fetch": "^3.3.2", - "pg": "8.12.0", - "react": "^18.3.1", - "react-dom": "^18.3.1", - "react-router-dom": "6.26.1", - "sharp": "^0.33.5", - "strapi-plugin-fuzzy-search": "^3.0.0", - "styled-components": "6.1.13" - }, - "author": { - "name": "A Strapi developer" - }, - "strapi": { - "uuid": "de6744d2-f4ac-43cf-aeba-224639f274d1" - }, - "engines": { - "node": "20.x.x", - "npm": ">=6.0.0" - }, - "packageManager": "pnpm@9.6.0", - "license": "MIT" -} diff --git a/services/strapi/pnpm-lock.yaml b/services/strapi/pnpm-lock.yaml deleted file mode 100644 index b90d087..0000000 --- a/services/strapi/pnpm-lock.yaml +++ /dev/null @@ -1,14433 +0,0 @@ -lockfileVersion: '9.0' - -settings: - autoInstallPeers: true - excludeLinksFromLockfile: false - -importers: - - .: - dependencies: - '@11ty/eleventy-fetch': - specifier: ^4.0.1 - version: 4.0.1 - '@aws-sdk/client-s3': - specifier: ^3.637.0 - version: 3.637.0 - '@mux/mux-node': - specifier: ^8.8.0 - version: 8.8.0 - '@paralleldrive/cuid2': - specifier: ^2.2.2 - version: 2.2.2 - '@strapi/plugin-cloud': - specifier: 4.25.9 - version: 4.25.9(vz4izpmlclysylnr54rpbz6mea) - '@strapi/plugin-i18n': - specifier: 4.25.9 - version: 4.25.9(22se43zwxmtuqqjkykcc76uoqe) - '@strapi/plugin-users-permissions': - specifier: 4.25.9 - version: 4.25.9(22se43zwxmtuqqjkykcc76uoqe) - '@strapi/strapi': - specifier: 4.25.9 - version: 4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0) - '@strapi/utils': - specifier: ^4.25.9 - version: 4.25.9 - canvas: - specifier: ^2.11.2 - version: 2.11.2 - date-fns: - specifier: ^3.6.0 - version: 3.6.0 - dotenv: - specifier: ^16.4.5 - version: 16.4.5 - lodash: - specifier: ^4.17.21 - version: 4.17.21 - node-fetch: - specifier: ^3.3.2 - version: 3.3.2 - pg: - specifier: 8.12.0 - version: 8.12.0 - react: - specifier: ^18.3.1 - version: 18.3.1 - react-dom: - specifier: ^18.3.1 - version: 18.3.1(react@18.3.1) - react-router-dom: - specifier: 6.26.1 - version: 6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - sharp: - specifier: ^0.33.5 - version: 0.33.5 - strapi-plugin-fuzzy-search: - specifier: ^3.0.0 - version: 3.0.0(@strapi/strapi@4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0))(@strapi/utils@4.25.9)(yup@0.32.9) - styled-components: - specifier: 6.1.13 - version: 6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - -packages: - - '@11ty/eleventy-fetch@4.0.1': - resolution: {integrity: sha512-yIiLM5ziBmg86i4TlXpBdcIygJHvh/GgPJyAiFOckO9H4y9cQDM8eIcJCUQ4Mum0NEVui/OjhEut2R08xw0vlQ==} - engines: {node: '>=14'} - - '@aws-crypto/crc32@5.2.0': - resolution: {integrity: sha512-nLbCWqQNgUiwwtFsen1AdzAtvuLRsQS8rYgMuxCrdKf9kOssamGLuPwyTY9wyYblNr9+1XM8v6zoDTPPSIeANg==} - engines: {node: '>=16.0.0'} - - '@aws-crypto/crc32c@5.2.0': - resolution: {integrity: sha512-+iWb8qaHLYKrNvGRbiYRHSdKRWhto5XlZUEBwDjYNf+ly5SVYG6zEoYIdxvf5R3zyeP16w4PLBn3rH1xc74Rag==} - - '@aws-crypto/sha1-browser@5.2.0': - resolution: {integrity: sha512-OH6lveCFfcDjX4dbAvCFSYUjJZjDr/3XJ3xHtjn3Oj5b9RjojQo8npoLeA/bNwkOkrSQ0wgrHzXk4tDRxGKJeg==} - - '@aws-crypto/sha256-browser@5.2.0': - resolution: {integrity: sha512-AXfN/lGotSQwu6HNcEsIASo7kWXZ5HYWvfOmSNKDsEqC4OashTp8alTmaz+F7TC2L083SFv5RdB+qU3Vs1kZqw==} - - '@aws-crypto/sha256-js@5.2.0': - resolution: {integrity: sha512-FFQQyu7edu4ufvIZ+OadFpHHOt+eSTBaYaki44c+akjg7qZg9oOQeLlk77F6tSYqjDAFClrHJk9tMf0HdVyOvA==} - engines: {node: '>=16.0.0'} - - '@aws-crypto/supports-web-crypto@5.2.0': - resolution: {integrity: sha512-iAvUotm021kM33eCdNfwIN//F77/IADDSs58i+MDaOqFrVjZo9bAal0NK7HurRuWLLpF1iLX7gbWrjHjeo+YFg==} - - '@aws-crypto/util@5.2.0': - resolution: {integrity: sha512-4RkU9EsI6ZpBve5fseQlGNUWKMa1RLPQ1dnjnQoe07ldfIzcsGb5hC5W0Dm7u423KWzawlrpbjXBrXCEv9zazQ==} - - '@aws-sdk/client-s3@3.637.0': - resolution: {integrity: sha512-y6UC94fsMvhKbf0dzfnjVP1HePeGjplfcYfilZU1COIJLyTkMcUv4XcT4I407CGIrvgEafONHkiC09ygqUauNA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/client-sso-oidc@3.637.0': - resolution: {integrity: sha512-27bHALN6Qb6m6KZmPvRieJ/QRlj1lyac/GT2Rn5kJpre8Mpp+yxrtvp3h9PjNBty4lCeFEENfY4dGNSozBuBcw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.637.0 - - '@aws-sdk/client-sso@3.637.0': - resolution: {integrity: sha512-+KjLvgX5yJYROWo3TQuwBJlHCY0zz9PsLuEolmXQn0BVK1L/m9GteZHtd+rEdAoDGBpE0Xqjy1oz5+SmtsaRUw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/client-sts@3.637.0': - resolution: {integrity: sha512-xUi7x4qDubtA8QREtlblPuAcn91GS/09YVEY/RwU7xCY0aqGuFwgszAANlha4OUIqva8oVj2WO4gJuG+iaSnhw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/core@3.635.0': - resolution: {integrity: sha512-i1x/E/sgA+liUE1XJ7rj1dhyXpAKO1UKFUcTTHXok2ARjWTvszHnSXMOsB77aPbmn0fUp1JTx2kHUAZ1LVt5Bg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-env@3.620.1': - resolution: {integrity: sha512-ExuILJ2qLW5ZO+rgkNRj0xiAipKT16Rk77buvPP8csR7kkCflT/gXTyzRe/uzIiETTxM7tr8xuO9MP/DQXqkfg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-http@3.635.0': - resolution: {integrity: sha512-iJyRgEjOCQlBMXqtwPLIKYc7Bsc6nqjrZybdMDenPDa+kmLg7xh8LxHsu9088e+2/wtLicE34FsJJIfzu3L82g==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-ini@3.637.0': - resolution: {integrity: sha512-h+PFCWfZ0Q3Dx84SppET/TFpcQHmxFW8/oV9ArEvMilw4EBN+IlxgbL0CnHwjHW64szcmrM0mbebjEfHf4FXmw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.637.0 - - '@aws-sdk/credential-provider-node@3.637.0': - resolution: {integrity: sha512-yoEhoxJJfs7sPVQ6Is939BDQJZpZCoUgKr/ySse4YKOZ24t4VqgHA6+wV7rYh+7IW24Rd91UTvEzSuHYTlxlNA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-process@3.620.1': - resolution: {integrity: sha512-hWqFMidqLAkaV9G460+1at6qa9vySbjQKKc04p59OT7lZ5cO5VH5S4aI05e+m4j364MBROjjk2ugNvfNf/8ILg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-sso@3.637.0': - resolution: {integrity: sha512-Mvz+h+e62/tl+dVikLafhv+qkZJ9RUb8l2YN/LeKMWkxQylPT83CPk9aimVhCV89zth1zpREArl97+3xsfgQvA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-web-identity@3.621.0': - resolution: {integrity: sha512-w7ASSyfNvcx7+bYGep3VBgC3K6vEdLmlpjT7nSIHxxQf+WSdvy+HynwJosrpZax0sK5q0D1Jpn/5q+r5lwwW6w==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.621.0 - - '@aws-sdk/middleware-bucket-endpoint@3.620.0': - resolution: {integrity: sha512-eGLL0W6L3HDb3OACyetZYOWpHJ+gLo0TehQKeQyy2G8vTYXqNTeqYhuI6up9HVjBzU9eQiULVQETmgQs7TFaRg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-expect-continue@3.620.0': - resolution: {integrity: sha512-QXeRFMLfyQ31nAHLbiTLtk0oHzG9QLMaof5jIfqcUwnOkO8YnQdeqzakrg1Alpy/VQ7aqzIi8qypkBe2KXZz0A==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-flexible-checksums@3.620.0': - resolution: {integrity: sha512-ftz+NW7qka2sVuwnnO1IzBku5ccP+s5qZGeRTPgrKB7OzRW85gthvIo1vQR2w+OwHFk7WJbbhhWwbCbktnP4UA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-host-header@3.620.0': - resolution: {integrity: sha512-VMtPEZwqYrII/oUkffYsNWY9PZ9xpNJpMgmyU0rlDQ25O1c0Hk3fJmZRe6pEkAJ0omD7kLrqGl1DUjQVxpd/Rg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-location-constraint@3.609.0': - resolution: {integrity: sha512-xzsdoTkszGVqGVPjUmgoP7TORiByLueMHieI1fhQL888WPdqctwAx3ES6d/bA9Q/i8jnc6hs+Fjhy8UvBTkE9A==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-logger@3.609.0': - resolution: {integrity: sha512-S62U2dy4jMDhDFDK5gZ4VxFdWzCtLzwbYyFZx2uvPYTECkepLUfzLic2BHg2Qvtu4QjX+oGE3P/7fwaGIsGNuQ==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-recursion-detection@3.620.0': - resolution: {integrity: sha512-nh91S7aGK3e/o1ck64sA/CyoFw+gAYj2BDOnoNa6ouyCrVJED96ZXWbhye/fz9SgmNUZR2g7GdVpiLpMKZoI5w==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-sdk-s3@3.635.0': - resolution: {integrity: sha512-RLdYJPEV4JL/7NBoFUs7VlP90X++5FlJdxHz0DzCjmiD3qCviKy+Cym3qg1gBgHwucs5XisuClxDrGokhAdTQw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-ssec@3.609.0': - resolution: {integrity: sha512-GZSD1s7+JswWOTamVap79QiDaIV7byJFssBW68GYjyRS5EBjNfwA/8s+6uE6g39R3ojyTbYOmvcANoZEhSULXg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-user-agent@3.637.0': - resolution: {integrity: sha512-EYo0NE9/da/OY8STDsK2LvM4kNa79DBsf4YVtaG4P5pZ615IeFsD8xOHZeuJmUrSMlVQ8ywPRX7WMucUybsKug==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/region-config-resolver@3.614.0': - resolution: {integrity: sha512-vDCeMXvic/LU0KFIUjpC3RiSTIkkvESsEfbVHiHH0YINfl8HnEqR5rj+L8+phsCeVg2+LmYwYxd5NRz4PHxt5g==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/signature-v4-multi-region@3.635.0': - resolution: {integrity: sha512-J6QY4/invOkpogCHjSaDON1hF03viPpOnsrzVuCvJMmclS/iG62R4EY0wq1alYll0YmSdmKlpJwHMWwGtqK63Q==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/token-providers@3.614.0': - resolution: {integrity: sha512-okItqyY6L9IHdxqs+Z116y5/nda7rHxLvROxtAJdLavWTYDydxrZstImNgGWTeVdmc0xX2gJCI77UYUTQWnhRw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sso-oidc': ^3.614.0 - - '@aws-sdk/types@3.609.0': - resolution: {integrity: sha512-+Tqnh9w0h2LcrUsdXyT1F8mNhXz+tVYBtP19LpeEGntmvHwa2XzvLUCWpoIAIVsHp5+HdB2X9Sn0KAtmbFXc2Q==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-arn-parser@3.568.0': - resolution: {integrity: sha512-XUKJWWo+KOB7fbnPP0+g/o5Ulku/X53t7i/h+sPHr5xxYTJJ9CYnbToo95mzxe7xWvkLrsNtJ8L+MnNn9INs2w==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-endpoints@3.637.0': - resolution: {integrity: sha512-pAqOKUHeVWHEXXDIp/qoMk/6jyxIb6GGjnK1/f8dKHtKIEs4tKsnnL563gceEvdad53OPXIt86uoevCcCzmBnw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-locate-window@3.568.0': - resolution: {integrity: sha512-3nh4TINkXYr+H41QaPelCceEB2FXP3fxp93YZXB/kqJvX0U9j0N0Uk45gvsjmEPzG8XxkPEeLIfT2I1M7A6Lig==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-user-agent-browser@3.609.0': - resolution: {integrity: sha512-fojPU+mNahzQ0YHYBsx0ZIhmMA96H+ZIZ665ObU9tl+SGdbLneVZVikGve+NmHTQwHzwkFsZYYnVKAkreJLAtA==} - - '@aws-sdk/util-user-agent-node@3.614.0': - resolution: {integrity: sha512-15ElZT88peoHnq5TEoEtZwoXTXRxNrk60TZNdpl/TUBJ5oNJ9Dqb5Z4ryb8ofN6nm9aFf59GVAerFDz8iUoHBA==} - engines: {node: '>=16.0.0'} - peerDependencies: - aws-crt: '>=1.0.0' - peerDependenciesMeta: - aws-crt: - optional: true - - '@aws-sdk/xml-builder@3.609.0': - resolution: {integrity: sha512-l9XxNcA4HX98rwCC2/KoiWcmEiRfZe4G+mYwDbCFT87JIMj6GBhLDkAzr/W8KAaA2IDr8Vc6J8fZPgVulxxfMA==} - engines: {node: '>=16.0.0'} - - '@babel/code-frame@7.24.7': - resolution: {integrity: sha512-BcYH1CVJBO9tvyIZ2jVeXgSIMvGZ2FDRvDdOIVQyuklNKSsx+eppDEBq/g47Ayw+RqNFE+URvOShmf+f/qwAlA==} - engines: {node: '>=6.9.0'} - - '@babel/generator@7.25.6': - resolution: {integrity: sha512-VPC82gr1seXOpkjAAKoLhP50vx4vGNlF4msF64dSFq1P8RfB+QAuJWGHPXXPc8QyfVWwwB/TNNU4+ayZmHNbZw==} - engines: {node: '>=6.9.0'} - - '@babel/helper-module-imports@7.24.7': - resolution: {integrity: sha512-8AyH3C+74cgCVVXow/myrynrAGv+nTVg5vKu2nZph9x7RcRwzmh0VFallJuFTZ9mx6u4eSdXZfcOzSqTUm0HCA==} - engines: {node: '>=6.9.0'} - - '@babel/helper-string-parser@7.24.8': - resolution: {integrity: sha512-pO9KhhRcuUyGnJWwyEgnRJTSIZHiT+vMD0kPeD+so0l7mxkMT19g3pjY9GTnHySck/hDzq+dtW/4VgnMkippsQ==} - engines: {node: '>=6.9.0'} - - '@babel/helper-validator-identifier@7.24.7': - resolution: {integrity: sha512-rR+PBcQ1SMQDDyF6X0wxtG8QyLCgUB0eRAGguqRLfkCA87l7yAP7ehq8SNj96OOGTO8OBV70KhuFYcIkHXOg0w==} - engines: {node: '>=6.9.0'} - - '@babel/highlight@7.24.7': - resolution: {integrity: sha512-EStJpq4OuY8xYfhGVXngigBJRWxftKX9ksiGDnmlY3o7B/V7KIAc9X4oiK87uPJSc/vs5L869bem5fhZa8caZw==} - engines: {node: '>=6.9.0'} - - '@babel/parser@7.25.6': - resolution: {integrity: sha512-trGdfBdbD0l1ZPmcJ83eNxB9rbEax4ALFTF7fN386TMYbeCQbyme5cOEXQhbGXKebwGaB/J52w1mrklMcbgy6Q==} - engines: {node: '>=6.0.0'} - hasBin: true - - '@babel/runtime-corejs3@7.25.6': - resolution: {integrity: sha512-Gz0Nrobx8szge6kQQ5Z5MX9L3ObqNwCQY1PSwSNzreFL7aHGxv8Fp2j3ETV6/wWdbiV+mW6OSm8oQhg3Tcsniw==} - engines: {node: '>=6.9.0'} - - '@babel/runtime@7.25.6': - resolution: {integrity: sha512-VBj9MYyDb9tuLq7yzqjgzt6Q+IBQLrGZfdjOekyEirZPHxXWoTSGUTMrpsfi58Up73d13NfYLv8HT9vmznjzhQ==} - engines: {node: '>=6.9.0'} - - '@babel/template@7.25.0': - resolution: {integrity: sha512-aOOgh1/5XzKvg1jvVz7AVrx2piJ2XBi227DHmbY6y+bM9H2FlN+IfecYu4Xl0cNiiVejlsCri89LUsbj8vJD9Q==} - engines: {node: '>=6.9.0'} - - '@babel/traverse@7.25.6': - resolution: {integrity: sha512-9Vrcx5ZW6UwK5tvqsj0nGpp/XzqthkT0dqIc9g1AdtygFToNtTF67XzYS//dm+SAK9cp3B9R4ZO/46p63SCjlQ==} - engines: {node: '>=6.9.0'} - - '@babel/types@7.25.6': - resolution: {integrity: sha512-/l42B1qxpG6RdfYf343Uw1vmDjeNhneUXtzhojE7pDgfpEypmRhI6j1kr17XCVv4Cgl9HdAiQY2x0GwKm7rWCw==} - engines: {node: '>=6.9.0'} - - '@casl/ability@6.5.0': - resolution: {integrity: sha512-3guc94ugr5ylZQIpJTLz0CDfwNi0mxKVECj1vJUPAvs+Lwunh/dcuUjwzc4MHM9D8JOYX0XUZMEPedpB3vIbOw==} - - '@codemirror/autocomplete@6.18.0': - resolution: {integrity: sha512-5DbOvBbY4qW5l57cjDsmmpDh3/TeK1vXfTHa+BUMrRzdWdcxKZ4U4V7vQaTtOpApNU4kLS4FQ6cINtLg245LXA==} - peerDependencies: - '@codemirror/language': ^6.0.0 - '@codemirror/state': ^6.0.0 - '@codemirror/view': ^6.0.0 - '@lezer/common': ^1.0.0 - - '@codemirror/commands@6.6.1': - resolution: {integrity: sha512-iBfKbyIoXS1FGdsKcZmnrxmbc8VcbMrSgD7AVrsnX+WyAYjmUDWvE93dt5D874qS4CCVu4O1JpbagHdXbbLiOw==} - - '@codemirror/lang-json@6.0.1': - resolution: {integrity: sha512-+T1flHdgpqDDlJZ2Lkil/rLiRy684WMLc74xUnjJH48GQdfJo/pudlTRreZmKwzP8/tGdKf83wlbAdOCzlJOGQ==} - - '@codemirror/language@6.10.2': - resolution: {integrity: sha512-kgbTYTo0Au6dCSc/TFy7fK3fpJmgHDv1sG1KNQKJXVi+xBTEeBPY/M30YXiU6mMXeH+YIDLsbrT4ZwNRdtF+SA==} - - '@codemirror/lint@6.8.1': - resolution: {integrity: sha512-IZ0Y7S4/bpaunwggW2jYqwLuHj0QtESf5xcROewY6+lDNwZ/NzvR4t+vpYgg9m7V8UXLPYqG+lu3DF470E5Oxg==} - - '@codemirror/search@6.5.6': - resolution: {integrity: sha512-rpMgcsh7o0GuCDUXKPvww+muLA1pDJaFrpq/CCHtpQJYz8xopu4D1hPcKRoDD0YlF8gZaqTNIRa4VRBWyhyy7Q==} - - '@codemirror/state@6.4.1': - resolution: {integrity: sha512-QkEyUiLhsJoZkbumGZlswmAhA7CBU02Wrz7zvH4SrcifbsqwlXShVXg65f3v/ts57W3dqyamEriMhij1Z3Zz4A==} - - '@codemirror/theme-one-dark@6.1.2': - resolution: {integrity: sha512-F+sH0X16j/qFLMAfbciKTxVOwkdAS336b7AXTKOZhy8BR3eH/RelsnLgLFINrpST63mmN2OuwUt0W2ndUgYwUA==} - - '@codemirror/view@6.33.0': - resolution: {integrity: sha512-AroaR3BvnjRW8fiZBalAaK+ZzB5usGgI014YKElYZvQdNH5ZIidHlO+cyf/2rWzyBFRkvG6VhiXeAEbC53P2YQ==} - - '@colors/colors@1.5.0': - resolution: {integrity: sha512-ooWCrlZP11i8GImSjTHYHLkvFDP48nS4+204nGb1RiX/WXYHmJA2III9/e2DWVabCESdW7hBAEzHRqUn9OUVvQ==} - engines: {node: '>=0.1.90'} - - '@colors/colors@1.6.0': - resolution: {integrity: sha512-Ir+AOibqzrIsL6ajt3Rz3LskB7OiMVHqltZmspbW/TJuTVuyOMirVqAkjfY6JISiLHgyNqicAC8AyHHGzNd/dA==} - engines: {node: '>=0.1.90'} - - '@dabh/diagnostics@2.0.3': - resolution: {integrity: sha512-hrlQOIi7hAfzsMqlGSFyVucrx38O+j6wiGOf//H2ecvIEqYN4ADBSS2iLMh5UFyDunCNniUIPk/q3riFv45xRA==} - - '@discoveryjs/json-ext@0.5.7': - resolution: {integrity: sha512-dBVuXR082gk3jsFp7Rd/JI4kytwGHecnCoTtXFb7DB6CNHp4rg5k1bhg0nWdLGLnOV71lmDzGQaLMy8iPLY0pw==} - engines: {node: '>=10.0.0'} - - '@emnapi/runtime@1.2.0': - resolution: {integrity: sha512-bV21/9LQmcQeCPEg3BDFtvwL6cwiTMksYNWQQ4KOxCZikEGalWtenoZ0wCiukJINlGCIi2KXx01g4FoH/LxpzQ==} - - '@emotion/babel-plugin@11.12.0': - resolution: {integrity: sha512-y2WQb+oP8Jqvvclh8Q55gLUyb7UFvgv7eJfsj7td5TToBrIUtPay2kMrZi4xjq9qw2vD0ZR5fSho0yqoFgX7Rw==} - - '@emotion/cache@11.13.1': - resolution: {integrity: sha512-iqouYkuEblRcXmylXIwwOodiEK5Ifl7JcX7o6V4jI3iW4mLXX3dmt5xwBtIkJiQEXFAI+pC8X0i67yiPkH9Ucw==} - - '@emotion/hash@0.9.2': - resolution: {integrity: sha512-MyqliTZGuOm3+5ZRSaaBGP3USLw6+EGykkwZns2EPC5g8jJ4z9OrdZY9apkl3+UP9+sdz76YYkwCKP5gh8iY3g==} - - '@emotion/is-prop-valid@1.2.2': - resolution: {integrity: sha512-uNsoYd37AFmaCdXlg6EYD1KaPOaRWRByMCYzbKUX4+hhMfrxdVSelShywL4JVaAeM/eHUOSprYBQls+/neX3pw==} - - '@emotion/memoize@0.8.1': - resolution: {integrity: sha512-W2P2c/VRW1/1tLox0mVUalvnWXxavmv/Oum2aPsRcoDJuob75FC3Y8FbpfLwUegRcxINtGUMPq0tFCvYNTBXNA==} - - '@emotion/memoize@0.9.0': - resolution: {integrity: sha512-30FAj7/EoJ5mwVPOWhAyCX+FPfMDrVecJAM+Iw9NRoSl4BBAQeqj4cApHHUXOVvIPgLVDsCFoz/hGD+5QQD1GQ==} - - '@emotion/react@11.13.3': - resolution: {integrity: sha512-lIsdU6JNrmYfJ5EbUCf4xW1ovy5wKQ2CkPRM4xogziOxH1nXxBSjpC9YqbFAP7circxMfYp+6x676BqWcEiixg==} - peerDependencies: - '@types/react': '*' - react: '>=16.8.0' - peerDependenciesMeta: - '@types/react': - optional: true - - '@emotion/serialize@1.3.1': - resolution: {integrity: sha512-dEPNKzBPU+vFPGa+z3axPRn8XVDetYORmDC0wAiej+TNcOZE70ZMJa0X7JdeoM6q/nWTMZeLpN/fTnD9o8MQBA==} - - '@emotion/sheet@1.4.0': - resolution: {integrity: sha512-fTBW9/8r2w3dXWYM4HCB1Rdp8NLibOw2+XELH5m5+AkWiL/KqYX6dc0kKYlaYyKjrQ6ds33MCdMPEwgs2z1rqg==} - - '@emotion/unitless@0.10.0': - resolution: {integrity: sha512-dFoMUuQA20zvtVTuxZww6OHoJYgrzfKM1t52mVySDJnMSEa08ruEvdYQbhvyu6soU+NeLVd3yKfTfT0NeV6qGg==} - - '@emotion/unitless@0.8.1': - resolution: {integrity: sha512-KOEGMu6dmJZtpadb476IsZBclKvILjopjUii3V+7MnXIQCYh8W3NgNcgwo21n9LXZX6EDIKvqfjYxXebDwxKmQ==} - - '@emotion/use-insertion-effect-with-fallbacks@1.1.0': - resolution: {integrity: sha512-+wBOcIV5snwGgI2ya3u99D7/FJquOIniQT1IKyDsBmEgwvpxMNeS65Oib7OnE2d2aY+3BU4OiH+0Wchf8yk3Hw==} - peerDependencies: - react: '>=16.8.0' - - '@emotion/utils@1.4.0': - resolution: {integrity: sha512-spEnrA1b6hDR/C68lC2M7m6ALPUHZC0lIY7jAS/B/9DuuO1ZP04eov8SMv/6fwRd8pzmsn2AuJEznRREWlQrlQ==} - - '@emotion/weak-memoize@0.4.0': - resolution: {integrity: sha512-snKqtPW01tN0ui7yu9rGv69aJXr/a/Ywvl11sUjNtEcRc+ng/mQriFL0wLXMef74iHa/EkftbDzU9F8iFbH+zg==} - - '@esbuild/aix-ppc64@0.19.11': - resolution: {integrity: sha512-FnzU0LyE3ySQk7UntJO4+qIiQgI7KoODnZg5xzXIrFJlKd2P2gwHsHY4927xj9y5PJmJSzULiUCWmv7iWnNa7g==} - engines: {node: '>=12'} - cpu: [ppc64] - os: [aix] - - '@esbuild/android-arm64@0.16.17': - resolution: {integrity: sha512-MIGl6p5sc3RDTLLkYL1MyL8BMRN4tLMRCn+yRJJmEDvYZ2M7tmAf80hx1kbNEUX2KJ50RRtxZ4JHLvCfuB6kBg==} - engines: {node: '>=12'} - cpu: [arm64] - os: [android] - - '@esbuild/android-arm64@0.19.11': - resolution: {integrity: sha512-aiu7K/5JnLj//KOnOfEZ0D90obUkRzDMyqd/wNAUQ34m4YUPVhRZpnqKV9uqDGxT7cToSDnIHsGooyIczu9T+Q==} - engines: {node: '>=12'} - cpu: [arm64] - os: [android] - - '@esbuild/android-arm@0.16.17': - resolution: {integrity: sha512-N9x1CMXVhtWEAMS7pNNONyA14f71VPQN9Cnavj1XQh6T7bskqiLLrSca4O0Vr8Wdcga943eThxnVp3JLnBMYtw==} - engines: {node: '>=12'} - cpu: [arm] - os: [android] - - '@esbuild/android-arm@0.19.11': - resolution: {integrity: sha512-5OVapq0ClabvKvQ58Bws8+wkLCV+Rxg7tUVbo9xu034Nm536QTII4YzhaFriQ7rMrorfnFKUsArD2lqKbFY4vw==} - engines: {node: '>=12'} - cpu: [arm] - os: [android] - - '@esbuild/android-x64@0.16.17': - resolution: {integrity: sha512-a3kTv3m0Ghh4z1DaFEuEDfz3OLONKuFvI4Xqczqx4BqLyuFaFkuaG4j2MtA6fuWEFeC5x9IvqnX7drmRq/fyAQ==} - engines: {node: '>=12'} - cpu: [x64] - os: [android] - - '@esbuild/android-x64@0.19.11': - resolution: {integrity: sha512-eccxjlfGw43WYoY9QgB82SgGgDbibcqyDTlk3l3C0jOVHKxrjdc9CTwDUQd0vkvYg5um0OH+GpxYvp39r+IPOg==} - engines: {node: '>=12'} - cpu: [x64] - os: [android] - - '@esbuild/darwin-arm64@0.16.17': - resolution: {integrity: sha512-/2agbUEfmxWHi9ARTX6OQ/KgXnOWfsNlTeLcoV7HSuSTv63E4DqtAc+2XqGw1KHxKMHGZgbVCZge7HXWX9Vn+w==} - engines: {node: '>=12'} - cpu: [arm64] - os: [darwin] - - '@esbuild/darwin-arm64@0.19.11': - resolution: {integrity: sha512-ETp87DRWuSt9KdDVkqSoKoLFHYTrkyz2+65fj9nfXsaV3bMhTCjtQfw3y+um88vGRKRiF7erPrh/ZuIdLUIVxQ==} - engines: {node: '>=12'} - cpu: [arm64] - os: [darwin] - - '@esbuild/darwin-x64@0.16.17': - resolution: {integrity: sha512-2By45OBHulkd9Svy5IOCZt376Aa2oOkiE9QWUK9fe6Tb+WDr8hXL3dpqi+DeLiMed8tVXspzsTAvd0jUl96wmg==} - engines: {node: '>=12'} - cpu: [x64] - os: [darwin] - - '@esbuild/darwin-x64@0.19.11': - resolution: {integrity: sha512-fkFUiS6IUK9WYUO/+22omwetaSNl5/A8giXvQlcinLIjVkxwTLSktbF5f/kJMftM2MJp9+fXqZ5ezS7+SALp4g==} - engines: {node: '>=12'} - cpu: [x64] - os: [darwin] - - '@esbuild/freebsd-arm64@0.16.17': - resolution: {integrity: sha512-mt+cxZe1tVx489VTb4mBAOo2aKSnJ33L9fr25JXpqQqzbUIw/yzIzi+NHwAXK2qYV1lEFp4OoVeThGjUbmWmdw==} - engines: {node: '>=12'} - cpu: [arm64] - os: [freebsd] - - '@esbuild/freebsd-arm64@0.19.11': - resolution: {integrity: sha512-lhoSp5K6bxKRNdXUtHoNc5HhbXVCS8V0iZmDvyWvYq9S5WSfTIHU2UGjcGt7UeS6iEYp9eeymIl5mJBn0yiuxA==} - engines: {node: '>=12'} - cpu: [arm64] - os: [freebsd] - - '@esbuild/freebsd-x64@0.16.17': - resolution: {integrity: sha512-8ScTdNJl5idAKjH8zGAsN7RuWcyHG3BAvMNpKOBaqqR7EbUhhVHOqXRdL7oZvz8WNHL2pr5+eIT5c65kA6NHug==} - engines: {node: '>=12'} - cpu: [x64] - os: [freebsd] - - '@esbuild/freebsd-x64@0.19.11': - resolution: {integrity: sha512-JkUqn44AffGXitVI6/AbQdoYAq0TEullFdqcMY/PCUZ36xJ9ZJRtQabzMA+Vi7r78+25ZIBosLTOKnUXBSi1Kw==} - engines: {node: '>=12'} - cpu: [x64] - os: [freebsd] - - '@esbuild/linux-arm64@0.16.17': - resolution: {integrity: sha512-7S8gJnSlqKGVJunnMCrXHU9Q8Q/tQIxk/xL8BqAP64wchPCTzuM6W3Ra8cIa1HIflAvDnNOt2jaL17vaW+1V0g==} - engines: {node: '>=12'} - cpu: [arm64] - os: [linux] - - '@esbuild/linux-arm64@0.19.11': - resolution: {integrity: sha512-LneLg3ypEeveBSMuoa0kwMpCGmpu8XQUh+mL8XXwoYZ6Be2qBnVtcDI5azSvh7vioMDhoJFZzp9GWp9IWpYoUg==} - engines: {node: '>=12'} - cpu: [arm64] - os: [linux] - - '@esbuild/linux-arm@0.16.17': - resolution: {integrity: sha512-iihzrWbD4gIT7j3caMzKb/RsFFHCwqqbrbH9SqUSRrdXkXaygSZCZg1FybsZz57Ju7N/SHEgPyaR0LZ8Zbe9gQ==} - engines: {node: '>=12'} - cpu: [arm] - os: [linux] - - '@esbuild/linux-arm@0.19.11': - resolution: {integrity: sha512-3CRkr9+vCV2XJbjwgzjPtO8T0SZUmRZla+UL1jw+XqHZPkPgZiyWvbDvl9rqAN8Zl7qJF0O/9ycMtjU67HN9/Q==} - engines: {node: '>=12'} - cpu: [arm] - os: [linux] - - '@esbuild/linux-ia32@0.16.17': - resolution: {integrity: sha512-kiX69+wcPAdgl3Lonh1VI7MBr16nktEvOfViszBSxygRQqSpzv7BffMKRPMFwzeJGPxcio0pdD3kYQGpqQ2SSg==} - engines: {node: '>=12'} - cpu: [ia32] - os: [linux] - - '@esbuild/linux-ia32@0.19.11': - resolution: {integrity: sha512-caHy++CsD8Bgq2V5CodbJjFPEiDPq8JJmBdeyZ8GWVQMjRD0sU548nNdwPNvKjVpamYYVL40AORekgfIubwHoA==} - engines: {node: '>=12'} - cpu: [ia32] - os: [linux] - - '@esbuild/linux-loong64@0.16.17': - resolution: {integrity: sha512-dTzNnQwembNDhd654cA4QhbS9uDdXC3TKqMJjgOWsC0yNCbpzfWoXdZvp0mY7HU6nzk5E0zpRGGx3qoQg8T2DQ==} - engines: {node: '>=12'} - cpu: [loong64] - os: [linux] - - '@esbuild/linux-loong64@0.19.11': - resolution: {integrity: sha512-ppZSSLVpPrwHccvC6nQVZaSHlFsvCQyjnvirnVjbKSHuE5N24Yl8F3UwYUUR1UEPaFObGD2tSvVKbvR+uT1Nrg==} - engines: {node: '>=12'} - cpu: [loong64] - os: [linux] - - '@esbuild/linux-mips64el@0.16.17': - resolution: {integrity: sha512-ezbDkp2nDl0PfIUn0CsQ30kxfcLTlcx4Foz2kYv8qdC6ia2oX5Q3E/8m6lq84Dj/6b0FrkgD582fJMIfHhJfSw==} - engines: {node: '>=12'} - cpu: [mips64el] - os: [linux] - - '@esbuild/linux-mips64el@0.19.11': - resolution: {integrity: sha512-B5x9j0OgjG+v1dF2DkH34lr+7Gmv0kzX6/V0afF41FkPMMqaQ77pH7CrhWeR22aEeHKaeZVtZ6yFwlxOKPVFyg==} - engines: {node: '>=12'} - cpu: [mips64el] - os: [linux] - - '@esbuild/linux-ppc64@0.16.17': - resolution: {integrity: sha512-dzS678gYD1lJsW73zrFhDApLVdM3cUF2MvAa1D8K8KtcSKdLBPP4zZSLy6LFZ0jYqQdQ29bjAHJDgz0rVbLB3g==} - engines: {node: '>=12'} - cpu: [ppc64] - os: [linux] - - '@esbuild/linux-ppc64@0.19.11': - resolution: {integrity: sha512-MHrZYLeCG8vXblMetWyttkdVRjQlQUb/oMgBNurVEnhj4YWOr4G5lmBfZjHYQHHN0g6yDmCAQRR8MUHldvvRDA==} - engines: {node: '>=12'} - cpu: [ppc64] - os: [linux] - - '@esbuild/linux-riscv64@0.16.17': - resolution: {integrity: sha512-ylNlVsxuFjZK8DQtNUwiMskh6nT0vI7kYl/4fZgV1llP5d6+HIeL/vmmm3jpuoo8+NuXjQVZxmKuhDApK0/cKw==} - engines: {node: '>=12'} - cpu: [riscv64] - os: [linux] - - '@esbuild/linux-riscv64@0.19.11': - resolution: {integrity: sha512-f3DY++t94uVg141dozDu4CCUkYW+09rWtaWfnb3bqe4w5NqmZd6nPVBm+qbz7WaHZCoqXqHz5p6CM6qv3qnSSQ==} - engines: {node: '>=12'} - cpu: [riscv64] - os: [linux] - - '@esbuild/linux-s390x@0.16.17': - resolution: {integrity: sha512-gzy7nUTO4UA4oZ2wAMXPNBGTzZFP7mss3aKR2hH+/4UUkCOyqmjXiKpzGrY2TlEUhbbejzXVKKGazYcQTZWA/w==} - engines: {node: '>=12'} - cpu: [s390x] - os: [linux] - - '@esbuild/linux-s390x@0.19.11': - resolution: {integrity: sha512-A5xdUoyWJHMMlcSMcPGVLzYzpcY8QP1RtYzX5/bS4dvjBGVxdhuiYyFwp7z74ocV7WDc0n1harxmpq2ePOjI0Q==} - engines: {node: '>=12'} - cpu: [s390x] - os: [linux] - - '@esbuild/linux-x64@0.16.17': - resolution: {integrity: sha512-mdPjPxfnmoqhgpiEArqi4egmBAMYvaObgn4poorpUaqmvzzbvqbowRllQ+ZgzGVMGKaPkqUmPDOOFQRUFDmeUw==} - engines: {node: '>=12'} - cpu: [x64] - os: [linux] - - '@esbuild/linux-x64@0.19.11': - resolution: {integrity: sha512-grbyMlVCvJSfxFQUndw5mCtWs5LO1gUlwP4CDi4iJBbVpZcqLVT29FxgGuBJGSzyOxotFG4LoO5X+M1350zmPA==} - engines: {node: '>=12'} - cpu: [x64] - os: [linux] - - '@esbuild/netbsd-x64@0.16.17': - resolution: {integrity: sha512-/PzmzD/zyAeTUsduZa32bn0ORug+Jd1EGGAUJvqfeixoEISYpGnAezN6lnJoskauoai0Jrs+XSyvDhppCPoKOA==} - engines: {node: '>=12'} - cpu: [x64] - os: [netbsd] - - '@esbuild/netbsd-x64@0.19.11': - resolution: {integrity: sha512-13jvrQZJc3P230OhU8xgwUnDeuC/9egsjTkXN49b3GcS5BKvJqZn86aGM8W9pd14Kd+u7HuFBMVtrNGhh6fHEQ==} - engines: {node: '>=12'} - cpu: [x64] - os: [netbsd] - - '@esbuild/openbsd-x64@0.16.17': - resolution: {integrity: sha512-2yaWJhvxGEz2RiftSk0UObqJa/b+rIAjnODJgv2GbGGpRwAfpgzyrg1WLK8rqA24mfZa9GvpjLcBBg8JHkoodg==} - engines: {node: '>=12'} - cpu: [x64] - os: [openbsd] - - '@esbuild/openbsd-x64@0.19.11': - resolution: {integrity: sha512-ysyOGZuTp6SNKPE11INDUeFVVQFrhcNDVUgSQVDzqsqX38DjhPEPATpid04LCoUr2WXhQTEZ8ct/EgJCUDpyNw==} - engines: {node: '>=12'} - cpu: [x64] - os: [openbsd] - - '@esbuild/sunos-x64@0.16.17': - resolution: {integrity: sha512-xtVUiev38tN0R3g8VhRfN7Zl42YCJvyBhRKw1RJjwE1d2emWTVToPLNEQj/5Qxc6lVFATDiy6LjVHYhIPrLxzw==} - engines: {node: '>=12'} - cpu: [x64] - os: [sunos] - - '@esbuild/sunos-x64@0.19.11': - resolution: {integrity: sha512-Hf+Sad9nVwvtxy4DXCZQqLpgmRTQqyFyhT3bZ4F2XlJCjxGmRFF0Shwn9rzhOYRB61w9VMXUkxlBy56dk9JJiQ==} - engines: {node: '>=12'} - cpu: [x64] - os: [sunos] - - '@esbuild/win32-arm64@0.16.17': - resolution: {integrity: sha512-ga8+JqBDHY4b6fQAmOgtJJue36scANy4l/rL97W+0wYmijhxKetzZdKOJI7olaBaMhWt8Pac2McJdZLxXWUEQw==} - engines: {node: '>=12'} - cpu: [arm64] - os: [win32] - - '@esbuild/win32-arm64@0.19.11': - resolution: {integrity: sha512-0P58Sbi0LctOMOQbpEOvOL44Ne0sqbS0XWHMvvrg6NE5jQ1xguCSSw9jQeUk2lfrXYsKDdOe6K+oZiwKPilYPQ==} - engines: {node: '>=12'} - cpu: [arm64] - os: [win32] - - '@esbuild/win32-ia32@0.16.17': - resolution: {integrity: sha512-WnsKaf46uSSF/sZhwnqE4L/F89AYNMiD4YtEcYekBt9Q7nj0DiId2XH2Ng2PHM54qi5oPrQ8luuzGszqi/veig==} - engines: {node: '>=12'} - cpu: [ia32] - os: [win32] - - '@esbuild/win32-ia32@0.19.11': - resolution: {integrity: sha512-6YOrWS+sDJDmshdBIQU+Uoyh7pQKrdykdefC1avn76ss5c+RN6gut3LZA4E2cH5xUEp5/cA0+YxRaVtRAb0xBg==} - engines: {node: '>=12'} - cpu: [ia32] - os: [win32] - - '@esbuild/win32-x64@0.16.17': - resolution: {integrity: sha512-y+EHuSchhL7FjHgvQL/0fnnFmO4T1bhvWANX6gcnqTjtnKWbTvUMCpGnv2+t+31d7RzyEAYAd4u2fnIhHL6N/Q==} - engines: {node: '>=12'} - cpu: [x64] - os: [win32] - - '@esbuild/win32-x64@0.19.11': - resolution: {integrity: sha512-vfkhltrjCAb603XaFhqhAF4LGDi2M4OrCRrFusyQ+iTLQ/o60QQXxc9cZC/FFpihBI9N1Grn6SMKVJ4KP7Fuiw==} - engines: {node: '>=12'} - cpu: [x64] - os: [win32] - - '@floating-ui/core@1.6.7': - resolution: {integrity: sha512-yDzVT/Lm101nQ5TCVeK65LtdN7Tj4Qpr9RTXJ2vPFLqtLxwOrpoxAHAJI8J3yYWUc40J0BDBheaitK5SJmno2g==} - - '@floating-ui/dom@1.6.10': - resolution: {integrity: sha512-fskgCFv8J8OamCmyun8MfjB1Olfn+uZKjOKZ0vhYF3gRmEUXcGOjxWL8bBr7i4kIuPZ2KD2S3EUIOxnjC8kl2A==} - - '@floating-ui/react-dom@2.1.1': - resolution: {integrity: sha512-4h84MJt3CHrtG18mGsXuLCHMrug49d7DFkU0RMIyshRveBeyV2hmV/pDaF2Uxtu8kgq5r46llp5E5FQiR0K2Yg==} - peerDependencies: - react: '>=16.8.0' - react-dom: '>=16.8.0' - - '@floating-ui/utils@0.2.7': - resolution: {integrity: sha512-X8R8Oj771YRl/w+c1HqAC1szL8zWQRwFvgDwT129k9ACdBoud/+/rX9V0qiMl6LWUdP9voC2nDVZYPMQQsb6eA==} - - '@formatjs/ecma402-abstract@1.14.3': - resolution: {integrity: sha512-SlsbRC/RX+/zg4AApWIFNDdkLtFbkq3LNoZWXZCE/nHVKqoIJyaoQyge/I0Y38vLxowUn9KTtXgusLD91+orbg==} - - '@formatjs/fast-memoize@2.0.1': - resolution: {integrity: sha512-M2GgV+qJn5WJQAYewz7q2Cdl6fobQa69S1AzSM2y0P68ZDbK5cWrJIcPCO395Of1ksftGZoOt4LYCO/j9BKBSA==} - - '@formatjs/icu-messageformat-parser@2.3.1': - resolution: {integrity: sha512-knF2AkAKN4Upv4oIiKY4Wd/dLH68TNMPgV/tJMu/T6FP9aQwbv8fpj7U3lkyniPaNVxvia56Gxax8MKOjtxLSQ==} - - '@formatjs/icu-skeleton-parser@1.3.18': - resolution: {integrity: sha512-ND1ZkZfmLPcHjAH1sVpkpQxA+QYfOX3py3SjKWMUVGDow18gZ0WPqz3F+pJLYQMpS2LnnQ5zYR2jPVYTbRwMpg==} - - '@formatjs/intl-displaynames@6.3.1': - resolution: {integrity: sha512-TlxguMDUbnFrJ4NA8fSyqXC62M7czvlRJ5mrJgtB91JVA+QPjjNdcRm1qPIC/DcU/pGUDcEzThn/x5A+jp15gg==} - - '@formatjs/intl-listformat@7.2.1': - resolution: {integrity: sha512-fRJFWLrGa7d25I4JSxNjKX29oXGcIXx8fJjgURnvs2C3ijS4gurUgFrUwLbv/2KfPfyJ5g567pz2INelNJZBdw==} - - '@formatjs/intl-localematcher@0.2.32': - resolution: {integrity: sha512-k/MEBstff4sttohyEpXxCmC3MqbUn9VvHGlZ8fauLzkbwXmVrEeyzS+4uhrvAk9DWU9/7otYWxyDox4nT/KVLQ==} - - '@formatjs/intl@2.7.1': - resolution: {integrity: sha512-se6vxidsN3PCmzqTsDd3YDT4IX9ZySPy39LYhF7x2ssNvlGMOuW3umkrIhKkXB7ZskqsJGY53LVCdiHsSwhGng==} - peerDependencies: - typescript: ^4.7 || 5 - peerDependenciesMeta: - typescript: - optional: true - - '@hapi/bourne@3.0.0': - resolution: {integrity: sha512-Waj1cwPXJDucOib4a3bAISsKJVb15MKi9IvmTI/7ssVEm6sywXGjVJDhl6/umt1pK1ZS7PacXU3A1PmFKHEZ2w==} - - '@img/sharp-darwin-arm64@0.33.5': - resolution: {integrity: sha512-UT4p+iz/2H4twwAoLCqfA9UH5pI6DggwKEGuaPy7nCVQ8ZsiY5PIcrRvD1DzuY3qYL07NtIQcWnBSY/heikIFQ==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm64] - os: [darwin] - - '@img/sharp-darwin-x64@0.33.5': - resolution: {integrity: sha512-fyHac4jIc1ANYGRDxtiqelIbdWkIuQaI84Mv45KvGRRxSAa7o7d1ZKAOBaYbnepLC1WqxfpimdeWfvqqSGwR2Q==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [x64] - os: [darwin] - - '@img/sharp-libvips-darwin-arm64@1.0.4': - resolution: {integrity: sha512-XblONe153h0O2zuFfTAbQYAX2JhYmDHeWikp1LM9Hul9gVPjFY427k6dFEcOL72O01QxQsWi761svJ/ev9xEDg==} - cpu: [arm64] - os: [darwin] - - '@img/sharp-libvips-darwin-x64@1.0.4': - resolution: {integrity: sha512-xnGR8YuZYfJGmWPvmlunFaWJsb9T/AO2ykoP3Fz/0X5XV2aoYBPkX6xqCQvUTKKiLddarLaxpzNe+b1hjeWHAQ==} - cpu: [x64] - os: [darwin] - - '@img/sharp-libvips-linux-arm64@1.0.4': - resolution: {integrity: sha512-9B+taZ8DlyyqzZQnoeIvDVR/2F4EbMepXMc/NdVbkzsJbzkUjhXv/70GQJ7tdLA4YJgNP25zukcxpX2/SueNrA==} - cpu: [arm64] - os: [linux] - - '@img/sharp-libvips-linux-arm@1.0.5': - resolution: {integrity: sha512-gvcC4ACAOPRNATg/ov8/MnbxFDJqf/pDePbBnuBDcjsI8PssmjoKMAz4LtLaVi+OnSb5FK/yIOamqDwGmXW32g==} - cpu: [arm] - os: [linux] - - '@img/sharp-libvips-linux-s390x@1.0.4': - resolution: {integrity: sha512-u7Wz6ntiSSgGSGcjZ55im6uvTrOxSIS8/dgoVMoiGE9I6JAfU50yH5BoDlYA1tcuGS7g/QNtetJnxA6QEsCVTA==} - cpu: [s390x] - os: [linux] - - '@img/sharp-libvips-linux-x64@1.0.4': - resolution: {integrity: sha512-MmWmQ3iPFZr0Iev+BAgVMb3ZyC4KeFc3jFxnNbEPas60e1cIfevbtuyf9nDGIzOaW9PdnDciJm+wFFaTlj5xYw==} - cpu: [x64] - os: [linux] - - '@img/sharp-libvips-linuxmusl-arm64@1.0.4': - resolution: {integrity: sha512-9Ti+BbTYDcsbp4wfYib8Ctm1ilkugkA/uscUn6UXK1ldpC1JjiXbLfFZtRlBhjPZ5o1NCLiDbg8fhUPKStHoTA==} - cpu: [arm64] - os: [linux] - - '@img/sharp-libvips-linuxmusl-x64@1.0.4': - resolution: {integrity: sha512-viYN1KX9m+/hGkJtvYYp+CCLgnJXwiQB39damAO7WMdKWlIhmYTfHjwSbQeUK/20vY154mwezd9HflVFM1wVSw==} - cpu: [x64] - os: [linux] - - '@img/sharp-linux-arm64@0.33.5': - resolution: {integrity: sha512-JMVv+AMRyGOHtO1RFBiJy/MBsgz0x4AWrT6QoEVVTyh1E39TrCUpTRI7mx9VksGX4awWASxqCYLCV4wBZHAYxA==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm64] - os: [linux] - - '@img/sharp-linux-arm@0.33.5': - resolution: {integrity: sha512-JTS1eldqZbJxjvKaAkxhZmBqPRGmxgu+qFKSInv8moZ2AmT5Yib3EQ1c6gp493HvrvV8QgdOXdyaIBrhvFhBMQ==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm] - os: [linux] - - '@img/sharp-linux-s390x@0.33.5': - resolution: {integrity: sha512-y/5PCd+mP4CA/sPDKl2961b+C9d+vPAveS33s6Z3zfASk2j5upL6fXVPZi7ztePZ5CuH+1kW8JtvxgbuXHRa4Q==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [s390x] - os: [linux] - - '@img/sharp-linux-x64@0.33.5': - resolution: {integrity: sha512-opC+Ok5pRNAzuvq1AG0ar+1owsu842/Ab+4qvU879ippJBHvyY5n2mxF1izXqkPYlGuP/M556uh53jRLJmzTWA==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [x64] - os: [linux] - - '@img/sharp-linuxmusl-arm64@0.33.5': - resolution: {integrity: sha512-XrHMZwGQGvJg2V/oRSUfSAfjfPxO+4DkiRh6p2AFjLQztWUuY/o8Mq0eMQVIY7HJ1CDQUJlxGGZRw1a5bqmd1g==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm64] - os: [linux] - - '@img/sharp-linuxmusl-x64@0.33.5': - resolution: {integrity: sha512-WT+d/cgqKkkKySYmqoZ8y3pxx7lx9vVejxW/W4DOFMYVSkErR+w7mf2u8m/y4+xHe7yY9DAXQMWQhpnMuFfScw==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [x64] - os: [linux] - - '@img/sharp-wasm32@0.33.5': - resolution: {integrity: sha512-ykUW4LVGaMcU9lu9thv85CbRMAwfeadCJHRsg2GmeRa/cJxsVY9Rbd57JcMxBkKHag5U/x7TSBpScF4U8ElVzg==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [wasm32] - - '@img/sharp-win32-ia32@0.33.5': - resolution: {integrity: sha512-T36PblLaTwuVJ/zw/LaH0PdZkRz5rd3SmMHX8GSmR7vtNSP5Z6bQkExdSK7xGWyxLw4sUknBuugTelgw2faBbQ==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [ia32] - os: [win32] - - '@img/sharp-win32-x64@0.33.5': - resolution: {integrity: sha512-MpY/o8/8kj+EcnxwvrP4aTJSWw/aZ7JIGR4aBeZkZw5B7/Jn+tY9/VNwtcoGmdT7GfggGIU4kygOMSbYnOrAbg==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [x64] - os: [win32] - - '@internationalized/date@3.5.5': - resolution: {integrity: sha512-H+CfYvOZ0LTJeeLOqm19E3uj/4YjrmOFtBufDHPfvtI80hFAMqtrp7oCACpe4Cil5l8S0Qu/9dYfZc/5lY8WQQ==} - - '@internationalized/number@3.5.3': - resolution: {integrity: sha512-rd1wA3ebzlp0Mehj5YTuTI50AQEx80gWFyHcQu+u91/5NgdwBecO8BH6ipPfE+lmQ9d63vpB3H9SHoIUiupllw==} - - '@isaacs/cliui@8.0.2': - resolution: {integrity: sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==} - engines: {node: '>=12'} - - '@jridgewell/gen-mapping@0.3.5': - resolution: {integrity: sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==} - engines: {node: '>=6.0.0'} - - '@jridgewell/resolve-uri@3.1.2': - resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==} - engines: {node: '>=6.0.0'} - - '@jridgewell/set-array@1.2.1': - resolution: {integrity: sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==} - engines: {node: '>=6.0.0'} - - '@jridgewell/source-map@0.3.6': - resolution: {integrity: sha512-1ZJTZebgqllO79ue2bm3rIGud/bOe0pP5BjSRCRxxYkEZS8STV7zN84UBbiYu7jy+eCKSnVIUgoWWE/tt+shMQ==} - - '@jridgewell/sourcemap-codec@1.5.0': - resolution: {integrity: sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==} - - '@jridgewell/trace-mapping@0.3.25': - resolution: {integrity: sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==} - - '@juggle/resize-observer@3.4.0': - resolution: {integrity: sha512-dfLbk+PwWvFzSxwk3n5ySL0hfBog779o8h68wK/7/APo/7cgyWp5jcXockbxdk5kFRkbeXWm4Fbi9FrdN381sA==} - - '@koa/cors@5.0.0': - resolution: {integrity: sha512-x/iUDjcS90W69PryLDIMgFyV21YLTnG9zOpPXS7Bkt2b8AsY3zZsIpOLBkYr9fBcF3HbkKaER5hOBZLfpLgYNw==} - engines: {node: '>= 14.0.0'} - - '@koa/router@10.1.1': - resolution: {integrity: sha512-ORNjq5z4EmQPriKbR0ER3k4Gh7YGNhWDL7JBW+8wXDrHLbWYKYSJaOJ9aN06npF5tbTxe2JBOsurpJDAvjiXKw==} - engines: {node: '>= 8.0.0'} - deprecated: '**IMPORTANT 10x+ PERFORMANCE UPGRADE**: Please upgrade to v12.0.1+ as we have fixed an issue with debuglog causing 10x slower router benchmark performance, see https://github.com/koajs/router/pull/173' - - '@lezer/common@1.2.1': - resolution: {integrity: sha512-yemX0ZD2xS/73llMZIK6KplkjIjf2EvAHcinDi/TfJ9hS25G0388+ClHt6/3but0oOxinTcQHJLDXh6w1crzFQ==} - - '@lezer/highlight@1.2.1': - resolution: {integrity: sha512-Z5duk4RN/3zuVO7Jq0pGLJ3qynpxUVsh7IbUbGj88+uV2ApSAn6kWg2au3iJb+0Zi7kKtqffIESgNcRXWZWmSA==} - - '@lezer/json@1.0.2': - resolution: {integrity: sha512-xHT2P4S5eeCYECyKNPhr4cbEL9tc8w83SPwRC373o9uEdrvGKTZoJVAGxpOsZckMlEh9W23Pc72ew918RWQOBQ==} - - '@lezer/lr@1.4.2': - resolution: {integrity: sha512-pu0K1jCIdnQ12aWNaAVU5bzi7Bd1w54J3ECgANPmYLtQKP0HBj2cE/5coBD66MT10xbtIuUr7tg0Shbsvk0mDA==} - - '@mapbox/node-pre-gyp@1.0.11': - resolution: {integrity: sha512-Yhlar6v9WQgUp/He7BdgzOz8lqMQ8sU+jkCq7Wx8Myc5YFJLbEe7lgui/V7G1qB1DJykHSGwreceSaD60Y0PUQ==} - hasBin: true - - '@mux/mux-node@8.8.0': - resolution: {integrity: sha512-7EMuTh62s3Ux+SrcKHoLvXgLDQzeHmbBnOf8xO9hEPLxU9SdV8XX7cqTkWnYlmVdP6ywIoBb4boQX2skI5qk5w==} - - '@noble/hashes@1.5.0': - resolution: {integrity: sha512-1j6kQFb7QRru7eKN3ZDvRcP13rugwdxZqCjbiAVZfIJwgj2A65UmT4TgARXGlXgnRkORLTDTrO19ZErt7+QXgA==} - engines: {node: ^14.21.3 || >=16} - - '@nodelib/fs.scandir@2.1.5': - resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==} - engines: {node: '>= 8'} - - '@nodelib/fs.stat@2.0.5': - resolution: {integrity: sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==} - engines: {node: '>= 8'} - - '@nodelib/fs.walk@1.2.8': - resolution: {integrity: sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==} - engines: {node: '>= 8'} - - '@paralleldrive/cuid2@2.2.2': - resolution: {integrity: sha512-ZOBkgDwEdoYVlSeRbYYXs0S9MejQofiVYoTbKzy/6GQa39/q5tQU2IX46+shYnUkpEl3wc+J6wRlar7r2EK2xA==} - - '@pkgjs/parseargs@0.11.0': - resolution: {integrity: sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==} - engines: {node: '>=14'} - - '@pkgr/utils@2.4.2': - resolution: {integrity: sha512-POgTXhjrTfbTV63DiFXav4lBHiICLKKwDeaKn9Nphwj7WH6m0hMMCaJkMyRWjgtPFyRKRVoMXXjczsTQRDEhYw==} - engines: {node: ^12.20.0 || ^14.18.0 || >=16.0.0} - - '@pmmmwh/react-refresh-webpack-plugin@0.5.11': - resolution: {integrity: sha512-7j/6vdTym0+qZ6u4XbSAxrWBGYSdCfTzySkj7WAFgDLmSyWlOrWvpyzxlFh5jtw9dn0oL/jtW+06XfFiisN3JQ==} - engines: {node: '>= 10.13'} - peerDependencies: - '@types/webpack': 4.x || 5.x - react-refresh: '>=0.10.0 <1.0.0' - sockjs-client: ^1.4.0 - type-fest: '>=0.17.0 <5.0.0' - webpack: '>=4.43.0 <6.0.0' - webpack-dev-server: 3.x || 4.x - webpack-hot-middleware: 2.x - webpack-plugin-serve: 0.x || 1.x - peerDependenciesMeta: - '@types/webpack': - optional: true - sockjs-client: - optional: true - type-fest: - optional: true - webpack-dev-server: - optional: true - webpack-hot-middleware: - optional: true - webpack-plugin-serve: - optional: true - - '@pnpm/config.env-replace@1.1.0': - resolution: {integrity: sha512-htyl8TWnKL7K/ESFa1oW2UB5lVDxuF5DpM7tBi6Hu2LNL3mWkIzNLG6N4zoCUP1lCKNxWy/3iu8mS8MvToGd6w==} - engines: {node: '>=12.22.0'} - - '@pnpm/network.ca-file@1.0.2': - resolution: {integrity: sha512-YcPQ8a0jwYU9bTdJDpXjMi7Brhkr1mXsXrUJvjqM2mQDgkRiz8jFaQGOdaLxgjtUfQgZhKy/O3cG/YwmgKaxLA==} - engines: {node: '>=12.22.0'} - - '@pnpm/npm-conf@2.3.1': - resolution: {integrity: sha512-c83qWb22rNRuB0UaVCI0uRPNRr8Z0FWnEIvT47jiHAmOIUHbBOg5XvV7pM5x+rKn9HRpjxquDbXYSXr3fAKFcw==} - engines: {node: '>=12'} - - '@polka/url@1.0.0-next.25': - resolution: {integrity: sha512-j7P6Rgr3mmtdkeDGTe0E/aYyWEWVtc5yFXtHCRHs28/jptDEWfaVOc5T7cblqy1XKPPfCxJc/8DwQ5YgLOZOVQ==} - - '@radix-ui/number@1.1.0': - resolution: {integrity: sha512-V3gRzhVNU1ldS5XhAPTom1fOIo4ccrjjJgmE+LI2h/WaFpHmx0MQApT+KZHnx8abG6Avtfcz4WoEciMnpFT3HQ==} - - '@radix-ui/primitive@1.0.1': - resolution: {integrity: sha512-yQ8oGX2GVsEYMWGxcovu1uGWPCxV5BFfeeYxqPmuAzUyLT9qmaMXSAhXpb0WrspIeqYzdJpkh2vHModJPgRIaw==} - - '@radix-ui/primitive@1.1.0': - resolution: {integrity: sha512-4Z8dn6Upk0qk4P74xBhZ6Hd/w0mPEzOOLxy4xiPXOXqjF7jZS0VAKk7/x/H6FyY2zCkYJqePf1G5KmkmNJ4RBA==} - - '@radix-ui/react-arrow@1.1.0': - resolution: {integrity: sha512-FmlW1rCg7hBpEBwFbjHwCW6AmWLQM6g/v0Sn8XbP9NvmSZ2San1FpQeyPtufzOMSIx7Y4dzjlHoifhp+7NkZhw==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-collection@1.0.3': - resolution: {integrity: sha512-3SzW+0PW7yBBoQlT8wNcGtaxaD0XSu0uLUFgrtHY08Acx05TaHaOmVLR73c0j/cqpDy53KBMO7s0dx2wmOIDIA==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 - react-dom: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-collection@1.1.0': - resolution: {integrity: sha512-GZsZslMJEyo1VKm5L1ZJY8tGDxZNPAoUeQUIbKeJfoi7Q4kmig5AsgLMYYuyYbfjd8fBmFORAIwYAkXMnXZgZw==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-compose-refs@1.0.1': - resolution: {integrity: sha512-fDSBgd44FKHa1FRMU59qBMPFcl2PZE+2nmqunj+BWFyYYjnhIDWL2ItDs3rrbJDQOtzt5nIebLCQc4QRfz6LJw==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-compose-refs@1.1.0': - resolution: {integrity: sha512-b4inOtiaOnYf9KWyO3jAeeCG6FeyfY6ldiEPanbUjWd+xIk5wZeHa8yVwmrJ2vderhu/BQvzCrJI0lHd+wIiqw==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-context@1.0.1': - resolution: {integrity: sha512-ebbrdFoYTcuZ0v4wG5tedGnp9tzcV8awzsxYph7gXUyvnNLuTIcCk1q17JEbnVhXAKG9oX3KtchwiMIAYp9NLg==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-context@1.1.0': - resolution: {integrity: sha512-OKrckBy+sMEgYM/sMmqmErVn0kZqrHPJze+Ql3DzYsDDp0hl0L62nx/2122/Bvps1qz645jlcu2tD9lrRSdf8A==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-direction@1.0.1': - resolution: {integrity: sha512-RXcvnXgyvYvBEOhCBuddKecVkoMiI10Jcm5cTI7abJRAHYfFxeu+FBQs/DvdxSYucxR5mna0dNsL6QFlds5TMA==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-direction@1.1.0': - resolution: {integrity: sha512-BUuBvgThEiAXh2DWu93XsT+a3aWrGqolGlqqw5VU1kG7p/ZH2cuDlM1sRLNnY3QcBS69UIz2mcKhMxDsdewhjg==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-dismissable-layer@1.1.0': - resolution: {integrity: sha512-/UovfmmXGptwGcBQawLzvn2jOfM0t4z3/uKffoBlj724+n3FvBbZ7M0aaBOmkp6pqFYpO4yx8tSVJjx3Fl2jig==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-dropdown-menu@2.1.1': - resolution: {integrity: sha512-y8E+x9fBq9qvteD2Zwa4397pUVhYsh9iq44b5RD5qu1GMJWBCBuVg1hMyItbc6+zH00TxGRqd9Iot4wzf3OoBQ==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-focus-guards@1.0.1': - resolution: {integrity: sha512-Rect2dWbQ8waGzhMavsIbmSVCgYxkXLxxR3ZvCX79JOglzdEy4JXMb98lq4hPxUbLr77nP0UOGf4rcMU+s1pUA==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-focus-guards@1.1.0': - resolution: {integrity: sha512-w6XZNUPVv6xCpZUqb/yN9DL6auvpGX3C/ee6Hdi16v2UUy25HV2Q5bcflsiDyT/g5RwbPQ/GIT1vLkeRb+ITBw==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-focus-scope@1.0.4': - resolution: {integrity: sha512-sL04Mgvf+FmyvZeYfNu1EPAaaxD+aw7cYeIB9L9Fvq8+urhltTRaEo5ysKOpHuKPclsZcSUMKlN05x4u+CINpA==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 - react-dom: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-focus-scope@1.1.0': - resolution: {integrity: sha512-200UD8zylvEyL8Bx+z76RJnASR2gRMuxlgFCPAe/Q/679a/r0eK3MBVYMb7vZODZcffZBdob1EGnky78xmVvcA==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-id@1.0.1': - resolution: {integrity: sha512-tI7sT/kqYp8p96yGWY1OAnLHrqDgzHefRBKQ2YAkBS5ja7QLcZ9Z/uY7bEjPUatf8RomoXM8/1sMj1IJaE5UzQ==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-id@1.1.0': - resolution: {integrity: sha512-EJUrI8yYh7WOjNOqpoJaf1jlFIH2LvtgAl+YcFqNCa+4hj64ZXmPkAKOFs/ukjz3byN6bdb/AVUqHkI8/uWWMA==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-menu@2.1.1': - resolution: {integrity: sha512-oa3mXRRVjHi6DZu/ghuzdylyjaMXLymx83irM7hTxutQbD+7IhPKdMdRHD26Rm+kHRrWcrUkkRPv5pd47a2xFQ==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-popper@1.2.0': - resolution: {integrity: sha512-ZnRMshKF43aBxVWPWvbj21+7TQCvhuULWJ4gNIKYpRlQt5xGRhLx66tMp8pya2UkGHTSlhpXwmjqltDYHhw7Vg==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-portal@1.1.1': - resolution: {integrity: sha512-A3UtLk85UtqhzFqtoC8Q0KvR2GbXF3mtPgACSazajqq6A41mEQgo53iPzY4i6BwDxlIFqWIhiQ2G729n+2aw/g==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-presence@1.1.0': - resolution: {integrity: sha512-Gq6wuRN/asf9H/E/VzdKoUtT8GC9PQc9z40/vEr0VCJ4u5XvvhWIrSsCB6vD2/cH7ugTdSfYq9fLJCcM00acrQ==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-primitive@1.0.3': - resolution: {integrity: sha512-yi58uVyoAcK/Nq1inRY56ZSjKypBNKTa/1mcL8qdl6oJeEaDbOldlzrGn7P6Q3Id5d+SYNGc5AJgc4vGhjs5+g==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 - react-dom: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-primitive@2.0.0': - resolution: {integrity: sha512-ZSpFm0/uHa8zTvKBDjLFWLo8dkr4MBsiDLz0g3gMUwqgLHz9rTaRRGYDgvZPtBJgYCBKXkS9fzmoySgr8CO6Cw==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-roving-focus@1.0.4': - resolution: {integrity: sha512-2mUg5Mgcu001VkGy+FfzZyzbmuUWzgWkj3rvv4yu+mLw03+mTzbxZHvfcGyFp2b8EkQeMkpRQ5FiA2Vr2O6TeQ==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 - react-dom: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-roving-focus@1.1.0': - resolution: {integrity: sha512-EA6AMGeq9AEeQDeSH0aZgG198qkfHSbvWTf1HvoDmOB5bBG/qTxjYMWUKMnYiV6J/iP/J8MEFSuB2zRU2n7ODA==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-separator@1.0.3': - resolution: {integrity: sha512-itYmTy/kokS21aiV5+Z56MZB54KrhPgn6eHDKkFeOLR34HMN2s8PaN47qZZAGnvupcjxHaFZnW4pQEh0BvvVuw==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 - react-dom: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-slot@1.0.2': - resolution: {integrity: sha512-YeTpuq4deV+6DusvVUW4ivBgnkHwECUu0BiN43L5UCDFgdhsRUWAghhTF5MbvNTPzmiFOx90asDSUjWuCNapwg==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-slot@1.1.0': - resolution: {integrity: sha512-FUCf5XMfmW4dtYl69pdS4DbxKy8nj4M7SafBgPllysxmdachynNflAdp/gCsnYWNDnge6tI9onzMp5ARYc1KNw==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-toggle-group@1.0.4': - resolution: {integrity: sha512-Uaj/M/cMyiyT9Bx6fOZO0SAG4Cls0GptBWiBmBxofmDbNVnYYoyRWj/2M/6VCi/7qcXFWnHhRUfdfZFvvkuu8A==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 - react-dom: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-toggle@1.0.3': - resolution: {integrity: sha512-Pkqg3+Bc98ftZGsl60CLANXQBBQ4W3mTFS9EJvNxKMZ7magklKV69/id1mlAlOFDDfHvlCms0fx8fA4CMKDJHg==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 - react-dom: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-toolbar@1.0.4': - resolution: {integrity: sha512-tBgmM/O7a07xbaEkYJWYTXkIdU/1pW4/KZORR43toC/4XWyBCURK0ei9kMUdp+gTPPKBgYLxXmRSH1EVcIDp8Q==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 - react-dom: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/react-use-callback-ref@1.0.1': - resolution: {integrity: sha512-D94LjX4Sp0xJFVaoQOd3OO9k7tpBYNOXdVhkltUbGv2Qb9OXdrg/CpsjlZv7ia14Sylv398LswWBVVu5nqKzAQ==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-use-callback-ref@1.1.0': - resolution: {integrity: sha512-CasTfvsy+frcFkbXtSJ2Zu9JHpN8TYKxkgJGWbjiZhFivxaeW7rMeZt7QELGVLaYVfFMsKHjb7Ak0nMEe+2Vfw==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-use-controllable-state@1.0.1': - resolution: {integrity: sha512-Svl5GY5FQeN758fWKrjM6Qb7asvXeiZltlT4U2gVfl8Gx5UAv2sMR0LWo8yhsIZh2oQ0eFdZ59aoOOMV7b47VA==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-use-controllable-state@1.1.0': - resolution: {integrity: sha512-MtfMVJiSr2NjzS0Aa90NPTnvTSg6C/JLCV7ma0W6+OMV78vd8OyRpID+Ng9LxzsPbLeuBnWBA1Nq30AtBIDChw==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-use-escape-keydown@1.1.0': - resolution: {integrity: sha512-L7vwWlR1kTTQ3oh7g1O0CBF3YCyyTj8NmhLR+phShpyA50HCfBFKVJTpshm9PzLiKmehsrQzTYTpX9HvmC9rhw==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-use-layout-effect@1.0.1': - resolution: {integrity: sha512-v/5RegiJWYdoCvMnITBkNNx6bCj20fiaJnWtRkU18yITptraXjffz5Qbn05uOiQnOvi+dbkznkoaMltz1GnszQ==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-use-layout-effect@1.1.0': - resolution: {integrity: sha512-+FPE0rOdziWSrH9athwI1R0HDVbWlEhd+FR+aSDk4uWGmSJ9Z54sdZVDQPZAinJhJXwfT+qnj969mCsT2gfm5w==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-use-previous@1.1.0': - resolution: {integrity: sha512-Z/e78qg2YFnnXcW88A4JmTtm4ADckLno6F7OXotmkQfeuCVaKuYzqAATPhVzl3delXE7CxIV8shofPn3jPc5Og==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-use-rect@1.1.0': - resolution: {integrity: sha512-0Fmkebhr6PiseyZlYAOtLS+nb7jLmpqTrJyv61Pe68MKYW6OWdRE2kI70TaYY27u7H0lajqM3hSMMLFq18Z7nQ==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-use-size@1.1.0': - resolution: {integrity: sha512-XW3/vWuIXHa+2Uwcc2ABSfcCledmXhhQPlGbfcRXbiUQI5Icjcg19BGCZVKKInYbvUCut/ufbbLLPFC5cbb1hw==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - - '@radix-ui/react-visually-hidden@1.1.0': - resolution: {integrity: sha512-N8MDZqtgCgG5S3aV60INAB475osJousYpZ4cTJ2cFbMpdHS5Y6loLTH8LPtkj2QN0x93J30HT/M3qJXM0+lyeQ==} - peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - - '@radix-ui/rect@1.1.0': - resolution: {integrity: sha512-A9+lCBZoaMJlVKcRBz2YByCG+Cp2t6nAnMnNba+XiWxnj6r4JUFqfsgwocMBZU9LPtdxC6wB56ySYpc7LQIoJg==} - - '@react-dnd/asap@5.0.2': - resolution: {integrity: sha512-WLyfoHvxhs0V9U+GTsGilGgf2QsPl6ZZ44fnv0/b8T3nQyvzxidxsg/ZltbWssbsRDlYW8UKSQMTGotuTotZ6A==} - - '@react-dnd/invariant@4.0.2': - resolution: {integrity: sha512-xKCTqAK/FFauOM9Ta2pswIyT3D8AQlfrYdOi/toTPEhqCuAs1v5tcJ3Y08Izh1cJ5Jchwy9SeAXmMg6zrKs2iw==} - - '@react-dnd/shallowequal@4.0.2': - resolution: {integrity: sha512-/RVXdLvJxLg4QKvMoM5WlwNR9ViO9z8B/qPcc+C0Sa/teJY7QG7kJ441DwzOjMYEY7GmU4dj5EcGHIkKZiQZCA==} - - '@reduxjs/toolkit@1.9.7': - resolution: {integrity: sha512-t7v8ZPxhhKgOKtU+uyJT13lu4vL7az5aFi4IdoDs/eS548edn2M8Ik9h8fxgvMjGoAUVFSt6ZC1P5cWmQ014QQ==} - peerDependencies: - react: ^16.9.0 || ^17.0.0 || ^18 - react-redux: ^7.2.1 || ^8.0.2 - peerDependenciesMeta: - react: - optional: true - react-redux: - optional: true - - '@remix-run/router@1.19.1': - resolution: {integrity: sha512-S45oynt/WH19bHbIXjtli6QmwNYvaz+vtnubvNpNDvUOoA/OWh6j1OikIP3G+v5GHdxyC6EXoChG3HgYGEUfcg==} - engines: {node: '>=14.0.0'} - - '@rollup/rollup-android-arm-eabi@4.21.2': - resolution: {integrity: sha512-fSuPrt0ZO8uXeS+xP3b+yYTCBUd05MoSp2N/MFOgjhhUhMmchXlpTQrTpI8T+YAwAQuK7MafsCOxW7VrPMrJcg==} - cpu: [arm] - os: [android] - - '@rollup/rollup-android-arm64@4.21.2': - resolution: {integrity: sha512-xGU5ZQmPlsjQS6tzTTGwMsnKUtu0WVbl0hYpTPauvbRAnmIvpInhJtgjj3mcuJpEiuUw4v1s4BimkdfDWlh7gA==} - cpu: [arm64] - os: [android] - - '@rollup/rollup-darwin-arm64@4.21.2': - resolution: {integrity: sha512-99AhQ3/ZMxU7jw34Sq8brzXqWH/bMnf7ZVhvLk9QU2cOepbQSVTns6qoErJmSiAvU3InRqC2RRZ5ovh1KN0d0Q==} - cpu: [arm64] - os: [darwin] - - '@rollup/rollup-darwin-x64@4.21.2': - resolution: {integrity: sha512-ZbRaUvw2iN/y37x6dY50D8m2BnDbBjlnMPotDi/qITMJ4sIxNY33HArjikDyakhSv0+ybdUxhWxE6kTI4oX26w==} - cpu: [x64] - os: [darwin] - - '@rollup/rollup-linux-arm-gnueabihf@4.21.2': - resolution: {integrity: sha512-ztRJJMiE8nnU1YFcdbd9BcH6bGWG1z+jP+IPW2oDUAPxPjo9dverIOyXz76m6IPA6udEL12reYeLojzW2cYL7w==} - cpu: [arm] - os: [linux] - - '@rollup/rollup-linux-arm-musleabihf@4.21.2': - resolution: {integrity: sha512-flOcGHDZajGKYpLV0JNc0VFH361M7rnV1ee+NTeC/BQQ1/0pllYcFmxpagltANYt8FYf9+kL6RSk80Ziwyhr7w==} - cpu: [arm] - os: [linux] - - '@rollup/rollup-linux-arm64-gnu@4.21.2': - resolution: {integrity: sha512-69CF19Kp3TdMopyteO/LJbWufOzqqXzkrv4L2sP8kfMaAQ6iwky7NoXTp7bD6/irKgknDKM0P9E/1l5XxVQAhw==} - cpu: [arm64] - os: [linux] - - '@rollup/rollup-linux-arm64-musl@4.21.2': - resolution: {integrity: sha512-48pD/fJkTiHAZTnZwR0VzHrao70/4MlzJrq0ZsILjLW/Ab/1XlVUStYyGt7tdyIiVSlGZbnliqmult/QGA2O2w==} - cpu: [arm64] - os: [linux] - - '@rollup/rollup-linux-powerpc64le-gnu@4.21.2': - resolution: {integrity: sha512-cZdyuInj0ofc7mAQpKcPR2a2iu4YM4FQfuUzCVA2u4HI95lCwzjoPtdWjdpDKyHxI0UO82bLDoOaLfpZ/wviyQ==} - cpu: [ppc64] - os: [linux] - - '@rollup/rollup-linux-riscv64-gnu@4.21.2': - resolution: {integrity: sha512-RL56JMT6NwQ0lXIQmMIWr1SW28z4E4pOhRRNqwWZeXpRlykRIlEpSWdsgNWJbYBEWD84eocjSGDu/XxbYeCmwg==} - cpu: [riscv64] - os: [linux] - - '@rollup/rollup-linux-s390x-gnu@4.21.2': - resolution: {integrity: sha512-PMxkrWS9z38bCr3rWvDFVGD6sFeZJw4iQlhrup7ReGmfn7Oukrr/zweLhYX6v2/8J6Cep9IEA/SmjXjCmSbrMQ==} - cpu: [s390x] - os: [linux] - - '@rollup/rollup-linux-x64-gnu@4.21.2': - resolution: {integrity: sha512-B90tYAUoLhU22olrafY3JQCFLnT3NglazdwkHyxNDYF/zAxJt5fJUB/yBoWFoIQ7SQj+KLe3iL4BhOMa9fzgpw==} - cpu: [x64] - os: [linux] - - '@rollup/rollup-linux-x64-musl@4.21.2': - resolution: {integrity: sha512-7twFizNXudESmC9oneLGIUmoHiiLppz/Xs5uJQ4ShvE6234K0VB1/aJYU3f/4g7PhssLGKBVCC37uRkkOi8wjg==} - cpu: [x64] - os: [linux] - - '@rollup/rollup-win32-arm64-msvc@4.21.2': - resolution: {integrity: sha512-9rRero0E7qTeYf6+rFh3AErTNU1VCQg2mn7CQcI44vNUWM9Ze7MSRS/9RFuSsox+vstRt97+x3sOhEey024FRQ==} - cpu: [arm64] - os: [win32] - - '@rollup/rollup-win32-ia32-msvc@4.21.2': - resolution: {integrity: sha512-5rA4vjlqgrpbFVVHX3qkrCo/fZTj1q0Xxpg+Z7yIo3J2AilW7t2+n6Q8Jrx+4MrYpAnjttTYF8rr7bP46BPzRw==} - cpu: [ia32] - os: [win32] - - '@rollup/rollup-win32-x64-msvc@4.21.2': - resolution: {integrity: sha512-6UUxd0+SKomjdzuAcp+HAmxw1FlGBnl1v2yEPSabtx4lBfdXHDVsW7+lQkgz9cNFJGY3AWR7+V8P5BqkD9L9nA==} - cpu: [x64] - os: [win32] - - '@rushstack/node-core-library@5.7.0': - resolution: {integrity: sha512-Ff9Cz/YlWu9ce4dmqNBZpA45AEya04XaBFIjV7xTVeEf+y/kTjEasmozqFELXlNG4ROdevss75JrrZ5WgufDkQ==} - peerDependencies: - '@types/node': '*' - peerDependenciesMeta: - '@types/node': - optional: true - - '@rushstack/terminal@0.14.0': - resolution: {integrity: sha512-juTKMAMpTIJKudeFkG5slD8Z/LHwNwGZLtU441l/u82XdTBfsP+LbGKJLCNwP5se+DMCT55GB8x9p6+C4UL7jw==} - peerDependencies: - '@types/node': '*' - peerDependenciesMeta: - '@types/node': - optional: true - - '@rushstack/ts-command-line@4.22.6': - resolution: {integrity: sha512-QSRqHT/IfoC5nk9zn6+fgyqOPXHME0BfchII9EUPR19pocsNp/xSbeBCbD3PIR2Lg+Q5qk7OFqk1VhWPMdKHJg==} - - '@sentry/core@6.19.7': - resolution: {integrity: sha512-tOfZ/umqB2AcHPGbIrsFLcvApdTm9ggpi/kQZFkej7kMphjT+SGBiQfYtjyg9jcRW+ilAR4JXC9BGKsdEQ+8Vw==} - engines: {node: '>=6'} - - '@sentry/hub@6.19.7': - resolution: {integrity: sha512-y3OtbYFAqKHCWezF0EGGr5lcyI2KbaXW2Ik7Xp8Mu9TxbSTuwTe4rTntwg8ngPjUQU3SUHzgjqVB8qjiGqFXCA==} - engines: {node: '>=6'} - - '@sentry/minimal@6.19.7': - resolution: {integrity: sha512-wcYmSJOdvk6VAPx8IcmZgN08XTXRwRtB1aOLZm+MVHjIZIhHoBGZJYTVQS/BWjldsamj2cX3YGbGXNunaCfYJQ==} - engines: {node: '>=6'} - - '@sentry/node@6.19.7': - resolution: {integrity: sha512-gtmRC4dAXKODMpHXKfrkfvyBL3cI8y64vEi3fDD046uqYcrWdgoQsffuBbxMAizc6Ez1ia+f0Flue6p15Qaltg==} - engines: {node: '>=6'} - - '@sentry/types@6.19.7': - resolution: {integrity: sha512-jH84pDYE+hHIbVnab3Hr+ZXr1v8QABfhx39KknxqKWr2l0oEItzepV0URvbEhB446lk/S/59230dlUUIBGsXbg==} - engines: {node: '>=6'} - - '@sentry/utils@6.19.7': - resolution: {integrity: sha512-z95ECmE3i9pbWoXQrD/7PgkBAzJYR+iXtPuTkpBjDKs86O3mT+PXOT3BAn79w2wkn7/i3vOGD2xVr1uiMl26dA==} - engines: {node: '>=6'} - - '@simov/deep-extend@1.0.0': - resolution: {integrity: sha512-Arv8/ZPcdKAMJnNF8cks35mPq1y3JnwH1lWpfWDKlJoj+Vw2xmA4+oL7m9GVHTgdX0mGFR7bCPTBTGbxhnfJJw==} - engines: {node: '>=4.0.0'} - - '@sindresorhus/is@4.6.0': - resolution: {integrity: sha512-t09vSN3MdfsyCHoFcTRCH/iUtG7OJ0CsjzB8cjAmKc/va/kIgeDI/TxsigdncE/4be734m0cvIYwNaV4i2XqAw==} - engines: {node: '>=10'} - - '@sindresorhus/slugify@1.1.0': - resolution: {integrity: sha512-ujZRbmmizX26yS/HnB3P9QNlNa4+UvHh+rIse3RbOXLp8yl6n1TxB4t7NHggtVgS8QmmOtzXo48kCxZGACpkPw==} - engines: {node: '>=10'} - - '@sindresorhus/transliterate@0.1.2': - resolution: {integrity: sha512-5/kmIOY9FF32nicXH+5yLNTX4NJ4atl7jRgqAJuIn/iyDFXBktOKDxCvyGE/EzmF4ngSUvjXxQUQlQiZ5lfw+w==} - engines: {node: '>=10'} - - '@smithy/abort-controller@3.1.1': - resolution: {integrity: sha512-MBJBiidoe+0cTFhyxT8g+9g7CeVccLM0IOKKUMCNQ1CNMJ/eIfoo0RTfVrXOONEI1UCN1W+zkiHSbzUNE9dZtQ==} - engines: {node: '>=16.0.0'} - - '@smithy/chunked-blob-reader-native@3.0.0': - resolution: {integrity: sha512-VDkpCYW+peSuM4zJip5WDfqvg2Mo/e8yxOv3VF1m11y7B8KKMKVFtmZWDe36Fvk8rGuWrPZHHXZ7rR7uM5yWyg==} - - '@smithy/chunked-blob-reader@3.0.0': - resolution: {integrity: sha512-sbnURCwjF0gSToGlsBiAmd1lRCmSn72nu9axfJu5lIx6RUEgHu6GwTMbqCdhQSi0Pumcm5vFxsi9XWXb2mTaoA==} - - '@smithy/config-resolver@3.0.5': - resolution: {integrity: sha512-SkW5LxfkSI1bUC74OtfBbdz+grQXYiPYolyu8VfpLIjEoN/sHVBlLeGXMQ1vX4ejkgfv6sxVbQJ32yF2cl1veA==} - engines: {node: '>=16.0.0'} - - '@smithy/core@2.4.0': - resolution: {integrity: sha512-cHXq+FneIF/KJbt4q4pjN186+Jf4ZB0ZOqEaZMBhT79srEyGDDBV31NqBRBjazz8ppQ1bJbDJMY9ba5wKFV36w==} - engines: {node: '>=16.0.0'} - - '@smithy/credential-provider-imds@3.2.0': - resolution: {integrity: sha512-0SCIzgd8LYZ9EJxUjLXBmEKSZR/P/w6l7Rz/pab9culE/RWuqelAKGJvn5qUOl8BgX8Yj5HWM50A5hiB/RzsgA==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-codec@3.1.2': - resolution: {integrity: sha512-0mBcu49JWt4MXhrhRAlxASNy0IjDRFU+aWNDRal9OtUJvJNiwDuyKMUONSOjLjSCeGwZaE0wOErdqULer8r7yw==} - - '@smithy/eventstream-serde-browser@3.0.6': - resolution: {integrity: sha512-2hM54UWQUOrki4BtsUI1WzmD13/SeaqT/AB3EUJKbcver/WgKNaiJ5y5F5XXuVe6UekffVzuUDrBZVAA3AWRpQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-config-resolver@3.0.3': - resolution: {integrity: sha512-NVTYjOuYpGfrN/VbRQgn31x73KDLfCXCsFdad8DiIc3IcdxL+dYA9zEQPyOP7Fy2QL8CPy2WE4WCUD+ZsLNfaQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-node@3.0.5': - resolution: {integrity: sha512-+upXvnHNyZP095s11jF5dhGw/Ihzqwl5G+/KtMnoQOpdfC3B5HYCcDVG9EmgkhJMXJlM64PyN5gjJl0uXFQehQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-universal@3.0.5': - resolution: {integrity: sha512-5u/nXbyoh1s4QxrvNre9V6vfyoLWuiVvvd5TlZjGThIikc3G+uNiG9uOTCWweSRjv1asdDIWK7nOmN7le4RYHQ==} - engines: {node: '>=16.0.0'} - - '@smithy/fetch-http-handler@3.2.4': - resolution: {integrity: sha512-kBprh5Gs5h7ug4nBWZi1FZthdqSM+T7zMmsZxx0IBvWUn7dK3diz2SHn7Bs4dQGFDk8plDv375gzenDoNwrXjg==} - - '@smithy/hash-blob-browser@3.1.2': - resolution: {integrity: sha512-hAbfqN2UbISltakCC2TP0kx4LqXBttEv2MqSPE98gVuDFMf05lU+TpC41QtqGP3Ff5A3GwZMPfKnEy0VmEUpmg==} - - '@smithy/hash-node@3.0.3': - resolution: {integrity: sha512-2ctBXpPMG+B3BtWSGNnKELJ7SH9e4TNefJS0cd2eSkOOROeBnnVBnAy9LtJ8tY4vUEoe55N4CNPxzbWvR39iBw==} - engines: {node: '>=16.0.0'} - - '@smithy/hash-stream-node@3.1.2': - resolution: {integrity: sha512-PBgDMeEdDzi6JxKwbfBtwQG9eT9cVwsf0dZzLXoJF4sHKHs5HEo/3lJWpn6jibfJwT34I1EBXpBnZE8AxAft6g==} - engines: {node: '>=16.0.0'} - - '@smithy/invalid-dependency@3.0.3': - resolution: {integrity: sha512-ID1eL/zpDULmHJbflb864k72/SNOZCADRc9i7Exq3RUNJw6raWUSlFEQ+3PX3EYs++bTxZB2dE9mEHTQLv61tw==} - - '@smithy/is-array-buffer@2.2.0': - resolution: {integrity: sha512-GGP3O9QFD24uGeAXYUjwSTXARoqpZykHadOmA8G5vfJPK0/DC67qa//0qvqrJzL1xc8WQWX7/yc7fwudjPHPhA==} - engines: {node: '>=14.0.0'} - - '@smithy/is-array-buffer@3.0.0': - resolution: {integrity: sha512-+Fsu6Q6C4RSJiy81Y8eApjEB5gVtM+oFKTffg+jSuwtvomJJrhUJBu2zS8wjXSgH/g1MKEWrzyChTBe6clb5FQ==} - engines: {node: '>=16.0.0'} - - '@smithy/md5-js@3.0.3': - resolution: {integrity: sha512-O/SAkGVwpWmelpj/8yDtsaVe6sINHLB1q8YE/+ZQbDxIw3SRLbTZuRaI10K12sVoENdnHqzPp5i3/H+BcZ3m3Q==} - - '@smithy/middleware-content-length@3.0.5': - resolution: {integrity: sha512-ILEzC2eyxx6ncej3zZSwMpB5RJ0zuqH7eMptxC4KN3f+v9bqT8ohssKbhNR78k/2tWW+KS5Spw+tbPF4Ejyqvw==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-endpoint@3.1.0': - resolution: {integrity: sha512-5y5aiKCEwg9TDPB4yFE7H6tYvGFf1OJHNczeY10/EFF8Ir8jZbNntQJxMWNfeQjC1mxPsaQ6mR9cvQbf+0YeMw==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-retry@3.0.15': - resolution: {integrity: sha512-iTMedvNt1ApdvkaoE8aSDuwaoc+BhvHqttbA/FO4Ty+y/S5hW6Ci/CTScG7vam4RYJWZxdTElc3MEfHRVH6cgQ==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-serde@3.0.3': - resolution: {integrity: sha512-puUbyJQBcg9eSErFXjKNiGILJGtiqmuuNKEYNYfUD57fUl4i9+mfmThtQhvFXU0hCVG0iEJhvQUipUf+/SsFdA==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-stack@3.0.3': - resolution: {integrity: sha512-r4klY9nFudB0r9UdSMaGSyjyQK5adUyPnQN/ZM6M75phTxOdnc/AhpvGD1fQUvgmqjQEBGCwpnPbDm8pH5PapA==} - engines: {node: '>=16.0.0'} - - '@smithy/node-config-provider@3.1.4': - resolution: {integrity: sha512-YvnElQy8HR4vDcAjoy7Xkx9YT8xZP4cBXcbJSgm/kxmiQu08DwUwj8rkGnyoJTpfl/3xYHH+d8zE+eHqoDCSdQ==} - engines: {node: '>=16.0.0'} - - '@smithy/node-http-handler@3.1.4': - resolution: {integrity: sha512-+UmxgixgOr/yLsUxcEKGH0fMNVteJFGkmRltYFHnBMlogyFdpzn2CwqWmxOrfJELhV34v0WSlaqG1UtE1uXlJg==} - engines: {node: '>=16.0.0'} - - '@smithy/property-provider@3.1.3': - resolution: {integrity: sha512-zahyOVR9Q4PEoguJ/NrFP4O7SMAfYO1HLhB18M+q+Z4KFd4V2obiMnlVoUFzFLSPeVt1POyNWneHHrZaTMoc/g==} - engines: {node: '>=16.0.0'} - - '@smithy/protocol-http@4.1.0': - resolution: {integrity: sha512-dPVoHYQ2wcHooGXg3LQisa1hH0e4y0pAddPMeeUPipI1tEOqL6A4N0/G7abeq+K8wrwSgjk4C0wnD1XZpJm5aA==} - engines: {node: '>=16.0.0'} - - '@smithy/querystring-builder@3.0.3': - resolution: {integrity: sha512-vyWckeUeesFKzCDaRwWLUA1Xym9McaA6XpFfAK5qI9DKJ4M33ooQGqvM4J+LalH4u/Dq9nFiC8U6Qn1qi0+9zw==} - engines: {node: '>=16.0.0'} - - '@smithy/querystring-parser@3.0.3': - resolution: {integrity: sha512-zahM1lQv2YjmznnfQsWbYojFe55l0SLG/988brlLv1i8z3dubloLF+75ATRsqPBboUXsW6I9CPGE5rQgLfY0vQ==} - engines: {node: '>=16.0.0'} - - '@smithy/service-error-classification@3.0.3': - resolution: {integrity: sha512-Jn39sSl8cim/VlkLsUhRFq/dKDnRUFlfRkvhOJaUbLBXUsLRLNf9WaxDv/z9BjuQ3A6k/qE8af1lsqcwm7+DaQ==} - engines: {node: '>=16.0.0'} - - '@smithy/shared-ini-file-loader@3.1.4': - resolution: {integrity: sha512-qMxS4hBGB8FY2GQqshcRUy1K6k8aBWP5vwm8qKkCT3A9K2dawUwOIJfqh9Yste/Bl0J2lzosVyrXDj68kLcHXQ==} - engines: {node: '>=16.0.0'} - - '@smithy/signature-v4@4.1.0': - resolution: {integrity: sha512-aRryp2XNZeRcOtuJoxjydO6QTaVhxx/vjaR+gx7ZjaFgrgPRyZ3HCTbfwqYj6ZWEBHkCSUfcaymKPURaByukag==} - engines: {node: '>=16.0.0'} - - '@smithy/smithy-client@3.2.0': - resolution: {integrity: sha512-pDbtxs8WOhJLJSeaF/eAbPgXg4VVYFlRcL/zoNYA5WbG3wBL06CHtBSg53ppkttDpAJ/hdiede+xApip1CwSLw==} - engines: {node: '>=16.0.0'} - - '@smithy/types@3.3.0': - resolution: {integrity: sha512-IxvBBCTFDHbVoK7zIxqA1ZOdc4QfM5HM7rGleCuHi7L1wnKv5Pn69xXJQ9hgxH60ZVygH9/JG0jRgtUncE3QUA==} - engines: {node: '>=16.0.0'} - - '@smithy/url-parser@3.0.3': - resolution: {integrity: sha512-pw3VtZtX2rg+s6HMs6/+u9+hu6oY6U7IohGhVNnjbgKy86wcIsSZwgHrFR+t67Uyxvp4Xz3p3kGXXIpTNisq8A==} - - '@smithy/util-base64@3.0.0': - resolution: {integrity: sha512-Kxvoh5Qtt0CDsfajiZOCpJxgtPHXOKwmM+Zy4waD43UoEMA+qPxxa98aE/7ZhdnBFZFXMOiBR5xbcaMhLtznQQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-body-length-browser@3.0.0': - resolution: {integrity: sha512-cbjJs2A1mLYmqmyVl80uoLTJhAcfzMOyPgjwAYusWKMdLeNtzmMz9YxNl3/jRLoxSS3wkqkf0jwNdtXWtyEBaQ==} - - '@smithy/util-body-length-node@3.0.0': - resolution: {integrity: sha512-Tj7pZ4bUloNUP6PzwhN7K386tmSmEET9QtQg0TgdNOnxhZvCssHji+oZTUIuzxECRfG8rdm2PMw2WCFs6eIYkA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-buffer-from@2.2.0': - resolution: {integrity: sha512-IJdWBbTcMQ6DA0gdNhh/BwrLkDR+ADW5Kr1aZmd4k3DIF6ezMV4R2NIAmT08wQJ3yUK82thHWmC/TnK/wpMMIA==} - engines: {node: '>=14.0.0'} - - '@smithy/util-buffer-from@3.0.0': - resolution: {integrity: sha512-aEOHCgq5RWFbP+UDPvPot26EJHjOC+bRgse5A8V3FSShqd5E5UN4qc7zkwsvJPPAVsf73QwYcHN1/gt/rtLwQA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-config-provider@3.0.0': - resolution: {integrity: sha512-pbjk4s0fwq3Di/ANL+rCvJMKM5bzAQdE5S/6RL5NXgMExFAi6UgQMPOm5yPaIWPpr+EOXKXRonJ3FoxKf4mCJQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-defaults-mode-browser@3.0.15': - resolution: {integrity: sha512-FZ4Psa3vjp8kOXcd3HJOiDPBCWtiilLl57r0cnNtq/Ga9RSDrM5ERL6xt+tO43+2af6Pn5Yp92x2n5vPuduNfg==} - engines: {node: '>= 10.0.0'} - - '@smithy/util-defaults-mode-node@3.0.15': - resolution: {integrity: sha512-KSyAAx2q6d0t6f/S4XB2+3+6aQacm3aLMhs9aLMqn18uYGUepbdssfogW5JQZpc6lXNBnp0tEnR5e9CEKmEd7A==} - engines: {node: '>= 10.0.0'} - - '@smithy/util-endpoints@2.0.5': - resolution: {integrity: sha512-ReQP0BWihIE68OAblC/WQmDD40Gx+QY1Ez8mTdFMXpmjfxSyz2fVQu3A4zXRfQU9sZXtewk3GmhfOHswvX+eNg==} - engines: {node: '>=16.0.0'} - - '@smithy/util-hex-encoding@3.0.0': - resolution: {integrity: sha512-eFndh1WEK5YMUYvy3lPlVmYY/fZcQE1D8oSf41Id2vCeIkKJXPcYDCZD+4+xViI6b1XSd7tE+s5AmXzz5ilabQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-middleware@3.0.3': - resolution: {integrity: sha512-l+StyYYK/eO3DlVPbU+4Bi06Jjal+PFLSMmlWM1BEwyLxZ3aKkf1ROnoIakfaA7mC6uw3ny7JBkau4Yc+5zfWw==} - engines: {node: '>=16.0.0'} - - '@smithy/util-retry@3.0.3': - resolution: {integrity: sha512-AFw+hjpbtVApzpNDhbjNG5NA3kyoMs7vx0gsgmlJF4s+yz1Zlepde7J58zpIRIsdjc+emhpAITxA88qLkPF26w==} - engines: {node: '>=16.0.0'} - - '@smithy/util-stream@3.1.3': - resolution: {integrity: sha512-FIv/bRhIlAxC0U7xM1BCnF2aDRPq0UaelqBHkM2lsCp26mcBbgI0tCVTv+jGdsQLUmAMybua/bjDsSu8RQHbmw==} - engines: {node: '>=16.0.0'} - - '@smithy/util-uri-escape@3.0.0': - resolution: {integrity: sha512-LqR7qYLgZTD7nWLBecUi4aqolw8Mhza9ArpNEQ881MJJIU2sE5iHCK6TdyqqzcDLy0OPe10IY4T8ctVdtynubg==} - engines: {node: '>=16.0.0'} - - '@smithy/util-utf8@2.3.0': - resolution: {integrity: sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A==} - engines: {node: '>=14.0.0'} - - '@smithy/util-utf8@3.0.0': - resolution: {integrity: sha512-rUeT12bxFnplYDe815GXbq/oixEGHfRFFtcTF3YdDi/JaENIM6aSYYLJydG83UNzLXeRI5K8abYd/8Sp/QM0kA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-waiter@3.1.2': - resolution: {integrity: sha512-4pP0EV3iTsexDx+8PPGAKCQpd/6hsQBaQhqWzU4hqKPHN5epPsxKbvUTIiYIHTxaKt6/kEaqPBpu/ufvfbrRzw==} - engines: {node: '>=16.0.0'} - - '@strapi/admin@4.25.9': - resolution: {integrity: sha512-j61GipixI7rNhDEhiWjY+pa7YVQaGOpHmrXHH8cSequI2w+2XF+hHH8zBpzNOuJ38e02DJ9dfugudaYx7cyxGQ==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - peerDependencies: - '@strapi/data-transfer': ^4.16.0 - '@strapi/strapi': ^4.3.4 - react: ^17.0.0 || ^18.0.0 - react-dom: ^17.0.0 || ^18.0.0 - react-router-dom: ^5.2.0 - styled-components: ^5.2.1 - - '@strapi/cloud-cli@4.25.9': - resolution: {integrity: sha512-QIinff6A2wMOgqWYvmBxV0ttVMKvWeX/whjnOcUlnTMUaZbq8OgC2usCxad+2uYKbo/qdLhkE9wrwbaZn9Zzhg==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - hasBin: true - - '@strapi/content-releases@4.25.9': - resolution: {integrity: sha512-N+BjsCBxJuI3Er1GFt64XBtPv/8UL3EUn21VbdqUIRk9MKOAXt7kwgj4veGkGQ6Nt4kxpMXRzRR5HN1IFqTcTQ==} - engines: {node: '>=16.0.0 <=20.x.x', npm: '>=6.0.0'} - peerDependencies: - '@strapi/admin': ^4.19.0 - '@strapi/strapi': ^4.15.1 - react: ^17.0.0 || ^18.0.0 - react-dom: ^17.0.0 || ^18.0.0 - react-router-dom: 5.3.4 - styled-components: 5.3.3 - - '@strapi/data-transfer@4.25.9': - resolution: {integrity: sha512-92CXvYV9B07VbpLzcEU4i8yZmI7wbC/AarFjEPB1Eajq/3vLZIB7pPwqWBYlPOf4kMsmabwZmeNQJZm4JjFalQ==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - peerDependencies: - '@strapi/strapi': ^4.14.4 - - '@strapi/database@4.25.9': - resolution: {integrity: sha512-y/li0iHY76wL9BhXVr4vvAtgnPt993adV7J+h560nrUYtHvj8YQp1GMYktC+4ZQMLaf01HZ+IkrFeqHTXZ8sXQ==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - - '@strapi/design-system@1.19.0': - resolution: {integrity: sha512-kEQNaRztIcr6I5Zh6mxtE/Nmkk1mylCS5s56ySKDdqOjWZw2BCbS72/J9k6r1RF1TLIDSXJN9r5dHR0ZKtWvBQ==} - peerDependencies: - '@strapi/icons': ^1.5.0 - react: ^17.0.0 || ^18.0.0 - react-dom: ^17.0.0 || ^18.0.0 - react-router-dom: ^5.2.0 - styled-components: ^5.2.1 - - '@strapi/generate-new@4.25.9': - resolution: {integrity: sha512-XsyjCNVGuuV0TgGbTK+y2xPKBkwfNg92Y5hkqznjJzagj0M83PMNONtKPG4CZY1LvqMMix0NPckCtSR5lFcNdA==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - - '@strapi/generators@4.25.9': - resolution: {integrity: sha512-0sl8xQgkoHQZRKwCN+h51GUnNbxwqFU98JQMvuEjhv7TBsJ3T/00RXDXMXTlID4H+DYipYnf6n4MRUhz0lWnag==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - - '@strapi/helper-plugin@4.25.9': - resolution: {integrity: sha512-SFHVvKTf/7TNenDa6/6aafMvYyUKWWos+rjwL8qHqG975we9IZIjiQHYytJY5ZzYYAnL7vsT86RKmTA96Kf+bQ==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - peerDependencies: - '@strapi/design-system': 1.19.0 - '@strapi/icons': 1.19.0 - react: ^17.0.0 || ^18.0.0 - react-dom: ^17.0.0 || ^18.0.0 - react-router-dom: ^5.2.0 - styled-components: ^5.2.1 - - '@strapi/icons@1.19.0': - resolution: {integrity: sha512-jcS7n3Ps+73fYGadxdaD6owazoDJKN1fHSG9dp8RX4RqkP6BfoHOX5j3aodVLmDX57Ksg6gy5JXf9xEml7nMpQ==} - peerDependencies: - react: ^17.0.0 || ^18.0.0 - react-dom: ^17.0.0 || ^18.0.0 - - '@strapi/logger@4.25.9': - resolution: {integrity: sha512-kyoOXVpQSNaxgWLfYeo7Zy9VUc4Q6PIRw9X/UHnr2cx5H/KWl/TEdvCDmE5m2jAr073ShKMyT5H5osOV1qY1Fg==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - - '@strapi/pack-up@4.23.0': - resolution: {integrity: sha512-hiSqUEEzks2JDai6bfvtvPHYaPhI6UnSifx9ZqBdC9Q551BYm1xt+1K7HJVeW0IPI4zLckZvCcGPHh/NeYyTPw==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - hasBin: true - - '@strapi/permissions@4.25.9': - resolution: {integrity: sha512-7pCIdVWP7Z9wB0omWwuLwk4FtP1EJuBKSJ1gOmOluUCHC21xleBsZbiqQU8HRTc5lDd+7/vlECJYKg+2biCmQw==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - - '@strapi/plugin-cloud@4.25.9': - resolution: {integrity: sha512-pBLIKbZlMrk9S0pzqhz/Zhf8+t+Rj2Sr3UDxRLKwfxm5sYBt0CC2x7rRdIupwdo6ebGcib1tDJbhxZ0SCtwQPQ==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - peerDependencies: - '@strapi/strapi': ^4.4.0 - react: ^17.0.0 || ^18.0.0 - react-dom: ^17.0.0 || ^18.0.0 - react-router-dom: 5.3.4 - styled-components: 5.3.3 - - '@strapi/plugin-content-manager@4.25.9': - resolution: {integrity: sha512-EeXkba/pUQSId5G9vw+wlpP12b8h6KxGQSMncnS55RhlWXXaxr7mBSC+LvxLO9Ka/ViaoIqc94QHNOXhaBjQ5A==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - - '@strapi/plugin-content-type-builder@4.25.9': - resolution: {integrity: sha512-GQx2I0ppZkmRKQhdO+tCxhE2HvzSi8mWSlKDwoBydk2aAuwJ4+8idu51lM3eayyfBuFnMo2KLiOu9Y8VszjndA==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - peerDependencies: - '@strapi/strapi': ^4.0.0 - react: ^17.0.0 || ^18.0.0 - react-dom: ^17.0.0 || ^18.0.0 - react-router-dom: ^5.2.0 - styled-components: ^5.2.1 - - '@strapi/plugin-email@4.25.9': - resolution: {integrity: sha512-vZj/QiZAlZuD6UhAq5nHJaEUYAyVQPMp1pJnpJsQNL9DpZ+FWuX3SVQKfdiNpE3v0Be9D9maDAeJFBYztw+YZA==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - peerDependencies: - koa: 2.13.4 - react: ^17.0.0 || ^18.0.0 - react-dom: ^17.0.0 || ^18.0.0 - react-router-dom: ^5.2.0 - styled-components: ^5.2.1 - - '@strapi/plugin-i18n@4.25.9': - resolution: {integrity: sha512-0Py2BJZmo9z1Leda7pDktwx5cyDyDAEEn7IoujUvuxnrjkqEoNMHh5+CiMeviYH4YkJwgKbFzpCkbDFDfH79FQ==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - peerDependencies: - '@strapi/strapi': ^4.0.0 - react: ^17.0.0 || ^18.0.0 - react-dom: ^17.0.0 || ^18.0.0 - react-router-dom: ^5.2.0 - styled-components: ^5.2.1 - - '@strapi/plugin-upload@4.25.9': - resolution: {integrity: sha512-ydNFsZR9mpJI4FO5kF+FC8n6kR80m1oeZcBroAd9NQsxiIeVHXz5jJlYmuwu07gg2VpbC8OuzsXZLw32elQLIQ==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - peerDependencies: - '@strapi/strapi': ^4.0.0 - react: ^17.0.0 || ^18.0.0 - react-dom: ^17.0.0 || ^18.0.0 - react-router-dom: ^5.2.0 - styled-components: ^5.2.1 - - '@strapi/plugin-users-permissions@4.25.9': - resolution: {integrity: sha512-t0C7sTTUbGABJl+SHVvch/QcWJtpjwzAG3OrT1rjPyVBiTA4bl0JzR0ZOO7hUFDNTq/Dvrfkvo7dJN3ETVK7VA==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - peerDependencies: - '@strapi/strapi': ^4.0.0 - react: ^17.0.0 || ^18.0.0 - react-dom: ^17.0.0 || ^18.0.0 - react-router-dom: ^5.2.0 - styled-components: ^5.2.1 - - '@strapi/provider-audit-logs-local@4.25.9': - resolution: {integrity: sha512-/+E+GdI3QMwiHEIfXcTR7+sD09gDuxcy9zAc+t9cXPwoOXQIWVJBRF2PcfGlefM6iUUX7KFi7v0VadioAGMqog==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - - '@strapi/provider-email-sendmail@4.25.9': - resolution: {integrity: sha512-SNSFReu/s9XM0lMFbdj1NQ3zwcjSZv3qfA9nblWaMWqV5MTCUKn/868X80pxB+d5SLNn9ix905buHXqClKv63A==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - - '@strapi/provider-upload-local@4.25.9': - resolution: {integrity: sha512-LMxgN7lrGisAg7oK99IXYmbM0OS861IhFYAnlCHPeOn3/4QEK6dqKULEo3jToxzzOkVl80RRgPk2aCDS73cDrw==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - - '@strapi/strapi@4.25.9': - resolution: {integrity: sha512-hh8j+/6mz88Ymv0gNqWslNGH1WR+Vbm1+xilOEEDGR87q+GDIY5kHpBojgYoCQzz0lTLMBtPPezSnZlvYuKRjg==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - hasBin: true - - '@strapi/types@4.25.9': - resolution: {integrity: sha512-XwiEcC3JVMLVIGy6vtAEcw8las4RDcj10h0Ydeyb9YOqxMO47wMZMHED5oTlUDa3C1iSPTHN8DbSoSu6eot6Bg==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - - '@strapi/typescript-utils@4.25.9': - resolution: {integrity: sha512-ZMqimhBfl9hTQmhuaRFpJGZaCD8zlUN0kB2FywEgmedAjuH2pO1H1IwoDvERNIg5GoFMEzxLlPxCftJTIvF2fQ==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - - '@strapi/ui-primitives@1.19.0': - resolution: {integrity: sha512-dEpmI0PpSH6VWuP/bBvRKI5lUpazdDAcxOpukoq2QDwUFbuZWywgW7a6O5nMnD4bLQtyNeYwd52J8Jqr9pNoQA==} - peerDependencies: - react: ^17.0.0 || ^18.0.0 - react-dom: ^17.0.0 || ^18.0.0 - - '@strapi/utils@4.25.9': - resolution: {integrity: sha512-tAcxiT0l0Of3FSsN1K2/BfpwyRP8ZbnjjpKZsODRj+OIWPAMFuEBYet4CFBpbu2jjPsc/maTtaSOBwNrJRN2bA==} - engines: {node: '>=18.0.0 <=20.x.x', npm: '>=6.0.0'} - - '@swc/core-darwin-arm64@1.7.23': - resolution: {integrity: sha512-yyOHPfti6yKlQulfVWMt7BVKst+SyEZYCWuQSGMn1KgmNCH/bYufRWfQXIhkGSj44ZkEepJmsJ8tDyIb4k5WyA==} - engines: {node: '>=10'} - cpu: [arm64] - os: [darwin] - - '@swc/core-darwin-x64@1.7.23': - resolution: {integrity: sha512-GzqHwQ0Y1VyjdI/bBKFX2GKm5HD3PIB6OhuAQtWZMTtEr2yIrlT0YK2T+XKh7oIg31JwxGBeQdBk3KTI7DARmQ==} - engines: {node: '>=10'} - cpu: [x64] - os: [darwin] - - '@swc/core-linux-arm-gnueabihf@1.7.23': - resolution: {integrity: sha512-qwX4gB41OS6/OZkHcpTqLFGsdmvoZyffnJIlgB/kZKwH3lfeJWzv6vx57zXtNpM/t7GoQEe0VZUVdmNjxSxBZw==} - engines: {node: '>=10'} - cpu: [arm] - os: [linux] - - '@swc/core-linux-arm64-gnu@1.7.23': - resolution: {integrity: sha512-TsrbUZdMaUwzI7+g/8rHPLWbntMKYSu5Bn5IBSqVKPeyqaXxNnlIUnWXgXcUcRAc+T+Y8ADfr7EiFz9iz5DuSA==} - engines: {node: '>=10'} - cpu: [arm64] - os: [linux] - - '@swc/core-linux-arm64-musl@1.7.23': - resolution: {integrity: sha512-JEdtwdthazKq4PBz53KSubwwK8MvqODAihGSAzc8u3Unq4ojcvaS8b0CwLBeD+kTQ78HpxOXTt3DsFIxpgaCAA==} - engines: {node: '>=10'} - cpu: [arm64] - os: [linux] - - '@swc/core-linux-x64-gnu@1.7.23': - resolution: {integrity: sha512-V51gFPWaVAHbI1yg9ahsoya3aB4uawye3SZ5uQWgcP7wdCdiv60dw4F5nuPJf5Z1oXD3U/BslXuamv8Oh9vXqQ==} - engines: {node: '>=10'} - cpu: [x64] - os: [linux] - - '@swc/core-linux-x64-musl@1.7.23': - resolution: {integrity: sha512-BBqQi4+UdeRqag3yM4IJjaHG4yc1o3l9ksENHToE0o/u2DT0FY5+K/DiYGZLC1JHbSFzNqRCYsa7DIzRtZ0A1A==} - engines: {node: '>=10'} - cpu: [x64] - os: [linux] - - '@swc/core-win32-arm64-msvc@1.7.23': - resolution: {integrity: sha512-JPk6pvCKncL6bXG7p+NLZf8PWx4FakVvKNdwGeMrYunb+yk1IZf7qf9LJk8+GDGF5QviDXPs8opZrTrfsW80fA==} - engines: {node: '>=10'} - cpu: [arm64] - os: [win32] - - '@swc/core-win32-ia32-msvc@1.7.23': - resolution: {integrity: sha512-2Whxi8d+bLQBzJcQ5qYPHlk02YYVGsMVav0fWk+FnX2z1QRREIu1L1xvrpi7gBpjXp6BIU40ya8GiKeekNT2bg==} - engines: {node: '>=10'} - cpu: [ia32] - os: [win32] - - '@swc/core-win32-x64-msvc@1.7.23': - resolution: {integrity: sha512-82fARk4/yJ40kwWKY/gdKDisPdtgJE9jgpl/vkNG3alyJxrCzuNM7+CtiKoYbXLeqM8GQTS3wlvCaJu9oQ8dag==} - engines: {node: '>=10'} - cpu: [x64] - os: [win32] - - '@swc/core@1.7.23': - resolution: {integrity: sha512-VDNkpDvDlreGh2E3tlDj8B3piiuLhhQA/7rIVZpiLUvG1YpucAa6N7iDXA7Gc/+Hah8spaCg/qvEaBkCmcIYCQ==} - engines: {node: '>=10'} - peerDependencies: - '@swc/helpers': '*' - peerDependenciesMeta: - '@swc/helpers': - optional: true - - '@swc/counter@0.1.3': - resolution: {integrity: sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ==} - - '@swc/helpers@0.5.13': - resolution: {integrity: sha512-UoKGxQ3r5kYI9dALKJapMmuK+1zWM/H17Z1+iwnNmzcJRnfFuevZs375TA5rW31pu4BS4NoSy1fRsexDXfWn5w==} - - '@swc/types@0.1.12': - resolution: {integrity: sha512-wBJA+SdtkbFhHjTMYH+dEH1y4VpfGdAc2Kw/LK09i9bXd/K6j6PkDcFCEzb6iVfZMkPRrl/q0e3toqTAJdkIVA==} - - '@szmarczak/http-timer@4.0.6': - resolution: {integrity: sha512-4BAffykYOgO+5nzBWYwE3W90sBgLJoUPRWWcL8wlyiM8IB8ipJz3UMJ9KXQd1RKQXpKp8Tutn80HZtWsu2u76w==} - engines: {node: '>=10'} - - '@types/argparse@1.0.38': - resolution: {integrity: sha512-ebDJ9b0e702Yr7pWgB0jzm+CX4Srzz8RcXtLJDJB+BSccqMa36uyH/zUsSYao5+BD1ytv3k3rPYCq4mAE1hsXA==} - - '@types/body-parser@1.19.5': - resolution: {integrity: sha512-fB3Zu92ucau0iQ0JMCFQE7b/dv8Ot07NI3KaZIkIUNXq82k4eBAqUaneXfleGY9JWskeS9y+u0nXMyspcuQrCg==} - - '@types/cacheable-request@6.0.3': - resolution: {integrity: sha512-IQ3EbTzGxIigb1I3qPZc1rWJnH0BmSKv5QYTalEwweFvyBDLSAe24zP0le/hyi7ecGfZVlIVAg4BZqb8WBwKqw==} - - '@types/connect@3.4.38': - resolution: {integrity: sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==} - - '@types/estree@1.0.5': - resolution: {integrity: sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==} - - '@types/express-serve-static-core@4.19.5': - resolution: {integrity: sha512-y6W03tvrACO72aijJ5uF02FRq5cgDR9lUxddQ8vyF+GvmjJQqbzDcJngEjURc+ZsG31VI3hODNZJ2URj86pzmg==} - - '@types/express@4.17.21': - resolution: {integrity: sha512-ejlPM315qwLpaQlQDTjPdsUFSc6ZsP4AN6AlWnogPjQ7CVi7PYF3YVz+CY3jE2pwYf7E/7HlDAN0rV2GxTG0HQ==} - - '@types/fined@1.1.5': - resolution: {integrity: sha512-2N93vadEGDFhASTIRbizbl4bNqpMOId5zZfj6hHqYZfEzEfO9onnU4Im8xvzo8uudySDveDHBOOSlTWf38ErfQ==} - - '@types/follow-redirects@1.14.4': - resolution: {integrity: sha512-GWXfsD0Jc1RWiFmMuMFCpXMzi9L7oPDVwxUnZdg89kDNnqsRfUKXEtUYtA98A6lig1WXH/CYY/fvPW9HuN5fTA==} - - '@types/formidable@1.2.8': - resolution: {integrity: sha512-6psvrUy5VDYb+yaPJReF1WrRsz+FBwyJutK9Twz1Efa27tm07bARNIkK2B8ZPWq80dXqpKfrxTO96xrtPp+AuA==} - - '@types/glob@7.2.0': - resolution: {integrity: sha512-ZUxbzKl0IfJILTS6t7ip5fQQM/J3TJYubDm3nMbgubNNYS62eXeUpoLUC8/7fJNiFYHTrGPQn7hspDUzIHX3UA==} - - '@types/hoist-non-react-statics@3.3.5': - resolution: {integrity: sha512-SbcrWzkKBw2cdwRTwQAswfpB9g9LJWfjtUeW/jvNwbhC8cpmmNYVePa+ncbUe0rGTQ7G3Ff6mYUN2VMfLVr+Sg==} - - '@types/html-minifier-terser@6.1.0': - resolution: {integrity: sha512-oh/6byDPnL1zeNXFrDXFLyZjkr1MsBG667IM792caf1L2UPOOMf65NFzjUH/ltyfwjAGfs1rsX1eftK0jC/KIg==} - - '@types/http-cache-semantics@4.0.4': - resolution: {integrity: sha512-1m0bIFVc7eJWyve9S0RnuRgcQqF/Xd5QsUZAZeQFr1Q3/p9JWoQQEqmVy+DPTNpGXwhgIetAoYF8JSc33q29QA==} - - '@types/http-errors@2.0.4': - resolution: {integrity: sha512-D0CFMMtydbJAegzOyHjtiKPLlvnm3iTZyZRSZoLq2mRhDdmLfIWOCYPfQJ4cu2erKghU++QvjcUjp/5h7hESpA==} - - '@types/inquirer@6.5.0': - resolution: {integrity: sha512-rjaYQ9b9y/VFGOpqBEXRavc3jh0a+e6evAbI31tMda8VlPaSy0AZJfXsvmIe3wklc7W6C3zCSfleuMXR7NOyXw==} - - '@types/interpret@1.1.3': - resolution: {integrity: sha512-uBaBhj/BhilG58r64mtDb/BEdH51HIQLgP5bmWzc5qCtFMja8dCk/IOJmk36j0lbi9QHwI6sbtUNGuqXdKCAtQ==} - - '@types/is-hotkey@0.1.10': - resolution: {integrity: sha512-RvC8KMw5BCac1NvRRyaHgMMEtBaZ6wh0pyPTBu7izn4Sj/AX9Y4aXU5c7rX8PnM/knsuUpC1IeoBkANtxBypsQ==} - - '@types/json-schema@7.0.15': - resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==} - - '@types/jsonwebtoken@9.0.6': - resolution: {integrity: sha512-/5hndP5dCjloafCXns6SZyESp3Ldq7YjH3zwzwczYnjxIT0Fqzk5ROSYVGfFyczIue7IUEj8hkvLbPoLQ18vQw==} - - '@types/keyv@3.1.4': - resolution: {integrity: sha512-BQ5aZNSCpj7D6K2ksrRCTmKRLEpnPvWDiLPfoGyhZ++8YtiK9d/3DBKPJgry359X/P1PfruyYwvnvwFjuEiEIg==} - - '@types/liftoff@2.5.1': - resolution: {integrity: sha512-nB3R6Q9CZcM07JgiTK6ibxqrG1reiHE+UX7em/W1DKwVBxDlfKWOefQjk4jubY5xX+GDxVsWR2KD1SenPby8ow==} - - '@types/lodash@4.17.7': - resolution: {integrity: sha512-8wTvZawATi/lsmNu10/j2hk1KEP0IvjubqPE3cu1Xz7xfXXt5oCq3SNUz4fMIP4XGF9Ky+Ue2tBA3hcS7LSBlA==} - - '@types/mime@1.3.5': - resolution: {integrity: sha512-/pyBZWSLD2n0dcHE3hq8s8ZvcETHtEuF+3E7XVt0Ig2nvsVQXdghHVcEkIWjy9A0wKfTn97a/PSDYohKIlnP/w==} - - '@types/minimatch@5.1.2': - resolution: {integrity: sha512-K0VQKziLUWkVKiRVrx4a40iPaxTUefQmjtkQofBkYRcoaaL/8rhwDWww9qWbrgicNOgnpIsMxyNIUM4+n6dUIA==} - - '@types/node-fetch@2.6.11': - resolution: {integrity: sha512-24xFj9R5+rfQJLRyM56qh+wnVSYhyXC2tkoBndtY0U+vubqNsYXGjufB2nn8Q6gt0LrARwL6UBtMCSVCwl4B1g==} - - '@types/node@18.19.48': - resolution: {integrity: sha512-7WevbG4ekUcRQSZzOwxWgi5dZmTak7FaxXDoW7xVxPBmKx1rTzfmRLkeCgJzcbBnOV2dkhAPc8cCeT6agocpjg==} - - '@types/node@22.5.2': - resolution: {integrity: sha512-acJsPTEqYqulZS/Yp/S3GgeE6GZ0qYODUR8aVr/DkhHQ8l9nd4j5x1/ZJy9/gHrRlFMqkO6i0I3E27Alu4jjPg==} - - '@types/normalize-package-data@2.4.4': - resolution: {integrity: sha512-37i+OaWTh9qeK4LSHPsyRC7NahnGotNuZvjLSgcPzblpHB3rrCJxAOgI5gCdKm7coonsaX1Of0ILiTcnZjbfxA==} - - '@types/parse-json@4.0.2': - resolution: {integrity: sha512-dISoDXWWQwUquiKsyZ4Ng+HX2KsPL7LyHKHQwgGFEA3IaKac4Obd+h2a/a6waisAoepJlBcx9paWqjA8/HVjCw==} - - '@types/progress-stream@2.0.5': - resolution: {integrity: sha512-5YNriuEZkHlFHHepLIaxzq3atGeav1qCTGzB74HKWpo66qjfostF+rHc785YYYHeBytve8ZG3ejg42jEIfXNiQ==} - - '@types/prop-types@15.7.12': - resolution: {integrity: sha512-5zvhXYtRNRluoE/jAp4GVsSduVUzNWKkOZrCDBWYtE7biZywwdC2AcEzg+cSMLFRfVgeAFqpfNabiPjxFddV1Q==} - - '@types/qs@6.9.15': - resolution: {integrity: sha512-uXHQKES6DQKKCLh441Xv/dwxOq1TVS3JPUMlEqoEglvlhR6Mxnlew/Xq/LRVHpLyk7iK3zODe1qYHIMltO7XGg==} - - '@types/range-parser@1.2.7': - resolution: {integrity: sha512-hKormJbkJqzQGhziax5PItDUTMAM9uE2XXQmM37dyd4hVM+5aVl7oVxMVUiVQn2oCQFN/LKCZdvSM0pFRqbSmQ==} - - '@types/react-transition-group@4.4.11': - resolution: {integrity: sha512-RM05tAniPZ5DZPzzNFP+DmrcOdD0efDUxMy3145oljWSl3x9ZV5vhme98gTxFrj2lhXvmGNnUiuDyJgY9IKkNA==} - - '@types/react@18.3.5': - resolution: {integrity: sha512-WeqMfGJLGuLCqHGYRGHxnKrXcTitc6L/nBUWfWPcTarG3t9PsquqUMuVeXZeca+mglY4Vo5GZjCi0A3Or2lnxA==} - - '@types/responselike@1.0.3': - resolution: {integrity: sha512-H/+L+UkTV33uf49PH5pCAUBVPNj2nDBXTN+qS1dOwyyg24l3CcicicCA7ca+HMvJBZcFgl5r8e+RR6elsb4Lyw==} - - '@types/send@0.17.4': - resolution: {integrity: sha512-x2EM6TJOybec7c52BX0ZspPodMsQUd5L6PRwOunVyVUhXiBSKf3AezDL8Dgvgt5o0UfKNfuA0eMLr2wLT4AiBA==} - - '@types/serve-static@1.15.7': - resolution: {integrity: sha512-W8Ym+h8nhuRwaKPaDw34QUkwsGi6Rc4yYqvKFo5rm2FUEhCFbzVWrxXUxuKK8TASjWsysJY0nsmNCGhCOIsrOw==} - - '@types/stylis@4.2.5': - resolution: {integrity: sha512-1Xve+NMN7FWjY14vLoY5tL3BVEQ/n42YLwaqJIPYhotZ9uBHt87VceMwWQpzmdEt2TNXIorIFG+YeCUUW7RInw==} - - '@types/through@0.0.33': - resolution: {integrity: sha512-HsJ+z3QuETzP3cswwtzt2vEIiHBk/dCcHGhbmG5X3ecnwFD/lPrMpliGXxSCg03L9AhrdwA4Oz/qfspkDW+xGQ==} - - '@types/triple-beam@1.3.5': - resolution: {integrity: sha512-6WaYesThRMCl19iryMYP7/x2OVgCtbIVflDGFpWnb9irXI3UjYE4AzmYuiUKY1AJstGijoY+MgUszMgRxIYTYw==} - - '@types/use-sync-external-store@0.0.3': - resolution: {integrity: sha512-EwmlvuaxPNej9+T4v5AuBPJa2x2UOJVdjCtDHgcDqitUeOtjnJKJ+apYjVcAoBEMjKW1VVFGZLUb5+qqa09XFA==} - - '@ucast/core@1.10.2': - resolution: {integrity: sha512-ons5CwXZ/51wrUPfoduC+cO7AS1/wRb0ybpQJ9RrssossDxVy4t49QxWoWgfBDvVKsz9VXzBk9z0wqTdZ+Cq8g==} - - '@ucast/js@3.0.4': - resolution: {integrity: sha512-TgG1aIaCMdcaEyckOZKQozn1hazE0w90SVdlpIJ/er8xVumE11gYAtSbw/LBeUnA4fFnFWTcw3t6reqseeH/4Q==} - - '@ucast/mongo2js@1.3.4': - resolution: {integrity: sha512-ahazOr1HtelA5AC1KZ9x0UwPMqqimvfmtSm/PRRSeKKeE5G2SCqTgwiNzO7i9jS8zA3dzXpKVPpXMkcYLnyItA==} - - '@ucast/mongo@2.4.3': - resolution: {integrity: sha512-XcI8LclrHWP83H+7H2anGCEeDq0n+12FU2mXCTz6/Tva9/9ddK/iacvvhCyW6cijAAOILmt0tWplRyRhVyZLsA==} - - '@uiw/codemirror-extensions-basic-setup@4.23.0': - resolution: {integrity: sha512-+k5nkRpUWGaHr1JWT8jcKsVewlXw5qBgSopm9LW8fZ6KnSNZBycz8kHxh0+WSvckmXEESGptkIsb7dlkmJT/hQ==} - peerDependencies: - '@codemirror/autocomplete': '>=6.0.0' - '@codemirror/commands': '>=6.0.0' - '@codemirror/language': '>=6.0.0' - '@codemirror/lint': '>=6.0.0' - '@codemirror/search': '>=6.0.0' - '@codemirror/state': '>=6.0.0' - '@codemirror/view': '>=6.0.0' - - '@uiw/react-codemirror@4.23.0': - resolution: {integrity: sha512-MnqTXfgeLA3fsUUQjqjJgemEuNyoGALgsExVm0NQAllAAi1wfj+IoKFeK+h3XXMlTFRCFYOUh4AHDv0YXJLsOg==} - peerDependencies: - '@babel/runtime': '>=7.11.0' - '@codemirror/state': '>=6.0.0' - '@codemirror/theme-one-dark': '>=6.0.0' - '@codemirror/view': '>=6.0.0' - codemirror: '>=6.0.0' - react: '>=16.8.0' - react-dom: '>=16.8.0' - - '@vitejs/plugin-react-swc@3.5.0': - resolution: {integrity: sha512-1PrOvAaDpqlCV+Up8RkAh9qaiUjoDUcjtttyhXDKw53XA6Ve16SOp6cCOpRs8Dj8DqUQs6eTW5YkLcLJjrXAig==} - peerDependencies: - vite: ^4 || ^5 - - '@webassemblyjs/ast@1.12.1': - resolution: {integrity: sha512-EKfMUOPRRUTy5UII4qJDGPpqfwjOmZ5jeGFwid9mnoqIFK+e0vqoi1qH56JpmZSzEL53jKnNzScdmftJyG5xWg==} - - '@webassemblyjs/floating-point-hex-parser@1.11.6': - resolution: {integrity: sha512-ejAj9hfRJ2XMsNHk/v6Fu2dGS+i4UaXBXGemOfQ/JfQ6mdQg/WXtwleQRLLS4OvfDhv8rYnVwH27YJLMyYsxhw==} - - '@webassemblyjs/helper-api-error@1.11.6': - resolution: {integrity: sha512-o0YkoP4pVu4rN8aTJgAyj9hC2Sv5UlkzCHhxqWj8butaLvnpdc2jOwh4ewE6CX0txSfLn/UYaV/pheS2Txg//Q==} - - '@webassemblyjs/helper-buffer@1.12.1': - resolution: {integrity: sha512-nzJwQw99DNDKr9BVCOZcLuJJUlqkJh+kVzVl6Fmq/tI5ZtEyWT1KZMyOXltXLZJmDtvLCDgwsyrkohEtopTXCw==} - - '@webassemblyjs/helper-numbers@1.11.6': - resolution: {integrity: sha512-vUIhZ8LZoIWHBohiEObxVm6hwP034jwmc9kuq5GdHZH0wiLVLIPcMCdpJzG4C11cHoQ25TFIQj9kaVADVX7N3g==} - - '@webassemblyjs/helper-wasm-bytecode@1.11.6': - resolution: {integrity: sha512-sFFHKwcmBprO9e7Icf0+gddyWYDViL8bpPjJJl0WHxCdETktXdmtWLGVzoHbqUcY4Be1LkNfwTmXOJUFZYSJdA==} - - '@webassemblyjs/helper-wasm-section@1.12.1': - resolution: {integrity: sha512-Jif4vfB6FJlUlSbgEMHUyk1j234GTNG9dBJ4XJdOySoj518Xj0oGsNi59cUQF4RRMS9ouBUxDDdyBVfPTypa5g==} - - '@webassemblyjs/ieee754@1.11.6': - resolution: {integrity: sha512-LM4p2csPNvbij6U1f19v6WR56QZ8JcHg3QIJTlSwzFcmx6WSORicYj6I63f9yU1kEUtrpG+kjkiIAkevHpDXrg==} - - '@webassemblyjs/leb128@1.11.6': - resolution: {integrity: sha512-m7a0FhE67DQXgouf1tbN5XQcdWoNgaAuoULHIfGFIEVKA6tu/edls6XnIlkmS6FrXAquJRPni3ZZKjw6FSPjPQ==} - - '@webassemblyjs/utf8@1.11.6': - resolution: {integrity: sha512-vtXf2wTQ3+up9Zsg8sa2yWiQpzSsMyXj0qViVP6xKGCUT8p8YJ6HqI7l5eCnWx1T/FYdsv07HQs2wTFbbof/RA==} - - '@webassemblyjs/wasm-edit@1.12.1': - resolution: {integrity: sha512-1DuwbVvADvS5mGnXbE+c9NfA8QRcZ6iKquqjjmR10k6o+zzsRVesil54DKexiowcFCPdr/Q0qaMgB01+SQ1u6g==} - - '@webassemblyjs/wasm-gen@1.12.1': - resolution: {integrity: sha512-TDq4Ojh9fcohAw6OIMXqiIcTq5KUXTGRkVxbSo1hQnSy6lAM5GSdfwWeSxpAo0YzgsgF182E/U0mDNhuA0tW7w==} - - '@webassemblyjs/wasm-opt@1.12.1': - resolution: {integrity: sha512-Jg99j/2gG2iaz3hijw857AVYekZe2SAskcqlWIZXjji5WStnOpVoat3gQfT/Q5tb2djnCjBtMocY/Su1GfxPBg==} - - '@webassemblyjs/wasm-parser@1.12.1': - resolution: {integrity: sha512-xikIi7c2FHXysxXe3COrVUPSheuBtpcfhbpFj4gmu7KRLYOzANztwUU0IbsqvMqzuNK2+glRGWCEqZo1WCLyAQ==} - - '@webassemblyjs/wast-printer@1.12.1': - resolution: {integrity: sha512-+X4WAlOisVWQMikjbcvY2e0rwPsKQ9F688lksZhBcPycBBuii3O7m8FACbDMWDojpAqvjIncrG8J0XHKyQfVeA==} - - '@xtuc/ieee754@1.2.0': - resolution: {integrity: sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA==} - - '@xtuc/long@4.2.2': - resolution: {integrity: sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ==} - - abbrev@1.1.1: - resolution: {integrity: sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==} - - abort-controller@3.0.0: - resolution: {integrity: sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==} - engines: {node: '>=6.5'} - - accepts@1.3.8: - resolution: {integrity: sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==} - engines: {node: '>= 0.6'} - - acorn-import-attributes@1.9.5: - resolution: {integrity: sha512-n02Vykv5uA3eHGM/Z2dQrcD56kL8TyDb2p1+0P83PClMnC/nc+anbQRhIOWnSq4Ke/KvDPrY3C9hDtC/A3eHnQ==} - peerDependencies: - acorn: ^8 - - acorn-walk@8.3.3: - resolution: {integrity: sha512-MxXdReSRhGO7VlFe1bRG/oI7/mdLV9B9JJT0N8vZOhF7gFRR5l3M8W9G8JxmKV+JC5mGqJ0QvqfSOLsCPa4nUw==} - engines: {node: '>=0.4.0'} - - acorn@8.12.1: - resolution: {integrity: sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==} - engines: {node: '>=0.4.0'} - hasBin: true - - addressparser@1.0.1: - resolution: {integrity: sha512-aQX7AISOMM7HFE0iZ3+YnD07oIeJqWGVnJ+ZIKaBZAk03ftmVYVqsGas/rbXKR21n4D/hKCSHypvcyOkds/xzg==} - - agent-base@6.0.2: - resolution: {integrity: sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==} - engines: {node: '>= 6.0.0'} - - agentkeepalive@4.5.0: - resolution: {integrity: sha512-5GG/5IbQQpC9FpkRGsSvZI5QYeSCzlJHdpBQntCsuTOxhKD8lqKhrleg2Yi7yvMIf82Ycmmqln9U8V9qwEiJew==} - engines: {node: '>= 8.0.0'} - - aggregate-error@3.1.0: - resolution: {integrity: sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==} - engines: {node: '>=8'} - - ajv-draft-04@1.0.0: - resolution: {integrity: sha512-mv00Te6nmYbRp5DCwclxtt7yV/joXJPGS7nM+97GdxvuttCOfgI3K4U25zboyeX0O+myI8ERluxQe5wljMmVIw==} - peerDependencies: - ajv: ^8.5.0 - peerDependenciesMeta: - ajv: - optional: true - - ajv-formats@2.1.1: - resolution: {integrity: sha512-Wx0Kx52hxE7C18hkMEggYlEifqWZtYaRgouJor+WMdPnQyEK13vgEWyVNup7SoeeoLMsr4kf5h6dOW11I15MUA==} - peerDependencies: - ajv: ^8.0.0 - peerDependenciesMeta: - ajv: - optional: true - - ajv-formats@3.0.1: - resolution: {integrity: sha512-8iUql50EUR+uUcdRQ3HDqa6EVyo3docL8g5WJ3FNcWmu62IbkGUue/pEyLBW8VGKKucTPgqeks4fIU1DA4yowQ==} - peerDependencies: - ajv: ^8.0.0 - peerDependenciesMeta: - ajv: - optional: true - - ajv-keywords@3.5.2: - resolution: {integrity: sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==} - peerDependencies: - ajv: ^6.9.1 - - ajv-keywords@5.1.0: - resolution: {integrity: sha512-YCS/JNFAUyr5vAuhk1DWm1CBxRHW9LbJ2ozWeemrIqpbsqKjHVxYPyi5GC0rjZIT5JxJ3virVTS8wk4i/Z+krw==} - peerDependencies: - ajv: ^8.8.2 - - ajv@6.12.6: - resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==} - - ajv@8.13.0: - resolution: {integrity: sha512-PRA911Blj99jR5RMeTunVbNXMF6Lp4vZXnk5GQjcnUWUTsrXtekg/pnmFFI2u/I36Y/2bITGS30GZCXei6uNkA==} - - ajv@8.17.1: - resolution: {integrity: sha512-B/gBuNg5SiMTrPkC+A2+cW0RszwxYmn6VYxB/inlBStS5nx6xHIt/ehKRhIMhqusl7a8LjQoZnjCs5vhwxOQ1g==} - - ansi-align@3.0.1: - resolution: {integrity: sha512-IOfwwBF5iczOjp/WeY4YxyjqAFMQoZufdQWDd19SEExbVLNXqvpzSJ/M7Za4/sCPmQ0+GRquoA7bGcINcxew6w==} - - ansi-escapes@4.3.2: - resolution: {integrity: sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==} - engines: {node: '>=8'} - - ansi-html-community@0.0.8: - resolution: {integrity: sha512-1APHAyr3+PCamwNw3bXCPp4HFLONZt/yIH0sZp0/469KWNTEy+qN5jQ3GVX6DMZ1UXAi34yVwtTeaG/HpBuuzw==} - engines: {'0': node >= 0.8.0} - hasBin: true - - ansi-regex@2.1.1: - resolution: {integrity: sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==} - engines: {node: '>=0.10.0'} - - ansi-regex@4.1.1: - resolution: {integrity: sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==} - engines: {node: '>=6'} - - ansi-regex@5.0.1: - resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==} - engines: {node: '>=8'} - - ansi-regex@6.0.1: - resolution: {integrity: sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==} - engines: {node: '>=12'} - - ansi-styles@2.2.1: - resolution: {integrity: sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA==} - engines: {node: '>=0.10.0'} - - ansi-styles@3.2.1: - resolution: {integrity: sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==} - engines: {node: '>=4'} - - ansi-styles@4.3.0: - resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} - engines: {node: '>=8'} - - ansi-styles@6.2.1: - resolution: {integrity: sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==} - engines: {node: '>=12'} - - any-promise@1.3.0: - resolution: {integrity: sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==} - - anymatch@3.1.3: - resolution: {integrity: sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==} - engines: {node: '>= 8'} - - aproba@2.0.0: - resolution: {integrity: sha512-lYe4Gx7QT+MKGbDsA+Z+he/Wtef0BiwDOlK/XkBrdfsh9J/jPPXbX0tE9x9cl27Tmu5gg3QUbUrQYa/y+KOHPQ==} - - are-we-there-yet@2.0.0: - resolution: {integrity: sha512-Ci/qENmwHnsYo9xKIcUJN5LeDKdJ6R1Z1j9V/J5wyq8nh/mYPEpIKJbBZXtZjG04HiK7zV/p6Vs9952MrMeUIw==} - engines: {node: '>=10'} - deprecated: This package is no longer supported. - - argparse@1.0.10: - resolution: {integrity: sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==} - - argparse@2.0.1: - resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==} - - aria-hidden@1.2.4: - resolution: {integrity: sha512-y+CcFFwelSXpLZk/7fMB2mUbGtX9lKycf1MWJ7CaTIERyitVlyQx6C+sxcROU2BAJ24OiZyK+8wj2i8AlBoS3A==} - engines: {node: '>=10'} - - arr-diff@4.0.0: - resolution: {integrity: sha512-YVIQ82gZPGBebQV/a8dar4AitzCQs0jjXwMPZllpXMaGjXPYVUawSxQrRsjhjupyVxEvbHgUmIhKVlND+j02kA==} - engines: {node: '>=0.10.0'} - - arr-flatten@1.1.0: - resolution: {integrity: sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==} - engines: {node: '>=0.10.0'} - - arr-union@3.1.0: - resolution: {integrity: sha512-sKpyeERZ02v1FeCZT8lrfJq5u6goHCtpTAzPwJYe7c8SPFOboNjNg1vz2L4VTn9T4PQxEx13TbXLmYUcS6Ug7Q==} - engines: {node: '>=0.10.0'} - - array-each@1.0.1: - resolution: {integrity: sha512-zHjL5SZa68hkKHBFBK6DJCTtr9sfTCPCaph/L7tMSLcTFgy+zX7E+6q5UArbtOtMBCtxdICpfTCspRse+ywyXA==} - engines: {node: '>=0.10.0'} - - array-slice@1.1.0: - resolution: {integrity: sha512-B1qMD3RBP7O8o0H2KbrXDyB0IccejMF15+87Lvlor12ONPRHP6gTjXMNkt/d3ZuOGbAe66hFmaCfECI24Ufp6w==} - engines: {node: '>=0.10.0'} - - array-union@2.1.0: - resolution: {integrity: sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==} - engines: {node: '>=8'} - - array-unique@0.3.2: - resolution: {integrity: sha512-SleRWjh9JUud2wH1hPs9rZBZ33H6T9HOiL0uwGnGx9FpE6wKGyfWugmbkEOIs6qWrZhg0LWeLziLrEwQJhs5mQ==} - engines: {node: '>=0.10.0'} - - asn1.js@5.4.1: - resolution: {integrity: sha512-+I//4cYPccV8LdmBLiX8CYvf9Sp3vQsrqu2QNXRcrbiWvcx/UdlFiqUJJzxRQxgsZmvhXhn4cSKeSmoFjVdupA==} - - assign-symbols@1.0.0: - resolution: {integrity: sha512-Q+JC7Whu8HhmTdBph/Tq59IoRtoy6KAm5zzPv00WdujX82lbAL8K7WVjne7vdCsAmbF4AYaDOPyO3k0kl8qIrw==} - engines: {node: '>=0.10.0'} - - async@3.2.6: - resolution: {integrity: sha512-htCUDlxyyCLMgaM3xXg0C0LW2xqfuQ6p05pCEIsXuyQ+a1koYKTuBMzRNwmybfLgvJDMd0r1LTn4+E0Ti6C2AA==} - - asynckit@0.4.0: - resolution: {integrity: sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==} - - atob@2.1.2: - resolution: {integrity: sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==} - engines: {node: '>= 4.5.0'} - hasBin: true - - axios@1.7.4: - resolution: {integrity: sha512-DukmaFRnY6AzAALSH4J2M3k6PkaC+MfaAGdEERRWcC9q3/TWQwLpHR8ZRLKTdQ3aBDL64EdluRDjJqKw+BPZEw==} - - b4a@1.6.6: - resolution: {integrity: sha512-5Tk1HLk6b6ctmjIkAcU/Ujv/1WqiDl0F0JdRCR80VsOcUlHcu7pWeWRlOqQLHfDEsVx9YH/aif5AG4ehoCtTmg==} - - babel-plugin-macros@3.1.0: - resolution: {integrity: sha512-Cg7TFGpIr01vOQNODXOOaGz2NpCU5gl8x1qJFbb6hbZxR7XrcE2vtbAsTAbJ7/xwJtUuJEw8K8Zr/AE0LHlesg==} - engines: {node: '>=10', npm: '>=6'} - - balanced-match@1.0.2: - resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} - - bare-events@2.4.2: - resolution: {integrity: sha512-qMKFd2qG/36aA4GwvKq8MxnPgCQAmBWmSyLWsJcbn8v03wvIPQ/hG1Ms8bPzndZxMDoHpxez5VOS+gC9Yi24/Q==} - - bare-fs@2.3.3: - resolution: {integrity: sha512-7RYKL+vZVCyAsMLi5SPu7QGauGGT8avnP/HO571ndEuV4MYdGXvLhtW67FuLPeEI8EiIY7zbbRR9x7x7HU0kgw==} - - bare-os@2.4.2: - resolution: {integrity: sha512-HZoJwzC+rZ9lqEemTMiO0luOePoGYNBgsLLgegKR/cljiJvcDNhDZQkzC+NC5Oh0aHbdBNSOHpghwMuB5tqhjg==} - - bare-path@2.1.3: - resolution: {integrity: sha512-lh/eITfU8hrj9Ru5quUp0Io1kJWIk1bTjzo7JH1P5dWmQ2EL4hFUlfI8FonAhSlgIfhn63p84CDY/x+PisgcXA==} - - bare-stream@2.2.0: - resolution: {integrity: sha512-+o9MG5bPRRBlkVSpfFlMag3n7wMaIZb4YZasU2+/96f+3HTQ4F9DKQeu3K/Sjz1W0umu6xvVq1ON0ipWdMlr3A==} - - base64-js@1.5.1: - resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==} - - base@0.11.2: - resolution: {integrity: sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg==} - engines: {node: '>=0.10.0'} - - bcryptjs@2.4.3: - resolution: {integrity: sha512-V/Hy/X9Vt7f3BbPJEi8BdVFMByHi+jNXrYkW3huaybV/kQ0KJg0Y6PkEMbn+zeT+i+SiKZ/HMqJGIIt4LZDqNQ==} - - big-integer@1.6.52: - resolution: {integrity: sha512-QxD8cf2eVqJOOz63z6JIN9BzvVs/dlySa5HGSBH5xtR8dPteIRQnBxxKqkNTiT6jbDTF6jAfrd4oMcND9RGbQg==} - engines: {node: '>=0.6'} - - big.js@5.2.2: - resolution: {integrity: sha512-vyL2OymJxmarO8gxMr0mhChsO9QGwhynfuu4+MHTAW6czfq9humCB7rKpUjDd9YUiDPU4mzpyupFSvOClAwbmQ==} - - binary-extensions@2.3.0: - resolution: {integrity: sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==} - engines: {node: '>=8'} - - bl@4.1.0: - resolution: {integrity: sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==} - - bn.js@4.12.0: - resolution: {integrity: sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==} - - boolbase@1.0.0: - resolution: {integrity: sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==} - - bowser@2.11.0: - resolution: {integrity: sha512-AlcaJBi/pqqJBIQ8U9Mcpc9i8Aqxn88Skv5d+xBX006BY5u8N3mGLHa5Lgppa7L/HfwgwLgZ6NYs+Ag6uUmJRA==} - - boxen@5.1.2: - resolution: {integrity: sha512-9gYgQKXx+1nP8mP7CzFyaUARhg7D3n1dF/FnErWmu9l6JvGpNUN278h0aSb+QjoiKSWG+iZ3uHrcqk0qrY9RQQ==} - engines: {node: '>=10'} - - bplist-parser@0.2.0: - resolution: {integrity: sha512-z0M+byMThzQmD9NILRniCUXYsYpjwnlO8N5uCFaCqIOpqRsJCrQL9NK3JsD67CN5a08nF5oIL2bD6loTdHOuKw==} - engines: {node: '>= 5.10.0'} - - brace-expansion@1.1.11: - resolution: {integrity: sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==} - - brace-expansion@2.0.1: - resolution: {integrity: sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==} - - braces@2.3.2: - resolution: {integrity: sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==} - engines: {node: '>=0.10.0'} - - braces@3.0.3: - resolution: {integrity: sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==} - engines: {node: '>=8'} - - broadcast-channel@3.7.0: - resolution: {integrity: sha512-cIAKJXAxGJceNZGTZSBzMxzyOn72cVgPnKx4dc6LRjQgbaJUQqhy5rzL3zbMxkMWsGKkv2hSFkPRMEXfoMZ2Mg==} - - brorand@1.1.0: - resolution: {integrity: sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w==} - - browserslist-to-esbuild@1.2.0: - resolution: {integrity: sha512-ftrrbI/VHBgEnmnSyhkqvQVMp6jAKybfs0qMIlm7SLBrQTGMsdCIP4q3BoKeLsZTBQllIQtY9kbxgRYV2WU47g==} - engines: {node: '>=12'} - - browserslist@4.23.3: - resolution: {integrity: sha512-btwCFJVjI4YWDNfau8RhZ+B1Q/VLoUITrm3RlP6y1tYGWIOa+InuYiRGXUBXo8nA1qKmHMyLB/iVQg5TT4eFoA==} - engines: {node: ^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7} - hasBin: true - - buffer-equal-constant-time@1.0.1: - resolution: {integrity: sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA==} - - buffer-from@1.1.2: - resolution: {integrity: sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==} - - buffer@5.7.1: - resolution: {integrity: sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==} - - buildmail@3.10.0: - resolution: {integrity: sha512-6e5sDN/pl3en5Klqdfyir7LEIBiFr9oqZuvYaEyVwjxpIbBZN+98e0j87Fz2Ukl8ud32rbk9VGOZAnsOZ7pkaA==} - deprecated: This project is unmaintained - - bundle-name@3.0.0: - resolution: {integrity: sha512-PKA4BeSvBpQKQ8iPOGCSiell+N8P+Tf1DlwqmYhpe2gAhKPHn8EYOxVT+ShuGmhg8lN8XiSlS80yiExKXrURlw==} - engines: {node: '>=12'} - - byte-size@7.0.1: - resolution: {integrity: sha512-crQdqyCwhokxwV1UyDzLZanhkugAgft7vt0qbbdt60C6Zf3CAiGmtUCylbtYwrU6loOUw3euGrNtW1J651ot1A==} - engines: {node: '>=10'} - - bytes@3.1.2: - resolution: {integrity: sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==} - engines: {node: '>= 0.8'} - - cache-base@1.0.1: - resolution: {integrity: sha512-AKcdTnFSWATd5/GCPRxr2ChwIJ85CeyrEyjRHlKxQ56d4XJMGym0uAiKn0xbLOGOl3+yRpOTi484dVCEc5AUzQ==} - engines: {node: '>=0.10.0'} - - cache-content-type@1.0.1: - resolution: {integrity: sha512-IKufZ1o4Ut42YUrZSo8+qnMTrFuKkvyoLXUywKz9GJ5BrhOFGhLdkx9sG4KAnVvbY6kEcSFjLQul+DVmBm2bgA==} - engines: {node: '>= 6.0.0'} - - cacheable-lookup@5.0.4: - resolution: {integrity: sha512-2/kNscPhpcxrOigMZzbiWF7dz8ilhb/nIHU3EyZiXWXpeq/au8qJ8VhdftMkty3n7Gj6HIGalQG8oiBNB3AJgA==} - engines: {node: '>=10.6.0'} - - cacheable-request@7.0.4: - resolution: {integrity: sha512-v+p6ongsrp0yTGbJXjgxPow2+DL93DASP4kXCDKb8/bwRtt9OEF3whggkkDkGNzgcWy2XaF4a8nZglC7uElscg==} - engines: {node: '>=8'} - - call-bind@1.0.7: - resolution: {integrity: sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==} - engines: {node: '>= 0.4'} - - callsites@3.1.0: - resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==} - engines: {node: '>=6'} - - camel-case@3.0.0: - resolution: {integrity: sha512-+MbKztAYHXPr1jNTSKQF52VpcFjwY5RkR7fxksV8Doo4KAYc5Fl4UJRgthBbTmEx8C54DqahhbLJkDwjI3PI/w==} - - camel-case@4.1.2: - resolution: {integrity: sha512-gxGWBrTT1JuMx6R+o5PTXMmUnhnVzLQ9SNutD4YqKtI6ap897t3tKECYla6gCWEkplXnlNybEkZg9GEGxKFCgw==} - - camelcase@6.3.0: - resolution: {integrity: sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==} - engines: {node: '>=10'} - - camelize@1.0.1: - resolution: {integrity: sha512-dU+Tx2fsypxTgtLoE36npi3UqcjSSMNYfkqgmoEhtZrraP5VWq0K7FkWVTYa8eMPtnU/G2txVsfdCJTn9uzpuQ==} - - caniuse-lite@1.0.30001655: - resolution: {integrity: sha512-jRGVy3iSGO5Uutn2owlb5gR6qsGngTw9ZTb4ali9f3glshcNmJ2noam4Mo9zia5P9Dk3jNNydy7vQjuE5dQmfg==} - - canvas@2.11.2: - resolution: {integrity: sha512-ItanGBMrmRV7Py2Z+Xhs7cT+FNt5K0vPL4p9EZ/UX/Mu7hFbkxSjKF2KVtPwX7UYWp7dRKnrTvReflgrItJbdw==} - engines: {node: '>=6'} - - chalk@1.1.3: - resolution: {integrity: sha512-U3lRVLMSlsCfjqYPbLyVv11M9CPW4I728d6TCKMAOJueEeB9/8o+eSsMnxPJD+Q+K909sdESg7C+tIkoH6on1A==} - engines: {node: '>=0.10.0'} - - chalk@2.4.2: - resolution: {integrity: sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==} - engines: {node: '>=4'} - - chalk@4.1.2: - resolution: {integrity: sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==} - engines: {node: '>=10'} - - change-case@3.1.0: - resolution: {integrity: sha512-2AZp7uJZbYEzRPsFoa+ijKdvp9zsrnnt6+yFokfwEpeJm0xuJDVoxiRCAaTzyJND8GJkofo2IcKWaUZ/OECVzw==} - - chardet@0.7.0: - resolution: {integrity: sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==} - - chokidar@3.5.3: - resolution: {integrity: sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==} - engines: {node: '>= 8.10.0'} - - chokidar@3.6.0: - resolution: {integrity: sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==} - engines: {node: '>= 8.10.0'} - - chownr@1.1.4: - resolution: {integrity: sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==} - - chownr@2.0.0: - resolution: {integrity: sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==} - engines: {node: '>=10'} - - chrome-trace-event@1.0.4: - resolution: {integrity: sha512-rNjApaLzuwaOTjCiT8lSDdGN1APCiqkChLMJxJPWLunPAt5fy8xgU9/jNOchV84wfIxrA0lRQB7oCT8jrn/wrQ==} - engines: {node: '>=6.0'} - - ci-info@3.8.0: - resolution: {integrity: sha512-eXTggHWSooYhq49F2opQhuHWgzucfF2YgODK4e1566GQs5BIfP30B0oenwBJHfWxAs2fyPB1s7Mg949zLf61Yw==} - engines: {node: '>=8'} - - class-utils@0.3.6: - resolution: {integrity: sha512-qOhPa/Fj7s6TY8H8esGu5QNpMMQxz79h+urzrNYN6mn+9BnxlDGf5QZ+XeCDsxSjPqsSR56XOZOJmpeurnLMeg==} - engines: {node: '>=0.10.0'} - - clean-css@5.3.3: - resolution: {integrity: sha512-D5J+kHaVb/wKSFcyyV75uCn8fiY4sV38XJoe4CUyGQ+mOU/fMVYUdH1hJC+CJQ5uY3EnW27SbJYS4X8BiLrAFg==} - engines: {node: '>= 10.0'} - - clean-stack@2.2.0: - resolution: {integrity: sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==} - engines: {node: '>=6'} - - cli-boxes@2.2.1: - resolution: {integrity: sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw==} - engines: {node: '>=6'} - - cli-cursor@2.1.0: - resolution: {integrity: sha512-8lgKz8LmCRYZZQDpRyT2m5rKJ08TnU4tR9FFFW2rxpxR1FzWi4PQ/NfyODchAatHaUgnSPVcx/R5w6NuTBzFiw==} - engines: {node: '>=4'} - - cli-cursor@3.1.0: - resolution: {integrity: sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==} - engines: {node: '>=8'} - - cli-progress@3.12.0: - resolution: {integrity: sha512-tRkV3HJ1ASwm19THiiLIXLO7Im7wlTuKnvkYaTkyoAPefqjNg7W7DHKUlGRxy9vxDvbyCYQkQozvptuMkGCg8A==} - engines: {node: '>=4'} - - cli-spinners@2.9.2: - resolution: {integrity: sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg==} - engines: {node: '>=6'} - - cli-table3@0.6.5: - resolution: {integrity: sha512-+W/5efTR7y5HRD7gACw9yQjqMVvEMLBHmboM/kPWam+H+Hmyrgjh6YncVKK122YZkXrLudzTuAukUw9FnMf7IQ==} - engines: {node: 10.* || >= 12.*} - - cli-width@3.0.0: - resolution: {integrity: sha512-FxqpkPPwu1HjuN93Omfm4h8uIanXofW0RxVEW3k5RKx+mJJYSthzNhp32Kzxxy3YAEZ/Dc/EWN1vZRY0+kOhbw==} - engines: {node: '>= 10'} - - cliui@7.0.4: - resolution: {integrity: sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==} - - cliui@8.0.1: - resolution: {integrity: sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==} - engines: {node: '>=12'} - - clone-response@1.0.3: - resolution: {integrity: sha512-ROoL94jJH2dUVML2Y/5PEDNaSHgeOdSDicUyS7izcF63G6sTc/FTjLub4b8Il9S8S0beOfYt0TaA5qvFK+w0wA==} - - clone@1.0.4: - resolution: {integrity: sha512-JQHZ2QMW6l3aH/j6xCqQThY/9OH4D/9ls34cgkUBiEeocRTU04tHfKPBsUK1PqZCUQM7GiA0IIXJSuXHI64Kbg==} - engines: {node: '>=0.8'} - - co-body@5.2.0: - resolution: {integrity: sha512-sX/LQ7LqUhgyaxzbe7IqwPeTr2yfpfUIQ/dgpKo6ZI4y4lpQA0YxAomWIY+7I7rHWcG02PG+OuPREzMW/5tszQ==} - - co-body@6.2.0: - resolution: {integrity: sha512-Kbpv2Yd1NdL1V/V4cwLVxraHDV6K8ayohr2rmH0J87Er8+zJjcTa6dAn9QMPC9CRgU8+aNajKbSf1TzDB1yKPA==} - engines: {node: '>=8.0.0'} - - co@4.6.0: - resolution: {integrity: sha512-QVb0dM5HvG+uaxitm8wONl7jltx8dqhfU33DcqtOZcLSVIKSDDLDi7+0LbAKiyI8hD9u42m2YxXSkMGWThaecQ==} - engines: {iojs: '>= 1.0.0', node: '>= 0.12.0'} - - codemirror@5.65.17: - resolution: {integrity: sha512-1zOsUx3lzAOu/gnMAZkQ9kpIHcPYOc9y1Fbm2UVk5UBPkdq380nhkelG0qUwm1f7wPvTbndu9ZYlug35EwAZRQ==} - - collection-visit@1.0.0: - resolution: {integrity: sha512-lNkKvzEeMBBjUGHZ+q6z9pSJla0KWAQPvtzhEV9+iGyQYG+pBpl7xKDhxoNSOZH2hhv0v5k0y2yAM4o4SjoSkw==} - engines: {node: '>=0.10.0'} - - color-convert@1.9.3: - resolution: {integrity: sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==} - - color-convert@2.0.1: - resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} - engines: {node: '>=7.0.0'} - - color-name@1.1.3: - resolution: {integrity: sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==} - - color-name@1.1.4: - resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} - - color-string@1.9.1: - resolution: {integrity: sha512-shrVawQFojnZv6xM40anx4CkoDP+fZsw/ZerEMsW/pyzsRbElpsL/DBVW7q3ExxwusdNXI3lXpuhEZkzs8p5Eg==} - - color-support@1.1.3: - resolution: {integrity: sha512-qiBjkpbMLO/HL68y+lh4q0/O1MZFj2RX6X/KmMa3+gJD3z+WwI1ZzDHysvqHGS3mP6mznPckpXmw1nI9cJjyRg==} - hasBin: true - - color@3.2.1: - resolution: {integrity: sha512-aBl7dZI9ENN6fUGC7mWpMTPNHmWUSNan9tuWN6ahh5ZLNk9baLJOnSMlrQkHcrfFgz2/RigjUVAjdx36VcemKA==} - - color@4.2.3: - resolution: {integrity: sha512-1rXeuUUiGGrykh+CeBdu5Ie7OJwinCgQY0bc7GCRxy5xVHy+moaqkpL/jqQq0MtQOeYcrqEz4abc5f0KtU7W4A==} - engines: {node: '>=12.5.0'} - - colorette@2.0.19: - resolution: {integrity: sha512-3tlv/dIP7FWvj3BsbHrGLJ6l/oKh1O3TcgBqMn+yyCagOxc23fyzDS6HypQbgxWbkpDnf52p1LuR4eWDQ/K9WQ==} - - colorette@2.0.20: - resolution: {integrity: sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w==} - - colorspace@1.1.4: - resolution: {integrity: sha512-BgvKJiuVu1igBUF2kEjRCZXol6wiiGbY5ipL/oVPwm0BL9sIpMIzM8IK7vwuxIIzOXMV3Ey5w+vxhm0rR/TN8w==} - - combined-stream@1.0.8: - resolution: {integrity: sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==} - engines: {node: '>= 0.8'} - - commander@10.0.1: - resolution: {integrity: sha512-y4Mg2tXshplEbSGzx7amzPwKKOCGuoSRP/CjEdwwk0FOGlUbq6lKuoyDZTNZkmxHdJtp54hdfY/JUrdL7Xfdug==} - engines: {node: '>=14'} - - commander@2.20.3: - resolution: {integrity: sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==} - - commander@7.2.0: - resolution: {integrity: sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==} - engines: {node: '>= 10'} - - commander@8.3.0: - resolution: {integrity: sha512-OkTL9umf+He2DZkUq8f8J9of7yL6RJKI24dVITBmNfZBmri9zYZQrKkuXiKhyfPSu8tUhnVBB1iKXevvnlR4Ww==} - engines: {node: '>= 12'} - - common-path-prefix@3.0.0: - resolution: {integrity: sha512-QE33hToZseCH3jS0qN96O/bSh3kaw/h+Tq7ngyY9eWDUnTlTNUyqfqvCXioLe5Na5jFsL78ra/wuBU4iuEgd4w==} - - component-emitter@1.3.1: - resolution: {integrity: sha512-T0+barUSQRTUQASh8bx02dl+DhF54GtIDY13Y3m9oWTklKbb3Wv974meRpeZ3lp1JpLVECWWNHC4vaG2XHXouQ==} - - compressible@2.0.18: - resolution: {integrity: sha512-AF3r7P5dWxL8MxyITRMlORQNaOA2IkAFaTr4k7BUumjPtRpGDTZpl0Pb1XCO6JeDCBdp126Cgs9sMxqSjgYyRg==} - engines: {node: '>= 0.6'} - - compute-scroll-into-view@1.0.20: - resolution: {integrity: sha512-UCB0ioiyj8CRjtrvaceBLqqhZCVP+1B8+NWQhmdsm0VXOJtobBCf1dBQmebCCo34qZmUwZfIH2MZLqNHazrfjg==} - - compute-scroll-into-view@3.1.0: - resolution: {integrity: sha512-rj8l8pD4bJ1nx+dAkMhV1xB5RuZEyVysfxJqB1pRchh1KVvwOv9b7CGB8ZfjTImVv2oF+sYMUkMZq6Na5Ftmbg==} - - concat-map@0.0.1: - resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} - - concurrently@8.2.2: - resolution: {integrity: sha512-1dP4gpXFhei8IOtlXRE/T/4H88ElHgTiUzh71YUmtjTEHMSRS2Z/fgOxHSxxusGHogsRfxNq1vyAwxSC+EVyDg==} - engines: {node: ^14.13.0 || >=16.0.0} - hasBin: true - - config-chain@1.1.13: - resolution: {integrity: sha512-qj+f8APARXHrM0hraqXYb2/bOVSV4PvJQlNZ/DVj0QrmNM2q2euizkeuVckQ57J+W0mRH6Hvi+k50M4Jul2VRQ==} - - configstore@5.0.1: - resolution: {integrity: sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==} - engines: {node: '>=8'} - - console-control-strings@1.1.0: - resolution: {integrity: sha512-ty/fTekppD2fIwRvnZAVdeOiGd1c7YXEixbgJTNzqcxJWKQnjJ/V1bNEEE6hygpM3WjwHFUVK6HTjWSzV4a8sQ==} - - constant-case@2.0.0: - resolution: {integrity: sha512-eS0N9WwmjTqrOmR3o83F5vW8Z+9R1HnVz3xmzT2PMFug9ly+Au/fxRWlEBSb6LcZwspSsEn9Xs1uw9YgzAg1EQ==} - - content-disposition@0.5.4: - resolution: {integrity: sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==} - engines: {node: '>= 0.6'} - - content-type@1.0.5: - resolution: {integrity: sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==} - engines: {node: '>= 0.6'} - - convert-source-map@1.9.0: - resolution: {integrity: sha512-ASFBup0Mz1uyiIjANan1jzLQami9z1PoYSZCiiYW2FczPbenXc45FZdBZLzOT+r6+iciuEModtmCti+hjaAk0A==} - - cookie-signature@1.2.1: - resolution: {integrity: sha512-78KWk9T26NhzXtuL26cIJ8/qNHANyJ/ZYrmEXFzUmhZdjpBv+DlWlOANRTGBt48YcyslsLrj0bMLFTmXvLRCOw==} - engines: {node: '>=6.6.0'} - - cookie@0.4.2: - resolution: {integrity: sha512-aSWTXFzaKWkvHO1Ny/s+ePFpvKsPnjc551iI41v3ny/ow6tBG5Vd+FuqGNhh1LxOmVzOlGUriIlOaokOvhaStA==} - engines: {node: '>= 0.6'} - - cookie@0.5.0: - resolution: {integrity: sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==} - engines: {node: '>= 0.6'} - - cookies@0.8.0: - resolution: {integrity: sha512-8aPsApQfebXnuI+537McwYsDtjVxGm8gTIzQI3FDW6t5t/DAhERxtnbEPN/8RX+uZthoz4eCOgloXaE5cYyNow==} - engines: {node: '>= 0.8'} - - copy-descriptor@0.1.1: - resolution: {integrity: sha512-XgZ0pFcakEUlbwQEVNg3+QAis1FyTL3Qel9FYy8pSkQqoG3PNoT0bOCQtOXcOkur21r2Eq2kI+IE+gsmAEVlYw==} - engines: {node: '>=0.10.0'} - - copy-to@2.0.1: - resolution: {integrity: sha512-3DdaFaU/Zf1AnpLiFDeNCD4TOWe3Zl2RZaTzUvWiIk5ERzcCodOE20Vqq4fzCbNoHURFHT4/us/Lfq+S2zyY4w==} - - copyfiles@2.4.1: - resolution: {integrity: sha512-fereAvAvxDrQDOXybk3Qu3dPbOoKoysFMWtkY3mv5BsL8//OSZVL5DCLYqgRfY5cWirgRzlC+WSrxp6Bo3eNZg==} - hasBin: true - - core-js-pure@3.38.1: - resolution: {integrity: sha512-BY8Etc1FZqdw1glX0XNOq2FDwfrg/VGqoZOZCdaL+UmdaqDwQwYXkMJT4t6In+zfEfOJDcM9T0KdbBeJg8KKCQ==} - - core-util-is@1.0.3: - resolution: {integrity: sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==} - - cosmiconfig@7.1.0: - resolution: {integrity: sha512-AdmX6xUzdNASswsFtmwSt7Vj8po9IuqXm0UXz7QKPuEUmPB4XyjGfaAr2PSuELMwkRMVH1EpIkX5bTZGRB3eCA==} - engines: {node: '>=10'} - - cosmiconfig@8.3.6: - resolution: {integrity: sha512-kcZ6+W5QzcJ3P1Mt+83OUv/oHFqZHIx8DuxG6eZ5RGMERoLqp4BuGjhHLYGK+Kf5XVkQvqBSmAy/nGWN3qDgEA==} - engines: {node: '>=14'} - peerDependencies: - typescript: '>=4.9.5' - peerDependenciesMeta: - typescript: - optional: true - - crc@3.8.0: - resolution: {integrity: sha512-iX3mfgcTMIq3ZKLIsVFAbv7+Mc10kxabAGQb8HvjA1o3T1PIYprbakQ65d3I+2HGHt6nSKkM9PYjgoJO2KcFBQ==} - - crelt@1.0.6: - resolution: {integrity: sha512-VQ2MBenTq1fWZUH9DJNGti7kKv6EeAuYr3cLwxUWhIu1baTaXh4Ib5W2CqHVqib4/MqbYGJqiL3Zb8GJZr3l4g==} - - cron-parser@4.9.0: - resolution: {integrity: sha512-p0SaNjrHOnQeR8/VnfGbmg9te2kfyYSQ7Sc/j/6DtPL3JQvKxmjO9TSjNFpujqV3vEYYBvNNvXSxzyksBWAx1Q==} - engines: {node: '>=12.0.0'} - - cropperjs@1.6.0: - resolution: {integrity: sha512-BzLU/ecrfsbflwxgu+o7sQTrTlo52pVRZkTVrugEK5uyj6n8qKwAHP4s6+DWHqlXLqQ5B9+cM2MKeXiNfAsF6Q==} - - cross-env@7.0.3: - resolution: {integrity: sha512-+/HKd6EgcQCJGh2PSjZuUitQBQynKor4wrFbRg4DtAgS1aWO+gU52xpH7M9ScGgXSYmAVS9bIJ8EzuaGw0oNAw==} - engines: {node: '>=10.14', npm: '>=6', yarn: '>=1'} - hasBin: true - - cross-spawn@7.0.3: - resolution: {integrity: sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==} - engines: {node: '>= 8'} - - crypto-random-string@2.0.0: - resolution: {integrity: sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==} - engines: {node: '>=8'} - - css-color-keywords@1.0.0: - resolution: {integrity: sha512-FyyrDHZKEjXDpNJYvVsV960FiqQyXc/LlYmsxl2BcdMb2WPx0OGRVgTg55rPSyLSNMqP52R9r8geSp7apN3Ofg==} - engines: {node: '>=4'} - - css-loader@6.11.0: - resolution: {integrity: sha512-CTJ+AEQJjq5NzLga5pE39qdiSV56F8ywCIsqNIRF0r7BDgWsN25aazToqAFg7ZrtA/U016xudB3ffgweORxX7g==} - engines: {node: '>= 12.13.0'} - peerDependencies: - '@rspack/core': 0.x || 1.x - webpack: ^5.0.0 - peerDependenciesMeta: - '@rspack/core': - optional: true - webpack: - optional: true - - css-select@4.3.0: - resolution: {integrity: sha512-wPpOYtnsVontu2mODhA19JrqWxNsfdatRKd64kmpRbQgh1KtItko5sTnEpPdpSaJszTOhEMlF/RPz28qj4HqhQ==} - - css-to-react-native@3.2.0: - resolution: {integrity: sha512-e8RKaLXMOFii+02mOlqwjbD00KSEKqblnpO9e++1aXS1fPQOpS1YoqdVHBqPjHNoxeF2mimzVqawm2KCbEdtHQ==} - - css-what@6.1.0: - resolution: {integrity: sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw==} - engines: {node: '>= 6'} - - cssesc@3.0.0: - resolution: {integrity: sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==} - engines: {node: '>=4'} - hasBin: true - - csstype@3.1.3: - resolution: {integrity: sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==} - - data-uri-to-buffer@4.0.1: - resolution: {integrity: sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A==} - engines: {node: '>= 12'} - - date-fns-tz@2.0.0: - resolution: {integrity: sha512-OAtcLdB9vxSXTWHdT8b398ARImVwQMyjfYGkKD2zaGpHseG2UPHbHjXELReErZFxWdSLph3c2zOaaTyHfOhERQ==} - peerDependencies: - date-fns: '>=2.0.0' - - date-fns@2.30.0: - resolution: {integrity: sha512-fnULvOpxnC5/Vg3NCiWelDsLiUc9bRwAPs/+LfTLNvetFCtCTN+yQz15C/fs4AwX1R9K5GLtLfn8QW+dWisaAw==} - engines: {node: '>=0.11'} - - date-fns@3.6.0: - resolution: {integrity: sha512-fRHTG8g/Gif+kSh50gaGEdToemgfj74aRX3swtiouboip5JDLAyDE9F11nHMIcvOaXeOC6D7SpNhi7uFyB7Uww==} - - debounce@1.2.1: - resolution: {integrity: sha512-XRRe6Glud4rd/ZGQfiV1ruXSfbvfJedlV9Y6zOlP+2K04vBYiJEte6stfFkCP03aMnY5tsipamumUjL14fofug==} - - debug@2.6.9: - resolution: {integrity: sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==} - peerDependencies: - supports-color: '*' - peerDependenciesMeta: - supports-color: - optional: true - - debug@3.2.7: - resolution: {integrity: sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==} - peerDependencies: - supports-color: '*' - peerDependenciesMeta: - supports-color: - optional: true - - debug@4.3.4: - resolution: {integrity: sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==} - engines: {node: '>=6.0'} - peerDependencies: - supports-color: '*' - peerDependenciesMeta: - supports-color: - optional: true - - debug@4.3.6: - resolution: {integrity: sha512-O/09Bd4Z1fBrU4VzkhFqVgpPzaGbw6Sm9FEkBT1A/YBXQFGuuSxa1dN2nxgxS34JmKXqYx8CZAwEVoJFImUXIg==} - engines: {node: '>=6.0'} - peerDependencies: - supports-color: '*' - peerDependenciesMeta: - supports-color: - optional: true - - decode-uri-component@0.2.2: - resolution: {integrity: sha512-FqUYQ+8o158GyGTrMFJms9qh3CqTKvAqgqsTnkLI8sKu0028orqBhxNMFkFen0zGyg6epACD32pjVk58ngIErQ==} - engines: {node: '>=0.10'} - - decompress-response@4.2.1: - resolution: {integrity: sha512-jOSne2qbyE+/r8G1VU+G/82LBs2Fs4LAsTiLSHOCOMZQl2OKZ6i8i4IyHemTe+/yIXOtTcRQMzPcgyhoFlqPkw==} - engines: {node: '>=8'} - - decompress-response@6.0.0: - resolution: {integrity: sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==} - engines: {node: '>=10'} - - decompress-response@7.0.0: - resolution: {integrity: sha512-6IvPrADQyyPGLpMnUh6kfKiqy7SrbXbjoUuZ90WMBJKErzv2pCiwlGEXjRX9/54OnTq+XFVnkOnOMzclLI5aEA==} - engines: {node: '>=10'} - - deep-equal@1.0.1: - resolution: {integrity: sha512-bHtC0iYvWhyaTzvV3CZgPeZQqCOBGyGsVV7v4eevpdkLHfiSrXUdBG+qAuSz4RI70sszvjQ1QSZ98An1yNwpSw==} - - deep-extend@0.6.0: - resolution: {integrity: sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==} - engines: {node: '>=4.0.0'} - - deepmerge@2.2.1: - resolution: {integrity: sha512-R9hc1Xa/NOBi9WRVUWg19rl1UB7Tt4kuPd+thNJgFZoxXsTz7ncaPaeIm+40oSGuP33DfMb4sZt1QIGiJzC4EA==} - engines: {node: '>=0.10.0'} - - deepmerge@4.3.1: - resolution: {integrity: sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==} - engines: {node: '>=0.10.0'} - - default-browser-id@3.0.0: - resolution: {integrity: sha512-OZ1y3y0SqSICtE8DE4S8YOE9UZOJ8wO16fKWVP5J1Qz42kV9jcnMVFrEE/noXb/ss3Q4pZIH79kxofzyNNtUNA==} - engines: {node: '>=12'} - - default-browser@4.0.0: - resolution: {integrity: sha512-wX5pXO1+BrhMkSbROFsyxUm0i/cJEScyNhA4PPxc41ICuv05ZZB/MX28s8aZx6xjmatvebIapF6hLEKEcpneUA==} - engines: {node: '>=14.16'} - - defaults@1.0.4: - resolution: {integrity: sha512-eFuaLoy/Rxalv2kr+lqMlUnrDWV+3j4pljOIJgLIhI058IQfWJ7vXhyEIHu+HtC738klGALYxOKDO0bQP3tg8A==} - - defer-to-connect@2.0.1: - resolution: {integrity: sha512-4tvttepXG1VaYGrRibk5EwJd1t4udunSOVMdLSAL6mId1ix438oPwPZMALY41FCijukO1L0twNcGsdzS7dHgDg==} - engines: {node: '>=10'} - - define-data-property@1.1.4: - resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==} - engines: {node: '>= 0.4'} - - define-lazy-prop@2.0.0: - resolution: {integrity: sha512-Ds09qNh8yw3khSjiJjiUInaGX9xlqZDY7JVryGxdxV7NPeuqQfplOpQ66yJFZut3jLa5zOwkXw1g9EI2uKh4Og==} - engines: {node: '>=8'} - - define-lazy-prop@3.0.0: - resolution: {integrity: sha512-N+MeXYoqr3pOgn8xfyRPREN7gHakLYjhsHhWGT3fWAiL4IkAt0iDw14QiiEm2bE30c5XX5q0FtAA3CK5f9/BUg==} - engines: {node: '>=12'} - - define-property@0.2.5: - resolution: {integrity: sha512-Rr7ADjQZenceVOAKop6ALkkRAmH1A4Gx9hV/7ZujPUN2rkATqFO0JZLZInbAjpZYoJ1gUx8MRMQVkYemcbMSTA==} - engines: {node: '>=0.10.0'} - - define-property@1.0.0: - resolution: {integrity: sha512-cZTYKFWspt9jZsMscWo8sc/5lbPC9Q0N5nBLgb+Yd915iL3udB1uFgS3B8YCx66UVHq018DAVFoee7x+gxggeA==} - engines: {node: '>=0.10.0'} - - define-property@2.0.2: - resolution: {integrity: sha512-jwK2UV4cnPpbcG7+VRARKTZPUWowwXA8bzH5NP6ud0oeAxyYPuGZUAC7hMugpCdz4BeSZl2Dl9k66CHJ/46ZYQ==} - engines: {node: '>=0.10.0'} - - del@5.1.0: - resolution: {integrity: sha512-wH9xOVHnczo9jN2IW68BabcecVPxacIA3g/7z6vhSU/4stOKQzeCRK0yD0A24WiAAUJmmVpWqrERcTxnLo3AnA==} - engines: {node: '>=8'} - - delayed-stream@1.0.0: - resolution: {integrity: sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==} - engines: {node: '>=0.4.0'} - - delegates@1.0.0: - resolution: {integrity: sha512-bd2L678uiWATM6m5Z1VzNCErI3jiGzt6HGY8OVICs40JQq/HALfbyNJmp0UDakEY4pMMaN0Ly5om/B1VI/+xfQ==} - - depd@1.1.2: - resolution: {integrity: sha512-7emPTl6Dpo6JRXOXjLRxck+FlLRX5847cLKEn00PLAgc3g2hTZZgr+e4c2v6QpSmLeFP3n5yUo7ft6avBK/5jQ==} - engines: {node: '>= 0.6'} - - depd@2.0.0: - resolution: {integrity: sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==} - engines: {node: '>= 0.8'} - - destroy@1.2.0: - resolution: {integrity: sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==} - engines: {node: '>= 0.8', npm: 1.2.8000 || >= 1.4.16} - - detect-file@1.0.0: - resolution: {integrity: sha512-DtCOLG98P007x7wiiOmfI0fi3eIKyWiLTGJ2MDnVi/E04lWGbf+JzrRHMm0rgIIZJGtHpKpbVgLWHrv8xXpc3Q==} - engines: {node: '>=0.10.0'} - - detect-indent@6.1.0: - resolution: {integrity: sha512-reYkTUJAZb9gUuZ2RvVCNhVHdg62RHnJ7WJl8ftMi4diZ6NWlciOzQN88pUhSELEwflJht4oQDv0F0BMlwaYtA==} - engines: {node: '>=8'} - - detect-indent@7.0.1: - resolution: {integrity: sha512-Mc7QhQ8s+cLrnUfU/Ji94vG/r8M26m8f++vyres4ZoojaRDpZ1eSIh/EpzLNwlWuvzSZ3UbDFspjFvTDXe6e/g==} - engines: {node: '>=12.20'} - - detect-libc@2.0.3: - resolution: {integrity: sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw==} - engines: {node: '>=8'} - - detect-newline@4.0.1: - resolution: {integrity: sha512-qE3Veg1YXzGHQhlA6jzebZN2qVf6NX+A7m7qlhCGG30dJixrAQhYOsJjsnBjJkCSmuOPpCk30145fr8FV0bzog==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - detect-node-es@1.1.0: - resolution: {integrity: sha512-ypdmJU/TbBby2Dxibuv7ZLW3Bs1QEmM7nHjEANfohJLvE0XVujisn1qPJcZxg+qDucsr+bP6fLD1rPS3AhJ7EQ==} - - detect-node@2.1.0: - resolution: {integrity: sha512-T0NIuQpnTvFDATNuHN5roPwSBG83rFsuO+MXXH9/3N1eFbn4wcPjttvjMLEPWJ0RGUYgQE7cGgS3tNxbqCGM7g==} - - dir-glob@3.0.1: - resolution: {integrity: sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==} - engines: {node: '>=8'} - - direction@1.0.4: - resolution: {integrity: sha512-GYqKi1aH7PJXxdhTeZBFrg8vUBeKXi+cNprXsC1kpJcbcVnV9wBsrOu1cQEdG0WeQwlfHiy3XvnKfIrJ2R0NzQ==} - hasBin: true - - dkim-signer@0.2.2: - resolution: {integrity: sha512-24OZ3cCA30UTRz+Plpg+ibfPq3h7tDtsJRg75Bo0pGakZePXcPBddY80bKi1Bi7Jsz7tL5Cw527mhCRDvNFgfg==} - - dnd-core@16.0.1: - resolution: {integrity: sha512-HK294sl7tbw6F6IeuK16YSBUoorvHpY8RHO+9yFfaJyCDVb6n7PRcezrOEOa2SBCqiYpemh5Jx20ZcjKdFAVng==} - - dom-converter@0.2.0: - resolution: {integrity: sha512-gd3ypIPfOMr9h5jIKq8E3sHOTCjeirnl0WK5ZdS1AW0Odt0b1PaWaHdJ4Qk4klv+YB9aJBS7mESXjFoDQPu6DA==} - - dom-helpers@5.2.1: - resolution: {integrity: sha512-nRCa7CK3VTrM2NmGkIy4cbK7IZlgBE/PYMn55rrXefr5xXDP0LdtfPnblFDoVdcAfslJ7or6iqAUnx0CCGIWQA==} - - dom-serializer@1.4.1: - resolution: {integrity: sha512-VHwB3KfrcOOkelEG2ZOfxqLZdfkil8PtJi4P8N2MMXucZq2yLp75ClViUlOVwyoHEDjYU433Aq+5zWP61+RGag==} - - dom-serializer@2.0.0: - resolution: {integrity: sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg==} - - domelementtype@2.3.0: - resolution: {integrity: sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==} - - domhandler@4.3.1: - resolution: {integrity: sha512-GrwoxYN+uWlzO8uhUXRl0P+kHE4GtVPfYzVLcUxPL7KNdHKj66vvlhiweIHqYYXWlw+T8iLMp42Lm67ghw4WMQ==} - engines: {node: '>= 4'} - - domhandler@5.0.3: - resolution: {integrity: sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==} - engines: {node: '>= 4'} - - domutils@2.8.0: - resolution: {integrity: sha512-w96Cjofp72M5IIhpjgobBimYEfoPjx1Vx0BSX9P30WBdZW2WIKU0T1Bd0kz2eNZ9ikjKgHbEyKx8BB6H1L3h3A==} - - domutils@3.1.0: - resolution: {integrity: sha512-H78uMmQtI2AhgDJjWeQmHwJJ2bLPD3GMmO7Zja/ZZh84wkm+4ut+IUnUdRa8uCGX88DiVx1j6FRe1XfxEgjEZA==} - - dot-case@2.1.1: - resolution: {integrity: sha512-HnM6ZlFqcajLsyudHq7LeeLDr2rFAVYtDv/hV5qchQEidSck8j9OPUsXY9KwJv/lHMtYlX4DjRQqwFYa+0r8Ug==} - - dot-case@3.0.4: - resolution: {integrity: sha512-Kv5nKlh6yRrdrGvxeJ2e5y2eRUpkUosIW4A2AS38zwSz27zu7ufDwQPi5Jhs3XAlGNetl3bmnGhQsMtkKJnj3w==} - - dot-prop@5.3.0: - resolution: {integrity: sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==} - engines: {node: '>=8'} - - dotenv@14.2.0: - resolution: {integrity: sha512-05POuPJyPpO6jqzTNweQFfAyMSD4qa4lvsMOWyTRTdpHKy6nnnN+IYWaXF+lHivhBH/ufDKlR4IWCAN3oPnHuw==} - engines: {node: '>=12'} - - dotenv@16.4.5: - resolution: {integrity: sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==} - engines: {node: '>=12'} - - duplexer@0.1.2: - resolution: {integrity: sha512-jtD6YG370ZCIi/9GTaJKQxWTZD045+4R4hTk/x1UyoqadyJ9x9CgSi1RlVDQF8U2sxLLSnFkCaMihqljHIWgMg==} - - eastasianwidth@0.2.0: - resolution: {integrity: sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==} - - ecdsa-sig-formatter@1.0.11: - resolution: {integrity: sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==} - - ee-first@1.1.1: - resolution: {integrity: sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==} - - electron-to-chromium@1.5.13: - resolution: {integrity: sha512-lbBcvtIJ4J6sS4tb5TLp1b4LyfCdMkwStzXPyAgVgTRAsep4bvrAGaBOP7ZJtQMNJpSQ9SqG4brWOroNaQtm7Q==} - - elliptic@6.5.7: - resolution: {integrity: sha512-ESVCtTwiA+XhY3wyh24QqRGBoP3rEdDUl3EDUUo9tft074fi19IrdpH7hLCMMP3CIj7jb3W96rn8lt/BqIlt5Q==} - - emittery@0.12.1: - resolution: {integrity: sha512-pYyW59MIZo0HxPFf+Vb3+gacUu0gxVS3TZwB2ClwkEZywgF9f9OJDoVmNLojTn0vKX3tO9LC+pdQEcLP4Oz/bQ==} - engines: {node: '>=12'} - - emoji-regex@8.0.0: - resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==} - - emoji-regex@9.2.2: - resolution: {integrity: sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==} - - emojis-list@3.0.0: - resolution: {integrity: sha512-/kyM18EfinwXZbno9FyUGeFh87KC8HRQBQGildHZbEuRyWFOmv1U10o9BBp8XVZDVNNuQKyIGIu5ZYAAXJ0V2Q==} - engines: {node: '>= 4'} - - enabled@2.0.0: - resolution: {integrity: sha512-AKrN98kuwOzMIdAizXGI86UFBoo26CL21UM763y1h/GMSJ4/OHU9k2YlsmBpyScFo/wbLzWQJBMCW4+IO3/+OQ==} - - encodeurl@1.0.2: - resolution: {integrity: sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==} - engines: {node: '>= 0.8'} - - end-of-stream@1.4.4: - resolution: {integrity: sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==} - - enhanced-resolve@5.17.1: - resolution: {integrity: sha512-LMHl3dXhTcfv8gM4kEzIUeTQ+7fpdA0l2tUf34BddXPkz2A5xJ5L/Pchd5BL6rdccM9QGvu0sWZzK1Z1t4wwyg==} - engines: {node: '>=10.13.0'} - - entities@2.1.0: - resolution: {integrity: sha512-hCx1oky9PFrJ611mf0ifBLBRW8lUUVRlFolb5gWRfIELabBlbp9xZvrqZLZAs+NxFnbfQoeGd8wDkygjg7U85w==} - - entities@2.2.0: - resolution: {integrity: sha512-p92if5Nz619I0w+akJrLZH0MX0Pb5DX39XOwQTtXSdQQOaYH03S1uIQp4mhOZtAXrxq4ViO67YTiLBo2638o9A==} - - entities@4.5.0: - resolution: {integrity: sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==} - engines: {node: '>=0.12'} - - error-ex@1.3.2: - resolution: {integrity: sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==} - - error-stack-parser@2.1.4: - resolution: {integrity: sha512-Sk5V6wVazPhq5MhpO+AUxJn5x7XSXGl1R93Vn7i+zS15KDVxQijejNCrz8340/2bgLBjR9GtEG8ZVKONDjcqGQ==} - - es-define-property@1.0.0: - resolution: {integrity: sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==} - engines: {node: '>= 0.4'} - - es-errors@1.3.0: - resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==} - engines: {node: '>= 0.4'} - - es-module-lexer@1.5.4: - resolution: {integrity: sha512-MVNK56NiMrOwitFB7cqDwq0CQutbw+0BvLshJSse0MUNU+y1FC3bUS/AQg7oUng+/wKrrki7JfmwtVHkVfPLlw==} - - esbuild-loader@2.21.0: - resolution: {integrity: sha512-k7ijTkCT43YBSZ6+fBCW1Gin7s46RrJ0VQaM8qA7lq7W+OLsGgtLyFV8470FzYi/4TeDexniTBTPTwZUnXXR5g==} - peerDependencies: - webpack: ^4.40.0 || ^5.0.0 - - esbuild-register@3.5.0: - resolution: {integrity: sha512-+4G/XmakeBAsvJuDugJvtyF1x+XJT4FMocynNpxrvEBViirpfUn2PgNpCHedfWhF4WokNsO/OvMKrmJOIJsI5A==} - peerDependencies: - esbuild: '>=0.12 <1' - - esbuild@0.16.17: - resolution: {integrity: sha512-G8LEkV0XzDMNwXKgM0Jwu3nY3lSTwSGY6XbxM9cr9+s0T/qSV1q1JVPBGzm3dcjhCic9+emZDmMffkwgPeOeLg==} - engines: {node: '>=12'} - hasBin: true - - esbuild@0.19.11: - resolution: {integrity: sha512-HJ96Hev2hX/6i5cDVwcqiJBBtuo9+FeIJOtZ9W1kA5M6AMJRHUZlpYZ1/SbEwtO0ioNAW8rUooVpC/WehY2SfA==} - engines: {node: '>=12'} - hasBin: true - - escalade@3.2.0: - resolution: {integrity: sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==} - engines: {node: '>=6'} - - escape-html@1.0.3: - resolution: {integrity: sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==} - - escape-string-regexp@1.0.5: - resolution: {integrity: sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==} - engines: {node: '>=0.8.0'} - - escape-string-regexp@2.0.0: - resolution: {integrity: sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==} - engines: {node: '>=8'} - - escape-string-regexp@4.0.0: - resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} - engines: {node: '>=10'} - - eslint-scope@5.1.1: - resolution: {integrity: sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==} - engines: {node: '>=8.0.0'} - - esm@3.2.25: - resolution: {integrity: sha512-U1suiZ2oDVWv4zPO56S0NcR5QriEahGtdN2OR6FiOG4WJvcjBVFB0qI4+eKoWFH483PKGuLuu6V8Z4T5g63UVA==} - engines: {node: '>=6'} - - esrecurse@4.3.0: - resolution: {integrity: sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==} - engines: {node: '>=4.0'} - - estraverse@4.3.0: - resolution: {integrity: sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==} - engines: {node: '>=4.0'} - - estraverse@5.3.0: - resolution: {integrity: sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==} - engines: {node: '>=4.0'} - - event-target-shim@5.0.1: - resolution: {integrity: sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==} - engines: {node: '>=6'} - - eventemitter3@4.0.7: - resolution: {integrity: sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==} - - events@3.3.0: - resolution: {integrity: sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==} - engines: {node: '>=0.8.x'} - - eventsource@2.0.2: - resolution: {integrity: sha512-IzUmBGPR3+oUG9dUeXynyNmf91/3zUSJg1lCktzKw47OXuhco54U3r9B7O4XX+Rb1Itm9OZ2b0RkTs10bICOxA==} - engines: {node: '>=12.0.0'} - - execa@5.1.1: - resolution: {integrity: sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==} - engines: {node: '>=10'} - - execa@7.2.0: - resolution: {integrity: sha512-UduyVP7TLB5IcAQl+OzLyLcS/l32W/GLg+AhHJ+ow40FOk2U3SAllPwR44v4vmdFwIWqpdwxxpQbF1n5ta9seA==} - engines: {node: ^14.18.0 || ^16.14.0 || >=18.0.0} - - expand-brackets@2.1.4: - resolution: {integrity: sha512-w/ozOKR9Obk3qoWeY/WDi6MFta9AoMR+zud60mdnbniMcBxRuFJyDt2LdX/14A1UABeqk+Uk+LDfUpvoGKppZA==} - engines: {node: '>=0.10.0'} - - expand-template@2.0.3: - resolution: {integrity: sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg==} - engines: {node: '>=6'} - - expand-tilde@2.0.2: - resolution: {integrity: sha512-A5EmesHW6rfnZ9ysHQjPdJRni0SRar0tjtG5MNtm9n5TUvsYU8oozprtRD4AqHxcZWWlVuAmQo2nWKfN9oyjTw==} - engines: {node: '>=0.10.0'} - - extend-shallow@2.0.1: - resolution: {integrity: sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==} - engines: {node: '>=0.10.0'} - - extend-shallow@3.0.2: - resolution: {integrity: sha512-BwY5b5Ql4+qZoefgMj2NUmx+tehVTH/Kf4k1ZEtOHNFcm2wSxMRo992l6X3TIgni2eZVTZ85xMOjF31fwZAj6Q==} - engines: {node: '>=0.10.0'} - - extend@3.0.2: - resolution: {integrity: sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==} - - external-editor@3.1.0: - resolution: {integrity: sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==} - engines: {node: '>=4'} - - extglob@2.0.4: - resolution: {integrity: sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==} - engines: {node: '>=0.10.0'} - - fast-deep-equal@3.1.3: - resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==} - - fast-fifo@1.3.2: - resolution: {integrity: sha512-/d9sfos4yxzpwkDkuN7k2SqFKtYNmCTzgfEpz82x34IM9/zc8KGxQoXg1liNC/izpRM/MBdt44Nmx41ZWqk+FQ==} - - fast-glob@3.3.2: - resolution: {integrity: sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==} - engines: {node: '>=8.6.0'} - - fast-json-stable-stringify@2.1.0: - resolution: {integrity: sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==} - - fast-safe-stringify@2.1.1: - resolution: {integrity: sha512-W+KJc2dmILlPplD/H4K9l9LcAHAfPtP6BY84uVLXQ6Evcz9Lcg33Y2z1IVblT6xdY54PXYVHEv+0Wpq8Io6zkA==} - - fast-uri@3.0.1: - resolution: {integrity: sha512-MWipKbbYiYI0UC7cl8m/i/IWTqfC8YXsqjzybjddLsFjStroQzsHXkc73JutMvBiXmOvapk+axIl79ig5t55Bw==} - - fast-xml-parser@4.4.1: - resolution: {integrity: sha512-xkjOecfnKGkSsOwtZ5Pz7Us/T6mrbPQrq0nh+aCO5V9nk5NLWmasAHumTKjiPJPWANe+kAZ84Jc8ooJkzZ88Sw==} - hasBin: true - - fastq@1.17.1: - resolution: {integrity: sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==} - - fecha@4.2.3: - resolution: {integrity: sha512-OP2IUU6HeYKJi3i0z4A19kHMQoLVs4Hc+DPqqxI2h/DPZHTm/vjsfC6P0b4jCMy14XizLBqvndQ+UilD7707Jw==} - - fetch-blob@3.2.0: - resolution: {integrity: sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ==} - engines: {node: ^12.20 || >= 14.13} - - figures@3.2.0: - resolution: {integrity: sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==} - engines: {node: '>=8'} - - fill-range@4.0.0: - resolution: {integrity: sha512-VcpLTWqWDiTerugjj8e3+esbg+skS3M9e54UuR3iCeIDMXCLTsAH8hTSzDQU/X6/6t3eYkOKoZSef2PlU6U1XQ==} - engines: {node: '>=0.10.0'} - - fill-range@7.1.1: - resolution: {integrity: sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==} - engines: {node: '>=8'} - - find-root@1.1.0: - resolution: {integrity: sha512-NKfW6bec6GfKc0SGx1e07QZY9PE99u0Bft/0rzSD5k3sO/vwkVUpDUKVm5Gpp5Ue3YfShPFTX2070tDs5kB9Ng==} - - find-up@3.0.0: - resolution: {integrity: sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==} - engines: {node: '>=6'} - - find-up@4.1.0: - resolution: {integrity: sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==} - engines: {node: '>=8'} - - find-up@5.0.0: - resolution: {integrity: sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==} - engines: {node: '>=10'} - - findup-sync@2.0.0: - resolution: {integrity: sha512-vs+3unmJT45eczmcAZ6zMJtxN3l/QXeccaXQx5cu/MeJMhewVfoWZqibRkOxPnmoR59+Zy5hjabfQc6JLSah4g==} - engines: {node: '>= 0.10'} - - fined@1.2.0: - resolution: {integrity: sha512-ZYDqPLGxDkDhDZBjZBb+oD1+j0rA4E0pXY50eplAAOPg2N/gUBSSk5IM1/QhPfyVo19lJ+CvXpqfvk+b2p/8Ng==} - engines: {node: '>= 0.10'} - - flagged-respawn@1.0.1: - resolution: {integrity: sha512-lNaHNVymajmk0OJMBn8fVUAU1BtDeKIqKoVhk4xAALB57aALg6b4W0MfJ/cUE0g9YBXy5XhSlPIpYIJ7HaY/3Q==} - engines: {node: '>= 0.10'} - - flat-cache@3.2.0: - resolution: {integrity: sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==} - engines: {node: ^10.12.0 || >=12.0.0} - - flatted@3.3.1: - resolution: {integrity: sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==} - - fn.name@1.1.0: - resolution: {integrity: sha512-GRnmB5gPyJpAhTQdSZTSp9uaPSvl09KoYcMQtsB9rQoOmzs9dH6ffeccH+Z+cv6P68Hu5bC6JjRh4Ah/mHSNRw==} - - follow-redirects@1.15.6: - resolution: {integrity: sha512-wWN62YITEaOpSK584EZXJafH1AGpO8RVgElfkuXbTOrPX4fIfOyEpW/CsiNd8JdYrAoOvafRTOEnvsO++qCqFA==} - engines: {node: '>=4.0'} - peerDependencies: - debug: '*' - peerDependenciesMeta: - debug: - optional: true - - for-in@1.0.2: - resolution: {integrity: sha512-7EwmXrOjyL+ChxMhmG5lnW9MPt1aIeZEwKhQzoBUdTV0N3zuwWDZYVJatDvZ2OyzPUvdIAZDsCetk3coyMfcnQ==} - engines: {node: '>=0.10.0'} - - for-own@1.0.0: - resolution: {integrity: sha512-0OABksIGrxKK8K4kynWkQ7y1zounQxP+CWnyclVwj81KW3vlLlGUx57DKGcP/LH216GzqnstnPocF16Nxs0Ycg==} - engines: {node: '>=0.10.0'} - - foreground-child@3.3.0: - resolution: {integrity: sha512-Ld2g8rrAyMYFXBhEqMz8ZAHBi4J4uS1i/CxGMDnjyFWddMXLVcDp051DZfu+t7+ab7Wv6SMqpWmyFIj5UbfFvg==} - engines: {node: '>=14'} - - fork-ts-checker-webpack-plugin@9.0.2: - resolution: {integrity: sha512-Uochze2R8peoN1XqlSi/rGUkDQpRogtLFocP9+PGu68zk1BDAKXfdeCdyVZpgTk8V8WFVQXdEz426VKjXLO1Gg==} - engines: {node: '>=12.13.0', yarn: '>=1.0.0'} - peerDependencies: - typescript: '>3.6.0' - webpack: ^5.11.0 - - form-data-encoder@1.7.2: - resolution: {integrity: sha512-qfqtYan3rxrnCk1VYaA4H+Ms9xdpPqvLZa6xmMgFvhO32x7/3J/ExcTd6qpxM0vH2GdMI+poehyBZvqfMTto8A==} - - form-data@4.0.0: - resolution: {integrity: sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==} - engines: {node: '>= 6'} - - formdata-node@4.4.1: - resolution: {integrity: sha512-0iirZp3uVDjVGt9p49aTaqjk84TrglENEDuqfdlZQ1roC9CWlPk6Avf8EEnZNcAqPonwkG35x4n3ww/1THYAeQ==} - engines: {node: '>= 12.20'} - - formdata-polyfill@4.0.10: - resolution: {integrity: sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g==} - engines: {node: '>=12.20.0'} - - formidable@1.2.6: - resolution: {integrity: sha512-KcpbcpuLNOwrEjnbpMC0gS+X8ciDoZE1kkqzat4a8vrprf+s9pKNQ/QIwWfbfs4ltgmFl3MD177SNTkve3BwGQ==} - deprecated: 'Please upgrade to latest, formidable@v2 or formidable@v3! Check these notes: https://bit.ly/2ZEqIau' - - formik@2.4.0: - resolution: {integrity: sha512-QZiWztt9fD84EYcF7Bmr431ZhIm1xUVgBACbTuJ6azPrUpVp7o6q+t9HJaIQsFZrMfcBPNBotYtDgyDpzQ3z0Q==} - peerDependencies: - react: '>=16.8.0' - - fractional-indexing@3.2.0: - resolution: {integrity: sha512-PcOxmqwYCW7O2ovKRU8OoQQj2yqTfEB/yeTYk4gPid6dN5ODRfU1hXd9tTVZzax/0NkO7AxpHykvZnT1aYp/BQ==} - engines: {node: ^14.13.1 || >=16.0.0} - - fragment-cache@0.2.1: - resolution: {integrity: sha512-GMBAbW9antB8iZRHLoGw0b3HANt57diZYFO/HL1JGIC1MjKrdmhxvrJbupnVvpys0zsz7yBApXdQyfepKly2kA==} - engines: {node: '>=0.10.0'} - - fresh@0.5.2: - resolution: {integrity: sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==} - engines: {node: '>= 0.6'} - - fs-constants@1.0.0: - resolution: {integrity: sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==} - - fs-extra@10.0.0: - resolution: {integrity: sha512-C5owb14u9eJwizKGdchcDUQeFtlSHHthBk8pbX9Vc1PFZrLombudjDnNns88aYslCyF6IY5SUw3Roz6xShcEIQ==} - engines: {node: '>=12'} - - fs-extra@7.0.1: - resolution: {integrity: sha512-YJDaCJZEnBmcbw13fvdAM9AwNOJwOzrE4pqMqBq5nFiEqXUqHwlK4B+3pUw6JNvfSPtX05xFHtYy/1ni01eGCw==} - engines: {node: '>=6 <7 || >=8'} - - fs-extra@8.1.0: - resolution: {integrity: sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==} - engines: {node: '>=6 <7 || >=8'} - - fs-jetpack@4.3.1: - resolution: {integrity: sha512-dbeOK84F6BiQzk2yqqCVwCPWTxAvVGJ3fMQc6E2wuEohS28mR6yHngbrKuVCK1KHRx/ccByDylqu4H5PCP2urQ==} - - fs-minipass@2.1.0: - resolution: {integrity: sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==} - engines: {node: '>= 8'} - - fs-monkey@1.0.6: - resolution: {integrity: sha512-b1FMfwetIKymC0eioW7mTywihSQE4oLzQn1dB6rZB5fx/3NpNEdAWeCSMB+60/AeT0TCXsxzAlcYVEFCTAksWg==} - - fs.realpath@1.0.0: - resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==} - - fsevents@2.3.3: - resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} - engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} - os: [darwin] - - function-bind@1.1.2: - resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} - - fuzzysort@3.0.2: - resolution: {integrity: sha512-ZyahVgxvckB1Qosn7YGWLDJJp2XlyaQ2WmZeI+d0AzW0AMqVYnz5N89G6KAKa6m/LOtv+kzJn4lhDF/yVg11Cg==} - - gauge@3.0.2: - resolution: {integrity: sha512-+5J6MS/5XksCuXq++uFRsnUd7Ovu1XenbeuIuNRJxYWjgQbPuFhT14lAvsWfqfAmnwluf1OwMjz39HjfLPci0Q==} - engines: {node: '>=10'} - deprecated: This package is no longer supported. - - get-caller-file@2.0.5: - resolution: {integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==} - engines: {node: 6.* || 8.* || >= 10.*} - - get-intrinsic@1.2.4: - resolution: {integrity: sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==} - engines: {node: '>= 0.4'} - - get-it@8.6.5: - resolution: {integrity: sha512-o1hjPwrb/icm3WJbCweTSq8mKuDfJlqwbFauI+Pdgid99at/BFaBXFBJZE+uqvHyOVARE4z680S44vrDm8SsCw==} - engines: {node: '>=14.0.0'} - - get-latest-version@5.1.0: - resolution: {integrity: sha512-Q6IBWr/zzw57zIkJmNhI23eRTw3nZ4BWWK034meLwOYU9L3J3IpXiyM73u2pYUwN6U7ahkerCwg2T0jlxiLwsw==} - engines: {node: '>=14.18'} - - get-nonce@1.0.1: - resolution: {integrity: sha512-FJhYRoDaiatfEkUK8HKlicmu/3SGFD51q3itKDGoSTysQJBnfOcxU5GxnhE1E6soB76MbT0MBtnKJuXyAx+96Q==} - engines: {node: '>=6'} - - get-package-type@0.1.0: - resolution: {integrity: sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==} - engines: {node: '>=8.0.0'} - - get-stdin@9.0.0: - resolution: {integrity: sha512-dVKBjfWisLAicarI2Sf+JuBE/DghV4UzNAVe9yhEJuzeREd3JhOTE9cUaJTeSa77fsbQUK3pcOpJfM59+VKZaA==} - engines: {node: '>=12'} - - get-stream@5.2.0: - resolution: {integrity: sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==} - engines: {node: '>=8'} - - get-stream@6.0.1: - resolution: {integrity: sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==} - engines: {node: '>=10'} - - get-value@2.0.6: - resolution: {integrity: sha512-Ln0UQDlxH1BapMu3GPtf7CuYNwRZf2gwCuPqbyG6pB8WfmFpzqcy4xtAaAMUhnNqjMKTiCPZG2oMT3YSx8U2NA==} - engines: {node: '>=0.10.0'} - - getopts@2.3.0: - resolution: {integrity: sha512-5eDf9fuSXwxBL6q5HX+dhDj+dslFGWzU5thZ9kNKUkcPtaPdatmUFKwHFrLb/uf/WpA4BHET+AX3Scl56cAjpA==} - - git-hooks-list@3.1.0: - resolution: {integrity: sha512-LF8VeHeR7v+wAbXqfgRlTSX/1BJR9Q1vEMR8JAz1cEg6GX07+zyj3sAdDvYjj/xnlIfVuGgj4qBei1K3hKH+PA==} - - git-up@7.0.0: - resolution: {integrity: sha512-ONdIrbBCFusq1Oy0sC71F5azx8bVkvtZtMJAsv+a6lz5YAmbNnLD6HAB4gptHZVLPR8S2/kVN6Gab7lryq5+lQ==} - - git-url-parse@13.1.0: - resolution: {integrity: sha512-5FvPJP/70WkIprlUZ33bm4UAaFdjcLkJLpWft1BeZKqwR0uhhNGoKwlUaPtVb4LxCSQ++erHapRak9kWGj+FCA==} - - github-from-package@0.0.0: - resolution: {integrity: sha512-SyHy3T1v2NUXn29OsWdxmK6RwHD+vkj3v8en8AOBZ1wBQ/hCAQ5bAQTD02kW4W9tUp/3Qh6J8r9EvntiyCmOOw==} - - glob-parent@5.1.2: - resolution: {integrity: sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==} - engines: {node: '>= 6'} - - glob-to-regexp@0.4.1: - resolution: {integrity: sha512-lkX1HJXwyMcprw/5YUZc2s7DrpAiHB21/V+E1rHUrVNokkvB6bqMzT0VfV6/86ZNabt1k14YOIaT7nDvOX3Iiw==} - - glob@10.4.2: - resolution: {integrity: sha512-GwMlUF6PkPo3Gk21UxkCohOv0PLcIXVtKyLlpEI28R/cO/4eNOdmLk3CMW1wROV/WR/EsZOWAfBbBOqYvs88/w==} - engines: {node: '>=16 || 14 >=14.18'} - hasBin: true - - glob@7.2.3: - resolution: {integrity: sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==} - deprecated: Glob versions prior to v9 are no longer supported - - glob@8.1.0: - resolution: {integrity: sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==} - engines: {node: '>=12'} - deprecated: Glob versions prior to v9 are no longer supported - - global-modules@1.0.0: - resolution: {integrity: sha512-sKzpEkf11GpOFuw0Zzjzmt4B4UZwjOcG757PPvrfhxcLFbq0wpsgpOqxpxtxFiCG4DtG93M6XRVbF2oGdev7bg==} - engines: {node: '>=0.10.0'} - - global-prefix@1.0.2: - resolution: {integrity: sha512-5lsx1NUDHtSjfg0eHlmYvZKv8/nVqX4ckFbM+FrGcQ+04KWcWFo9P5MxPZYSzUvyzmdTbI7Eix8Q4IbELDqzKg==} - engines: {node: '>=0.10.0'} - - globals@11.12.0: - resolution: {integrity: sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==} - engines: {node: '>=4'} - - globby@10.0.2: - resolution: {integrity: sha512-7dUi7RvCoT/xast/o/dLN53oqND4yk0nsHkhRgn9w65C4PofCLOoJ39iSOg+qVDdWQPIEj+eszMHQ+aLVwwQSg==} - engines: {node: '>=8'} - - globby@13.2.2: - resolution: {integrity: sha512-Y1zNGV+pzQdh7H39l9zgB4PJqjRNqydvdYCDG4HFXM4XuvSaQQlEc91IU1yALL8gUTDomgBAfz3XJdmUS+oo0w==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - gopd@1.0.1: - resolution: {integrity: sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==} - - got@11.8.6: - resolution: {integrity: sha512-6tfZ91bOr7bOXnK7PRDCGBLa1H4U080YHNaAQ2KsMGlLEzRbk44nsZF2E1IeRc3vtJHPVbKCYgdFbaGO2ljd8g==} - engines: {node: '>=10.19.0'} - - graceful-fs@4.2.10: - resolution: {integrity: sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==} - - graceful-fs@4.2.11: - resolution: {integrity: sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==} - - grant-koa@5.4.8: - resolution: {integrity: sha512-Kw8np9AL3Z3mZuvoSUklHJpTe3xx7iLBDauRyIwwbDLRr/5Ll6APmOFHixXj+Vw+LGEnreTxO35CyhAf9oBUMA==} - engines: {node: '>=8.0.0'} - peerDependencies: - koa: '>=2.0.0' - - grant@5.4.22: - resolution: {integrity: sha512-DEi+/JjXT84mmFYhSmv+SX14v+3Z7vuCIYAMwtdPCTXHMSLhWqSYqWAMXDUQZuV7yaJv2d84AYnkCFNooLKBsA==} - engines: {node: '>=12.0.0'} - - gzip-size@6.0.0: - resolution: {integrity: sha512-ax7ZYomf6jqPTQ4+XCpUGyXKHk5WweS+e05MBO4/y3WJ5RkmPXNKvX+bx1behVILVwr6JSQvZAku021CHPXG3Q==} - engines: {node: '>=10'} - - handlebars@4.7.8: - resolution: {integrity: sha512-vafaFqs8MZkRrSX7sFVUdo3ap/eNiLnb4IakshzvP56X5Nr1iGKAIqdX6tMlm6HcNRIkr6AxO5jFEoJzzpT8aQ==} - engines: {node: '>=0.4.7'} - hasBin: true - - has-ansi@2.0.0: - resolution: {integrity: sha512-C8vBJ8DwUCx19vhm7urhTuUsr4/IyP6l4VzNQDv+ryHQObW3TTTp9yB68WpYgRe2bbaGuZ/se74IqFeVnMnLZg==} - engines: {node: '>=0.10.0'} - - has-flag@3.0.0: - resolution: {integrity: sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==} - engines: {node: '>=4'} - - has-flag@4.0.0: - resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} - engines: {node: '>=8'} - - has-property-descriptors@1.0.2: - resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==} - - has-proto@1.0.3: - resolution: {integrity: sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==} - engines: {node: '>= 0.4'} - - has-symbols@1.0.3: - resolution: {integrity: sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==} - engines: {node: '>= 0.4'} - - has-tostringtag@1.0.2: - resolution: {integrity: sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==} - engines: {node: '>= 0.4'} - - has-unicode@2.0.1: - resolution: {integrity: sha512-8Rf9Y83NBReMnx0gFzA8JImQACstCYWUplepDa9xprwwtmgEZUF0h/i5xSA625zB/I37EtrswSST6OXxwaaIJQ==} - - has-value@0.3.1: - resolution: {integrity: sha512-gpG936j8/MzaeID5Yif+577c17TxaDmhuyVgSwtnL/q8UUTySg8Mecb+8Cf1otgLoD7DDH75axp86ER7LFsf3Q==} - engines: {node: '>=0.10.0'} - - has-value@1.0.0: - resolution: {integrity: sha512-IBXk4GTsLYdQ7Rvt+GRBrFSVEkmuOUy4re0Xjd9kJSUQpnTrWR4/y9RpfexN9vkAPMFuQoeWKwqzPozRTlasGw==} - engines: {node: '>=0.10.0'} - - has-values@0.1.4: - resolution: {integrity: sha512-J8S0cEdWuQbqD9//tlZxiMuMNmxB8PlEwvYwuxsTmR1G5RXUePEX/SJn7aD0GMLieuZYSwNH0cQuJGwnYunXRQ==} - engines: {node: '>=0.10.0'} - - has-values@1.0.0: - resolution: {integrity: sha512-ODYZC64uqzmtfGMEAX/FvZiRyWLpAC3vYnNunURUnkGVTS+mI0smVsWaPydRBsE3g+ok7h960jChO8mFcWlHaQ==} - engines: {node: '>=0.10.0'} - - hash.js@1.1.7: - resolution: {integrity: sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==} - - hasown@2.0.2: - resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} - engines: {node: '>= 0.4'} - - he@1.2.0: - resolution: {integrity: sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==} - hasBin: true - - header-case@1.0.1: - resolution: {integrity: sha512-i0q9mkOeSuhXw6bGgiQCCBgY/jlZuV/7dZXyZ9c6LcBrqwvT8eT719E9uxE5LiZftdl+z81Ugbg/VvXV4OJOeQ==} - - helmet@6.2.0: - resolution: {integrity: sha512-DWlwuXLLqbrIOltR6tFQXShj/+7Cyp0gLi6uAb8qMdFh/YBBFbKSgQ6nbXmScYd8emMctuthmgIa7tUfo9Rtyg==} - engines: {node: '>=14.0.0'} - - highlight.js@10.7.3: - resolution: {integrity: sha512-tzcUFauisWKNHaRkN4Wjl/ZA07gENAjFl3J/c480dprkGTg5EQstgaNFqBfUqCq54kZRIEcreTsAgF/m2quD7A==} - - history@4.10.1: - resolution: {integrity: sha512-36nwAD620w12kuzPAsyINPWJqlNbij+hpK1k9XRloDtym8mxzGYl2c17LnV6IAGB2Dmg4tEa7G7DlawS0+qjew==} - - hmac-drbg@1.0.1: - resolution: {integrity: sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==} - - hoist-non-react-statics@3.3.2: - resolution: {integrity: sha512-/gGivxi8JPKWNm/W0jSmzcMPpfpPLc3dY/6GxhX2hQ9iGj3aDfklV4ET7NjKpSinLpJ5vafa9iiGIEZg10SfBw==} - - homedir-polyfill@1.0.3: - resolution: {integrity: sha512-eSmmWE5bZTK2Nou4g0AI3zZ9rswp7GRKoKXS1BLUkvPviOqs4YTN1djQIqrXy9k5gEtdLPy86JjRwsNM9tnDcA==} - engines: {node: '>=0.10.0'} - - hosted-git-info@2.8.9: - resolution: {integrity: sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==} - - html-entities@2.5.2: - resolution: {integrity: sha512-K//PSRMQk4FZ78Kyau+mZurHn3FH0Vwr+H36eE0rPbeYkRRi9YxceYPhuN60UwWorxyKHhqoAJl2OFKa4BVtaA==} - - html-escaper@2.0.2: - resolution: {integrity: sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==} - - html-minifier-terser@6.1.0: - resolution: {integrity: sha512-YXxSlJBZTP7RS3tWnQw74ooKa6L9b9i9QYXY21eUEvhZ3u9XLfv6OnFsQq6RxkhHygsaUMvYsZRV5rU/OVNZxw==} - engines: {node: '>=12'} - hasBin: true - - html-webpack-plugin@5.6.0: - resolution: {integrity: sha512-iwaY4wzbe48AfKLZ/Cc8k0L+FKG6oSNRaZ8x5A/T/IVDGyXcbHncM9TdDa93wn0FsSm82FhTKW7f3vS61thXAw==} - engines: {node: '>=10.13.0'} - peerDependencies: - '@rspack/core': 0.x || 1.x - webpack: ^5.20.0 - peerDependenciesMeta: - '@rspack/core': - optional: true - webpack: - optional: true - - htmlparser2@6.1.0: - resolution: {integrity: sha512-gyyPk6rgonLFEDGoeRgQNaEUvdJ4ktTmmUh/h2t7s+M8oPpIPxgNACWa+6ESR57kXstwqPiCut0V8NRpcwgU7A==} - - htmlparser2@8.0.2: - resolution: {integrity: sha512-GYdjWKDkbRLkZ5geuHs5NY1puJ+PXwP7+fHPRz06Eirsb9ugf6d8kkXav6ADhcODhFFPMIXyxkxSuMf3D6NCFA==} - - http-assert@1.5.0: - resolution: {integrity: sha512-uPpH7OKX4H25hBmU6G1jWNaqJGpTXxey+YOUizJUAgu0AjLUeC8D73hTrhvDS5D+GJN1DN1+hhc/eF/wpxtp0w==} - engines: {node: '>= 0.8'} - - http-cache-semantics@4.1.1: - resolution: {integrity: sha512-er295DKPVsV82j5kw1Gjt+ADA/XYHsajl82cGNQG2eyoPkvgUhX+nDIyelzhIWbbsXP39EHcI6l5tYs2FYqYXQ==} - - http-errors@1.6.3: - resolution: {integrity: sha512-lks+lVC8dgGyh97jxvxeYTWQFvh4uw4yC12gVl63Cg30sjPX4wuGcdkICVXDAESr6OJGjqGA8Iz5mkeN6zlD7A==} - engines: {node: '>= 0.6'} - - http-errors@1.8.1: - resolution: {integrity: sha512-Kpk9Sm7NmI+RHhnj6OIWDI1d6fIoFAtFt9RLaTMRlg/8w49juAStsrBgp0Dp4OdxdVbRIeKhtCUvoi/RuAhO4g==} - engines: {node: '>= 0.6'} - - http-errors@2.0.0: - resolution: {integrity: sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==} - engines: {node: '>= 0.8'} - - http2-wrapper@1.0.3: - resolution: {integrity: sha512-V+23sDMr12Wnz7iTcDeJr3O6AIxlnvT/bmaAAAP/Xda35C90p9599p0F1eHR/N1KILWSoWVAiOMFjBBXaXSMxg==} - engines: {node: '>=10.19.0'} - - https-proxy-agent@5.0.1: - resolution: {integrity: sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==} - engines: {node: '>= 6'} - - human-signals@2.1.0: - resolution: {integrity: sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==} - engines: {node: '>=10.17.0'} - - human-signals@4.3.1: - resolution: {integrity: sha512-nZXjEF2nbo7lIw3mgYjItAfgQXog3OjJogSbKa2CQIIvSGWcKgeJnQlNXip6NglNzYH45nSRiEVimMvYL8DDqQ==} - engines: {node: '>=14.18.0'} - - humanize-ms@1.2.1: - resolution: {integrity: sha512-Fl70vYtsAFb/C06PTS9dZBo7ihau+Tu/DNCk/OyHhea07S+aeMWpFFkUaXRa8fI+ScZbEI8dfSxwY7gxZ9SAVQ==} - - iconv-lite@0.4.13: - resolution: {integrity: sha512-QwVuTNQv7tXC5mMWFX5N5wGjmybjNBBD8P3BReTkPmipoxTUFgWM2gXNvldHQr6T14DH0Dh6qBVg98iJt7u4mQ==} - engines: {node: '>=0.8.0'} - - iconv-lite@0.4.15: - resolution: {integrity: sha512-RGR+c9Lm+tLsvU57FTJJtdbv2hQw42Yl2n26tVIBaYmZzLN+EGfroUugN/z9nJf9kOXd49hBmpoGr4FEm+A4pw==} - engines: {node: '>=0.10.0'} - - iconv-lite@0.4.24: - resolution: {integrity: sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==} - engines: {node: '>=0.10.0'} - - icss-utils@5.1.0: - resolution: {integrity: sha512-soFhflCVWLfRNOPU3iv5Z9VUdT44xFRbzjLsEzSr5AQmgqPMTHdU3PMT1Cf1ssx8fLNJDA1juftYl+PUcv3MqA==} - engines: {node: ^10 || ^12 || >= 14} - peerDependencies: - postcss: ^8.1.0 - - ieee754@1.2.1: - resolution: {integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==} - - ignore-by-default@1.0.1: - resolution: {integrity: sha512-Ius2VYcGNk7T90CppJqcIkS5ooHUZyIQK+ClZfMfMNFEF9VSE73Fq+906u/CWu92x4gzZMWOwfFYckPObzdEbA==} - - ignore-walk@3.0.4: - resolution: {integrity: sha512-PY6Ii8o1jMRA1z4F2hRkH/xN59ox43DavKvD3oDpfurRlOJyAHpifIwpbdv1n4jt4ov0jSpw3kQ4GhJnpBL6WQ==} - - ignore@5.3.2: - resolution: {integrity: sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==} - engines: {node: '>= 4'} - - immer@9.0.19: - resolution: {integrity: sha512-eY+Y0qcsB4TZKwgQzLaE/lqYMlKhv5J9dyd2RhhtGhNo2njPXDqU9XPfcNfa3MIDsdtZt5KlkIsirlo4dHsWdQ==} - - immer@9.0.21: - resolution: {integrity: sha512-bc4NBHqOqSfRW7POMkHd51LvClaeMXpm8dx0e8oE2GORbq5aRK7Bxl4FyzVLdGtLmvLKL7BTDBG5ACQm4HWjTA==} - - import-fresh@3.3.0: - resolution: {integrity: sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==} - engines: {node: '>=6'} - - import-lazy@4.0.0: - resolution: {integrity: sha512-rKtvo6a868b5Hu3heneU+L4yEQ4jYKLtjpnPeUdK7h0yzXGmyBTypknlkCvHFBqfX9YlorEiMM6Dnq/5atfHkw==} - engines: {node: '>=8'} - - imurmurhash@0.1.4: - resolution: {integrity: sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==} - engines: {node: '>=0.8.19'} - - indent-string@4.0.0: - resolution: {integrity: sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==} - engines: {node: '>=8'} - - inflation@2.1.0: - resolution: {integrity: sha512-t54PPJHG1Pp7VQvxyVCJ9mBbjG3Hqryges9bXoOO6GExCPa+//i/d5GSuFtpx3ALLd7lgIAur6zrIlBQyJuMlQ==} - engines: {node: '>= 0.8.0'} - - inflight@1.0.6: - resolution: {integrity: sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==} - deprecated: This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful. - - inherits@2.0.3: - resolution: {integrity: sha512-x00IRNXNy63jwGkJmzPigoySHbaqpNuzKbBOmzK+g2OdZpQ9w+sxCN+VSB3ja7IAge2OP2qpfxTjeNcyjmW1uw==} - - inherits@2.0.4: - resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} - - ini@1.3.8: - resolution: {integrity: sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==} - - ini@2.0.0: - resolution: {integrity: sha512-7PnF4oN3CvZF23ADhA5wRaYEQpJ8qygSkbtTXWBeXWXmEVRXK+1ITciHWwHhsjv1TmW0MgacIv6hEi5pX5NQdA==} - engines: {node: '>=10'} - - ini@4.1.1: - resolution: {integrity: sha512-QQnnxNyfvmHFIsj7gkPcYymR8Jdw/o7mp5ZFihxn6h8Ci6fh3Dx4E1gPjpQEpIuPo9XVNY/ZUwh4BPMjGyL01g==} - engines: {node: ^14.17.0 || ^16.13.0 || >=18.0.0} - - inquirer@7.3.3: - resolution: {integrity: sha512-JG3eIAj5V9CwcGvuOmoo6LB9kbAYT8HXffUl6memuszlwDC/qvFAJw49XJ5NROSFNPxp3iQg1GqkFhaY/CR0IA==} - engines: {node: '>=8.0.0'} - - inquirer@8.2.5: - resolution: {integrity: sha512-QAgPDQMEgrDssk1XiwwHoOGYF9BAbUcc1+j+FhEvaOt8/cKRqyLn0U5qA6F74fGhTMGxf92pOvPBeh29jQJDTQ==} - engines: {node: '>=12.0.0'} - - interpret@1.4.0: - resolution: {integrity: sha512-agE4QfB2Lkp9uICn7BAqoscw4SZP9kTE2hxiFI3jBPmXJfdqiahTbUuKGsMoN2GtqL9AxhYioAcVvgsb1HvRbA==} - engines: {node: '>= 0.10'} - - interpret@2.2.0: - resolution: {integrity: sha512-Ju0Bz/cEia55xDwUWEa8+olFpCiQoypjnQySseKtmjNrnps3P+xfpUmGr90T7yjlVJmOtybRvPXhKMbHr+fWnw==} - engines: {node: '>= 0.10'} - - intl-messageformat@10.3.4: - resolution: {integrity: sha512-/FxUIrlbPtuykSNX85CB5sp2FjLVeTmdD7TfRkVFPft2n4FgcSlAcilFytYiFAEmPHc+0PvpLCIPXeaGFzIvOg==} - - invariant@2.2.4: - resolution: {integrity: sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==} - - is-absolute@1.0.0: - resolution: {integrity: sha512-dOWoqflvcydARa360Gvv18DZ/gRuHKi2NU/wU5X1ZFzdYfH29nkiNZsF3mp4OJ3H4yo9Mx8A/uAGNzpzPN3yBA==} - engines: {node: '>=0.10.0'} - - is-accessor-descriptor@1.0.1: - resolution: {integrity: sha512-YBUanLI8Yoihw923YeFUS5fs0fF2f5TSFTNiYAAzhhDscDa3lEqYuz1pDOEP5KvX94I9ey3vsqjJcLVFVU+3QA==} - engines: {node: '>= 0.10'} - - is-arrayish@0.2.1: - resolution: {integrity: sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==} - - is-arrayish@0.3.2: - resolution: {integrity: sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==} - - is-binary-path@2.1.0: - resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==} - engines: {node: '>=8'} - - is-buffer@1.1.6: - resolution: {integrity: sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==} - - is-class-hotfix@0.0.6: - resolution: {integrity: sha512-0n+pzCC6ICtVr/WXnN2f03TK/3BfXY7me4cjCAqT8TYXEl0+JBRoqBo94JJHXcyDSLUeWbNX8Fvy5g5RJdAstQ==} - - is-core-module@2.15.1: - resolution: {integrity: sha512-z0vtXSwucUJtANQWldhbtbt7BnL0vxiFjIdDLAatwhDYty2bad6s+rijD6Ri4YuYJubLzIJLUidCh09e1djEVQ==} - engines: {node: '>= 0.4'} - - is-data-descriptor@1.0.1: - resolution: {integrity: sha512-bc4NlCDiCr28U4aEsQ3Qs2491gVq4V8G7MQyws968ImqjKuYtTJXrl7Vq7jsN7Ly/C3xj5KWFrY7sHNeDkAzXw==} - engines: {node: '>= 0.4'} - - is-descriptor@0.1.7: - resolution: {integrity: sha512-C3grZTvObeN1xud4cRWl366OMXZTj0+HGyk4hvfpx4ZHt1Pb60ANSXqCK7pdOTeUQpRzECBSTphqvD7U+l22Eg==} - engines: {node: '>= 0.4'} - - is-descriptor@1.0.3: - resolution: {integrity: sha512-JCNNGbwWZEVaSPtS45mdtrneRWJFp07LLmykxeFV5F6oBvNF8vHSfJuJgoT472pSfk+Mf8VnlrspaFBHWM8JAw==} - engines: {node: '>= 0.4'} - - is-docker@2.2.1: - resolution: {integrity: sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==} - engines: {node: '>=8'} - hasBin: true - - is-docker@3.0.0: - resolution: {integrity: sha512-eljcgEDlEns/7AXFosB5K/2nCM4P7FQPkGc/DWLy5rmFEWvZayGrik1d9/QIY5nJ4f9YsVvBkA6kJpHn9rISdQ==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - hasBin: true - - is-extendable@0.1.1: - resolution: {integrity: sha512-5BMULNob1vgFX6EjQw5izWDxrecWK9AM72rugNr0TFldMOi0fj6Jk+zeKIt0xGj4cEfQIJth4w3OKWOJ4f+AFw==} - engines: {node: '>=0.10.0'} - - is-extendable@1.0.1: - resolution: {integrity: sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==} - engines: {node: '>=0.10.0'} - - is-extglob@2.1.1: - resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} - engines: {node: '>=0.10.0'} - - is-fullwidth-code-point@3.0.0: - resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==} - engines: {node: '>=8'} - - is-generator-function@1.0.10: - resolution: {integrity: sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==} - engines: {node: '>= 0.4'} - - is-glob@3.1.0: - resolution: {integrity: sha512-UFpDDrPgM6qpnFNI+rh/p3bUaq9hKLZN8bMUWzxmcnZVS3omf4IPK+BrewlnWjO1WmUsMYuSjKh4UJuV4+Lqmw==} - engines: {node: '>=0.10.0'} - - is-glob@4.0.3: - resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==} - engines: {node: '>=0.10.0'} - - is-hotkey@0.1.8: - resolution: {integrity: sha512-qs3NZ1INIS+H+yeo7cD9pDfwYV/jqRh1JG9S9zYrNudkoUQg7OL7ziXqRKu+InFjUIDoP2o6HIkLYMh1pcWgyQ==} - - is-inside-container@1.0.0: - resolution: {integrity: sha512-KIYLCCJghfHZxqjYBE7rEy0OBuTd5xCHS7tHVgvCLkx7StIoaxwNW3hCALgEUjFfeRk+MG/Qxmp/vtETEF3tRA==} - engines: {node: '>=14.16'} - hasBin: true - - is-interactive@1.0.0: - resolution: {integrity: sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==} - engines: {node: '>=8'} - - is-localhost-ip@2.0.0: - resolution: {integrity: sha512-vlgs2cSgMOfnKU8c1ewgKPyum9rVrjjLLW2HBdL5i0iAJjOs8NY55ZBd/hqUTaYR0EO9CKZd3hVSC2HlIbygTQ==} - engines: {node: '>=12'} - - is-lower-case@1.1.3: - resolution: {integrity: sha512-+5A1e/WJpLLXZEDlgz4G//WYSHyQBD32qa4Jd3Lw06qQlv3fJHnp3YIHjTQSGzHMgzmVKz2ZP3rBxTHkPw/lxA==} - - is-number@3.0.0: - resolution: {integrity: sha512-4cboCqIpliH+mAvFNegjZQ4kgKc3ZUhQVr3HvWbSh5q3WH2v82ct+T2Y1hdU5Gdtorx/cLifQjqCbL7bpznLTg==} - engines: {node: '>=0.10.0'} - - is-number@7.0.0: - resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==} - engines: {node: '>=0.12.0'} - - is-obj@2.0.0: - resolution: {integrity: sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==} - engines: {node: '>=8'} - - is-path-cwd@2.2.0: - resolution: {integrity: sha512-w942bTcih8fdJPJmQHFzkS76NEP8Kzzvmw92cXsazb8intwLqPibPPdXf4ANdKV3rYMuuQYGIWtvz9JilB3NFQ==} - engines: {node: '>=6'} - - is-path-inside@3.0.3: - resolution: {integrity: sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==} - engines: {node: '>=8'} - - is-plain-obj@4.1.0: - resolution: {integrity: sha512-+Pgi+vMuUNkJyExiMBt5IlFoMyKnr5zhJ4Uspz58WOhBF5QoIZkFyNHIbBAtHwzVAgk5RtndVNsDRN61/mmDqg==} - engines: {node: '>=12'} - - is-plain-object@2.0.4: - resolution: {integrity: sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==} - engines: {node: '>=0.10.0'} - - is-plain-object@5.0.0: - resolution: {integrity: sha512-VRSzKkbMm5jMDoKLbltAkFQ5Qr7VDiTFGXxYFXXowVj387GeGNOCsOH6Msy00SGZ3Fp84b1Naa1psqgcCIEP5Q==} - engines: {node: '>=0.10.0'} - - is-relative@1.0.0: - resolution: {integrity: sha512-Kw/ReK0iqwKeu0MITLFuj0jbPAmEiOsIwyIXvvbfa6QfmN9pkD1M+8pdk7Rl/dTKbH34/XBFMbgD4iMJhLQbGA==} - engines: {node: '>=0.10.0'} - - is-retry-allowed@2.2.0: - resolution: {integrity: sha512-XVm7LOeLpTW4jV19QSH38vkswxoLud8sQ57YwJVTPWdiaI9I8keEhGFpBlslyVsgdQy4Opg8QOLb8YRgsyZiQg==} - engines: {node: '>=10'} - - is-ssh@1.4.0: - resolution: {integrity: sha512-x7+VxdxOdlV3CYpjvRLBv5Lo9OJerlYanjwFrPR9fuGPjCiNiCzFgAWpiLAohSbsnH4ZAys3SBh+hq5rJosxUQ==} - - is-stream@2.0.1: - resolution: {integrity: sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==} - engines: {node: '>=8'} - - is-stream@3.0.0: - resolution: {integrity: sha512-LnQR4bZ9IADDRSkvpqMGvt/tEJWclzklNgSw48V5EAaAeDd6qGvN8ei6k5p0tvxSR171VmGyHuTiAOfxAbr8kA==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - is-type-of@1.4.0: - resolution: {integrity: sha512-EddYllaovi5ysMLMEN7yzHEKh8A850cZ7pykrY1aNRQGn/CDjRDE9qEWbIdt7xGEVJmjBXzU/fNnC4ABTm8tEQ==} - - is-typedarray@1.0.0: - resolution: {integrity: sha512-cyA56iCMHAh5CdzjJIa4aohJyeO1YbwLi3Jc35MmRU6poroFjIGZzUzupGiRPOjgHg9TLu43xbpwXk523fMxKA==} - - is-unc-path@1.0.0: - resolution: {integrity: sha512-mrGpVd0fs7WWLfVsStvgF6iEJnbjDFZh9/emhRDcGWTduTfNHd9CHeUwH3gYIjdbwo4On6hunkztwOaAw0yllQ==} - engines: {node: '>=0.10.0'} - - is-unicode-supported@0.1.0: - resolution: {integrity: sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==} - engines: {node: '>=10'} - - is-upper-case@1.1.2: - resolution: {integrity: sha512-GQYSJMgfeAmVwh9ixyk888l7OIhNAGKtY6QA+IrWlu9MDTCaXmeozOZ2S9Knj7bQwBO/H6J2kb+pbyTUiMNbsw==} - - is-windows@1.0.2: - resolution: {integrity: sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==} - engines: {node: '>=0.10.0'} - - is-wsl@2.2.0: - resolution: {integrity: sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==} - engines: {node: '>=8'} - - isarray@0.0.1: - resolution: {integrity: sha512-D2S+3GLxWH+uhrNEcoh/fnmYeP8E8/zHl644d/jdA0g2uyXvy3sb0qxotE+ne0LtccHknQzWwZEzhak7oJ0COQ==} - - isarray@1.0.0: - resolution: {integrity: sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==} - - isbinaryfile@4.0.10: - resolution: {integrity: sha512-iHrqe5shvBUcFbmZq9zOQHBoeOhZJu6RQGrDpBgenUm/Am+F3JM2MgQj+rK3Z601fzrL5gLZWtAPH2OBaSVcyw==} - engines: {node: '>= 8.0.0'} - - isexe@2.0.0: - resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} - - isobject@2.1.0: - resolution: {integrity: sha512-+OUdGJlgjOBZDfxnDjYYG6zp487z0JGNQq3cYQYg5f5hKR+syHMsaztzGeml/4kGG55CSpKSpWTY+jYGgsHLgA==} - engines: {node: '>=0.10.0'} - - isobject@3.0.1: - resolution: {integrity: sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg==} - engines: {node: '>=0.10.0'} - - isstream@0.1.2: - resolution: {integrity: sha512-Yljz7ffyPbrLpLngrMtZ7NduUgVvi6wG9RJ9IUcyCd59YQ911PBJphODUcbOVbqYfxe1wuYf/LJ8PauMRwsM/g==} - - jackspeak@3.4.3: - resolution: {integrity: sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==} - - jest-worker@27.5.1: - resolution: {integrity: sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg==} - engines: {node: '>= 10.13.0'} - - jju@1.4.0: - resolution: {integrity: sha512-8wb9Yw966OSxApiCt0K3yNJL8pnNeIv+OEq2YMidz4FKP6nonSRoOXc80iXY4JaN2FC11B9qsNmDsm+ZOfMROA==} - - jose@4.15.9: - resolution: {integrity: sha512-1vUQX+IdDMVPj4k8kOxgUqlcK518yluMuGZwqlr44FS1ppZB/5GWh4rZG89erpOBOJjU/OBsnCVFfapsRz6nEA==} - - joycon@3.1.1: - resolution: {integrity: sha512-34wB/Y7MW7bzjKRjUKTa46I2Z7eV62Rkhva+KkopW7Qvv/OSWBqvkSY7vusOPrNuZcUG3tApvdVgNB8POj3SPw==} - engines: {node: '>=10'} - - js-cookie@2.2.1: - resolution: {integrity: sha512-HvdH2LzI/EAZcUwA8+0nKNtWHqS+ZmijLA30RwZA0bo7ToCckjK5MkGhjED9KoRcXO6BaGI3I9UIzSA1FKFPOQ==} - - js-sha3@0.8.0: - resolution: {integrity: sha512-gF1cRrHhIzNfToc802P800N8PpXS+evLLXfsVpowqmAFR9uwbi89WvXg2QspOmXL8QL86J4T1EpFu+yUkwJY3Q==} - - js-tokens@4.0.0: - resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} - - js-yaml@4.1.0: - resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==} - hasBin: true - - jsesc@2.5.2: - resolution: {integrity: sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==} - engines: {node: '>=4'} - hasBin: true - - json-buffer@3.0.1: - resolution: {integrity: sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==} - - json-parse-even-better-errors@2.3.1: - resolution: {integrity: sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==} - - json-schema-traverse@0.4.1: - resolution: {integrity: sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==} - - json-schema-traverse@1.0.0: - resolution: {integrity: sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==} - - json5@2.2.3: - resolution: {integrity: sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==} - engines: {node: '>=6'} - hasBin: true - - jsonfile@4.0.0: - resolution: {integrity: sha512-m6F1R3z8jjlf2imQHS2Qez5sjKWQzbuuhuJ/FKYFRZvPE3PuHcSMVZzfsLhGVOkfd20obL5SWEBew5ShlquNxg==} - - jsonfile@6.1.0: - resolution: {integrity: sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==} - - jsonwebtoken@9.0.0: - resolution: {integrity: sha512-tuGfYXxkQGDPnLJ7SibiQgVgeDgfbPq2k2ICcbgqW8WxWLBAxKQM/ZCu/IT8SOSwmaYl4dpTFCW5xZv7YbbWUw==} - engines: {node: '>=12', npm: '>=6'} - - jwa@1.4.1: - resolution: {integrity: sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==} - - jwa@2.0.0: - resolution: {integrity: sha512-jrZ2Qx916EA+fq9cEAeCROWPTfCwi1IVHqT2tapuqLEVVDKFDENFw1oL+MwrTvH6msKxsd1YTDVw6uKEcsrLEA==} - - jwk-to-pem@2.0.5: - resolution: {integrity: sha512-L90jwellhO8jRKYwbssU9ifaMVqajzj3fpRjDKcsDzrslU9syRbFqfkXtT4B89HYAap+xsxNcxgBSB09ig+a7A==} - - jwks-rsa@3.1.0: - resolution: {integrity: sha512-v7nqlfezb9YfHHzYII3ef2a2j1XnGeSE/bK3WfumaYCqONAIstJbrEGapz4kadScZzEt7zYCN7bucj8C0Mv/Rg==} - engines: {node: '>=14'} - - jws@3.2.2: - resolution: {integrity: sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==} - - jws@4.0.0: - resolution: {integrity: sha512-KDncfTmOZoOMTFG4mBlG0qUIOlc03fmzH+ru6RgYVZhPkyiy/92Owlt/8UEN+a4TXR1FQetfIpJE8ApdvdVxTg==} - - keygrip@1.1.0: - resolution: {integrity: sha512-iYSchDJ+liQ8iwbSI2QqsQOvqv58eJCEanyJPJi+Khyu8smkcKSFUCbPwzFcL7YVtZ6eONjqRX/38caJ7QjRAQ==} - engines: {node: '>= 0.6'} - - keyv@4.5.4: - resolution: {integrity: sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==} - - kind-of@3.2.2: - resolution: {integrity: sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==} - engines: {node: '>=0.10.0'} - - kind-of@4.0.0: - resolution: {integrity: sha512-24XsCxmEbRwEDbz/qz3stgin8TTzZ1ESR56OMCN0ujYg+vRutNSiOj9bHH9u85DKgXguraugV5sFuvbD4FW/hw==} - engines: {node: '>=0.10.0'} - - kind-of@6.0.3: - resolution: {integrity: sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==} - engines: {node: '>=0.10.0'} - - kleur@3.0.3: - resolution: {integrity: sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==} - engines: {node: '>=6'} - - knex@2.5.0: - resolution: {integrity: sha512-h6Ru3PJmZjCDUEqLgwQ/RJUu06Bz7MTzY6sD90udLIa9qwtC7Rnicr7TBiWSaswZmDqk4EZ8xysdg1fkvhYM6w==} - engines: {node: '>=12'} - hasBin: true - peerDependencies: - better-sqlite3: '*' - mysql: '*' - mysql2: '*' - pg: '*' - pg-native: '*' - sqlite3: '*' - tedious: '*' - peerDependenciesMeta: - better-sqlite3: - optional: true - mysql: - optional: true - mysql2: - optional: true - pg: - optional: true - pg-native: - optional: true - sqlite3: - optional: true - tedious: - optional: true - - koa-body@4.2.0: - resolution: {integrity: sha512-wdGu7b9amk4Fnk/ytH8GuWwfs4fsB5iNkY8kZPpgQVb04QZSv85T0M8reb+cJmvLE8cjPYvBzRikD3s6qz8OoA==} - - koa-bodyparser@4.4.1: - resolution: {integrity: sha512-kBH3IYPMb+iAXnrxIhXnW+gXV8OTzCu8VPDqvcDHW9SQrbkHmqPQtiZwrltNmSq6/lpipHnT7k7PsjlVD7kK0w==} - engines: {node: '>=8.0.0'} - - koa-compose@4.1.0: - resolution: {integrity: sha512-8ODW8TrDuMYvXRwra/Kh7/rJo9BtOfPc6qO8eAfC80CnCvSjSl0bkRM24X6/XBBEyj0v1nRUQ1LyOy3dbqOWXw==} - - koa-compress@5.1.0: - resolution: {integrity: sha512-G3Ppo9jrUwlchp6qdoRgQNMiGZtM0TAHkxRZQ7EoVvIG8E47J4nAsMJxXHAUQ+0oc7t0MDxSdONWTFcbzX7/Bg==} - engines: {node: '>= 8.0.0'} - - koa-convert@2.0.0: - resolution: {integrity: sha512-asOvN6bFlSnxewce2e/DK3p4tltyfC4VM7ZwuTuepI7dEQVcvpyFuBcEARu1+Hxg8DIwytce2n7jrZtRlPrARA==} - engines: {node: '>= 10'} - - koa-favicon@2.1.0: - resolution: {integrity: sha512-LvukcooYjxKtnZq0RXdBup+JDhaHwLgnLlDHB/xvjwQEjbc4rbp/0WkmOzpOvaHujc+fIwPear0dpKX1V+dHVg==} - - koa-helmet@7.0.2: - resolution: {integrity: sha512-AvzS6VuEfFgbAm0mTUnkk/BpMarMcs5A56g+f0sfrJ6m63wII48d2GDrnUQGp0Nj+RR950vNtgqXm9UJSe7GOg==} - engines: {node: '>= 14.0.0'} - - koa-ip@2.1.3: - resolution: {integrity: sha512-QLVBByImwDq9enZXVOD3Astk876B7N0IYta7Kik4iyNB462rVzBB1/LD0Ek1F+v9nGUTHBFyhh8043EIlskK9Q==} - - koa-is-json@1.0.0: - resolution: {integrity: sha512-+97CtHAlWDx0ndt0J8y3P12EWLwTLMXIfMnYDev3wOTwH/RpBGMlfn4bDXlMEg1u73K6XRE9BbUp+5ZAYoRYWw==} - - koa-passport@5.0.0: - resolution: {integrity: sha512-eNGg3TGgZ4ydm9DYCOqaa0ySSA/44BS6X+v4CKjP/nHOoXlADRonHsZvS3QWok6EV0ZL0V7FhfWxRYfD2B5kTQ==} - engines: {node: '>= 4'} - - koa-range@0.3.0: - resolution: {integrity: sha512-Ich3pCz6RhtbajYXRWjIl6O5wtrLs6kE3nkXc9XmaWe+MysJyZO7K4L3oce1Jpg/iMgCbj+5UCiMm/rqVtcDIg==} - engines: {node: '>=7'} - - koa-send@5.0.1: - resolution: {integrity: sha512-tmcyQ/wXXuxpDxyNXv5yNNkdAMdFRqwtegBXUaowiQzUKqJehttS0x2j0eOZDQAyloAth5w6wwBImnFzkUz3pQ==} - engines: {node: '>= 8'} - - koa-session@6.4.0: - resolution: {integrity: sha512-h/dxmSOvNEXpHQPRs4TV03TZVFyZIjmYQiTAW5JBFTYBOZ0VdpZ8QEE6Dud75g8z9JNGXi3m++VqRmqToB+c2A==} - engines: {node: '>=8.0.0'} - - koa-static@5.0.0: - resolution: {integrity: sha512-UqyYyH5YEXaJrf9S8E23GoJFQZXkBVJ9zYYMPGz919MSX1KuvAcycIuS0ci150HCoPf4XQVhQ84Qf8xRPWxFaQ==} - engines: {node: '>= 7.6.0'} - - koa2-ratelimit@1.1.3: - resolution: {integrity: sha512-gdrIw6m/D7pmScScL4dz50qLbRR3UGqvO1Vuy2dc7hVIuFAl1OVTnu6WFyEJ5GbfyLZFaCMWzRw6t4krvzvUTg==} - engines: {node: '>=7.10.1'} - peerDependencies: - mongoose: '>= 5' - redis: '>= 4.0.0' - sequelize: '>=5.8.7' - peerDependenciesMeta: - mongoose: - optional: true - redis: - optional: true - sequelize: - optional: true - - koa@2.13.4: - resolution: {integrity: sha512-43zkIKubNbnrULWlHdN5h1g3SEKXOEzoAlRsHOTFpnlDu8JlAOZSMJBLULusuXRequboiwJcj5vtYXKB3k7+2g==} - engines: {node: ^4.8.4 || ^6.10.1 || ^7.10.1 || >= 8.1.4} - - kuler@2.0.0: - resolution: {integrity: sha512-Xq9nH7KlWZmXAtodXDDRE7vs6DU1gTU8zYDHDiWLSip45Egwq3plLHzPn27NgvzL2r1LMPC1vdqh98sQxtqj4A==} - - libbase64@0.1.0: - resolution: {integrity: sha512-B91jifmFw1DKEqEWstSpg1PbtUbBzR4yQAPT86kCQXBtud1AJVA+Z6RSklSrqmKe4q2eiEufgnhqJKPgozzfIQ==} - - libmime@2.1.0: - resolution: {integrity: sha512-4be2R6/jOasyPTw0BkpIZBVk2cElqjdIdS0PRPhbOCV4wWuL/ZcYYpN1BCTVB+6eIQ0uuAwp5hQTHFrM5Joa8w==} - - libmime@2.1.3: - resolution: {integrity: sha512-ABr2f4O+K99sypmkF/yPz2aXxUFHEZzv+iUkxItCeKZWHHXdQPpDXd6rV1kBBwL4PserzLU09EIzJ2lxC9hPfQ==} - - libqp@1.1.0: - resolution: {integrity: sha512-4Rgfa0hZpG++t1Vi2IiqXG9Ad1ig4QTmtuZF946QJP4bPqOYC78ixUXgz5TW/wE7lNaNKlplSYTxQ+fR2KZ0EA==} - - liftoff@2.5.0: - resolution: {integrity: sha512-01zfGFqfORP1CGmZZP2Zn51zsqz4RltDi0RDOhbGoLYdUT5Lw+I2gX6QdwXhPITF6hPOHEOp+At6/L24hIg9WQ==} - engines: {node: '>= 0.8'} - - limiter@1.1.5: - resolution: {integrity: sha512-FWWMIEOxz3GwUI4Ts/IvgVy6LPvoMPgjMdQ185nN6psJyBJ4yOpzqm695/h5umdLJg2vW3GR5iG11MAkR2AzJA==} - - lines-and-columns@1.2.4: - resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} - - linkify-it@3.0.3: - resolution: {integrity: sha512-ynTsyrFSdE5oZ/O9GEf00kPngmOfVwazR5GKDq6EYfhlpFug3J2zybX56a2PRRpc9P+FuSoGNAwjlbDs9jJBPQ==} - - loader-runner@4.3.0: - resolution: {integrity: sha512-3R/1M+yS3j5ou80Me59j7F9IMs4PXs3VqRrm0TU3AbKPxlmpoY1TNscJV/oGJXo8qCatFGTfDbY6W6ipGOYXfg==} - engines: {node: '>=6.11.5'} - - loader-utils@2.0.4: - resolution: {integrity: sha512-xXqpXoINfFhgua9xiqD8fPFHgkoq1mmmpE92WlDbm9rNRd/EbRb+Gqf908T2DMfuHjjJlksiK2RbHVOdD/MqSw==} - engines: {node: '>=8.9.0'} - - locate-path@3.0.0: - resolution: {integrity: sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==} - engines: {node: '>=6'} - - locate-path@5.0.0: - resolution: {integrity: sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==} - engines: {node: '>=8'} - - locate-path@6.0.0: - resolution: {integrity: sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==} - engines: {node: '>=10'} - - lodash-es@4.17.21: - resolution: {integrity: sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw==} - - lodash.clonedeep@4.5.0: - resolution: {integrity: sha512-H5ZhCF25riFd9uB5UCkVKo61m3S/xZk1x4wA6yp/L3RFP6Z/eHH1ymQcGLo7J3GMPfm0V/7m1tryHuGVxpqEBQ==} - - lodash.deburr@4.1.0: - resolution: {integrity: sha512-m/M1U1f3ddMCs6Hq2tAsYThTBDaAKFDX3dwDo97GEYzamXi9SqUpjWi/Rrj/gf3X2n8ktwgZrlP1z6E3v/IExQ==} - - lodash.get@4.4.2: - resolution: {integrity: sha512-z+Uw/vLuy6gQe8cfaFWD7p0wVv8fJl3mbzXh33RS+0oW2wvUqiRXiQ69gLWSLpgB5/6sU+r6BlQR0MBILadqTQ==} - - lodash.isplainobject@4.0.6: - resolution: {integrity: sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==} - - lodash@4.17.21: - resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==} - - log-symbols@2.2.0: - resolution: {integrity: sha512-VeIAFslyIerEJLXHziedo2basKbMKtTw3vfn5IzG0XTjhAVEJyNHnL2p7vc+wBDSdQuUpNw3M2u6xb9QsAY5Eg==} - engines: {node: '>=4'} - - log-symbols@4.1.0: - resolution: {integrity: sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==} - engines: {node: '>=10'} - - logform@2.6.1: - resolution: {integrity: sha512-CdaO738xRapbKIMVn2m4F6KTj4j7ooJ8POVnebSgKo3KBz5axNXRAL7ZdRjIV6NOr2Uf4vjtRkxrFETOioCqSA==} - engines: {node: '>= 12.0.0'} - - long-timeout@0.1.1: - resolution: {integrity: sha512-BFRuQUqc7x2NWxfJBCyUrN8iYUYznzL9JROmRz1gZ6KlOIgmoD+njPVbb+VNn2nGMKggMsK79iUNErillsrx7w==} - - loose-envify@1.4.0: - resolution: {integrity: sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==} - hasBin: true - - lower-case-first@1.0.2: - resolution: {integrity: sha512-UuxaYakO7XeONbKrZf5FEgkantPf5DUqDayzP5VXZrtRPdH86s4kN47I8B3TW10S4QKiE3ziHNf3kRN//okHjA==} - - lower-case@1.1.4: - resolution: {integrity: sha512-2Fgx1Ycm599x+WGpIYwJOvsjmXFzTSc34IwDWALRA/8AopUKAVPwfJ+h5+f85BCp0PWmmJcWzEpxOpoXycMpdA==} - - lower-case@2.0.2: - resolution: {integrity: sha512-7fm3l3NAF9WfN6W3JOmf5drwpVqX78JtoGJ3A6W0a6ZnldM41w2fV5D490psKFTpMds8TJse/eHLFFsNHHjHgg==} - - lowercase-keys@2.0.0: - resolution: {integrity: sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==} - engines: {node: '>=8'} - - lru-cache@10.4.3: - resolution: {integrity: sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==} - - lru-cache@6.0.0: - resolution: {integrity: sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==} - engines: {node: '>=10'} - - lru-memoizer@2.3.0: - resolution: {integrity: sha512-GXn7gyHAMhO13WSKrIiNfztwxodVsP8IoZ3XfrJV4yH2x0/OeTO/FIaAHTY5YekdGgW94njfuKmyyt1E0mR6Ug==} - - lru_map@0.3.3: - resolution: {integrity: sha512-Pn9cox5CsMYngeDbmChANltQl+5pi6XmTrraMSzhPmMBbmgcxmqWry0U3PGapCU1yB4/LqCcom7qhHZiF/jGfQ==} - - luxon@3.5.0: - resolution: {integrity: sha512-rh+Zjr6DNfUYR3bPwJEnuwDdqMbxZW7LOQfUN4B54+Cl+0o5zaU9RJ6bcidfDtC1cWCZXQ+nvX8bf6bAji37QQ==} - engines: {node: '>=12'} - - mailcomposer@3.12.0: - resolution: {integrity: sha512-zBeDoKUTNI8IAsazoMQFt3eVSVRtDtgrvBjBVdBjxDEX+5KLlKtEFCrBXnxPhs8aTYufUS1SmbFnGpjHS53deg==} - deprecated: This project is unmaintained - - make-dir@3.1.0: - resolution: {integrity: sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==} - engines: {node: '>=8'} - - make-iterator@1.0.1: - resolution: {integrity: sha512-pxiuXh0iVEq7VM7KMIhs5gxsfxCux2URptUQaXo4iZZJxBAzTPOLE2BumO5dbfVYq/hBJFBR/a1mFDmOx5AGmw==} - engines: {node: '>=0.10.0'} - - map-cache@0.2.2: - resolution: {integrity: sha512-8y/eV9QQZCiyn1SprXSrCmqJN0yNRATe+PO8ztwqrvrbdRLA3eYJF0yaR0YayLWkMbsQSKWS9N2gPcGEc4UsZg==} - engines: {node: '>=0.10.0'} - - map-visit@1.0.0: - resolution: {integrity: sha512-4y7uGv8bd2WdM9vpQsiQNo41Ln1NvhvDRuVt0k2JZQ+ezN2uaQes7lZeZ+QQUHOLQAtDaBJ+7wCbi+ab/KFs+w==} - engines: {node: '>=0.10.0'} - - markdown-it-abbr@1.0.4: - resolution: {integrity: sha512-ZeA4Z4SaBbYysZap5iZcxKmlPL6bYA8grqhzJIHB1ikn7njnzaP8uwbtuXc4YXD5LicI4/2Xmc0VwmSiFV04gg==} - - markdown-it-container@3.0.0: - resolution: {integrity: sha512-y6oKTq4BB9OQuY/KLfk/O3ysFhB3IMYoIWhGJEidXt1NQFocFK2sA2t0NYZAMyMShAGL6x5OPIbrmXPIqaN9rw==} - - markdown-it-deflist@2.1.0: - resolution: {integrity: sha512-3OuqoRUlSxJiuQYu0cWTLHNhhq2xtoSFqsZK8plANg91+RJQU1ziQ6lA2LzmFAEes18uPBsHZpcX6We5l76Nzg==} - - markdown-it-emoji@2.0.2: - resolution: {integrity: sha512-zLftSaNrKuYl0kR5zm4gxXjHaOI3FAOEaloKmRA5hijmJZvSjmxcokOLlzycb/HXlUFWzXqpIEoyEMCE4i9MvQ==} - - markdown-it-footnote@3.0.3: - resolution: {integrity: sha512-YZMSuCGVZAjzKMn+xqIco9d1cLGxbELHZ9do/TSYVzraooV8ypsppKNmUJ0fVH5ljkCInQAtFpm8Rb3eXSrt5w==} - - markdown-it-ins@3.0.1: - resolution: {integrity: sha512-32SSfZqSzqyAmmQ4SHvhxbFqSzPDqsZgMHDwxqPzp+v+t8RsmqsBZRG+RfRQskJko9PfKC2/oxyOs4Yg/CfiRw==} - - markdown-it-mark@3.0.1: - resolution: {integrity: sha512-HyxjAu6BRsdt6Xcv6TKVQnkz/E70TdGXEFHRYBGLncRE9lBFwDNLVtFojKxjJWgJ+5XxUwLaHXy+2sGBbDn+4A==} - - markdown-it-sub@1.0.0: - resolution: {integrity: sha512-z2Rm/LzEE1wzwTSDrI+FlPEveAAbgdAdPhdWarq/ZGJrGW/uCQbKAnhoCsE4hAbc3SEym26+W2z/VQB0cQiA9Q==} - - markdown-it-sup@1.0.0: - resolution: {integrity: sha512-E32m0nV9iyhRR7CrhnzL5msqic7rL1juWre6TQNxsnApg7Uf+F97JOKxUijg5YwXz86lZ0mqfOnutoryyNdntQ==} - - markdown-it@12.3.2: - resolution: {integrity: sha512-TchMembfxfNVpHkbtriWltGWc+m3xszaRD0CZup7GFFhzIgQqxIfn3eGj1yZpfuflzPvfkt611B2Q/Bsk1YnGg==} - hasBin: true - - match-sorter@6.3.4: - resolution: {integrity: sha512-jfZW7cWS5y/1xswZo8VBOdudUiSd9nifYRWphc9M5D/ee4w4AoXLgBEdRbgVaxbMuagBPeUC5y2Hi8DO6o9aDg==} - - mdurl@1.0.1: - resolution: {integrity: sha512-/sKlQJCBYVY9Ers9hqzKou4H6V5UWc/M59TH2dvkt+84itfnq7uFOMLpOiOS4ujvHP4etln18fmIxA5R5fll0g==} - - media-typer@0.3.0: - resolution: {integrity: sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==} - engines: {node: '>= 0.6'} - - memfs@3.5.3: - resolution: {integrity: sha512-UERzLsxzllchadvbPs5aolHh65ISpKpM+ccLbOJ8/vvpBKmAWf+la7dXFy7Mr0ySHbdHrFv5kGFCUHHe6GFEmw==} - engines: {node: '>= 4.0.0'} - - memoize-one@5.2.1: - resolution: {integrity: sha512-zYiwtZUcYyXKo/np96AGZAckk+FWWsUdJ3cHGGmld7+AhvcWmQyGCYUh1hc4Q/pkOhb65dQR/pqCyK0cOaHz4Q==} - - memoize-one@6.0.0: - resolution: {integrity: sha512-rkpe71W0N0c0Xz6QD0eJETuWAJGnJ9afsl1srmwPrI+yBCkge5EycXXbYRyvL29zZVUWQCY7InPRCv3GDXuZNw==} - - merge-stream@2.0.0: - resolution: {integrity: sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==} - - merge2@1.4.1: - resolution: {integrity: sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==} - engines: {node: '>= 8'} - - methods@1.1.2: - resolution: {integrity: sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==} - engines: {node: '>= 0.6'} - - micromatch@3.1.10: - resolution: {integrity: sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==} - engines: {node: '>=0.10.0'} - - micromatch@4.0.8: - resolution: {integrity: sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==} - engines: {node: '>=8.6'} - - microseconds@0.2.0: - resolution: {integrity: sha512-n7DHHMjR1avBbSpsTBj6fmMGh2AGrifVV4e+WYc3Q9lO+xnSZ3NyhcBND3vzzatt05LFhoKFRxrIyklmLlUtyA==} - - mime-db@1.52.0: - resolution: {integrity: sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==} - engines: {node: '>= 0.6'} - - mime-db@1.53.0: - resolution: {integrity: sha512-oHlN/w+3MQ3rba9rqFr6V/ypF10LSkdwUysQL7GkXoTgIWeV+tcXGA852TBxH+gsh8UWoyhR1hKcoMJTuWflpg==} - engines: {node: '>= 0.6'} - - mime-types@2.1.35: - resolution: {integrity: sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==} - engines: {node: '>= 0.6'} - - mimic-fn@1.2.0: - resolution: {integrity: sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==} - engines: {node: '>=4'} - - mimic-fn@2.1.0: - resolution: {integrity: sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==} - engines: {node: '>=6'} - - mimic-fn@4.0.0: - resolution: {integrity: sha512-vqiC06CuhBTUdZH+RYl8sFrL096vA45Ok5ISO6sE/Mr1jRbGH4Csnhi8f3wKVl7x8mO4Au7Ir9D3Oyv1VYMFJw==} - engines: {node: '>=12'} - - mimic-response@1.0.1: - resolution: {integrity: sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==} - engines: {node: '>=4'} - - mimic-response@2.1.0: - resolution: {integrity: sha512-wXqjST+SLt7R009ySCglWBCFpjUygmCIfD790/kVbiGmUgfYGuB14PiTd5DwVxSV4NcYHjzMkoj5LjQZwTQLEA==} - engines: {node: '>=8'} - - mimic-response@3.1.0: - resolution: {integrity: sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==} - engines: {node: '>=10'} - - mini-css-extract-plugin@2.7.7: - resolution: {integrity: sha512-+0n11YGyRavUR3IlaOzJ0/4Il1avMvJ1VJfhWfCn24ITQXhRr1gghbhhrda6tgtNcpZaWKdSuwKq20Jb7fnlyw==} - engines: {node: '>= 12.13.0'} - peerDependencies: - webpack: ^5.0.0 - - minimalistic-assert@1.0.1: - resolution: {integrity: sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==} - - minimalistic-crypto-utils@1.0.1: - resolution: {integrity: sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg==} - - minimatch@3.1.2: - resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==} - - minimatch@5.1.6: - resolution: {integrity: sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==} - engines: {node: '>=10'} - - minimatch@9.0.3: - resolution: {integrity: sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==} - engines: {node: '>=16 || 14 >=14.17'} - - minimatch@9.0.5: - resolution: {integrity: sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==} - engines: {node: '>=16 || 14 >=14.17'} - - minimist@1.2.8: - resolution: {integrity: sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==} - - minipass@3.3.6: - resolution: {integrity: sha512-DxiNidxSEK+tHG6zOIklvNOwm3hvCrbUrdtzY74U6HKTJxvIDfOUL5W5P2Ghd3DTkhhKPYGqeNUIh5qcM4YBfw==} - engines: {node: '>=8'} - - minipass@4.2.8: - resolution: {integrity: sha512-fNzuVyifolSLFL4NzpF+wEF4qrgqaaKX0haXPQEdQ7NKAN+WecoKMHV09YcuL/DHxrUsYQOK3MiuDf7Ip2OXfQ==} - engines: {node: '>=8'} - - minipass@5.0.0: - resolution: {integrity: sha512-3FnjYuehv9k6ovOEbyOswadCDPX1piCfhV8ncmYtHOjuPwylVWsghTLo7rabjC3Rx5xD4HDx8Wm1xnMF7S5qFQ==} - engines: {node: '>=8'} - - minipass@7.1.2: - resolution: {integrity: sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==} - engines: {node: '>=16 || 14 >=14.17'} - - minizlib@2.1.2: - resolution: {integrity: sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==} - engines: {node: '>= 8'} - - mixin-deep@1.3.2: - resolution: {integrity: sha512-WRoDn//mXBiJ1H40rqa3vH0toePwSsGb45iInWlTySa+Uu4k3tYUSxa2v1KqAiLtvlrSzaExqS1gtk96A9zvEA==} - engines: {node: '>=0.10.0'} - - mkdirp-classic@0.5.3: - resolution: {integrity: sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==} - - mkdirp@0.5.6: - resolution: {integrity: sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==} - hasBin: true - - mkdirp@1.0.4: - resolution: {integrity: sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==} - engines: {node: '>=10'} - hasBin: true - - mrmime@2.0.0: - resolution: {integrity: sha512-eu38+hdgojoyq63s+yTpN4XMBdt5l8HhMhc4VKLO9KM5caLIBvUm4thi7fFaxyTmCKeNnXZ5pAlBwCUnhA09uw==} - engines: {node: '>=10'} - - ms@2.0.0: - resolution: {integrity: sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==} - - ms@2.1.2: - resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==} - - ms@2.1.3: - resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} - - multistream@4.1.0: - resolution: {integrity: sha512-J1XDiAmmNpRCBfIWJv+n0ymC4ABcf/Pl+5YvC5B/D2f/2+8PtHvCNxMPKiQcZyi922Hq69J2YOpb1pTywfifyw==} - - mute-stream@0.0.8: - resolution: {integrity: sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==} - - mz@2.7.0: - resolution: {integrity: sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==} - - nan@2.20.0: - resolution: {integrity: sha512-bk3gXBZDGILuuo/6sKtr0DQmSThYHLtNCdSdXk9YkxD/jK6X2vmCyyXBBxyqZ4XcnzTyYEAThfX3DCEnLf6igw==} - - nano-time@1.0.0: - resolution: {integrity: sha512-flnngywOoQ0lLQOTRNexn2gGSNuM9bKj9RZAWSzhQ+UJYaAFG9bac4DW9VHjUAzrOaIcajHybCTHe/bkvozQqA==} - - nanoclone@0.2.1: - resolution: {integrity: sha512-wynEP02LmIbLpcYw8uBKpcfF6dmg2vcpKqxeH5UcoKEYdExslsdUA4ugFauuaeYdTB76ez6gJW8XAZ6CgkXYxA==} - - nanoid@3.3.7: - resolution: {integrity: sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==} - engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1} - hasBin: true - - nanomatch@1.2.13: - resolution: {integrity: sha512-fpoe2T0RbHwBTBUOftAfBPaDEi06ufaUai0mE6Yn1kacc3SnTErfb/h+X94VXzI64rKFHYImXSvdwGGCmwOqCA==} - engines: {node: '>=0.10.0'} - - napi-build-utils@1.0.2: - resolution: {integrity: sha512-ONmRUqK7zj7DWX0D9ADe03wbwOBZxNAfF20PlGfCWQcD3+/MakShIHrMqx9YwPTfxDdF1zLeL+RGZiR9kGMLdg==} - - negotiator@0.6.3: - resolution: {integrity: sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==} - engines: {node: '>= 0.6'} - - neo-async@2.6.2: - resolution: {integrity: sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==} - - no-case@2.3.2: - resolution: {integrity: sha512-rmTZ9kz+f3rCvK2TD1Ue/oZlns7OGoIWP4fc3llxxRXlOkHKoWPPWJOfFYpITabSow43QJbRIoHQXtt10VldyQ==} - - no-case@3.0.4: - resolution: {integrity: sha512-fgAN3jGAh+RoxUGZHTSOLJIqUc2wmoBwGR4tbpNAKmmovFoWq0OdRkb0VkldReO2a2iBT/OEulG9XSUc10r3zg==} - - node-abi@3.67.0: - resolution: {integrity: sha512-bLn/fU/ALVBE9wj+p4Y21ZJWYFjUXLXPi/IewyLZkx3ApxKDNBWCKdReeKOtD8dWpOdDCeMyLh6ZewzcLsG2Nw==} - engines: {node: '>=10'} - - node-abort-controller@3.1.1: - resolution: {integrity: sha512-AGK2yQKIjRuqnc6VkX2Xj5d+QW8xZ87pa1UK6yA6ouUyuxfHuMP6umE5QK7UmTeOAymo+Zx1Fxiuw9rVx8taHQ==} - - node-addon-api@6.1.0: - resolution: {integrity: sha512-+eawOlIgy680F0kBzPUNFhMZGtJ1YmqM6l4+Crf4IkImjYrO/mqPwRMh352g23uIaQKFItcQ64I7KMaJxHgAVA==} - - node-domexception@1.0.0: - resolution: {integrity: sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==} - engines: {node: '>=10.5.0'} - - node-fetch@2.7.0: - resolution: {integrity: sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==} - engines: {node: 4.x || >=6.0.0} - peerDependencies: - encoding: ^0.1.0 - peerDependenciesMeta: - encoding: - optional: true - - node-fetch@3.3.2: - resolution: {integrity: sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - node-machine-id@1.1.12: - resolution: {integrity: sha512-QNABxbrPa3qEIfrE6GOJ7BYIuignnJw7iQ2YPbc3Nla1HzRJjXzZOiikfF8m7eAMfichLt3M4VgLOetqgDmgGQ==} - - node-plop@0.26.3: - resolution: {integrity: sha512-Cov028YhBZ5aB7MdMWJEmwyBig43aGL5WT4vdoB28Oitau1zZAcHUn8Sgfk9HM33TqhtLJ9PlM/O0Mv+QpV/4Q==} - engines: {node: '>=8.9.4'} - - node-releases@2.0.18: - resolution: {integrity: sha512-d9VeXT4SJ7ZeOqGX6R5EM022wpL+eWPooLI+5UpWn2jCT1aosUQEhQP214x33Wkwx3JQMvIm+tIoVOdodFS40g==} - - node-schedule@2.1.1: - resolution: {integrity: sha512-OXdegQq03OmXEjt2hZP33W2YPs/E5BcFQks46+G2gAxs4gHOIVD1u7EqlYLYSKsaIpyKCK9Gbk0ta1/gjRSMRQ==} - engines: {node: '>=6'} - - nodemailer-fetch@1.6.0: - resolution: {integrity: sha512-P7S5CEVGAmDrrpn351aXOLYs1R/7fD5NamfMCHyi6WIkbjS2eeZUB/TkuvpOQr0bvRZicVqo59+8wbhR3yrJbQ==} - - nodemailer-shared@1.1.0: - resolution: {integrity: sha512-68xW5LSyPWv8R0GLm6veAvm7E+XFXkVgvE3FW0FGxNMMZqMkPFeGDVALfR1DPdSfcoO36PnW7q5AAOgFImEZGg==} - - nodemon@3.0.2: - resolution: {integrity: sha512-9qIN2LNTrEzpOPBaWHTm4Asy1LxXLSickZStAQ4IZe7zsoIpD/A7LWxhZV3t4Zu352uBcqVnRsDXSMR2Sc3lTA==} - engines: {node: '>=10'} - hasBin: true - - noms@0.0.0: - resolution: {integrity: sha512-lNDU9VJaOPxUmXcLb+HQFeUgQQPtMI24Gt6hgfuMHRJgMRHMF/qZ4HJD3GDru4sSw9IQl2jPjAYnQrdIeLbwow==} - - nopt@5.0.0: - resolution: {integrity: sha512-Tbj67rffqceeLpcRXrT7vKAN8CwfPeIBgM7E6iBkmKLV7bEMwpGgYLGv0jACUsECaa/vuxP0IjEont6umdMgtQ==} - engines: {node: '>=6'} - hasBin: true - - normalize-package-data@2.5.0: - resolution: {integrity: sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==} - - normalize-path@3.0.0: - resolution: {integrity: sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==} - engines: {node: '>=0.10.0'} - - normalize-url@6.1.0: - resolution: {integrity: sha512-DlL+XwOy3NxAQ8xuC0okPgK46iuVNAK01YN7RueYBqqFeGsBjV9XmCAzAdgt+667bCl5kPh9EqKKDwnaPG1I7A==} - engines: {node: '>=10'} - - npm-bundled@1.1.2: - resolution: {integrity: sha512-x5DHup0SuyQcmL3s7Rx/YQ8sbw/Hzg0rj48eN0dV7hf5cmQq5PXIeioroH3raV1QC1yh3uTYuMThvEQF3iKgGQ==} - - npm-normalize-package-bin@1.0.1: - resolution: {integrity: sha512-EPfafl6JL5/rU+ot6P3gRSCpPDW5VmIzX959Ob1+ySFUuuYHWHekXpwdUZcKP5C+DS4GEtdJluwBjnsNDl+fSA==} - - npm-packlist@2.2.2: - resolution: {integrity: sha512-Jt01acDvJRhJGthnUJVF/w6gumWOZxO7IkpY/lsX9//zqQgnF7OJaxgQXcerd4uQOLu7W5bkb4mChL9mdfm+Zg==} - engines: {node: '>=10'} - hasBin: true - - npm-run-path@4.0.1: - resolution: {integrity: sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==} - engines: {node: '>=8'} - - npm-run-path@5.3.0: - resolution: {integrity: sha512-ppwTtiJZq0O/ai0z7yfudtBpWIoxM8yE6nHi1X47eFR2EWORqfbu6CnPlNsjeN683eT0qG6H/Pyf9fCcvjnnnQ==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - npmlog@5.0.1: - resolution: {integrity: sha512-AqZtDUWOMKs1G/8lwylVjrdYgqA4d9nu8hc+0gzRxlDb1I10+FHBGMXs6aiQHFdCUUlqH99MUMuLfzWDNDtfxw==} - deprecated: This package is no longer supported. - - nth-check@2.1.1: - resolution: {integrity: sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==} - - oauth-sign@0.9.0: - resolution: {integrity: sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==} - - object-assign@4.1.1: - resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} - engines: {node: '>=0.10.0'} - - object-copy@0.1.0: - resolution: {integrity: sha512-79LYn6VAb63zgtmAteVOWo9Vdj71ZVBy3Pbse+VqxDpEP83XuujMrGqHIwAXJ5I/aM0zU7dIyIAhifVTPrNItQ==} - engines: {node: '>=0.10.0'} - - object-inspect@1.13.2: - resolution: {integrity: sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==} - engines: {node: '>= 0.4'} - - object-visit@1.0.1: - resolution: {integrity: sha512-GBaMwwAVK9qbQN3Scdo0OyvgPW7l3lnaVMj84uTOZlswkX0KpF6fyDBJhtTthf7pymztoN36/KEr1DyhF96zEA==} - engines: {node: '>=0.10.0'} - - object.defaults@1.1.0: - resolution: {integrity: sha512-c/K0mw/F11k4dEUBMW8naXUuBuhxRCfG7W+yFy8EcijU/rSmazOUd1XAEEe6bC0OuXY4HUKjTJv7xbxIMqdxrA==} - engines: {node: '>=0.10.0'} - - object.map@1.0.1: - resolution: {integrity: sha512-3+mAJu2PLfnSVGHwIWubpOFLscJANBKuB/6A4CxBstc4aqwQY0FWcsppuy4jU5GSB95yES5JHSI+33AWuS4k6w==} - engines: {node: '>=0.10.0'} - - object.pick@1.3.0: - resolution: {integrity: sha512-tqa/UMy/CCoYmj+H5qc07qvSL9dqcs/WZENZ1JbtWBlATP+iVOe778gE6MSijnyCnORzDuX6hU+LA4SZ09YjFQ==} - engines: {node: '>=0.10.0'} - - oblivious-set@1.0.0: - resolution: {integrity: sha512-z+pI07qxo4c2CulUHCDf9lcqDlMSo72N/4rLUpRXf6fu+q8vjt8y0xS+Tlf8NTJDdTXHbdeO1n3MlbctwEoXZw==} - - on-finished@2.4.1: - resolution: {integrity: sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==} - engines: {node: '>= 0.8'} - - once@1.4.0: - resolution: {integrity: sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==} - - one-time@1.0.0: - resolution: {integrity: sha512-5DXOiRKwuSEcQ/l0kGCF6Q3jcADFv5tSmRaJck/OqkVFcOzutB134KRSfF0xDrL39MNnqxbHBbUUcjZIhTgb2g==} - - onetime@2.0.1: - resolution: {integrity: sha512-oyyPpiMaKARvvcgip+JV+7zci5L8D1W9RZIz2l1o08AM3pfspitVWnPt3mzHcBPp12oYMTy0pqrFs/C+m3EwsQ==} - engines: {node: '>=4'} - - onetime@5.1.2: - resolution: {integrity: sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==} - engines: {node: '>=6'} - - onetime@6.0.0: - resolution: {integrity: sha512-1FlR+gjXK7X+AsAHso35MnyN5KqGwJRi/31ft6x0M194ht7S+rWAvd7PHss9xSKMzE0asv1pyIHaJYq+BbacAQ==} - engines: {node: '>=12'} - - only@0.0.2: - resolution: {integrity: sha512-Fvw+Jemq5fjjyWz6CpKx6w9s7xxqo3+JCyM0WXWeCSOboZ8ABkyvP8ID4CZuChA/wxSx+XSJmdOm8rGVyJ1hdQ==} - - open@8.4.0: - resolution: {integrity: sha512-XgFPPM+B28FtCCgSb9I+s9szOC1vZRSwgWsRUA5ylIxRTgKozqjOCrVOqGsYABPYK5qnfqClxZTFBa8PKt2v6Q==} - engines: {node: '>=12'} - - open@9.1.0: - resolution: {integrity: sha512-OS+QTnw1/4vrf+9hh1jc1jnYjzSG4ttTBB8UxOwAnInG3Uo4ssetzC1ihqaIHjLJnA5GGlRl6QlZXOTQhRBUvg==} - engines: {node: '>=14.16'} - - opener@1.5.2: - resolution: {integrity: sha512-ur5UIdyw5Y7yEj9wLzhqXiy6GZ3Mwx0yGI+5sMn2r0N0v3cKJvUmFH5yPP+WXh9e0xfyzyJX95D8l088DNFj7A==} - hasBin: true - - ora@3.4.0: - resolution: {integrity: sha512-eNwHudNbO1folBP3JsZ19v9azXWtQZjICdr3Q0TDPIaeBQ3mXLrh54wM+er0+hSp+dWKf+Z8KM58CYzEyIYxYg==} - engines: {node: '>=6'} - - ora@5.4.1: - resolution: {integrity: sha512-5b6Y85tPxZZ7QytO+BQzysW31HJku27cRIlkbAXaNx+BdcVi+LlRFmVXzeF6a7JCwJpyw5c4b+YSVImQIrBpuQ==} - engines: {node: '>=10'} - - os-paths@7.4.0: - resolution: {integrity: sha512-Ux1J4NUqC6tZayBqLN1kUlDAEvLiQlli/53sSddU4IN+h+3xxnv2HmRSMpVSvr1hvJzotfMs3ERvETGK+f4OwA==} - engines: {node: '>= 4.0'} - - os-tmpdir@1.0.2: - resolution: {integrity: sha512-D2FR03Vir7FIu45XBY20mTb+/ZSWB00sjU9jdQXt83gDrI4Ztz5Fs7/yy74g2N5SVQY4xY1qDr4rNddwYRVX0g==} - engines: {node: '>=0.10.0'} - - outdent@0.8.0: - resolution: {integrity: sha512-KiOAIsdpUTcAXuykya5fnVVT+/5uS0Q1mrkRHcF89tpieSmY33O/tmc54CqwA+bfhbtEfZUNLHaPUiB9X3jt1A==} - - p-cancelable@2.1.1: - resolution: {integrity: sha512-BZOr3nRQHOntUjTrH8+Lh54smKHoHyur8We1V8DSMVrl5A2malOOwuJRnKRDjSnkoeBh4at6BwEnb5I7Jl31wg==} - engines: {node: '>=8'} - - p-finally@1.0.0: - resolution: {integrity: sha512-LICb2p9CB7FS+0eR1oqWnHhp0FljGLZCWBE9aix0Uye9W8LTQPwMTYVGWQWIw9RdQiDg4+epXQODwIYJtSJaow==} - engines: {node: '>=4'} - - p-limit@2.3.0: - resolution: {integrity: sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==} - engines: {node: '>=6'} - - p-limit@3.1.0: - resolution: {integrity: sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==} - engines: {node: '>=10'} - - p-locate@3.0.0: - resolution: {integrity: sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==} - engines: {node: '>=6'} - - p-locate@4.1.0: - resolution: {integrity: sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==} - engines: {node: '>=8'} - - p-locate@5.0.0: - resolution: {integrity: sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==} - engines: {node: '>=10'} - - p-map@3.0.0: - resolution: {integrity: sha512-d3qXVTF/s+W+CdJ5A29wywV2n8CQQYahlgz2bFiA+4eVNJbHJodPZ+/gXwPGh0bOqA+j8S+6+ckmvLGPk1QpxQ==} - engines: {node: '>=8'} - - p-map@4.0.0: - resolution: {integrity: sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==} - engines: {node: '>=10'} - - p-queue@6.6.2: - resolution: {integrity: sha512-RwFpb72c/BhQLEXIZ5K2e+AhgNVmIejGlTgiB9MzZ0e93GRvqZ7uSi0dvRF7/XIXDeNkra2fNHBxTyPDGySpjQ==} - engines: {node: '>=8'} - - p-timeout@3.2.0: - resolution: {integrity: sha512-rhIwUycgwwKcP9yTOOFK/AKsAopjjCakVqLHePO3CC6Mir1Z99xT+R63jZxAT5lFZLa2inS5h+ZS2GvR99/FBg==} - engines: {node: '>=8'} - - p-try@2.2.0: - resolution: {integrity: sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==} - engines: {node: '>=6'} - - package-json-from-dist@1.0.0: - resolution: {integrity: sha512-dATvCeZN/8wQsGywez1mzHtTlP22H8OEfPrVMLNr4/eGa+ijtLn/6M5f0dY8UKNrC2O9UCU6SSoG3qRKnt7STw==} - - package-json@7.0.0: - resolution: {integrity: sha512-CHJqc94AA8YfSLHGQT3DbvSIuE12NLFekpM4n7LRrAd3dOJtA911+4xe9q6nC3/jcKraq7nNS9VxgtT0KC+diA==} - engines: {node: '>=12'} - - param-case@2.1.1: - resolution: {integrity: sha512-eQE845L6ot89sk2N8liD8HAuH4ca6Vvr7VWAWwt7+kvvG5aBcPmmphQ68JsEG2qa9n1TykS2DLeMt363AAH8/w==} - - param-case@3.0.4: - resolution: {integrity: sha512-RXlj7zCYokReqWpOPH9oYivUzLYZ5vAPIfEmCTNViosC78F8F0H9y7T7gG2M39ymgutxF5gcFEsyZQSph9Bp3A==} - - parent-module@1.0.1: - resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} - engines: {node: '>=6'} - - parse-filepath@1.0.2: - resolution: {integrity: sha512-FwdRXKCohSVeXqwtYonZTXtbGJKrn+HNyWDYVcp5yuJlesTwNH4rsmRZ+GrKAPJ5bLpRxESMeS+Rl0VCHRvB2Q==} - engines: {node: '>=0.8'} - - parse-json@5.2.0: - resolution: {integrity: sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==} - engines: {node: '>=8'} - - parse-passwd@1.0.0: - resolution: {integrity: sha512-1Y1A//QUXEZK7YKz+rD9WydcE1+EuPr6ZBgKecAB8tmoW6UFv0NREVJe1p+jRxtThkcbbKkfwIbWJe/IeE6m2Q==} - engines: {node: '>=0.10.0'} - - parse-path@7.0.0: - resolution: {integrity: sha512-Euf9GG8WT9CdqwuWJGdf3RkUcTBArppHABkO7Lm8IzRQp0e2r/kkFnmhu4TSK30Wcu5rVAZLmfPKSBBi9tWFog==} - - parse-srcset@1.0.2: - resolution: {integrity: sha512-/2qh0lav6CmI15FzA3i/2Bzk2zCgQhGMkvhOhKNcBVQ1ldgpbfiNTVslmooUmWJcADi1f1kIeynbDRVzNlfR6Q==} - - parse-url@8.1.0: - resolution: {integrity: sha512-xDvOoLU5XRrcOZvnI6b8zA6n9O9ejNk/GExuz1yBuWUGn9KA97GI6HTs6u02wKara1CeVmZhH+0TZFdWScR89w==} - - parseurl@1.3.3: - resolution: {integrity: sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==} - engines: {node: '>= 0.8'} - - pascal-case@2.0.1: - resolution: {integrity: sha512-qjS4s8rBOJa2Xm0jmxXiyh1+OFf6ekCWOvUaRgAQSktzlTbMotS0nmG9gyYAybCWBcuP4fsBeRCKNwGBnMe2OQ==} - - pascal-case@3.1.2: - resolution: {integrity: sha512-uWlGT3YSnK9x3BQJaOdcZwrnV6hPpd8jFH1/ucpiLRPh/2zCVJKS19E4GvYHvaCcACn3foXZ0cLB9Wrx1KGe5g==} - - pascalcase@0.1.1: - resolution: {integrity: sha512-XHXfu/yOQRy9vYOtUDVMN60OEJjW013GoObG1o+xwQTpB9eYJX/BjXMsdW13ZDPruFhYYn0AG22w0xgQMwl3Nw==} - engines: {node: '>=0.10.0'} - - passport-local@1.0.0: - resolution: {integrity: sha512-9wCE6qKznvf9mQYYbgJ3sVOHmCWoUNMVFoZzNoznmISbhnNNPhN9xfY3sLmScHMetEJeoY7CXwfhCe7argfQow==} - engines: {node: '>= 0.4.0'} - - passport-strategy@1.0.0: - resolution: {integrity: sha512-CB97UUvDKJde2V0KDWWB3lyf6PC3FaZP7YxZ2G8OAtn9p4HI9j9JLP9qjOGZFvyl8uwNT8qM+hGnz/n16NI7oA==} - engines: {node: '>= 0.4.0'} - - passport@0.6.0: - resolution: {integrity: sha512-0fe+p3ZnrWRW74fe8+SvCyf4a3Pb2/h7gFkQ8yTJpAO50gDzlfjZUZTO1k5Eg9kUct22OxHLqDZoKUWRHOh9ug==} - engines: {node: '>= 0.4.0'} - - path-case@2.1.1: - resolution: {integrity: sha512-Ou0N05MioItesaLr9q8TtHVWmJ6fxWdqKB2RohFmNWVyJ+2zeKIeDNWAN6B/Pe7wpzWChhZX6nONYmOnMeJQ/Q==} - - path-exists@3.0.0: - resolution: {integrity: sha512-bpC7GYwiDYQ4wYLe+FA8lhRjhQCMcQGuSgGGqDkg/QerRWw9CmGRT0iSOVRSZJ29NMLZgIzqaljJ63oaL4NIJQ==} - engines: {node: '>=4'} - - path-exists@4.0.0: - resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==} - engines: {node: '>=8'} - - path-is-absolute@1.0.1: - resolution: {integrity: sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==} - engines: {node: '>=0.10.0'} - - path-key@3.1.1: - resolution: {integrity: sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==} - engines: {node: '>=8'} - - path-key@4.0.0: - resolution: {integrity: sha512-haREypq7xkM7ErfgIyA0z+Bj4AGKlMSdlQE2jvJo6huWD1EdkKYV+G/T4nq0YEF2vgTT8kqMFKo1uHn950r4SQ==} - engines: {node: '>=12'} - - path-parse@1.0.7: - resolution: {integrity: sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==} - - path-root-regex@0.1.2: - resolution: {integrity: sha512-4GlJ6rZDhQZFE0DPVKh0e9jmZ5egZfxTkp7bcRDuPlJXbAwhxcl2dINPUAsjLdejqaLsCeg8axcLjIbvBjN4pQ==} - engines: {node: '>=0.10.0'} - - path-root@0.1.1: - resolution: {integrity: sha512-QLcPegTHF11axjfojBIoDygmS2E3Lf+8+jI6wOVmNVenrKSo3mFdSGiIgdSHenczw3wPtlVMQaFVwGmM7BJdtg==} - engines: {node: '>=0.10.0'} - - path-scurry@1.11.1: - resolution: {integrity: sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==} - engines: {node: '>=16 || 14 >=14.18'} - - path-to-regexp@6.2.2: - resolution: {integrity: sha512-GQX3SSMokngb36+whdpRXE+3f9V8UzyAorlYvOGx87ufGHehNTn5lCxrKtLyZ4Yl/wEKnNnr98ZzOwwDZV5ogw==} - - path-type@4.0.0: - resolution: {integrity: sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==} - engines: {node: '>=8'} - - pause@0.0.1: - resolution: {integrity: sha512-KG8UEiEVkR3wGEb4m5yZkVCzigAD+cVEJck2CzYZO37ZGJfctvVptVO192MwrtPhzONn6go8ylnOdMhKqi4nfg==} - - pg-cloudflare@1.1.1: - resolution: {integrity: sha512-xWPagP/4B6BgFO+EKz3JONXv3YDgvkbVrGw2mTo3D6tVDQRh1e7cqVGvyR3BE+eQgAvx1XhW/iEASj4/jCWl3Q==} - - pg-connection-string@2.6.1: - resolution: {integrity: sha512-w6ZzNu6oMmIzEAYVw+RLK0+nqHPt8K3ZnknKi+g48Ak2pr3dtljJW3o+D/n2zzCG07Zoe9VOX3aiKpj+BN0pjg==} - - pg-connection-string@2.6.4: - resolution: {integrity: sha512-v+Z7W/0EO707aNMaAEfiGnGL9sxxumwLl2fJvCQtMn9Fxsg+lPpPkdcyBSv/KFgpGdYkMfn+EI1Or2EHjpgLCA==} - - pg-int8@1.0.1: - resolution: {integrity: sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw==} - engines: {node: '>=4.0.0'} - - pg-pool@3.6.2: - resolution: {integrity: sha512-Htjbg8BlwXqSBQ9V8Vjtc+vzf/6fVUuak/3/XXKA9oxZprwW3IMDQTGHP+KDmVL7rtd+R1QjbnCFPuTHm3G4hg==} - peerDependencies: - pg: '>=8.0' - - pg-protocol@1.6.1: - resolution: {integrity: sha512-jPIlvgoD63hrEuihvIg+tJhoGjUsLPn6poJY9N5CnlPd91c2T18T/9zBtLxZSb1EhYxBRoZJtzScCaWlYLtktg==} - - pg-types@2.2.0: - resolution: {integrity: sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==} - engines: {node: '>=4'} - - pg@8.12.0: - resolution: {integrity: sha512-A+LHUSnwnxrnL/tZ+OLfqR1SxLN3c/pgDztZ47Rpbsd4jUytsTtwQo/TLPRzPJMp/1pbhYVhH9cuSZLAajNfjQ==} - engines: {node: '>= 8.0.0'} - peerDependencies: - pg-native: '>=3.0.1' - peerDependenciesMeta: - pg-native: - optional: true - - pgpass@1.0.5: - resolution: {integrity: sha512-FdW9r/jQZhSeohs1Z3sI1yxFQNFvMcnmfuj4WBMUTxOrAyLMaTcE1aAMBiTlbMNaXvBCQuVi0R7hd8udDSP7ug==} - - picocolors@1.1.0: - resolution: {integrity: sha512-TQ92mBOW0l3LeMeyLV6mzy/kWr8lkd/hp3mTg7wYK7zJhuBStmGMBG0BdeDZS/dZx1IukaX6Bk11zcln25o1Aw==} - - picomatch@2.3.1: - resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==} - engines: {node: '>=8.6'} - - pkg-up@3.1.0: - resolution: {integrity: sha512-nDywThFk1i4BQK4twPQ6TA4RT8bDY96yeuCVBWL3ePARCiEKDRSrNGbFIgUJpLp+XeIR65v8ra7WuJOFUBtkMA==} - engines: {node: '>=8'} - - plop@2.7.6: - resolution: {integrity: sha512-IgnYAsC3Ni7t1cDU7wH2151CD22YhMxH8PFh+iPzCf+WuGEWXslJ5t1Tpr0N/gjL23CAV/HbLAWug2IPM2YrHg==} - engines: {node: '>=8.9.4'} - hasBin: true - - pluralize@8.0.0: - resolution: {integrity: sha512-Nc3IT5yHzflTfbjgqWcCPpo7DaKy4FnpB0l/zCAW0Tc7jxAiuqSxHasntB3D7887LSrA93kDJ9IXovxJYxyLCA==} - engines: {node: '>=4'} - - pony-cause@2.1.11: - resolution: {integrity: sha512-M7LhCsdNbNgiLYiP4WjsfLUuFmCfnjdF6jKe2R9NKl4WFN+HZPGHJZ9lnLP7f9ZnKe3U9nuWD0szirmj+migUg==} - engines: {node: '>=12.0.0'} - - posix-character-classes@0.1.1: - resolution: {integrity: sha512-xTgYBc3fuo7Yt7JbiuFxSYGToMoz8fLoE6TC9Wx1P/u+LfeThMOAqmuyECnlBaaJb+u1m9hHiXUEtwW4OzfUJg==} - engines: {node: '>=0.10.0'} - - postcss-modules-extract-imports@3.1.0: - resolution: {integrity: sha512-k3kNe0aNFQDAZGbin48pL2VNidTF0w4/eASDsxlyspobzU3wZQLOGj7L9gfRe0Jo9/4uud09DsjFNH7winGv8Q==} - engines: {node: ^10 || ^12 || >= 14} - peerDependencies: - postcss: ^8.1.0 - - postcss-modules-local-by-default@4.0.5: - resolution: {integrity: sha512-6MieY7sIfTK0hYfafw1OMEG+2bg8Q1ocHCpoWLqOKj3JXlKu4G7btkmM/B7lFubYkYWmRSPLZi5chid63ZaZYw==} - engines: {node: ^10 || ^12 || >= 14} - peerDependencies: - postcss: ^8.1.0 - - postcss-modules-scope@3.2.0: - resolution: {integrity: sha512-oq+g1ssrsZOsx9M96c5w8laRmvEu9C3adDSjI8oTcbfkrTE8hx/zfyobUoWIxaKPO8bt6S62kxpw5GqypEw1QQ==} - engines: {node: ^10 || ^12 || >= 14} - peerDependencies: - postcss: ^8.1.0 - - postcss-modules-values@4.0.0: - resolution: {integrity: sha512-RDxHkAiEGI78gS2ofyvCsu7iycRv7oqw5xMWn9iMoR0N/7mf9D50ecQqUo5BZ9Zh2vH4bCUR/ktCqbB9m8vJjQ==} - engines: {node: ^10 || ^12 || >= 14} - peerDependencies: - postcss: ^8.1.0 - - postcss-selector-parser@6.1.2: - resolution: {integrity: sha512-Q8qQfPiZ+THO/3ZrOrO0cJJKfpYCagtMUkXbnEfmgUjwXg6z/WBeOyS9APBBPCTSiDV+s4SwQGu8yFsiMRIudg==} - engines: {node: '>=4'} - - postcss-value-parser@4.2.0: - resolution: {integrity: sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==} - - postcss@8.4.38: - resolution: {integrity: sha512-Wglpdk03BSfXkHoQa3b/oulrotAkwrlLDRSOb9D0bN86FdRyE9lppSp33aHNPgBa0JKCoB+drFLZkQoRRYae5A==} - engines: {node: ^10 || ^12 || >=14} - - postcss@8.4.44: - resolution: {integrity: sha512-Aweb9unOEpQ3ezu4Q00DPvvM2ZTUitJdNKeP/+uQgr1IBIqu574IaZoURId7BKtWMREwzKa9OgzPzezWGPWFQw==} - engines: {node: ^10 || ^12 || >=14} - - postgres-array@2.0.0: - resolution: {integrity: sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA==} - engines: {node: '>=4'} - - postgres-bytea@1.0.0: - resolution: {integrity: sha512-xy3pmLuQqRBZBXDULy7KbaitYqLcmxigw14Q5sj8QBVLqEwXfeybIKVWiqAXTlcvdvb0+xkOtDbfQMOf4lST1w==} - engines: {node: '>=0.10.0'} - - postgres-date@1.0.7: - resolution: {integrity: sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q==} - engines: {node: '>=0.10.0'} - - postgres-interval@1.2.0: - resolution: {integrity: sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==} - engines: {node: '>=0.10.0'} - - prebuild-install@7.1.2: - resolution: {integrity: sha512-UnNke3IQb6sgarcZIDU3gbMeTp/9SSU1DAIkil7PrqG1vZlBtY5msYccSKSHDqa3hNg436IXK+SNImReuA1wEQ==} - engines: {node: '>=10'} - hasBin: true - - prettier-plugin-packagejson@2.4.5: - resolution: {integrity: sha512-glG71jE1gO3y5+JNAhC8X+4yrlN28rub6Aj461SKbaPie9RgMiHKcInH2Moi2VGOfkTXaEHBhg4uVMBqa+kBUA==} - peerDependencies: - prettier: '>= 1.16.0' - peerDependenciesMeta: - prettier: - optional: true - - prettier@2.8.4: - resolution: {integrity: sha512-vIS4Rlc2FNh0BySk3Wkd6xmwxB0FpOndW5fisM5H8hsZSxU2VWVB5CWIkIjWvrHjIhxk2g3bfMKM87zNTrZddw==} - engines: {node: '>=10.13.0'} - hasBin: true - - pretty-error@4.0.0: - resolution: {integrity: sha512-AoJ5YMAcXKYxKhuJGdcvse+Voc6v1RgnsR3nWcYU7q4t6z0Q6T86sv5Zq8VIRbOWWFpvdGE83LtdSMNd+6Y0xw==} - - process-nextick-args@2.0.1: - resolution: {integrity: sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==} - - progress-stream@2.0.0: - resolution: {integrity: sha512-xJwOWR46jcXUq6EH9yYyqp+I52skPySOeHfkxOZ2IY1AiBi/sFJhbhAKHoV3OTw/omQ45KTio9215dRJ2Yxd3Q==} - - prompts@2.4.2: - resolution: {integrity: sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==} - engines: {node: '>= 6'} - - prop-types@15.8.1: - resolution: {integrity: sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==} - - property-expr@2.0.6: - resolution: {integrity: sha512-SVtmxhRE/CGkn3eZY1T6pC8Nln6Fr/lu1mKSgRud0eC73whjGfoAogbn78LkD8aFL0zz3bAFerKSnOl7NlErBA==} - - proto-list@1.2.4: - resolution: {integrity: sha512-vtK/94akxsTMhe0/cbfpR+syPuszcuwhqVjJq26CuNDgFGj682oRBXOP5MJpv2r7JtE8MsiepGIqvvOTBwn2vA==} - - protocols@2.0.1: - resolution: {integrity: sha512-/XJ368cyBJ7fzLMwLKv1e4vLxOju2MNAIokcr7meSaNcVbWz/CPcW22cP04mwxOErdA5mwjA8Q6w/cdAQxVn7Q==} - - proxy-from-env@1.1.0: - resolution: {integrity: sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==} - - pstree.remy@1.1.8: - resolution: {integrity: sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==} - - pump@3.0.0: - resolution: {integrity: sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==} - - punycode@2.3.1: - resolution: {integrity: sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==} - engines: {node: '>=6'} - - purest@4.0.2: - resolution: {integrity: sha512-Uq6kdia8zGVHOb/0zAOb7FvKFMKeyeTZTLEwpO0JR3cIFEkpH6asv3ls9M9URDjHiYIdgAPmht5ecSbvPacfyg==} - engines: {node: '>=12.0.0'} - - qs@6.11.1: - resolution: {integrity: sha512-0wsrzgTz/kAVIeuxSjnpGC56rzYtr6JT/2BwEvMaPhFIoYa1aGO8LbzuU1R0uUYQkLpWBTOj0l/CLAJB64J6nQ==} - engines: {node: '>=0.6'} - - qs@6.13.0: - resolution: {integrity: sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==} - engines: {node: '>=0.6'} - - queue-microtask@1.2.3: - resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==} - - queue-tick@1.0.1: - resolution: {integrity: sha512-kJt5qhMxoszgU/62PLP1CJytzd2NKetjSRnyuj31fDd3Rlcz3fzlFdFLD1SItunPwyqEOkca6GbV612BWfaBag==} - - quick-lru@5.1.1: - resolution: {integrity: sha512-WuyALRjWPDGtt/wzJiadO5AXY+8hZ80hVpe6MyivgraREW751X3SbhRvG3eLKOYN+8VEvqLcf3wdnt44Z4S4SA==} - engines: {node: '>=10'} - - randombytes@2.1.0: - resolution: {integrity: sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==} - - range-parser@1.2.1: - resolution: {integrity: sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==} - engines: {node: '>= 0.6'} - - raw-body@2.5.2: - resolution: {integrity: sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==} - engines: {node: '>= 0.8'} - - rc@1.2.8: - resolution: {integrity: sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==} - hasBin: true - - react-dnd-html5-backend@16.0.1: - resolution: {integrity: sha512-Wu3dw5aDJmOGw8WjH1I1/yTH+vlXEL4vmjk5p+MHxP8HuHJS1lAGeIdG/hze1AvNeXWo/JgULV87LyQOr+r5jw==} - - react-dnd@16.0.1: - resolution: {integrity: sha512-QeoM/i73HHu2XF9aKksIUuamHPDvRglEwdHL4jsp784BgUuWcg6mzfxT0QDdQz8Wj0qyRKx2eMg8iZtWvU4E2Q==} - peerDependencies: - '@types/hoist-non-react-statics': '>= 3.3.1' - '@types/node': '>= 12' - '@types/react': '>= 16' - react: '>= 16.14' - peerDependenciesMeta: - '@types/hoist-non-react-statics': - optional: true - '@types/node': - optional: true - '@types/react': - optional: true - - react-dom@18.3.1: - resolution: {integrity: sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==} - peerDependencies: - react: ^18.3.1 - - react-error-boundary@3.1.4: - resolution: {integrity: sha512-uM9uPzZJTF6wRQORmSrvOIgt4lJ9MC1sNgEOj2XGsDTRE4kmpWxg7ENK9EWNKJRMAOY9z0MuF4yIfl6gp4sotA==} - engines: {node: '>=10', npm: '>=6'} - peerDependencies: - react: '>=16.13.1' - - react-fast-compare@2.0.4: - resolution: {integrity: sha512-suNP+J1VU1MWFKcyt7RtjiSWUjvidmQSlqu+eHslq+342xCbGTYmC0mEhPCOHxlW0CywylOC1u2DFAT+bv4dBw==} - - react-fast-compare@3.2.2: - resolution: {integrity: sha512-nsO+KSNgo1SbJqJEYRE9ERzo7YtYbou/OqjSQKxV7jcKox7+usiUVZOAC+XnDOABXggQTno0Y1CpVnuWEc1boQ==} - - react-helmet@6.1.0: - resolution: {integrity: sha512-4uMzEY9nlDlgxr61NL3XbKRy1hEkXmKNXhjbAIOVw5vcFrsdYbH2FEwcNyWvWinl103nXgzYNlns9ca+8kFiWw==} - peerDependencies: - react: '>=16.3.0' - - react-intl@6.4.1: - resolution: {integrity: sha512-/aT5595AEMZ+Pjmt8W2R5/ZkYJmyyd6jTzHzqhJ1LnfeG36+N5huBtykxYhHqLc1BrIRQ1fTX1orYC0Ej5ojtg==} - peerDependencies: - react: ^16.6.0 || 17 || 18 - typescript: ^4.7 || 5 - peerDependenciesMeta: - typescript: - optional: true - - react-is@16.13.1: - resolution: {integrity: sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==} - - react-is@18.3.1: - resolution: {integrity: sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==} - - react-query@3.39.3: - resolution: {integrity: sha512-nLfLz7GiohKTJDuT4us4X3h/8unOh+00MLb2yJoGTPjxKs2bc1iDhkNx2bd5MKklXnOD3NrVZ+J2UXujA5In4g==} - peerDependencies: - react: ^16.8.0 || ^17.0.0 || ^18.0.0 - react-dom: '*' - react-native: '*' - peerDependenciesMeta: - react-dom: - optional: true - react-native: - optional: true - - react-redux@8.1.1: - resolution: {integrity: sha512-5W0QaKtEhj+3bC0Nj0NkqkhIv8gLADH/2kYFMTHxCVqQILiWzLv6MaLuV5wJU3BQEdHKzTfcvPN0WMS6SC1oyA==} - peerDependencies: - '@types/react': ^16.8 || ^17.0 || ^18.0 - '@types/react-dom': ^16.8 || ^17.0 || ^18.0 - react: ^16.8 || ^17.0 || ^18.0 - react-dom: ^16.8 || ^17.0 || ^18.0 - react-native: '>=0.59' - redux: ^4 || ^5.0.0-beta.0 - peerDependenciesMeta: - '@types/react': - optional: true - '@types/react-dom': - optional: true - react-dom: - optional: true - react-native: - optional: true - redux: - optional: true - - react-refresh@0.14.0: - resolution: {integrity: sha512-wViHqhAd8OHeLS/IRMJjTSDHF3U9eWi62F/MledQGPdJGDhodXJ9PBLNGr6WWL7qlH12Mt3TyTpbS+hGXMjCzQ==} - engines: {node: '>=0.10.0'} - - react-remove-scroll-bar@2.3.6: - resolution: {integrity: sha512-DtSYaao4mBmX+HDo5YWYdBWQwYIQQshUV/dVxFxK+KM26Wjwp1gZ6rv6OC3oujI6Bfu6Xyg3TwK533AQutsn/g==} - engines: {node: '>=10'} - peerDependencies: - '@types/react': ^16.8.0 || ^17.0.0 || ^18.0.0 - react: ^16.8.0 || ^17.0.0 || ^18.0.0 - peerDependenciesMeta: - '@types/react': - optional: true - - react-remove-scroll@2.5.10: - resolution: {integrity: sha512-m3zvBRANPBw3qxVVjEIPEQinkcwlFZ4qyomuWVpNJdv4c6MvHfXV0C3L9Jx5rr3HeBHKNRX+1jreB5QloDIJjA==} - engines: {node: '>=10'} - peerDependencies: - '@types/react': ^16.8.0 || ^17.0.0 || ^18.0.0 - react: ^16.8.0 || ^17.0.0 || ^18.0.0 - peerDependenciesMeta: - '@types/react': - optional: true - - react-remove-scroll@2.5.7: - resolution: {integrity: sha512-FnrTWO4L7/Bhhf3CYBNArEG/yROV0tKmTv7/3h9QCFvH6sndeFf1wPqOcbFVu5VAulS5dV1wGT3GZZ/1GawqiA==} - engines: {node: '>=10'} - peerDependencies: - '@types/react': ^16.8.0 || ^17.0.0 || ^18.0.0 - react: ^16.8.0 || ^17.0.0 || ^18.0.0 - peerDependenciesMeta: - '@types/react': - optional: true - - react-router-dom@6.26.1: - resolution: {integrity: sha512-veut7m41S1fLql4pLhxeSW3jlqs+4MtjRLj0xvuCEXsxusJCbs6I8yn9BxzzDX2XDgafrccY6hwjmd/bL54tFw==} - engines: {node: '>=14.0.0'} - peerDependencies: - react: '>=16.8' - react-dom: '>=16.8' - - react-router@6.26.1: - resolution: {integrity: sha512-kIwJveZNwp7teQRI5QmwWo39A5bXRyqpH0COKKmPnyD2vBvDwgFXSqDUYtt1h+FEyfnE8eXr7oe0MxRzVwCcvQ==} - engines: {node: '>=14.0.0'} - peerDependencies: - react: '>=16.8' - - react-select@5.7.0: - resolution: {integrity: sha512-lJGiMxCa3cqnUr2Jjtg9YHsaytiZqeNOKeibv6WF5zbK/fPegZ1hg3y/9P1RZVLhqBTs0PfqQLKuAACednYGhQ==} - peerDependencies: - react: ^16.8.0 || ^17.0.0 || ^18.0.0 - react-dom: ^16.8.0 || ^17.0.0 || ^18.0.0 - - react-side-effect@2.1.2: - resolution: {integrity: sha512-PVjOcvVOyIILrYoyGEpDN3vmYNLdy1CajSFNt4TDsVQC5KpTijDvWVoR+/7Rz2xT978D8/ZtFceXxzsPwZEDvw==} - peerDependencies: - react: ^16.3.0 || ^17.0.0 || ^18.0.0 - - react-style-singleton@2.2.1: - resolution: {integrity: sha512-ZWj0fHEMyWkHzKYUr2Bs/4zU6XLmq9HsgBURm7g5pAVfyn49DgUiNgY2d4lXRlYSiCif9YBGpQleewkcqddc7g==} - engines: {node: '>=10'} - peerDependencies: - '@types/react': ^16.8.0 || ^17.0.0 || ^18.0.0 - react: ^16.8.0 || ^17.0.0 || ^18.0.0 - peerDependenciesMeta: - '@types/react': - optional: true - - react-transition-group@4.4.5: - resolution: {integrity: sha512-pZcd1MCJoiKiBR2NRxeCRg13uCXbydPnmB4EOeRrY7480qNWO8IIgQG6zlDkm6uRMsURXPuKq0GWtiM59a5Q6g==} - peerDependencies: - react: '>=16.6.0' - react-dom: '>=16.6.0' - - react-window@1.8.8: - resolution: {integrity: sha512-D4IiBeRtGXziZ1n0XklnFGu7h9gU684zepqyKzgPNzrsrk7xOCxni+TCckjg2Nr/DiaEEGVVmnhYSlT2rB47dQ==} - engines: {node: '>8.0.0'} - peerDependencies: - react: ^15.0.0 || ^16.0.0 || ^17.0.0 || ^18.0.0 - react-dom: ^15.0.0 || ^16.0.0 || ^17.0.0 || ^18.0.0 - - react@18.3.1: - resolution: {integrity: sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ==} - engines: {node: '>=0.10.0'} - - read-pkg-up@7.0.1: - resolution: {integrity: sha512-zK0TB7Xd6JpCLmlLmufqykGE+/TlOePD6qKClNW7hHDKFh/J7/7gCWGR7joEQEW1bKq3a3yUZSObOoWLFQ4ohg==} - engines: {node: '>=8'} - - read-pkg@5.2.0: - resolution: {integrity: sha512-Ug69mNOpfvKDAc2Q8DRpMjjzdtrnv9HcSMX+4VsZxD1aZ6ZzrIE7rlzXBtWTyhULSMKg076AW6WR5iZpD0JiOg==} - engines: {node: '>=8'} - - readable-stream@1.0.34: - resolution: {integrity: sha512-ok1qVCJuRkNmvebYikljxJA/UEsKwLl2nI1OmaqAu4/UE+h0wKCHok4XkL/gvi39OacXvw59RJUOFUkDib2rHg==} - - readable-stream@2.3.8: - resolution: {integrity: sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==} - - readable-stream@3.6.2: - resolution: {integrity: sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==} - engines: {node: '>= 6'} - - readdirp@3.6.0: - resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==} - engines: {node: '>=8.10.0'} - - rechoir@0.6.2: - resolution: {integrity: sha512-HFM8rkZ+i3zrV+4LQjwQ0W+ez98pApMGM3HUrN04j3CqzPOzl9nmP15Y8YXNm8QHGv/eacOVEjqhmWpkRV0NAw==} - engines: {node: '>= 0.10'} - - rechoir@0.8.0: - resolution: {integrity: sha512-/vxpCXddiX8NGfGO/mTafwjq4aFa/71pvamip0++IQk3zG8cbCj0fifNPrjjF1XMXUne91jL9OoxmdykoEtifQ==} - engines: {node: '>= 10.13.0'} - - redux-thunk@2.4.2: - resolution: {integrity: sha512-+P3TjtnP0k/FEjcBL5FZpoovtvrTNT/UXd4/sluaSyrURlSlhLSzEdfsTBW7WsKB6yPvgd7q/iZPICFjW4o57Q==} - peerDependencies: - redux: ^4 - - redux@4.2.1: - resolution: {integrity: sha512-LAUYz4lc+Do8/g7aeRa8JkyDErK6ekstQaqWQrNRW//MY1TvCEpMtpTWvlQ+FPbWCx+Xixu/6SHt5N0HR+SB4w==} - - regenerator-runtime@0.14.1: - resolution: {integrity: sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==} - - regex-not@1.0.2: - resolution: {integrity: sha512-J6SDjUgDxQj5NusnOtdFxDwN/+HWykR8GELwctJ7mdqhcyy1xEc4SRFHUXvxTp661YaVKAjfRLZ9cCqS6tn32A==} - engines: {node: '>=0.10.0'} - - registry-auth-token@4.2.2: - resolution: {integrity: sha512-PC5ZysNb42zpFME6D/XlIgtNGdTl8bBOCw90xQLVMpzuuubJKYDWFAEuUNc+Cn8Z8724tg2SDhDRrkVEsqfDMg==} - engines: {node: '>=6.0.0'} - - registry-auth-token@5.0.2: - resolution: {integrity: sha512-o/3ikDxtXaA59BmZuZrJZDJv8NMDGSj+6j6XaeBmHw8eY1i1qd9+6H+LjVvQXx3HN6aRCGa1cUdJ9RaJZUugnQ==} - engines: {node: '>=14'} - - registry-url@5.1.0: - resolution: {integrity: sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==} - engines: {node: '>=8'} - - relateurl@0.2.7: - resolution: {integrity: sha512-G08Dxvm4iDN3MLM0EsP62EDV9IuhXPR6blNz6Utcp7zyV3tr4HVNINt6MpaRWbxoOHT3Q7YN2P+jaHX8vUbgog==} - engines: {node: '>= 0.10'} - - remove-accents@0.5.0: - resolution: {integrity: sha512-8g3/Otx1eJaVD12e31UbJj1YzdtVvzH85HV7t+9MJYk/u3XmkOUJ5Ys9wQrf9PCPK8+xn4ymzqYCiZl6QWKn+A==} - - renderkid@3.0.0: - resolution: {integrity: sha512-q/7VIQA8lmM1hF+jn+sFSPWGlMkSAeNYcPLmDQx2zzuiDfaLrOmumR8iaUKlenFgh0XRPIUeSPlH3A+AW3Z5pg==} - - repeat-element@1.1.4: - resolution: {integrity: sha512-LFiNfRcSu7KK3evMyYOuCzv3L10TW7yC1G2/+StMjK8Y6Vqd2MG7r/Qjw4ghtuCOjFvlnms/iMmLqpvW/ES/WQ==} - engines: {node: '>=0.10.0'} - - repeat-string@1.6.1: - resolution: {integrity: sha512-PV0dzCYDNfRi1jCDbJzpW7jNNDRuCOG/jI5ctQcGKt/clZD+YcPS3yIlWuTJMmESC8aevCFmWJy5wjAFgNqN6w==} - engines: {node: '>=0.10'} - - request-compose@2.1.6: - resolution: {integrity: sha512-S07L+2VbJB32WddD/o/PnYGKym63zLVbymygVWXvt8L79VAngcjAxhHaGuFOICLxEV90EasEPzqPKKHPspXP8w==} - engines: {node: '>=12.0.0'} - - request-ip@3.3.0: - resolution: {integrity: sha512-cA6Xh6e0fDBBBwH77SLJaJPBmD3nWVAcF9/XAcsrIHdjhFzFiB5aNQFytdjCGPezU3ROwrR11IddKAM08vohxA==} - - request-multipart@1.0.0: - resolution: {integrity: sha512-dazx88T19dIKFNc0XdlZV8H46D2RmNFdR4mipcbrFOaN70PSSSMM3urVY+eVbrpraf/fHXccxFhLvG1wkSUtKQ==} - engines: {node: '>=8.0.0'} - - request-oauth@1.0.1: - resolution: {integrity: sha512-85THTg1RgOYtqQw42JON6AqvHLptlj1biw265Tsq4fD4cPdUvhDB2Qh9NTv17yCD322ROuO9aOmpc4GyayGVBA==} - engines: {node: '>=8.0.0'} - - require-directory@2.1.1: - resolution: {integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==} - engines: {node: '>=0.10.0'} - - require-from-string@2.0.2: - resolution: {integrity: sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==} - engines: {node: '>=0.10.0'} - - reselect@4.1.8: - resolution: {integrity: sha512-ab9EmR80F/zQTMNeneUr4cv+jSwPJgIlvEmVwLerwrWVbpLlBuls9XHzIeTFy4cegU2NHBp3va0LKOzU5qFEYQ==} - - resolve-alpn@1.2.1: - resolution: {integrity: sha512-0a1F4l73/ZFZOakJnQ3FvkJ2+gSTQWz/r2KE5OdDY0TxPm5h4GkqkWWfM47T7HsbnOtcJVEF4epCVy6u7Q3K+g==} - - resolve-cwd@3.0.0: - resolution: {integrity: sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==} - engines: {node: '>=8'} - - resolve-dir@1.0.1: - resolution: {integrity: sha512-R7uiTjECzvOsWSfdM0QKFNBVFcK27aHOUwdvK53BcW8zqnGdYp0Fbj82cy54+2A4P2tFM22J5kRfe1R+lM/1yg==} - engines: {node: '>=0.10.0'} - - resolve-from@4.0.0: - resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==} - engines: {node: '>=4'} - - resolve-from@5.0.0: - resolution: {integrity: sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==} - engines: {node: '>=8'} - - resolve-path@1.4.0: - resolution: {integrity: sha512-i1xevIst/Qa+nA9olDxLWnLk8YZbi8R/7JPbCMcgyWaFR6bKWaexgJgEB5oc2PKMjYdrHynyz0NY+if+H98t1w==} - engines: {node: '>= 0.8'} - - resolve-pathname@3.0.0: - resolution: {integrity: sha512-C7rARubxI8bXFNB/hqcp/4iUeIXJhJZvFPFPiSPRnhU5UPxzMFIl+2E6yY6c4k9giDJAhtV+enfA+G89N6Csng==} - - resolve-url@0.2.1: - resolution: {integrity: sha512-ZuF55hVUQaaczgOIwqWzkEcEidmlD/xl44x1UZnhOXcYuFN2S6+rcxpG+C1N3So0wvNI3DmJICUFfu2SxhBmvg==} - deprecated: https://github.com/lydell/resolve-url#deprecated - - resolve@1.22.8: - resolution: {integrity: sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==} - hasBin: true - - responselike@2.0.1: - resolution: {integrity: sha512-4gl03wn3hj1HP3yzgdI7d3lCkF95F21Pz4BPGvKHinyQzALR5CapwC8yIi0Rh58DEMQ/SguC03wFj2k0M/mHhw==} - - restore-cursor@2.0.0: - resolution: {integrity: sha512-6IzJLuGi4+R14vwagDHX+JrXmPVtPpn4mffDJ1UdR7/Edm87fl6yi8mMBIVvFtJaNTUvjughmW4hwLhRG7gC1Q==} - engines: {node: '>=4'} - - restore-cursor@3.1.0: - resolution: {integrity: sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==} - engines: {node: '>=8'} - - ret@0.1.15: - resolution: {integrity: sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==} - engines: {node: '>=0.12'} - - reusify@1.0.4: - resolution: {integrity: sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==} - engines: {iojs: '>=1.0.0', node: '>=0.10.0'} - - rimraf@2.7.1: - resolution: {integrity: sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==} - deprecated: Rimraf versions prior to v4 are no longer supported - hasBin: true - - rimraf@3.0.2: - resolution: {integrity: sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==} - deprecated: Rimraf versions prior to v4 are no longer supported - hasBin: true - - rollup@4.21.2: - resolution: {integrity: sha512-e3TapAgYf9xjdLvKQCkQTnbTKd4a6jwlpQSJJFokHGaX2IVjoEqkIIhiQfqsi0cdwlOD+tQGuOd5AJkc5RngBw==} - engines: {node: '>=18.0.0', npm: '>=8.0.0'} - hasBin: true - - run-applescript@5.0.0: - resolution: {integrity: sha512-XcT5rBksx1QdIhlFOCtgZkB99ZEouFZ1E2Kc2LHqNW13U3/74YGdkQRmThTwxy4QIyookibDKYZOPqX//6BlAg==} - engines: {node: '>=12'} - - run-async@2.4.1: - resolution: {integrity: sha512-tvVnVv01b8c1RrA6Ep7JkStj85Guv/YrMcwqYQnwjsAS2cTmmPGBBjAjpCW7RrSodNSoE2/qg9O4bceNvUuDgQ==} - engines: {node: '>=0.12.0'} - - run-parallel@1.2.0: - resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} - - rxjs@6.6.7: - resolution: {integrity: sha512-hTdwr+7yYNIT5n4AMYp85KA6yw2Va0FLa3Rguvbpa4W3I5xynaBZo41cM3XM+4Q6fRMj3sBYIR1VAmZMXYJvRQ==} - engines: {npm: '>=2.0.0'} - - rxjs@7.8.1: - resolution: {integrity: sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg==} - - safe-buffer@5.1.2: - resolution: {integrity: sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==} - - safe-buffer@5.2.1: - resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} - - safe-regex@1.1.0: - resolution: {integrity: sha512-aJXcif4xnaNUzvUuC5gcb46oTS7zvg4jpMTnuqtrEPlR3vFr4pxtdTwaF1Qs3Enjn9HK+ZlwQui+a7z0SywIzg==} - - safe-stable-stringify@2.5.0: - resolution: {integrity: sha512-b3rppTKm9T+PsVCBEOUR46GWI7fdOs00VKZ1+9c1EWDaDMvjQc6tUwuFyIprgGgTcWoVHSKrU8H31ZHA2e0RHA==} - engines: {node: '>=10'} - - safer-buffer@2.1.2: - resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==} - - sanitize-html@2.13.0: - resolution: {integrity: sha512-Xff91Z+4Mz5QiNSLdLWwjgBDm5b1RU6xBT0+12rapjiaR7SwfRdjw8f+6Rir2MXKLrDicRFHdb51hGOAxmsUIA==} - - scheduler@0.23.2: - resolution: {integrity: sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==} - - schema-utils@3.3.0: - resolution: {integrity: sha512-pN/yOAvcC+5rQ5nERGuwrjLlYvLTbCibnZ1I7B1LaiAz9BRBlE9GMgE/eqV30P7aJQUf7Ddimy/RsbYO/GrVGg==} - engines: {node: '>= 10.13.0'} - - schema-utils@4.2.0: - resolution: {integrity: sha512-L0jRsrPpjdckP3oPug3/VxNKt2trR8TcabrM6FOAAlvC/9Phcmm+cuAgTlxBqdBR1WJx7Naj9WHw+aOmheSVbw==} - engines: {node: '>= 12.13.0'} - - scroll-into-view-if-needed@2.2.31: - resolution: {integrity: sha512-dGCXy99wZQivjmjIqihaBQNjryrz5rueJY7eHfTdyWEiR4ttYpsajb14rn9s5d4DY4EcY6+4+U/maARBXJedkA==} - - semver@5.7.2: - resolution: {integrity: sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==} - hasBin: true - - semver@6.3.1: - resolution: {integrity: sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==} - hasBin: true - - semver@7.5.4: - resolution: {integrity: sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==} - engines: {node: '>=10'} - hasBin: true - - semver@7.6.3: - resolution: {integrity: sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==} - engines: {node: '>=10'} - hasBin: true - - sendmail@1.6.1: - resolution: {integrity: sha512-lIhvnjSi5e5jL8wA1GPP6j2QVlx6JOEfmdn0QIfmuJdmXYGmJ375kcOU0NSm/34J+nypm4sa1AXrYE5w3uNIIA==} - engines: {node: '>=6.0.0'} - - sentence-case@2.1.1: - resolution: {integrity: sha512-ENl7cYHaK/Ktwk5OTD+aDbQ3uC8IByu/6Bkg+HDv8Mm+XnBnppVNalcfJTNsp1ibstKh030/JKQQWglDvtKwEQ==} - - serialize-javascript@6.0.2: - resolution: {integrity: sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g==} - - set-blocking@2.0.0: - resolution: {integrity: sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw==} - - set-function-length@1.2.2: - resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==} - engines: {node: '>= 0.4'} - - set-value@2.0.1: - resolution: {integrity: sha512-JxHc1weCN68wRY0fhCoXpyK55m/XPHafOmK4UWD7m2CI14GMcFypt4w/0+NV5f/ZMby2F6S2wwA7fgynh9gWSw==} - engines: {node: '>=0.10.0'} - - setprototypeof@1.1.0: - resolution: {integrity: sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ==} - - setprototypeof@1.2.0: - resolution: {integrity: sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==} - - shallowequal@1.1.0: - resolution: {integrity: sha512-y0m1JoUZSlPAjXVtPPW70aZWfIL/dSP7AFkRnniLCrK/8MDKog3TySTBmckD+RObVxH0v4Tox67+F14PdED2oQ==} - - sharp@0.32.6: - resolution: {integrity: sha512-KyLTWwgcR9Oe4d9HwCwNM2l7+J0dUQwn/yf7S0EnTtb0eVS4RxO0eUSvxPtzT4F3SY+C4K6fqdv/DO27sJ/v/w==} - engines: {node: '>=14.15.0'} - - sharp@0.33.5: - resolution: {integrity: sha512-haPVm1EkS9pgvHrQ/F3Xy+hgcuMV0Wm9vfIBSiwZ05k+xgb0PkBQpGsAA/oWdDobNaZTH5ppvHtzCFbnSEwHVw==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - - shebang-command@2.0.0: - resolution: {integrity: sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==} - engines: {node: '>=8'} - - shebang-regex@3.0.0: - resolution: {integrity: sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==} - engines: {node: '>=8'} - - shell-quote@1.8.1: - resolution: {integrity: sha512-6j1W9l1iAs/4xYBI1SYOVZyFcCis9b4KCLQ8fgAGG07QvzaRLVVRQvAy85yNmmZSjYjg4MWh4gNvlPujU/5LpA==} - - side-channel@1.0.6: - resolution: {integrity: sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==} - engines: {node: '>= 0.4'} - - sift@16.0.1: - resolution: {integrity: sha512-Wv6BjQ5zbhW7VFefWusVP33T/EM0vYikCaQ2qR8yULbsilAT8/wQaXvuQ3ptGLpoKx+lihJE3y2UTgKDyyNHZQ==} - - signal-exit@3.0.7: - resolution: {integrity: sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==} - - signal-exit@4.1.0: - resolution: {integrity: sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==} - engines: {node: '>=14'} - - simple-concat@1.0.1: - resolution: {integrity: sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==} - - simple-get@3.1.1: - resolution: {integrity: sha512-CQ5LTKGfCpvE1K0n2us+kuMPbk/q0EKl82s4aheV9oXjFEz6W/Y7oQFVJuU6QG77hRT4Ghb5RURteF5vnWjupA==} - - simple-get@4.0.1: - resolution: {integrity: sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA==} - - simple-swizzle@0.2.2: - resolution: {integrity: sha512-JA//kQgZtbuY83m+xT+tXJkmJncGMTFT+C+g2h2R9uxkYIrE2yy9sgmcLhCnw57/WSD+Eh3J97FPEDFnbXnDUg==} - - simple-update-notifier@2.0.0: - resolution: {integrity: sha512-a2B9Y0KlNXl9u/vsW6sTIu9vGEpfKu2wRV6l1H3XEas/0gUIzGzBoP/IouTcUQbm9JWZLH3COxyn03TYlFax6w==} - engines: {node: '>=10'} - - sirv@2.0.4: - resolution: {integrity: sha512-94Bdh3cC2PKrbgSOUqTiGPWVZeSiXfKOVZNJniWoqrWrRkB1CJzBU3NEbiTsPcYy1lDsANA/THzS+9WBiy5nfQ==} - engines: {node: '>= 10'} - - sisteransi@1.0.5: - resolution: {integrity: sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==} - - slash@3.0.0: - resolution: {integrity: sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==} - engines: {node: '>=8'} - - slash@4.0.0: - resolution: {integrity: sha512-3dOsAHXXUkQTpOYcoAxLIorMTp4gIQr5IW3iVb7A7lFIp0VHhnynm9izx6TssdrIcVIESAlVjtnO2K8bg+Coew==} - engines: {node: '>=12'} - - slate-history@0.93.0: - resolution: {integrity: sha512-Gr1GMGPipRuxIz41jD2/rbvzPj8eyar56TVMyJBvBeIpQSSjNISssvGNDYfJlSWM8eaRqf6DAcxMKzsLCYeX6g==} - peerDependencies: - slate: '>=0.65.3' - - slate-react@0.98.3: - resolution: {integrity: sha512-p1BnF9eRyRM0i5hkgOb11KgmpWLQm9Zyp6jVkOAj5fPdIGheKhg48Z7aWKrayeJ4nmRyi/NjRZz/io5hQcphmw==} - peerDependencies: - react: '>=16.8.0' - react-dom: '>=16.8.0' - slate: '>=0.65.3' - - slate@0.94.1: - resolution: {integrity: sha512-GH/yizXr1ceBoZ9P9uebIaHe3dC/g6Plpf9nlUwnvoyf6V1UOYrRwkabtOCd3ZfIGxomY4P7lfgLr7FPH8/BKA==} - - snake-case@2.1.0: - resolution: {integrity: sha512-FMR5YoPFwOLuh4rRz92dywJjyKYZNLpMn1R5ujVpIYkbA9p01fq8RMg0FkO4M+Yobt4MjHeLTJVm5xFFBHSV2Q==} - - snapdragon-node@2.1.1: - resolution: {integrity: sha512-O27l4xaMYt/RSQ5TR3vpWCAB5Kb/czIcqUFOM/C4fYcLnbZUc1PkjTAMjof2pBWaSTwOUd6qUHcFGVGj7aIwnw==} - engines: {node: '>=0.10.0'} - - snapdragon-util@3.0.1: - resolution: {integrity: sha512-mbKkMdQKsjX4BAL4bRYTj21edOf8cN7XHdYUJEe+Zn99hVEYcMvKPct1IqNe7+AZPirn8BCDOQBHQZknqmKlZQ==} - engines: {node: '>=0.10.0'} - - snapdragon@0.8.2: - resolution: {integrity: sha512-FtyOnWN/wCHTVXOMwvSv26d+ko5vWlIDD6zoUJ7LW8vh+ZBC8QdljveRP+crNrtBwioEUWy/4dMtbBjA4ioNlg==} - engines: {node: '>=0.10.0'} - - sort-object-keys@1.1.3: - resolution: {integrity: sha512-855pvK+VkU7PaKYPc+Jjnmt4EzejQHyhhF33q31qG8x7maDzkeFhAAThdCYay11CISO+qAMwjOBP+fPZe0IPyg==} - - sort-package-json@2.5.1: - resolution: {integrity: sha512-vx/KoZxm8YNMUqdlw7SGTfqR5pqZ/sUfgOuRtDILiOy/3AvzhAibyUe2cY3OpLs3oRSow9up4yLVtQaM24rbDQ==} - hasBin: true - - sorted-array-functions@1.3.0: - resolution: {integrity: sha512-2sqgzeFlid6N4Z2fUQ1cvFmTOLRi/sEDzSQ0OKYchqgoPmQBVyM3959qYx3fpS6Esef80KjmpgPeEr028dP3OA==} - - source-list-map@2.0.1: - resolution: {integrity: sha512-qnQ7gVMxGNxsiL4lEuJwe/To8UnK7fAnmbGEEH8RpLouuKbeEm0lhbQVFIrNSuB+G7tVrAlVsZgETT5nljf+Iw==} - - source-map-js@1.2.0: - resolution: {integrity: sha512-itJW8lvSA0TXEphiRoawsCksnlf8SyvmFzIhltqAHluXd88pkCd+cXJVHTDwdCr0IzwptSm035IHQktUu1QUMg==} - engines: {node: '>=0.10.0'} - - source-map-resolve@0.5.3: - resolution: {integrity: sha512-Htz+RnsXWk5+P2slx5Jh3Q66vhQj1Cllm0zvnaY98+NFx+Dv2CF/f5O/t8x+KaNdrdIAsruNzoh/KpialbqAnw==} - deprecated: See https://github.com/lydell/source-map-resolve#deprecated - - source-map-support@0.5.21: - resolution: {integrity: sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==} - - source-map-url@0.4.1: - resolution: {integrity: sha512-cPiFOTLUKvJFIg4SKVScy4ilPPW6rFgMgfuZJPNoDuMs3nC1HbMUycBoJw77xFIp6z1UJQJOfx6C9GMH80DiTw==} - deprecated: See https://github.com/lydell/source-map-url#deprecated - - source-map@0.5.7: - resolution: {integrity: sha512-LbrmJOMUSdEVxIKvdcJzQC+nQhe8FUZQTXQy6+I75skNgn3OoQ0DZA8YnFa7gp8tqtL3KPf1kmo0R5DoApeSGQ==} - engines: {node: '>=0.10.0'} - - source-map@0.6.1: - resolution: {integrity: sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==} - engines: {node: '>=0.10.0'} - - source-map@0.7.4: - resolution: {integrity: sha512-l3BikUxvPOcn5E74dZiq5BGsTb5yEwhaTSzccU6t4sDOH8NWJCstKO5QT2CvtFoK6F0saL7p9xHAqHOlCPJygA==} - engines: {node: '>= 8'} - - spawn-command@0.0.2: - resolution: {integrity: sha512-zC8zGoGkmc8J9ndvml8Xksr1Amk9qBujgbF0JAIWO7kXr43w0h/0GJNM/Vustixu+YE8N/MTrQ7N31FvHUACxQ==} - - spdx-correct@3.2.0: - resolution: {integrity: sha512-kN9dJbvnySHULIluDHy32WHRUu3Og7B9sbY7tsFLctQkIqnMh3hErYgdMjTYuqmcXX+lK5T1lnUt3G7zNswmZA==} - - spdx-exceptions@2.5.0: - resolution: {integrity: sha512-PiU42r+xO4UbUS1buo3LPJkjlO7430Xn5SVAhdpzzsPHsjbYVflnnFdATgabnLude+Cqu25p6N+g2lw/PFsa4w==} - - spdx-expression-parse@3.0.1: - resolution: {integrity: sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==} - - spdx-license-ids@3.0.20: - resolution: {integrity: sha512-jg25NiDV/1fLtSgEgyvVyDunvaNHbuwF9lfNV17gSmPFAlYzdfNBlLtLzXTevwkPj7DhGbmN9VnmJIgLnhvaBw==} - - speedometer@1.0.0: - resolution: {integrity: sha512-lgxErLl/7A5+vgIIXsh9MbeukOaCb2axgQ+bKCdIE+ibNT4XNYGNCR1qFEGq6F+YDASXK3Fh/c5FgtZchFolxw==} - - split-string@3.1.0: - resolution: {integrity: sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==} - engines: {node: '>=0.10.0'} - - split2@4.2.0: - resolution: {integrity: sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==} - engines: {node: '>= 10.x'} - - sprintf-js@1.0.3: - resolution: {integrity: sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==} - - stack-trace@0.0.10: - resolution: {integrity: sha512-KGzahc7puUKkzyMt+IqAep+TVNbKP+k2Lmwhub39m1AsTSkaDutx56aDCo+HLDzf/D26BIHTJWNiTG1KAJiQCg==} - - stackframe@1.3.4: - resolution: {integrity: sha512-oeVtt7eWQS+Na6F//S4kJ2K2VbRlS9D43mAlMyVpVWovy9o+jfgH8O9agzANzaiLjclA0oYzUXEM4PurhSUChw==} - - static-extend@0.1.2: - resolution: {integrity: sha512-72E9+uLc27Mt718pMHt9VMNiAL4LMsmDbBva8mxWUCkT07fSzEGMYUCk0XWY6lp0j6RBAG4cJ3mWuZv2OE3s0g==} - engines: {node: '>=0.10.0'} - - statuses@1.5.0: - resolution: {integrity: sha512-OpZ3zP+jT1PI7I8nemJX4AKmAX070ZkYPVWV/AaKTJl+tXCTGyVdC1a4SL8RUQYEwk/f34ZX8UTykN68FwrqAA==} - engines: {node: '>= 0.6'} - - statuses@2.0.1: - resolution: {integrity: sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==} - engines: {node: '>= 0.8'} - - strapi-plugin-fuzzy-search@3.0.0: - resolution: {integrity: sha512-8qtgfHbpDfAzjGU4RnE63a19+NGtgtpDIow5gEMdgN4vkG/YlhaE4ffenla67iaCuQGuNM2qsgwaEgdq3qxFOg==} - engines: {node: '>=18.x.x <=20.x.x', npm: '>=6.0.0'} - peerDependencies: - '@strapi/strapi': ^4.25.0 - '@strapi/utils': ^4.25.0 - yup: ^0.32.9 - - stream-chain@2.2.5: - resolution: {integrity: sha512-1TJmBx6aSWqZ4tx7aTpBDXK0/e2hhcNSTV8+CbFJtDjbb+I1mZ8lHit0Grw9GRT+6JbIrrDd8esncgBi8aBXGA==} - - stream-json@1.8.0: - resolution: {integrity: sha512-HZfXngYHUAr1exT4fxlbc1IOce1RYxp2ldeaf97LYCOPSoOqY/1Psp7iGvpb+6JIOgkra9zDYnPX01hGAHzEPw==} - - stream-slice@0.1.2: - resolution: {integrity: sha512-QzQxpoacatkreL6jsxnVb7X5R/pGw9OUv2qWTYWnmLpg4NdN31snPy/f3TdQE1ZUXaThRvj1Zw4/OGg0ZkaLMA==} - - streamx@2.20.0: - resolution: {integrity: sha512-ZGd1LhDeGFucr1CUCTBOS58ZhEendd0ttpGT3usTvosS4ntIwKN9LJFp+OeCSprsCPL14BXVRZlHGRY1V9PVzQ==} - - string-argv@0.3.2: - resolution: {integrity: sha512-aqD2Q0144Z+/RqG52NeHEkZauTAUWJO8c6yTftGJKO3Tja5tUgIfmIl6kExvhtxSDP7fXB6DvzkfMpCd/F3G+Q==} - engines: {node: '>=0.6.19'} - - string-width@4.2.3: - resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} - engines: {node: '>=8'} - - string-width@5.1.2: - resolution: {integrity: sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==} - engines: {node: '>=12'} - - string_decoder@0.10.31: - resolution: {integrity: sha512-ev2QzSzWPYmy9GuqfIVildA4OdcGLeFZQrq5ys6RtiuF+RQQiZWr8TZNyAcuVXyQRYfEO+MsoB/1BuQVhOJuoQ==} - - string_decoder@1.1.1: - resolution: {integrity: sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==} - - string_decoder@1.3.0: - resolution: {integrity: sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==} - - strip-ansi@3.0.1: - resolution: {integrity: sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==} - engines: {node: '>=0.10.0'} - - strip-ansi@5.2.0: - resolution: {integrity: sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==} - engines: {node: '>=6'} - - strip-ansi@6.0.1: - resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} - engines: {node: '>=8'} - - strip-ansi@7.1.0: - resolution: {integrity: sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==} - engines: {node: '>=12'} - - strip-final-newline@2.0.0: - resolution: {integrity: sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==} - engines: {node: '>=6'} - - strip-final-newline@3.0.0: - resolution: {integrity: sha512-dOESqjYr96iWYylGObzd39EuNTa5VJxyvVAEm5Jnh7KGo75V43Hk1odPQkNDyXNmUR6k+gEiDVXnjB8HJ3crXw==} - engines: {node: '>=12'} - - strip-json-comments@2.0.1: - resolution: {integrity: sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==} - engines: {node: '>=0.10.0'} - - strnum@1.0.5: - resolution: {integrity: sha512-J8bbNyKKXl5qYcR36TIO8W3mVGVHrmmxsd5PAItGkmyzwJvybiw2IVq5nqd0i4LSNSkB/sx9VHllbfFdr9k1JA==} - - style-loader@3.3.4: - resolution: {integrity: sha512-0WqXzrsMTyb8yjZJHDqwmnwRJvhALK9LfRtRc6B4UTWe8AijYLZYZ9thuJTZc2VfQWINADW/j+LiJnfy2RoC1w==} - engines: {node: '>= 12.13.0'} - peerDependencies: - webpack: ^5.0.0 - - style-mod@4.1.2: - resolution: {integrity: sha512-wnD1HyVqpJUI2+eKZ+eo1UwghftP6yuFheBqqe+bWCotBjC2K1YnteJILRMs3SM4V/0dLEW1SC27MWP5y+mwmw==} - - styled-components@6.1.13: - resolution: {integrity: sha512-M0+N2xSnAtwcVAQeFEsGWFFxXDftHUD7XrKla06QbpUMmbmtFBMMTcKWvFXtWxuD5qQkB8iU5gk6QASlx2ZRMw==} - engines: {node: '>= 16'} - peerDependencies: - react: '>= 16.8.0' - react-dom: '>= 16.8.0' - - stylis@4.2.0: - resolution: {integrity: sha512-Orov6g6BB1sDfYgzWfTHDOxamtX1bE/zo104Dh9e6fqJ3PooipYyfJ0pUmrZO2wAvO8YbEyeFrkV91XTsGMSrw==} - - stylis@4.3.2: - resolution: {integrity: sha512-bhtUjWd/z6ltJiQwg0dUfxEJ+W+jdqQd8TbWLWyeIJHlnsqmGLRFFd8e5mA0AZi/zx90smXRlN66YMTcaSFifg==} - - supports-color@2.0.0: - resolution: {integrity: sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g==} - engines: {node: '>=0.8.0'} - - supports-color@5.5.0: - resolution: {integrity: sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==} - engines: {node: '>=4'} - - supports-color@7.2.0: - resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==} - engines: {node: '>=8'} - - supports-color@8.1.1: - resolution: {integrity: sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==} - engines: {node: '>=10'} - - supports-preserve-symlinks-flag@1.0.0: - resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==} - engines: {node: '>= 0.4'} - - swap-case@1.1.2: - resolution: {integrity: sha512-BAmWG6/bx8syfc6qXPprof3Mn5vQgf5dwdUNJhsNqU9WdPt5P+ES/wQ5bxfijy8zwZgZZHslC3iAsxsuQMCzJQ==} - - synckit@0.8.5: - resolution: {integrity: sha512-L1dapNV6vu2s/4Sputv8xGsCdAVlb5nRDMFU/E27D44l5U6cw1g0dGd45uLc+OXjNMmF4ntiMdCimzcjFKQI8Q==} - engines: {node: ^14.18.0 || >=16.0.0} - - tapable@2.2.1: - resolution: {integrity: sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ==} - engines: {node: '>=6'} - - tar-fs@2.1.1: - resolution: {integrity: sha512-V0r2Y9scmbDRLCNex/+hYzvp/zyYjvFbHPNgVTKfQvVrb6guiE/fxP+XblDNR011utopbkex2nM4dHNV6GDsng==} - - tar-fs@3.0.6: - resolution: {integrity: sha512-iokBDQQkUyeXhgPYaZxmczGPhnhXZ0CmrqI+MOb/WFGS9DW5wnfrLgtjUJBvz50vQ3qfRwJ62QVoCFu8mPVu5w==} - - tar-stream@2.2.0: - resolution: {integrity: sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==} - engines: {node: '>=6'} - - tar-stream@3.1.7: - resolution: {integrity: sha512-qJj60CXt7IU1Ffyc3NJMjh6EkuCFej46zUqJ4J7pqYlThyd9bO0XBTmcOIhSzZJVWfsLks0+nle/j538YAW9RQ==} - - tar@6.1.13: - resolution: {integrity: sha512-jdIBIN6LTIe2jqzay/2vtYLlBHa3JF42ot3h1dW8Q0PaAG4v8rm0cvpVePtau5C6OKXGGcgO9q2AMNSWxiLqKw==} - engines: {node: '>=10'} - - tar@6.2.1: - resolution: {integrity: sha512-DZ4yORTwrbTj/7MZYq2w+/ZFdI6OZ/f9SFHR+71gIVUZhOQPHzVCLpvRnPgyaMpfWxxk/4ONva3GQSyNIKRv6A==} - engines: {node: '>=10'} - - tarn@3.0.2: - resolution: {integrity: sha512-51LAVKUSZSVfI05vjPESNc5vwqqZpbXCsU+/+wxlOrUjk2SnFTt97v9ZgQrD4YmxYW1Px6w2KjaDitCfkvgxMQ==} - engines: {node: '>=8.0.0'} - - terser-webpack-plugin@5.3.10: - resolution: {integrity: sha512-BKFPWlPDndPs+NGGCr1U59t0XScL5317Y0UReNrHaw9/FwhPENlq6bfgs+4yPfyP51vqC1bQ4rp1EfXW5ZSH9w==} - engines: {node: '>= 10.13.0'} - peerDependencies: - '@swc/core': '*' - esbuild: '*' - uglify-js: '*' - webpack: ^5.1.0 - peerDependenciesMeta: - '@swc/core': - optional: true - esbuild: - optional: true - uglify-js: - optional: true - - terser@5.31.6: - resolution: {integrity: sha512-PQ4DAriWzKj+qgehQ7LK5bQqCFNMmlhjR2PFFLuqGCpuCAauxemVBWwWOxo3UIwWQx8+Pr61Df++r76wDmkQBg==} - engines: {node: '>=10'} - hasBin: true - - text-decoder@1.1.1: - resolution: {integrity: sha512-8zll7REEv4GDD3x4/0pW+ppIxSNs7H1J10IKFZsuOMscumCdM2a+toDGLPA3T+1+fLBql4zbt5z83GEQGGV5VA==} - - text-hex@1.0.0: - resolution: {integrity: sha512-uuVGNWzgJ4yhRaNSiubPY7OjISw4sw4E5Uv0wbjp+OzcbmVU/rsT8ujgcXJhn9ypzsgr5vlzpPqP+MBBKcGvbg==} - - thenify-all@1.6.0: - resolution: {integrity: sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==} - engines: {node: '>=0.8'} - - thenify@3.3.1: - resolution: {integrity: sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==} - - through2@2.0.5: - resolution: {integrity: sha512-/mrRod8xqpA+IHSLyGCQ2s8SPHiCDEeQJSep1jqLYeEUClOFG2Qsh+4FU6G9VeqpZnGW/Su8LQGc4YKni5rYSQ==} - - through@2.3.8: - resolution: {integrity: sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg==} - - tildify@2.0.0: - resolution: {integrity: sha512-Cc+OraorugtXNfs50hU9KS369rFXCfgGLpfCfvlc+Ud5u6VWmUQsOAa9HbTvheQdYnrdJqqv1e5oIqXppMYnSw==} - engines: {node: '>=8'} - - tiny-invariant@1.0.6: - resolution: {integrity: sha512-FOyLWWVjG+aC0UqG76V53yAWdXfH8bO6FNmyZOuUrzDzK8DI3/JRY25UD7+g49JWM1LXwymsKERB+DzI0dTEQA==} - - tiny-invariant@1.3.3: - resolution: {integrity: sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg==} - - tiny-warning@1.0.3: - resolution: {integrity: sha512-lBN9zLN/oAf68o3zNXYrdCt1kP8WsiGW8Oo2ka41b2IM5JL/S1CTyX1rW0mb/zSuJun0ZUrDxx4sqvYS2FWzPA==} - - title-case@2.1.1: - resolution: {integrity: sha512-EkJoZ2O3zdCz3zJsYCsxyq2OC5hrxR9mfdd5I+w8h/tmFfeOxJ+vvkxsKxdmN0WtS9zLdHEgfgVOiMVgv+Po4Q==} - - titleize@3.0.0: - resolution: {integrity: sha512-KxVu8EYHDPBdUYdKZdKtU2aj2XfEx9AfjXxE/Aj0vT06w2icA09Vus1rh6eSu1y01akYg6BjIK/hxyLJINoMLQ==} - engines: {node: '>=12'} - - tmp@0.0.33: - resolution: {integrity: sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==} - engines: {node: '>=0.6.0'} - - to-fast-properties@2.0.0: - resolution: {integrity: sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==} - engines: {node: '>=4'} - - to-object-path@0.3.0: - resolution: {integrity: sha512-9mWHdnGRuh3onocaHzukyvCZhzvr6tiflAy/JRFXcJX0TjgfWA9pk9t8CMbzmBE4Jfw58pXbkngtBtqYxzNEyg==} - engines: {node: '>=0.10.0'} - - to-regex-range@2.1.1: - resolution: {integrity: sha512-ZZWNfCjUokXXDGXFpZehJIkZqq91BcULFq/Pi7M5i4JnxXdhMKAK682z8bCW3o8Hj1wuuzoKcW3DfVzaP6VuNg==} - engines: {node: '>=0.10.0'} - - to-regex-range@5.0.1: - resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} - engines: {node: '>=8.0'} - - to-regex@3.0.2: - resolution: {integrity: sha512-FWtleNAtZ/Ki2qtqej2CXTOayOH9bHDQF+Q48VpWyDXjbYxA4Yz8iDB31zXOBUlOHHKidDbqGVrTUvQMPmBGBw==} - engines: {node: '>=0.10.0'} - - toidentifier@1.0.1: - resolution: {integrity: sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==} - engines: {node: '>=0.6'} - - toposort@2.0.2: - resolution: {integrity: sha512-0a5EOkAUp8D4moMi2W8ZF8jcga7BgZd91O/yabJCFY8az+XSzeGyTKs0Aoo897iV1Nj6guFq8orWDS96z91oGg==} - - totalist@3.0.1: - resolution: {integrity: sha512-sf4i37nQ2LBx4m3wB74y+ubopq6W/dIzXg0FDGjsYnZHVa1Da8FH853wlL2gtUhg+xJXjfk3kUZS3BRoQeoQBQ==} - engines: {node: '>=6'} - - touch@3.1.1: - resolution: {integrity: sha512-r0eojU4bI8MnHr8c5bNo7lJDdI2qXlWWJk6a9EAFG7vbhTjElYhBVS3/miuE0uOuoLdb8Mc/rVfsmm6eo5o9GA==} - hasBin: true - - tr46@0.0.3: - resolution: {integrity: sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==} - - transliteration@2.3.5: - resolution: {integrity: sha512-HAGI4Lq4Q9dZ3Utu2phaWgtm3vB6PkLUFqWAScg/UW+1eZ/Tg6Exo4oC0/3VUol/w4BlefLhUUSVBr/9/ZGQOw==} - engines: {node: '>=6.0.0'} - hasBin: true - - tree-kill@1.2.2: - resolution: {integrity: sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==} - hasBin: true - - triple-beam@1.4.1: - resolution: {integrity: sha512-aZbgViZrg1QNcG+LULa7nhZpJTZSLm/mXnHXnbAbjmN5aSa0y7V+wvv6+4WaBtpISJzThKy+PIPxc1Nq1EJ9mg==} - engines: {node: '>= 14.0.0'} - - tslib@1.14.1: - resolution: {integrity: sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==} - - tslib@2.6.2: - resolution: {integrity: sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==} - - tslib@2.7.0: - resolution: {integrity: sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA==} - - tsscmp@1.0.6: - resolution: {integrity: sha512-LxhtAkPDTkVCMQjt2h6eBVY28KCjikZqZfMcC15YBeNjkgUpdCfBu5HoiOTDu86v6smE8yOjyEktJ8hlbANHQA==} - engines: {node: '>=0.6.x'} - - tunnel-agent@0.6.0: - resolution: {integrity: sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==} - - type-fest@0.20.2: - resolution: {integrity: sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==} - engines: {node: '>=10'} - - type-fest@0.21.3: - resolution: {integrity: sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==} - engines: {node: '>=10'} - - type-fest@0.6.0: - resolution: {integrity: sha512-q+MB8nYR1KDLrgr4G5yemftpMC7/QLqVndBmEEdqzmNj5dcFOO4Oo8qlwZE3ULT3+Zim1F8Kq4cBnikNhlCMlg==} - engines: {node: '>=8'} - - type-fest@0.8.1: - resolution: {integrity: sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==} - engines: {node: '>=8'} - - type-fest@2.19.0: - resolution: {integrity: sha512-RAH822pAdBgcNMAfWnCBU3CFZcfZ/i1eZjwFU/dsLKumyuuP3niueg2UAukXYF0E2AAoc82ZSSf9J0WQBinzHA==} - engines: {node: '>=12.20'} - - type-is@1.6.18: - resolution: {integrity: sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==} - engines: {node: '>= 0.6'} - - typedarray-to-buffer@3.1.5: - resolution: {integrity: sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==} - - typescript@5.2.2: - resolution: {integrity: sha512-mI4WrpHsbCIcwT9cF4FZvr80QUeKvsUsUvKDoR+X/7XHQH98xYD8YHZg7ANtz2GtZt/CBq2QJ0thkGJMHfqc1w==} - engines: {node: '>=14.17'} - hasBin: true - - uc.micro@1.0.6: - resolution: {integrity: sha512-8Y75pvTYkLJW2hWQHXxoqRgV7qb9B+9vFEtidML+7koHUFapnVJAZ6cKs+Qjz5Aw3aZWHMC6u0wJE3At+nSGwA==} - - uglify-js@3.19.3: - resolution: {integrity: sha512-v3Xu+yuwBXisp6QYTcH4UbH+xYJXqnq2m/LtQVWKWzYc1iehYnLixoQDN9FH6/j9/oybfd6W9Ghwkl8+UMKTKQ==} - engines: {node: '>=0.8.0'} - hasBin: true - - umzug@3.2.1: - resolution: {integrity: sha512-XyWQowvP9CKZycKc/Zg9SYWrAWX/gJCE799AUTFqk8yC3tp44K1xWr3LoFF0MNEjClKOo1suCr5ASnoy+KltdA==} - engines: {node: '>=12'} - - unc-path-regex@0.1.2: - resolution: {integrity: sha512-eXL4nmJT7oCpkZsHZUOJo8hcX3GbsiDOa0Qu9F646fi8dT3XuSVopVqAcEiVzSKKH7UoDti23wNX3qGFxcW5Qg==} - engines: {node: '>=0.10.0'} - - undefsafe@2.0.5: - resolution: {integrity: sha512-WxONCrssBM8TSPRqN5EmsjVrsv4A8X12J4ArBiiayv3DyyG3ZlIg6yysuuSYdZsVz3TKcTg2fd//Ujd4CHV1iA==} - - undici-types@5.26.5: - resolution: {integrity: sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==} - - undici-types@6.19.8: - resolution: {integrity: sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==} - - union-value@1.0.1: - resolution: {integrity: sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==} - engines: {node: '>=0.10.0'} - - unique-string@2.0.0: - resolution: {integrity: sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==} - engines: {node: '>=8'} - - universalify@0.1.2: - resolution: {integrity: sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==} - engines: {node: '>= 4.0.0'} - - universalify@2.0.1: - resolution: {integrity: sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw==} - engines: {node: '>= 10.0.0'} - - unload@2.2.0: - resolution: {integrity: sha512-B60uB5TNBLtN6/LsgAf3udH9saB5p7gqJwcFfbOEZ8BcBHnGwCf6G/TGiEqkRAxX7zAFIUtzdrXQSdL3Q/wqNA==} - - unpipe@1.0.0: - resolution: {integrity: sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==} - engines: {node: '>= 0.8'} - - unset-value@1.0.0: - resolution: {integrity: sha512-PcA2tsuGSF9cnySLHTLSh2qrQiJ70mn+r+Glzxv2TWZblxsxCC52BDlZoPCsz7STd9pN7EZetkWZBAvk4cgZdQ==} - engines: {node: '>=0.10.0'} - - untildify@4.0.0: - resolution: {integrity: sha512-KK8xQ1mkzZeg9inewmFVDNkg3l5LUhoq9kN6iWYB/CC9YMG8HA+c1Q8HwDe6dEX7kErrEVNVBO3fWsVq5iDgtw==} - engines: {node: '>=8'} - - update-browserslist-db@1.1.0: - resolution: {integrity: sha512-EdRAaAyk2cUE1wOf2DkEhzxqOQvFOoRJFNS6NeyJ01Gp2beMRpBAINjM2iDXE3KCuKhwnvHIQCJm6ThL2Z+HzQ==} - hasBin: true - peerDependencies: - browserslist: '>= 4.21.0' - - upper-case-first@1.1.2: - resolution: {integrity: sha512-wINKYvI3Db8dtjikdAqoBbZoP6Q+PZUyfMR7pmwHzjC2quzSkUq5DmPrTtPEqHaz8AGtmsB4TqwapMTM1QAQOQ==} - - upper-case@1.1.3: - resolution: {integrity: sha512-WRbjgmYzgXkCV7zNVpy5YgrHgbBv126rMALQQMrmzOVC4GM2waQ9x7xtm8VU+1yF2kWyPzI9zbZ48n4vSxwfSA==} - - uri-js@4.4.1: - resolution: {integrity: sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==} - - urix@0.1.0: - resolution: {integrity: sha512-Am1ousAhSLBeB9cG/7k7r2R0zj50uDRlZHPGbazid5s9rlF1F/QKYObEKSIunSjIOkJZqwRRLpvewjEkM7pSqg==} - deprecated: Please see https://github.com/lydell/urix#deprecated - - url-join@4.0.1: - resolution: {integrity: sha512-jk1+QP6ZJqyOiuEI9AEWQfju/nB2Pw466kbA0LEZljHwKeMgd9WrAEgEGxjPDD2+TNbbb37rTyhEfrCXfuKXnA==} - - use-callback-ref@1.3.2: - resolution: {integrity: sha512-elOQwe6Q8gqZgDA8mrh44qRTQqpIHDcZ3hXTLjBe1i4ph8XpNJnO+aQf3NaG+lriLopI4HMx9VjQLfPQ6vhnoA==} - engines: {node: '>=10'} - peerDependencies: - '@types/react': ^16.8.0 || ^17.0.0 || ^18.0.0 - react: ^16.8.0 || ^17.0.0 || ^18.0.0 - peerDependenciesMeta: - '@types/react': - optional: true - - use-isomorphic-layout-effect@1.1.2: - resolution: {integrity: sha512-49L8yCO3iGT/ZF9QttjwLF/ZD9Iwto5LnH5LmEdk/6cFmXddqi2ulF0edxTwjj+7mqvpVVGQWvbXZdn32wRSHA==} - peerDependencies: - '@types/react': '*' - react: ^16.8.0 || ^17.0.0 || ^18.0.0 - peerDependenciesMeta: - '@types/react': - optional: true - - use-sidecar@1.1.2: - resolution: {integrity: sha512-epTbsLuzZ7lPClpz2TyryBfztm7m+28DlEv2ZCQ3MDr5ssiwyOwGH/e5F9CkfWjJ1t4clvI58yF822/GUkjjhw==} - engines: {node: '>=10'} - peerDependencies: - '@types/react': ^16.9.0 || ^17.0.0 || ^18.0.0 - react: ^16.8.0 || ^17.0.0 || ^18.0.0 - peerDependenciesMeta: - '@types/react': - optional: true - - use-sync-external-store@1.2.2: - resolution: {integrity: sha512-PElTlVMwpblvbNqQ82d2n6RjStvdSoNe9FG28kNfz3WiXilJm4DdNkEzRhCZuIDwY8U08WVihhGR5iRqAwfDiw==} - peerDependencies: - react: ^16.8.0 || ^17.0.0 || ^18.0.0 - - use@3.1.1: - resolution: {integrity: sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ==} - engines: {node: '>=0.10.0'} - - user-home@1.1.1: - resolution: {integrity: sha512-aggiKfEEubv3UwRNqTzLInZpAOmKzwdHqEBmW/hBA/mt99eg+b4VrX6i+IRLxU8+WJYfa33rGwRseg4eElUgsQ==} - engines: {node: '>=0.10.0'} - hasBin: true - - util-deprecate@1.0.2: - resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==} - - utila@0.4.0: - resolution: {integrity: sha512-Z0DbgELS9/L/75wZbro8xAnT50pBVFQZ+hUEueGDU5FN51YSCYM+jdxsfCiHjwNP/4LCDD0i/graKpeBnOXKRA==} - - utils-merge@1.0.1: - resolution: {integrity: sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==} - engines: {node: '>= 0.4.0'} - - uuid@8.3.2: - resolution: {integrity: sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==} - hasBin: true - - uuid@9.0.1: - resolution: {integrity: sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==} - hasBin: true - - v8flags@2.1.1: - resolution: {integrity: sha512-SKfhk/LlaXzvtowJabLZwD4K6SGRYeoxA7KJeISlUMAB/NT4CBkZjMq3WceX2Ckm4llwqYVo8TICgsDYCBU2tA==} - engines: {node: '>= 0.10.0'} - - validate-npm-package-license@3.0.4: - resolution: {integrity: sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==} - - value-equal@1.0.1: - resolution: {integrity: sha512-NOJ6JZCAWr0zlxZt+xqCHNTEKOsrks2HQd4MqhP1qy4z1SkbEP467eNx6TgDKXMvUOb+OENfJCZwM+16n7fRfw==} - - vary@1.1.2: - resolution: {integrity: sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==} - engines: {node: '>= 0.8'} - - vite@5.0.13: - resolution: {integrity: sha512-/9ovhv2M2dGTuA+dY93B9trfyWMDRQw2jdVBhHNP6wr0oF34wG2i/N55801iZIpgUpnHDm4F/FabGQLyc+eOgg==} - engines: {node: ^18.0.0 || >=20.0.0} - hasBin: true - peerDependencies: - '@types/node': ^18.0.0 || >=20.0.0 - less: '*' - lightningcss: ^1.21.0 - sass: '*' - stylus: '*' - sugarss: '*' - terser: ^5.4.0 - peerDependenciesMeta: - '@types/node': - optional: true - less: - optional: true - lightningcss: - optional: true - sass: - optional: true - stylus: - optional: true - sugarss: - optional: true - terser: - optional: true - - w3c-keyname@2.2.8: - resolution: {integrity: sha512-dpojBhNsCNN7T82Tm7k26A6G9ML3NkhDsnw9n/eoxSRlVBB4CEtIQ/KTCLI2Fwf3ataSXRhYFkQi3SlnFwPvPQ==} - - watchpack@2.4.2: - resolution: {integrity: sha512-TnbFSbcOCcDgjZ4piURLCbJ3nJhznVh9kw6F6iokjiFPl8ONxe9A6nMDVXDiNbrSfLILs6vB07F7wLBrwPYzJw==} - engines: {node: '>=10.13.0'} - - wcwidth@1.0.1: - resolution: {integrity: sha512-XHPEwS0q6TaxcvG85+8EYkbiCux2XtWG2mkc47Ng2A77BQu9+DqIOJldST4HgPkuea7dvKSj5VgX3P1d4rW8Tg==} - - web-streams-polyfill@3.3.3: - resolution: {integrity: sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw==} - engines: {node: '>= 8'} - - web-streams-polyfill@4.0.0-beta.3: - resolution: {integrity: sha512-QW95TCTaHmsYfHDybGMwO5IJIM93I/6vTRk+daHTWFPhwh+C8Cg7j7XyKrwrj8Ib6vYXe0ocYNrmzY4xAAN6ug==} - engines: {node: '>= 14'} - - webidl-conversions@3.0.1: - resolution: {integrity: sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==} - - webpack-bundle-analyzer@4.10.2: - resolution: {integrity: sha512-vJptkMm9pk5si4Bv922ZbKLV8UTT4zib4FPgXMhgzUny0bfDDkLXAVQs3ly3fS4/TN9ROFtb0NFrm04UXFE/Vw==} - engines: {node: '>= 10.13.0'} - hasBin: true - - webpack-dev-middleware@6.1.2: - resolution: {integrity: sha512-Wu+EHmX326YPYUpQLKmKbTyZZJIB8/n6R09pTmB03kJmnMsVPTo9COzHZFr01txwaCAuZvfBJE4ZCHRcKs5JaQ==} - engines: {node: '>= 14.15.0'} - peerDependencies: - webpack: ^5.0.0 - peerDependenciesMeta: - webpack: - optional: true - - webpack-hot-middleware@2.26.0: - resolution: {integrity: sha512-okzjec5sAEy4t+7rzdT8eRyxsk0FDSmBPN2KwX4Qd+6+oQCfe5Ve07+u7cJvofgB+B4w5/4dO4Pz0jhhHyyPLQ==} - - webpack-sources@1.4.3: - resolution: {integrity: sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==} - - webpack-sources@3.2.3: - resolution: {integrity: sha512-/DyMEOrDgLKKIG0fmvtz+4dUX/3Ghozwgm6iPp8KRhvn+eQf9+Q7GWxVNMk3+uCPWfdXYC4ExGBckIXdFEfH1w==} - engines: {node: '>=10.13.0'} - - webpack@5.94.0: - resolution: {integrity: sha512-KcsGn50VT+06JH/iunZJedYGUJS5FGjow8wb9c0v5n1Om8O1g4L6LjtfxwlXIATopoQu+vOXXa7gYisWxCoPyg==} - engines: {node: '>=10.13.0'} - hasBin: true - peerDependencies: - webpack-cli: '*' - peerDependenciesMeta: - webpack-cli: - optional: true - - whatwg-url@5.0.0: - resolution: {integrity: sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==} - - which@1.3.1: - resolution: {integrity: sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==} - hasBin: true - - which@2.0.2: - resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==} - engines: {node: '>= 8'} - hasBin: true - - wide-align@1.1.5: - resolution: {integrity: sha512-eDMORYaPNZ4sQIuuYPDHdQvf4gyCF9rEEV/yPxGfwPkRodwEgiMUUXTx/dex+Me0wxx53S+NgUHaP7y3MGlDmg==} - - widest-line@3.1.0: - resolution: {integrity: sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==} - engines: {node: '>=8'} - - winston-transport@4.7.1: - resolution: {integrity: sha512-wQCXXVgfv/wUPOfb2x0ruxzwkcZfxcktz6JIMUaPLmcNhO4bZTwA/WtDWK74xV3F2dKu8YadrFv0qhwYjVEwhA==} - engines: {node: '>= 12.0.0'} - - winston@3.10.0: - resolution: {integrity: sha512-nT6SIDaE9B7ZRO0u3UvdrimG0HkB7dSTAgInQnNR2SOPJ4bvq5q79+pXLftKmP52lJGW15+H5MCK0nM9D3KB/g==} - engines: {node: '>= 12.0.0'} - - wordwrap@1.0.0: - resolution: {integrity: sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==} - - wrap-ansi@7.0.0: - resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} - engines: {node: '>=10'} - - wrap-ansi@8.1.0: - resolution: {integrity: sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==} - engines: {node: '>=12'} - - wrappy@1.0.2: - resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==} - - write-file-atomic@3.0.3: - resolution: {integrity: sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==} - - ws@7.5.10: - resolution: {integrity: sha512-+dbF1tHwZpXcbOJdVOkzLDxZP1ailvSxM6ZweXTegylPny803bFhA+vqBYw4s31NSAk4S2Qz+AKXK9a4wkdjcQ==} - engines: {node: '>=8.3.0'} - peerDependencies: - bufferutil: ^4.0.1 - utf-8-validate: ^5.0.2 - peerDependenciesMeta: - bufferutil: - optional: true - utf-8-validate: - optional: true - - ws@8.13.0: - resolution: {integrity: sha512-x9vcZYTrFPC7aSIbj7sRCYo7L/Xb8Iy+pW0ng0wt2vCJv7M9HOMy0UoN3rr+IFC7hb7vXoqS+P9ktyLLLhO+LA==} - engines: {node: '>=10.0.0'} - peerDependencies: - bufferutil: ^4.0.1 - utf-8-validate: '>=5.0.2' - peerDependenciesMeta: - bufferutil: - optional: true - utf-8-validate: - optional: true - - xdg-app-paths@8.3.0: - resolution: {integrity: sha512-mgxlWVZw0TNWHoGmXq+NC3uhCIc55dDpAlDkMQUaIAcQzysb0kxctwv//fvuW61/nAAeUBJMQ8mnZjMmuYwOcQ==} - engines: {node: '>= 4.0'} - - xdg-basedir@4.0.0: - resolution: {integrity: sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==} - engines: {node: '>=8'} - - xdg-portable@10.6.0: - resolution: {integrity: sha512-xrcqhWDvtZ7WLmt8G4f3hHy37iK7D2idtosRgkeiSPZEPmBShp0VfmRBLWAPC6zLF48APJ21yfea+RfQMF4/Aw==} - engines: {node: '>= 4.0'} - - xtend@4.0.2: - resolution: {integrity: sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==} - engines: {node: '>=0.4'} - - y18n@5.0.8: - resolution: {integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==} - engines: {node: '>=10'} - - yalc@1.0.0-pre.53: - resolution: {integrity: sha512-tpNqBCpTXplnduzw5XC+FF8zNJ9L/UXmvQyyQj7NKrDNavbJtHvzmZplL5ES/RCnjX7JR7W9wz5GVDXVP3dHUQ==} - hasBin: true - - yallist@4.0.0: - resolution: {integrity: sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==} - - yaml@1.10.2: - resolution: {integrity: sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==} - engines: {node: '>= 6'} - - yargs-parser@20.2.9: - resolution: {integrity: sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==} - engines: {node: '>=10'} - - yargs-parser@21.1.1: - resolution: {integrity: sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==} - engines: {node: '>=12'} - - yargs@16.2.0: - resolution: {integrity: sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==} - engines: {node: '>=10'} - - yargs@17.7.2: - resolution: {integrity: sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==} - engines: {node: '>=12'} - - ylru@1.4.0: - resolution: {integrity: sha512-2OQsPNEmBCvXuFlIni/a+Rn+R2pHW9INm0BxXJ4hVDA8TirqMj+J/Rp9ItLatT/5pZqWwefVrTQcHpixsxnVlA==} - engines: {node: '>= 4.0.0'} - - yocto-queue@0.1.0: - resolution: {integrity: sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==} - engines: {node: '>=10'} - - yup@0.32.9: - resolution: {integrity: sha512-Ci1qN+i2H0XpY7syDQ0k5zKQ/DoxO0LzPg8PAR/X4Mpj6DqaeCoIYEEjDJwhArh3Fa7GWbQQVDZKeXYlSH4JMg==} - engines: {node: '>=10'} - -snapshots: - - '@11ty/eleventy-fetch@4.0.1': - dependencies: - debug: 4.3.6 - flat-cache: 3.2.0 - node-fetch: 2.7.0 - p-queue: 6.6.2 - transitivePeerDependencies: - - encoding - - supports-color - - '@aws-crypto/crc32@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/crc32c@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/sha1-browser@5.2.0': - dependencies: - '@aws-crypto/supports-web-crypto': 5.2.0 - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-locate-window': 3.568.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-crypto/sha256-browser@5.2.0': - dependencies: - '@aws-crypto/sha256-js': 5.2.0 - '@aws-crypto/supports-web-crypto': 5.2.0 - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-locate-window': 3.568.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-crypto/sha256-js@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/supports-web-crypto@5.2.0': - dependencies: - tslib: 2.7.0 - - '@aws-crypto/util@5.2.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-sdk/client-s3@3.637.0': - dependencies: - '@aws-crypto/sha1-browser': 5.2.0 - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-bucket-endpoint': 3.620.0 - '@aws-sdk/middleware-expect-continue': 3.620.0 - '@aws-sdk/middleware-flexible-checksums': 3.620.0 - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-location-constraint': 3.609.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-sdk-s3': 3.635.0 - '@aws-sdk/middleware-ssec': 3.609.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/signature-v4-multi-region': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@aws-sdk/xml-builder': 3.609.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/eventstream-serde-browser': 3.0.6 - '@smithy/eventstream-serde-config-resolver': 3.0.3 - '@smithy/eventstream-serde-node': 3.0.5 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-blob-browser': 3.1.2 - '@smithy/hash-node': 3.0.3 - '@smithy/hash-stream-node': 3.1.2 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/md5-js': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-stream': 3.1.3 - '@smithy/util-utf8': 3.0.0 - '@smithy/util-waiter': 3.1.2 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sso@3.637.0': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sts@3.637.0': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/core@3.635.0': - dependencies: - '@smithy/core': 2.4.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-middleware': 3.0.3 - fast-xml-parser: 4.4.1 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-env@3.620.1': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-http@3.635.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-stream': 3.1.3 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-ini@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/credential-provider-env': 3.620.1 - '@aws-sdk/credential-provider-http': 3.635.0 - '@aws-sdk/credential-provider-process': 3.620.1 - '@aws-sdk/credential-provider-sso': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/credential-provider-web-identity': 3.621.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - aws-crt - - '@aws-sdk/credential-provider-node@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/credential-provider-env': 3.620.1 - '@aws-sdk/credential-provider-http': 3.635.0 - '@aws-sdk/credential-provider-ini': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/credential-provider-process': 3.620.1 - '@aws-sdk/credential-provider-sso': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/credential-provider-web-identity': 3.621.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - '@aws-sdk/client-sts' - - aws-crt - - '@aws-sdk/credential-provider-process@3.620.1': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-sso@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))': - dependencies: - '@aws-sdk/client-sso': 3.637.0 - '@aws-sdk/token-providers': 3.614.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - aws-crt - - '@aws-sdk/credential-provider-web-identity@3.621.0(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-bucket-endpoint@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-arn-parser': 3.568.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-expect-continue@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-flexible-checksums@3.620.0': - dependencies: - '@aws-crypto/crc32': 5.2.0 - '@aws-crypto/crc32c': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@smithy/is-array-buffer': 3.0.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-host-header@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-location-constraint@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-logger@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-recursion-detection@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-sdk-s3@3.635.0': - dependencies: - '@aws-sdk/core': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-arn-parser': 3.568.0 - '@smithy/core': 2.4.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-stream': 3.1.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-ssec@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-user-agent@3.637.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/region-config-resolver@3.614.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@aws-sdk/signature-v4-multi-region@3.635.0': - dependencies: - '@aws-sdk/middleware-sdk-s3': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/token-providers@3.614.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))': - dependencies: - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/types@3.609.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/util-arn-parser@3.568.0': - dependencies: - tslib: 2.7.0 - - '@aws-sdk/util-endpoints@3.637.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - '@smithy/util-endpoints': 2.0.5 - tslib: 2.7.0 - - '@aws-sdk/util-locate-window@3.568.0': - dependencies: - tslib: 2.7.0 - - '@aws-sdk/util-user-agent-browser@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - bowser: 2.11.0 - tslib: 2.7.0 - - '@aws-sdk/util-user-agent-node@3.614.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/xml-builder@3.609.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@babel/code-frame@7.24.7': - dependencies: - '@babel/highlight': 7.24.7 - picocolors: 1.1.0 - - '@babel/generator@7.25.6': - dependencies: - '@babel/types': 7.25.6 - '@jridgewell/gen-mapping': 0.3.5 - '@jridgewell/trace-mapping': 0.3.25 - jsesc: 2.5.2 - - '@babel/helper-module-imports@7.24.7': - dependencies: - '@babel/traverse': 7.25.6 - '@babel/types': 7.25.6 - transitivePeerDependencies: - - supports-color - - '@babel/helper-string-parser@7.24.8': {} - - '@babel/helper-validator-identifier@7.24.7': {} - - '@babel/highlight@7.24.7': - dependencies: - '@babel/helper-validator-identifier': 7.24.7 - chalk: 2.4.2 - js-tokens: 4.0.0 - picocolors: 1.1.0 - - '@babel/parser@7.25.6': - dependencies: - '@babel/types': 7.25.6 - - '@babel/runtime-corejs3@7.25.6': - dependencies: - core-js-pure: 3.38.1 - regenerator-runtime: 0.14.1 - - '@babel/runtime@7.25.6': - dependencies: - regenerator-runtime: 0.14.1 - - '@babel/template@7.25.0': - dependencies: - '@babel/code-frame': 7.24.7 - '@babel/parser': 7.25.6 - '@babel/types': 7.25.6 - - '@babel/traverse@7.25.6': - dependencies: - '@babel/code-frame': 7.24.7 - '@babel/generator': 7.25.6 - '@babel/parser': 7.25.6 - '@babel/template': 7.25.0 - '@babel/types': 7.25.6 - debug: 4.3.6 - globals: 11.12.0 - transitivePeerDependencies: - - supports-color - - '@babel/types@7.25.6': - dependencies: - '@babel/helper-string-parser': 7.24.8 - '@babel/helper-validator-identifier': 7.24.7 - to-fast-properties: 2.0.0 - - '@casl/ability@6.5.0': - dependencies: - '@ucast/mongo2js': 1.3.4 - - '@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1)': - dependencies: - '@codemirror/language': 6.10.2 - '@codemirror/state': 6.4.1 - '@codemirror/view': 6.33.0 - '@lezer/common': 1.2.1 - - '@codemirror/commands@6.6.1': - dependencies: - '@codemirror/language': 6.10.2 - '@codemirror/state': 6.4.1 - '@codemirror/view': 6.33.0 - '@lezer/common': 1.2.1 - - '@codemirror/lang-json@6.0.1': - dependencies: - '@codemirror/language': 6.10.2 - '@lezer/json': 1.0.2 - - '@codemirror/language@6.10.2': - dependencies: - '@codemirror/state': 6.4.1 - '@codemirror/view': 6.33.0 - '@lezer/common': 1.2.1 - '@lezer/highlight': 1.2.1 - '@lezer/lr': 1.4.2 - style-mod: 4.1.2 - - '@codemirror/lint@6.8.1': - dependencies: - '@codemirror/state': 6.4.1 - '@codemirror/view': 6.33.0 - crelt: 1.0.6 - - '@codemirror/search@6.5.6': - dependencies: - '@codemirror/state': 6.4.1 - '@codemirror/view': 6.33.0 - crelt: 1.0.6 - - '@codemirror/state@6.4.1': {} - - '@codemirror/theme-one-dark@6.1.2': - dependencies: - '@codemirror/language': 6.10.2 - '@codemirror/state': 6.4.1 - '@codemirror/view': 6.33.0 - '@lezer/highlight': 1.2.1 - - '@codemirror/view@6.33.0': - dependencies: - '@codemirror/state': 6.4.1 - style-mod: 4.1.2 - w3c-keyname: 2.2.8 - - '@colors/colors@1.5.0': {} - - '@colors/colors@1.6.0': {} - - '@dabh/diagnostics@2.0.3': - dependencies: - colorspace: 1.1.4 - enabled: 2.0.0 - kuler: 2.0.0 - - '@discoveryjs/json-ext@0.5.7': {} - - '@emnapi/runtime@1.2.0': - dependencies: - tslib: 2.7.0 - optional: true - - '@emotion/babel-plugin@11.12.0': - dependencies: - '@babel/helper-module-imports': 7.24.7 - '@babel/runtime': 7.25.6 - '@emotion/hash': 0.9.2 - '@emotion/memoize': 0.9.0 - '@emotion/serialize': 1.3.1 - babel-plugin-macros: 3.1.0 - convert-source-map: 1.9.0 - escape-string-regexp: 4.0.0 - find-root: 1.1.0 - source-map: 0.5.7 - stylis: 4.2.0 - transitivePeerDependencies: - - supports-color - - '@emotion/cache@11.13.1': - dependencies: - '@emotion/memoize': 0.9.0 - '@emotion/sheet': 1.4.0 - '@emotion/utils': 1.4.0 - '@emotion/weak-memoize': 0.4.0 - stylis: 4.2.0 - - '@emotion/hash@0.9.2': {} - - '@emotion/is-prop-valid@1.2.2': - dependencies: - '@emotion/memoize': 0.8.1 - - '@emotion/memoize@0.8.1': {} - - '@emotion/memoize@0.9.0': {} - - '@emotion/react@11.13.3(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - '@emotion/babel-plugin': 11.12.0 - '@emotion/cache': 11.13.1 - '@emotion/serialize': 1.3.1 - '@emotion/use-insertion-effect-with-fallbacks': 1.1.0(react@18.3.1) - '@emotion/utils': 1.4.0 - '@emotion/weak-memoize': 0.4.0 - hoist-non-react-statics: 3.3.2 - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - transitivePeerDependencies: - - supports-color - - '@emotion/serialize@1.3.1': - dependencies: - '@emotion/hash': 0.9.2 - '@emotion/memoize': 0.9.0 - '@emotion/unitless': 0.10.0 - '@emotion/utils': 1.4.0 - csstype: 3.1.3 - - '@emotion/sheet@1.4.0': {} - - '@emotion/unitless@0.10.0': {} - - '@emotion/unitless@0.8.1': {} - - '@emotion/use-insertion-effect-with-fallbacks@1.1.0(react@18.3.1)': - dependencies: - react: 18.3.1 - - '@emotion/utils@1.4.0': {} - - '@emotion/weak-memoize@0.4.0': {} - - '@esbuild/aix-ppc64@0.19.11': - optional: true - - '@esbuild/android-arm64@0.16.17': - optional: true - - '@esbuild/android-arm64@0.19.11': - optional: true - - '@esbuild/android-arm@0.16.17': - optional: true - - '@esbuild/android-arm@0.19.11': - optional: true - - '@esbuild/android-x64@0.16.17': - optional: true - - '@esbuild/android-x64@0.19.11': - optional: true - - '@esbuild/darwin-arm64@0.16.17': - optional: true - - '@esbuild/darwin-arm64@0.19.11': - optional: true - - '@esbuild/darwin-x64@0.16.17': - optional: true - - '@esbuild/darwin-x64@0.19.11': - optional: true - - '@esbuild/freebsd-arm64@0.16.17': - optional: true - - '@esbuild/freebsd-arm64@0.19.11': - optional: true - - '@esbuild/freebsd-x64@0.16.17': - optional: true - - '@esbuild/freebsd-x64@0.19.11': - optional: true - - '@esbuild/linux-arm64@0.16.17': - optional: true - - '@esbuild/linux-arm64@0.19.11': - optional: true - - '@esbuild/linux-arm@0.16.17': - optional: true - - '@esbuild/linux-arm@0.19.11': - optional: true - - '@esbuild/linux-ia32@0.16.17': - optional: true - - '@esbuild/linux-ia32@0.19.11': - optional: true - - '@esbuild/linux-loong64@0.16.17': - optional: true - - '@esbuild/linux-loong64@0.19.11': - optional: true - - '@esbuild/linux-mips64el@0.16.17': - optional: true - - '@esbuild/linux-mips64el@0.19.11': - optional: true - - '@esbuild/linux-ppc64@0.16.17': - optional: true - - '@esbuild/linux-ppc64@0.19.11': - optional: true - - '@esbuild/linux-riscv64@0.16.17': - optional: true - - '@esbuild/linux-riscv64@0.19.11': - optional: true - - '@esbuild/linux-s390x@0.16.17': - optional: true - - '@esbuild/linux-s390x@0.19.11': - optional: true - - '@esbuild/linux-x64@0.16.17': - optional: true - - '@esbuild/linux-x64@0.19.11': - optional: true - - '@esbuild/netbsd-x64@0.16.17': - optional: true - - '@esbuild/netbsd-x64@0.19.11': - optional: true - - '@esbuild/openbsd-x64@0.16.17': - optional: true - - '@esbuild/openbsd-x64@0.19.11': - optional: true - - '@esbuild/sunos-x64@0.16.17': - optional: true - - '@esbuild/sunos-x64@0.19.11': - optional: true - - '@esbuild/win32-arm64@0.16.17': - optional: true - - '@esbuild/win32-arm64@0.19.11': - optional: true - - '@esbuild/win32-ia32@0.16.17': - optional: true - - '@esbuild/win32-ia32@0.19.11': - optional: true - - '@esbuild/win32-x64@0.16.17': - optional: true - - '@esbuild/win32-x64@0.19.11': - optional: true - - '@floating-ui/core@1.6.7': - dependencies: - '@floating-ui/utils': 0.2.7 - - '@floating-ui/dom@1.6.10': - dependencies: - '@floating-ui/core': 1.6.7 - '@floating-ui/utils': 0.2.7 - - '@floating-ui/react-dom@2.1.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@floating-ui/dom': 1.6.10 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - - '@floating-ui/utils@0.2.7': {} - - '@formatjs/ecma402-abstract@1.14.3': - dependencies: - '@formatjs/intl-localematcher': 0.2.32 - tslib: 2.7.0 - - '@formatjs/fast-memoize@2.0.1': - dependencies: - tslib: 2.7.0 - - '@formatjs/icu-messageformat-parser@2.3.1': - dependencies: - '@formatjs/ecma402-abstract': 1.14.3 - '@formatjs/icu-skeleton-parser': 1.3.18 - tslib: 2.7.0 - - '@formatjs/icu-skeleton-parser@1.3.18': - dependencies: - '@formatjs/ecma402-abstract': 1.14.3 - tslib: 2.7.0 - - '@formatjs/intl-displaynames@6.3.1': - dependencies: - '@formatjs/ecma402-abstract': 1.14.3 - '@formatjs/intl-localematcher': 0.2.32 - tslib: 2.7.0 - - '@formatjs/intl-listformat@7.2.1': - dependencies: - '@formatjs/ecma402-abstract': 1.14.3 - '@formatjs/intl-localematcher': 0.2.32 - tslib: 2.7.0 - - '@formatjs/intl-localematcher@0.2.32': - dependencies: - tslib: 2.7.0 - - '@formatjs/intl@2.7.1(typescript@5.2.2)': - dependencies: - '@formatjs/ecma402-abstract': 1.14.3 - '@formatjs/fast-memoize': 2.0.1 - '@formatjs/icu-messageformat-parser': 2.3.1 - '@formatjs/intl-displaynames': 6.3.1 - '@formatjs/intl-listformat': 7.2.1 - intl-messageformat: 10.3.4 - tslib: 2.7.0 - optionalDependencies: - typescript: 5.2.2 - - '@hapi/bourne@3.0.0': {} - - '@img/sharp-darwin-arm64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-darwin-arm64': 1.0.4 - optional: true - - '@img/sharp-darwin-x64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-darwin-x64': 1.0.4 - optional: true - - '@img/sharp-libvips-darwin-arm64@1.0.4': - optional: true - - '@img/sharp-libvips-darwin-x64@1.0.4': - optional: true - - '@img/sharp-libvips-linux-arm64@1.0.4': - optional: true - - '@img/sharp-libvips-linux-arm@1.0.5': - optional: true - - '@img/sharp-libvips-linux-s390x@1.0.4': - optional: true - - '@img/sharp-libvips-linux-x64@1.0.4': - optional: true - - '@img/sharp-libvips-linuxmusl-arm64@1.0.4': - optional: true - - '@img/sharp-libvips-linuxmusl-x64@1.0.4': - optional: true - - '@img/sharp-linux-arm64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linux-arm64': 1.0.4 - optional: true - - '@img/sharp-linux-arm@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linux-arm': 1.0.5 - optional: true - - '@img/sharp-linux-s390x@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linux-s390x': 1.0.4 - optional: true - - '@img/sharp-linux-x64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linux-x64': 1.0.4 - optional: true - - '@img/sharp-linuxmusl-arm64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linuxmusl-arm64': 1.0.4 - optional: true - - '@img/sharp-linuxmusl-x64@0.33.5': - optionalDependencies: - '@img/sharp-libvips-linuxmusl-x64': 1.0.4 - optional: true - - '@img/sharp-wasm32@0.33.5': - dependencies: - '@emnapi/runtime': 1.2.0 - optional: true - - '@img/sharp-win32-ia32@0.33.5': - optional: true - - '@img/sharp-win32-x64@0.33.5': - optional: true - - '@internationalized/date@3.5.5': - dependencies: - '@swc/helpers': 0.5.13 - - '@internationalized/number@3.5.3': - dependencies: - '@swc/helpers': 0.5.13 - - '@isaacs/cliui@8.0.2': - dependencies: - string-width: 5.1.2 - string-width-cjs: string-width@4.2.3 - strip-ansi: 7.1.0 - strip-ansi-cjs: strip-ansi@6.0.1 - wrap-ansi: 8.1.0 - wrap-ansi-cjs: wrap-ansi@7.0.0 - - '@jridgewell/gen-mapping@0.3.5': - dependencies: - '@jridgewell/set-array': 1.2.1 - '@jridgewell/sourcemap-codec': 1.5.0 - '@jridgewell/trace-mapping': 0.3.25 - - '@jridgewell/resolve-uri@3.1.2': {} - - '@jridgewell/set-array@1.2.1': {} - - '@jridgewell/source-map@0.3.6': - dependencies: - '@jridgewell/gen-mapping': 0.3.5 - '@jridgewell/trace-mapping': 0.3.25 - - '@jridgewell/sourcemap-codec@1.5.0': {} - - '@jridgewell/trace-mapping@0.3.25': - dependencies: - '@jridgewell/resolve-uri': 3.1.2 - '@jridgewell/sourcemap-codec': 1.5.0 - - '@juggle/resize-observer@3.4.0': {} - - '@koa/cors@5.0.0': - dependencies: - vary: 1.1.2 - - '@koa/router@10.1.1': - dependencies: - debug: 4.3.4(supports-color@5.5.0) - http-errors: 1.8.1 - koa-compose: 4.1.0 - methods: 1.1.2 - path-to-regexp: 6.2.2 - transitivePeerDependencies: - - supports-color - - '@lezer/common@1.2.1': {} - - '@lezer/highlight@1.2.1': - dependencies: - '@lezer/common': 1.2.1 - - '@lezer/json@1.0.2': - dependencies: - '@lezer/common': 1.2.1 - '@lezer/highlight': 1.2.1 - '@lezer/lr': 1.4.2 - - '@lezer/lr@1.4.2': - dependencies: - '@lezer/common': 1.2.1 - - '@mapbox/node-pre-gyp@1.0.11': - dependencies: - detect-libc: 2.0.3 - https-proxy-agent: 5.0.1 - make-dir: 3.1.0 - node-fetch: 2.7.0 - nopt: 5.0.0 - npmlog: 5.0.1 - rimraf: 3.0.2 - semver: 7.6.3 - tar: 6.2.1 - transitivePeerDependencies: - - encoding - - supports-color - - '@mux/mux-node@8.8.0': - dependencies: - '@types/node': 18.19.48 - '@types/node-fetch': 2.6.11 - '@types/qs': 6.9.15 - abort-controller: 3.0.0 - agentkeepalive: 4.5.0 - form-data-encoder: 1.7.2 - formdata-node: 4.4.1 - jose: 4.15.9 - node-fetch: 2.7.0 - qs: 6.13.0 - web-streams-polyfill: 3.3.3 - transitivePeerDependencies: - - encoding - - '@noble/hashes@1.5.0': {} - - '@nodelib/fs.scandir@2.1.5': - dependencies: - '@nodelib/fs.stat': 2.0.5 - run-parallel: 1.2.0 - - '@nodelib/fs.stat@2.0.5': {} - - '@nodelib/fs.walk@1.2.8': - dependencies: - '@nodelib/fs.scandir': 2.1.5 - fastq: 1.17.1 - - '@paralleldrive/cuid2@2.2.2': - dependencies: - '@noble/hashes': 1.5.0 - - '@pkgjs/parseargs@0.11.0': - optional: true - - '@pkgr/utils@2.4.2': - dependencies: - cross-spawn: 7.0.3 - fast-glob: 3.3.2 - is-glob: 4.0.3 - open: 9.1.0 - picocolors: 1.1.0 - tslib: 2.7.0 - - '@pmmmwh/react-refresh-webpack-plugin@0.5.11(react-refresh@0.14.0)(type-fest@2.19.0)(webpack-hot-middleware@2.26.0)(webpack@5.94.0(esbuild@0.19.11))': - dependencies: - ansi-html-community: 0.0.8 - common-path-prefix: 3.0.0 - core-js-pure: 3.38.1 - error-stack-parser: 2.1.4 - find-up: 5.0.0 - html-entities: 2.5.2 - loader-utils: 2.0.4 - react-refresh: 0.14.0 - schema-utils: 3.3.0 - source-map: 0.7.4 - webpack: 5.94.0(esbuild@0.19.11) - optionalDependencies: - type-fest: 2.19.0 - webpack-hot-middleware: 2.26.0 - - '@pnpm/config.env-replace@1.1.0': {} - - '@pnpm/network.ca-file@1.0.2': - dependencies: - graceful-fs: 4.2.10 - - '@pnpm/npm-conf@2.3.1': - dependencies: - '@pnpm/config.env-replace': 1.1.0 - '@pnpm/network.ca-file': 1.0.2 - config-chain: 1.1.13 - - '@polka/url@1.0.0-next.25': {} - - '@radix-ui/number@1.1.0': {} - - '@radix-ui/primitive@1.0.1': - dependencies: - '@babel/runtime': 7.25.6 - - '@radix-ui/primitive@1.1.0': {} - - '@radix-ui/react-arrow@1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@radix-ui/react-primitive': 2.0.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-collection@1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - '@radix-ui/react-compose-refs': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-context': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-primitive': 1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-slot': 1.0.2(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-collection@1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-context': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-primitive': 2.0.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-slot': 1.1.0(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-compose-refs@1.0.1(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-compose-refs@1.1.0(@types/react@18.3.5)(react@18.3.1)': - dependencies: - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-context@1.0.1(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-context@1.1.0(@types/react@18.3.5)(react@18.3.1)': - dependencies: - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-direction@1.0.1(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-direction@1.1.0(@types/react@18.3.5)(react@18.3.1)': - dependencies: - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-dismissable-layer@1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@radix-ui/primitive': 1.1.0 - '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-primitive': 2.0.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-use-escape-keydown': 1.1.0(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-dropdown-menu@2.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@radix-ui/primitive': 1.1.0 - '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-context': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-id': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-menu': 2.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-primitive': 2.0.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-use-controllable-state': 1.1.0(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-focus-guards@1.0.1(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-focus-guards@1.1.0(@types/react@18.3.5)(react@18.3.1)': - dependencies: - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-focus-scope@1.0.4(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - '@radix-ui/react-compose-refs': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-primitive': 1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-use-callback-ref': 1.0.1(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-focus-scope@1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-primitive': 2.0.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-id@1.0.1(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - '@radix-ui/react-use-layout-effect': 1.0.1(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-id@1.1.0(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-menu@2.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@radix-ui/primitive': 1.1.0 - '@radix-ui/react-collection': 1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-context': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-direction': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-dismissable-layer': 1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-focus-guards': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-focus-scope': 1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-id': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-popper': 1.2.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-portal': 1.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-presence': 1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-primitive': 2.0.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-roving-focus': 1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-slot': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.5)(react@18.3.1) - aria-hidden: 1.2.4 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - react-remove-scroll: 2.5.7(@types/react@18.3.5)(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-popper@1.2.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@floating-ui/react-dom': 2.1.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-arrow': 1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-context': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-primitive': 2.0.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-use-rect': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-use-size': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/rect': 1.1.0 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-portal@1.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@radix-ui/react-primitive': 2.0.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-presence@1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-primitive@1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - '@radix-ui/react-slot': 1.0.2(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-primitive@2.0.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@radix-ui/react-slot': 1.1.0(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-roving-focus@1.0.4(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - '@radix-ui/primitive': 1.0.1 - '@radix-ui/react-collection': 1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-compose-refs': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-context': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-direction': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-id': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-primitive': 1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-use-callback-ref': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-use-controllable-state': 1.0.1(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-roving-focus@1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@radix-ui/primitive': 1.1.0 - '@radix-ui/react-collection': 1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-context': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-direction': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-id': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-primitive': 2.0.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-use-controllable-state': 1.1.0(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-separator@1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - '@radix-ui/react-primitive': 1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-slot@1.0.2(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - '@radix-ui/react-compose-refs': 1.0.1(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-slot@1.1.0(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-toggle-group@1.0.4(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - '@radix-ui/primitive': 1.0.1 - '@radix-ui/react-context': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-direction': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-primitive': 1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-roving-focus': 1.0.4(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-toggle': 1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-use-controllable-state': 1.0.1(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-toggle@1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - '@radix-ui/primitive': 1.0.1 - '@radix-ui/react-primitive': 1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-use-controllable-state': 1.0.1(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-toolbar@1.0.4(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - '@radix-ui/primitive': 1.0.1 - '@radix-ui/react-context': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-direction': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-primitive': 1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-roving-focus': 1.0.4(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-separator': 1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-toggle-group': 1.0.4(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-use-callback-ref@1.0.1(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-use-callback-ref@1.1.0(@types/react@18.3.5)(react@18.3.1)': - dependencies: - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-use-controllable-state@1.0.1(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - '@radix-ui/react-use-callback-ref': 1.0.1(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-use-controllable-state@1.1.0(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-use-escape-keydown@1.1.0(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-use-layout-effect@1.0.1(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-use-layout-effect@1.1.0(@types/react@18.3.5)(react@18.3.1)': - dependencies: - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-use-previous@1.1.0(@types/react@18.3.5)(react@18.3.1)': - dependencies: - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-use-rect@1.1.0(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@radix-ui/rect': 1.1.0 - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-use-size@1.1.0(@types/react@18.3.5)(react@18.3.1)': - dependencies: - '@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.5)(react@18.3.1) - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/react-visually-hidden@1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@radix-ui/react-primitive': 2.0.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - '@radix-ui/rect@1.1.0': {} - - '@react-dnd/asap@5.0.2': {} - - '@react-dnd/invariant@4.0.2': {} - - '@react-dnd/shallowequal@4.0.2': {} - - '@reduxjs/toolkit@1.9.7(react-redux@8.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(redux@4.2.1))(react@18.3.1)': - dependencies: - immer: 9.0.21 - redux: 4.2.1 - redux-thunk: 2.4.2(redux@4.2.1) - reselect: 4.1.8 - optionalDependencies: - react: 18.3.1 - react-redux: 8.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(redux@4.2.1) - - '@remix-run/router@1.19.1': {} - - '@rollup/rollup-android-arm-eabi@4.21.2': - optional: true - - '@rollup/rollup-android-arm64@4.21.2': - optional: true - - '@rollup/rollup-darwin-arm64@4.21.2': - optional: true - - '@rollup/rollup-darwin-x64@4.21.2': - optional: true - - '@rollup/rollup-linux-arm-gnueabihf@4.21.2': - optional: true - - '@rollup/rollup-linux-arm-musleabihf@4.21.2': - optional: true - - '@rollup/rollup-linux-arm64-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-arm64-musl@4.21.2': - optional: true - - '@rollup/rollup-linux-powerpc64le-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-riscv64-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-s390x-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-x64-gnu@4.21.2': - optional: true - - '@rollup/rollup-linux-x64-musl@4.21.2': - optional: true - - '@rollup/rollup-win32-arm64-msvc@4.21.2': - optional: true - - '@rollup/rollup-win32-ia32-msvc@4.21.2': - optional: true - - '@rollup/rollup-win32-x64-msvc@4.21.2': - optional: true - - '@rushstack/node-core-library@5.7.0(@types/node@22.5.2)': - dependencies: - ajv: 8.13.0 - ajv-draft-04: 1.0.0(ajv@8.13.0) - ajv-formats: 3.0.1(ajv@8.13.0) - fs-extra: 7.0.1 - import-lazy: 4.0.0 - jju: 1.4.0 - resolve: 1.22.8 - semver: 7.5.4 - optionalDependencies: - '@types/node': 22.5.2 - - '@rushstack/terminal@0.14.0(@types/node@22.5.2)': - dependencies: - '@rushstack/node-core-library': 5.7.0(@types/node@22.5.2) - supports-color: 8.1.1 - optionalDependencies: - '@types/node': 22.5.2 - - '@rushstack/ts-command-line@4.22.6(@types/node@22.5.2)': - dependencies: - '@rushstack/terminal': 0.14.0(@types/node@22.5.2) - '@types/argparse': 1.0.38 - argparse: 1.0.10 - string-argv: 0.3.2 - transitivePeerDependencies: - - '@types/node' - - '@sentry/core@6.19.7': - dependencies: - '@sentry/hub': 6.19.7 - '@sentry/minimal': 6.19.7 - '@sentry/types': 6.19.7 - '@sentry/utils': 6.19.7 - tslib: 1.14.1 - - '@sentry/hub@6.19.7': - dependencies: - '@sentry/types': 6.19.7 - '@sentry/utils': 6.19.7 - tslib: 1.14.1 - - '@sentry/minimal@6.19.7': - dependencies: - '@sentry/hub': 6.19.7 - '@sentry/types': 6.19.7 - tslib: 1.14.1 - - '@sentry/node@6.19.7': - dependencies: - '@sentry/core': 6.19.7 - '@sentry/hub': 6.19.7 - '@sentry/types': 6.19.7 - '@sentry/utils': 6.19.7 - cookie: 0.4.2 - https-proxy-agent: 5.0.1 - lru_map: 0.3.3 - tslib: 1.14.1 - transitivePeerDependencies: - - supports-color - - '@sentry/types@6.19.7': {} - - '@sentry/utils@6.19.7': - dependencies: - '@sentry/types': 6.19.7 - tslib: 1.14.1 - - '@simov/deep-extend@1.0.0': {} - - '@sindresorhus/is@4.6.0': {} - - '@sindresorhus/slugify@1.1.0': - dependencies: - '@sindresorhus/transliterate': 0.1.2 - escape-string-regexp: 4.0.0 - - '@sindresorhus/transliterate@0.1.2': - dependencies: - escape-string-regexp: 2.0.0 - lodash.deburr: 4.1.0 - - '@smithy/abort-controller@3.1.1': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/chunked-blob-reader-native@3.0.0': - dependencies: - '@smithy/util-base64': 3.0.0 - tslib: 2.7.0 - - '@smithy/chunked-blob-reader@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/config-resolver@3.0.5': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@smithy/core@2.4.0': - dependencies: - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/credential-provider-imds@3.2.0': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - tslib: 2.7.0 - - '@smithy/eventstream-codec@3.1.2': - dependencies: - '@aws-crypto/crc32': 5.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-hex-encoding': 3.0.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-browser@3.0.6': - dependencies: - '@smithy/eventstream-serde-universal': 3.0.5 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-config-resolver@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-node@3.0.5': - dependencies: - '@smithy/eventstream-serde-universal': 3.0.5 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-universal@3.0.5': - dependencies: - '@smithy/eventstream-codec': 3.1.2 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/fetch-http-handler@3.2.4': - dependencies: - '@smithy/protocol-http': 4.1.0 - '@smithy/querystring-builder': 3.0.3 - '@smithy/types': 3.3.0 - '@smithy/util-base64': 3.0.0 - tslib: 2.7.0 - - '@smithy/hash-blob-browser@3.1.2': - dependencies: - '@smithy/chunked-blob-reader': 3.0.0 - '@smithy/chunked-blob-reader-native': 3.0.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/hash-node@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/hash-stream-node@3.1.2': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/invalid-dependency@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/is-array-buffer@2.2.0': - dependencies: - tslib: 2.7.0 - - '@smithy/is-array-buffer@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/md5-js@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/middleware-content-length@3.0.5': - dependencies: - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/middleware-endpoint@3.1.0': - dependencies: - '@smithy/middleware-serde': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@smithy/middleware-retry@3.0.15': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/service-error-classification': 3.0.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - tslib: 2.7.0 - uuid: 9.0.1 - - '@smithy/middleware-serde@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/middleware-stack@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/node-config-provider@3.1.4': - dependencies: - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/node-http-handler@3.1.4': - dependencies: - '@smithy/abort-controller': 3.1.1 - '@smithy/protocol-http': 4.1.0 - '@smithy/querystring-builder': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/property-provider@3.1.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/protocol-http@4.1.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/querystring-builder@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-uri-escape': 3.0.0 - tslib: 2.7.0 - - '@smithy/querystring-parser@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/service-error-classification@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - - '@smithy/shared-ini-file-loader@3.1.4': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/signature-v4@4.1.0': - dependencies: - '@smithy/is-array-buffer': 3.0.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-hex-encoding': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-uri-escape': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/smithy-client@3.2.0': - dependencies: - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-stack': 3.0.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-stream': 3.1.3 - tslib: 2.7.0 - - '@smithy/types@3.3.0': - dependencies: - tslib: 2.7.0 - - '@smithy/url-parser@3.0.3': - dependencies: - '@smithy/querystring-parser': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-base64@3.0.0': - dependencies: - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-body-length-browser@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-body-length-node@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-buffer-from@2.2.0': - dependencies: - '@smithy/is-array-buffer': 2.2.0 - tslib: 2.7.0 - - '@smithy/util-buffer-from@3.0.0': - dependencies: - '@smithy/is-array-buffer': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-config-provider@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-defaults-mode-browser@3.0.15': - dependencies: - '@smithy/property-provider': 3.1.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - bowser: 2.11.0 - tslib: 2.7.0 - - '@smithy/util-defaults-mode-node@3.0.15': - dependencies: - '@smithy/config-resolver': 3.0.5 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-endpoints@2.0.5': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-hex-encoding@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-middleware@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-retry@3.0.3': - dependencies: - '@smithy/service-error-classification': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-stream@3.1.3': - dependencies: - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-base64': 3.0.0 - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-hex-encoding': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-uri-escape@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-utf8@2.3.0': - dependencies: - '@smithy/util-buffer-from': 2.2.0 - tslib: 2.7.0 - - '@smithy/util-utf8@3.0.0': - dependencies: - '@smithy/util-buffer-from': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-waiter@3.1.2': - dependencies: - '@smithy/abort-controller': 3.1.1 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@strapi/admin@4.25.9(kgi7qayoxmh7jdvj26fzdxk4ry)': - dependencies: - '@casl/ability': 6.5.0 - '@pmmmwh/react-refresh-webpack-plugin': 0.5.11(react-refresh@0.14.0)(type-fest@2.19.0)(webpack-hot-middleware@2.26.0)(webpack@5.94.0(esbuild@0.19.11)) - '@radix-ui/react-context': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-toolbar': 1.0.4(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@reduxjs/toolkit': 1.9.7(react-redux@8.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(redux@4.2.1))(react@18.3.1) - '@strapi/data-transfer': 4.25.9(@strapi/strapi@4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0))(@types/node@22.5.2)(pg@8.12.0) - '@strapi/design-system': 1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)) - '@strapi/helper-plugin': 4.25.9(@strapi/design-system@1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)))(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(debug@4.3.4)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(typescript@5.2.2) - '@strapi/icons': 1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@strapi/permissions': 4.25.9 - '@strapi/provider-audit-logs-local': 4.25.9 - '@strapi/strapi': 4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0) - '@strapi/types': 4.25.9(@types/node@22.5.2)(pg@8.12.0) - '@strapi/typescript-utils': 4.25.9 - '@strapi/utils': 4.25.9 - '@vitejs/plugin-react-swc': 3.5.0(@swc/helpers@0.5.13)(vite@5.0.13(@types/node@22.5.2)(terser@5.31.6)) - axios: 1.7.4(debug@4.3.4) - bcryptjs: 2.4.3 - boxen: 5.1.2 - browserslist: 4.23.3 - browserslist-to-esbuild: 1.2.0 - chalk: 4.1.2 - chokidar: 3.5.3 - codemirror5: codemirror@5.65.17 - cross-env: 7.0.3 - css-loader: 6.11.0(webpack@5.94.0(esbuild@0.19.11)) - date-fns: 2.30.0 - dotenv: 14.2.0 - esbuild: 0.19.11 - esbuild-loader: 2.21.0(webpack@5.94.0(esbuild@0.19.11)) - esbuild-register: 3.5.0(esbuild@0.19.11) - execa: 5.1.1 - fast-deep-equal: 3.1.3 - find-root: 1.1.0 - fork-ts-checker-webpack-plugin: 9.0.2(typescript@5.2.2)(webpack@5.94.0(esbuild@0.19.11)) - formik: 2.4.0(react@18.3.1) - fractional-indexing: 3.2.0 - fs-extra: 10.0.0 - highlight.js: 10.7.3 - history: 4.10.1 - html-webpack-plugin: 5.6.0(webpack@5.94.0(esbuild@0.19.11)) - immer: 9.0.19 - inquirer: 8.2.5 - invariant: 2.2.4 - is-localhost-ip: 2.0.0 - js-cookie: 2.2.1 - jsonwebtoken: 9.0.0 - koa: 2.13.4 - koa-bodyparser: 4.4.1 - koa-compose: 4.1.0 - koa-passport: 5.0.0 - koa-static: 5.0.0 - koa2-ratelimit: 1.1.3 - lodash: 4.17.21 - markdown-it: 12.3.2 - markdown-it-abbr: 1.0.4 - markdown-it-container: 3.0.0 - markdown-it-deflist: 2.1.0 - markdown-it-emoji: 2.0.2 - markdown-it-footnote: 3.0.3 - markdown-it-ins: 3.0.1 - markdown-it-mark: 3.0.1 - markdown-it-sub: 1.0.0 - markdown-it-sup: 1.0.0 - mini-css-extract-plugin: 2.7.7(webpack@5.94.0(esbuild@0.19.11)) - node-schedule: 2.1.1 - ora: 5.4.1 - outdent: 0.8.0 - p-map: 4.0.0 - passport-local: 1.0.0 - pluralize: 8.0.0 - prettier: 2.8.4 - prop-types: 15.8.1 - punycode: 2.3.1 - qs: 6.11.1 - react: 18.3.1 - react-dnd: 16.0.1(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(react@18.3.1) - react-dnd-html5-backend: 16.0.1 - react-dom: 18.3.1(react@18.3.1) - react-error-boundary: 3.1.4(react@18.3.1) - react-helmet: 6.1.0(react@18.3.1) - react-intl: 6.4.1(react@18.3.1)(typescript@5.2.2) - react-is: 18.3.1 - react-query: 3.39.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - react-redux: 8.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(redux@4.2.1) - react-refresh: 0.14.0 - react-router-dom: 6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - react-select: 5.7.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - react-window: 1.8.8(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - read-pkg-up: 7.0.1 - resolve-from: 5.0.0 - rimraf: 3.0.2 - sanitize-html: 2.13.0 - semver: 7.5.4 - sift: 16.0.1 - slate: 0.94.1 - slate-history: 0.93.0(slate@0.94.1) - slate-react: 0.98.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(slate@0.94.1) - style-loader: 3.3.4(webpack@5.94.0(esbuild@0.19.11)) - styled-components: 6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - typescript: 5.2.2 - vite: 5.0.13(@types/node@22.5.2)(terser@5.31.6) - webpack: 5.94.0(esbuild@0.19.11) - webpack-bundle-analyzer: 4.10.2 - webpack-dev-middleware: 6.1.2(webpack@5.94.0(esbuild@0.19.11)) - webpack-hot-middleware: 2.26.0 - yup: 0.32.9 - transitivePeerDependencies: - - '@babel/runtime' - - '@codemirror/autocomplete' - - '@codemirror/language' - - '@codemirror/lint' - - '@codemirror/search' - - '@codemirror/state' - - '@codemirror/theme-one-dark' - - '@codemirror/view' - - '@rspack/core' - - '@swc/core' - - '@swc/helpers' - - '@types/hoist-non-react-statics' - - '@types/node' - - '@types/react' - - '@types/react-dom' - - '@types/webpack' - - better-sqlite3 - - bufferutil - - codemirror - - debug - - encoding - - less - - lightningcss - - mongoose - - mysql - - mysql2 - - pg - - pg-native - - react-native - - redis - - redux - - sass - - sequelize - - sockjs-client - - sqlite3 - - stylus - - sugarss - - supports-color - - tedious - - terser - - type-fest - - uglify-js - - utf-8-validate - - webpack-cli - - webpack-dev-server - - webpack-plugin-serve - - '@strapi/cloud-cli@4.25.9(debug@4.3.4)': - dependencies: - '@strapi/utils': 4.25.9 - axios: 1.7.4(debug@4.3.4) - chalk: 4.1.2 - cli-progress: 3.12.0 - commander: 8.3.0 - eventsource: 2.0.2 - fast-safe-stringify: 2.1.1 - fs-extra: 10.0.0 - inquirer: 8.2.5 - jsonwebtoken: 9.0.0 - jwks-rsa: 3.1.0 - lodash: 4.17.21 - minimatch: 9.0.3 - open: 8.4.0 - ora: 5.4.1 - pkg-up: 3.1.0 - tar: 6.1.13 - xdg-app-paths: 8.3.0 - yup: 0.32.9 - transitivePeerDependencies: - - debug - - supports-color - - '@strapi/content-releases@4.25.9(q7vptuqps424wczizevu7bwk2q)': - dependencies: - '@reduxjs/toolkit': 1.9.7(react-redux@8.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(redux@4.2.1))(react@18.3.1) - '@strapi/admin': 4.25.9(kgi7qayoxmh7jdvj26fzdxk4ry) - '@strapi/design-system': 1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)) - '@strapi/helper-plugin': 4.25.9(@strapi/design-system@1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)))(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(debug@4.3.4)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(typescript@5.2.2) - '@strapi/icons': 1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@strapi/strapi': 4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0) - '@strapi/types': 4.25.9(@types/node@22.5.2)(pg@8.12.0) - '@strapi/utils': 4.25.9 - axios: 1.7.4(debug@4.3.4) - date-fns: 2.30.0 - date-fns-tz: 2.0.0(date-fns@2.30.0) - formik: 2.4.0(react@18.3.1) - lodash: 4.17.21 - node-schedule: 2.1.1 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - react-intl: 6.4.1(react@18.3.1)(typescript@5.2.2) - react-redux: 8.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(redux@4.2.1) - react-router-dom: 6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - styled-components: 6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - yup: 0.32.9 - transitivePeerDependencies: - - '@babel/runtime' - - '@codemirror/autocomplete' - - '@codemirror/language' - - '@codemirror/lint' - - '@codemirror/search' - - '@codemirror/state' - - '@codemirror/theme-one-dark' - - '@codemirror/view' - - '@types/node' - - '@types/react' - - '@types/react-dom' - - better-sqlite3 - - codemirror - - debug - - encoding - - mysql - - mysql2 - - pg - - pg-native - - react-native - - redux - - sqlite3 - - supports-color - - tedious - - typescript - - '@strapi/data-transfer@4.25.9(@strapi/strapi@4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0))(@types/node@22.5.2)(pg@8.12.0)': - dependencies: - '@strapi/logger': 4.25.9 - '@strapi/strapi': 4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0) - '@strapi/types': 4.25.9(@types/node@22.5.2)(pg@8.12.0) - '@strapi/utils': 4.25.9 - chalk: 4.1.2 - cli-table3: 0.6.5 - commander: 8.3.0 - fs-extra: 10.0.0 - inquirer: 8.2.5 - lodash: 4.17.21 - ora: 5.4.1 - resolve-cwd: 3.0.0 - semver: 7.5.4 - stream-chain: 2.2.5 - stream-json: 1.8.0 - tar: 6.1.13 - tar-stream: 2.2.0 - ws: 8.13.0 - transitivePeerDependencies: - - '@types/node' - - better-sqlite3 - - bufferutil - - encoding - - mysql - - mysql2 - - pg - - pg-native - - sqlite3 - - supports-color - - tedious - - utf-8-validate - - '@strapi/database@4.25.9(@types/node@22.5.2)(pg@8.12.0)': - dependencies: - '@strapi/utils': 4.25.9 - date-fns: 2.30.0 - debug: 4.3.4(supports-color@5.5.0) - fs-extra: 10.0.0 - knex: 2.5.0(pg@8.12.0) - lodash: 4.17.21 - semver: 7.5.4 - umzug: 3.2.1(@types/node@22.5.2) - transitivePeerDependencies: - - '@types/node' - - better-sqlite3 - - mysql - - mysql2 - - pg - - pg-native - - sqlite3 - - supports-color - - tedious - - '@strapi/design-system@1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))': - dependencies: - '@codemirror/lang-json': 6.0.1 - '@floating-ui/react-dom': 2.1.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@internationalized/date': 3.5.5 - '@internationalized/number': 3.5.3 - '@radix-ui/react-dismissable-layer': 1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-dropdown-menu': 2.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-focus-scope': 1.0.4(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@strapi/icons': 1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@strapi/ui-primitives': 1.19.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@uiw/react-codemirror': 4.23.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - aria-hidden: 1.2.4 - compute-scroll-into-view: 3.1.0 - prop-types: 15.8.1 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - react-remove-scroll: 2.5.10(@types/react@18.3.5)(react@18.3.1) - react-router-dom: 6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - styled-components: 6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - transitivePeerDependencies: - - '@babel/runtime' - - '@codemirror/autocomplete' - - '@codemirror/language' - - '@codemirror/lint' - - '@codemirror/search' - - '@codemirror/state' - - '@codemirror/theme-one-dark' - - '@codemirror/view' - - '@types/react' - - '@types/react-dom' - - codemirror - - '@strapi/generate-new@4.25.9': - dependencies: - '@sentry/node': 6.19.7 - chalk: 4.1.2 - execa: 5.1.1 - fs-extra: 10.0.0 - inquirer: 8.2.5 - lodash: 4.17.21 - node-fetch: 2.7.0 - node-machine-id: 1.1.12 - ora: 5.4.1 - semver: 7.5.4 - tar: 6.1.13 - transitivePeerDependencies: - - encoding - - supports-color - - '@strapi/generators@4.25.9': - dependencies: - '@sindresorhus/slugify': 1.1.0 - '@strapi/typescript-utils': 4.25.9 - '@strapi/utils': 4.25.9 - chalk: 4.1.2 - copyfiles: 2.4.1 - fs-extra: 10.0.0 - node-plop: 0.26.3 - plop: 2.7.6 - pluralize: 8.0.0 - transitivePeerDependencies: - - supports-color - - '@strapi/helper-plugin@4.25.9(@strapi/design-system@1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)))(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(debug@4.3.4)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(typescript@5.2.2)': - dependencies: - '@strapi/design-system': 1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)) - '@strapi/icons': 1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - axios: 1.7.4(debug@4.3.4) - date-fns: 2.30.0 - formik: 2.4.0(react@18.3.1) - immer: 9.0.19 - lodash: 4.17.21 - qs: 6.11.1 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - react-helmet: 6.1.0(react@18.3.1) - react-intl: 6.4.1(react@18.3.1)(typescript@5.2.2) - react-query: 3.39.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - react-router-dom: 6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - react-select: 5.7.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - styled-components: 6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - transitivePeerDependencies: - - '@types/react' - - debug - - react-native - - supports-color - - typescript - - '@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - - '@strapi/logger@4.25.9': - dependencies: - lodash: 4.17.21 - winston: 3.10.0 - - '@strapi/pack-up@4.23.0(@swc/helpers@0.5.13)(@types/node@22.5.2)(debug@4.3.4)(terser@5.31.6)': - dependencies: - '@vitejs/plugin-react-swc': 3.5.0(@swc/helpers@0.5.13)(vite@5.0.13(@types/node@22.5.2)(terser@5.31.6)) - boxen: 5.1.2 - browserslist-to-esbuild: 1.2.0 - chalk: 4.1.2 - chokidar: 3.5.3 - commander: 8.3.0 - esbuild: 0.19.11 - esbuild-register: 3.5.0(esbuild@0.19.11) - get-latest-version: 5.1.0(debug@4.3.4) - git-url-parse: 13.1.0 - ini: 4.1.1 - ora: 5.4.1 - outdent: 0.8.0 - pkg-up: 3.1.0 - prettier: 2.8.4 - prettier-plugin-packagejson: 2.4.5(prettier@2.8.4) - prompts: 2.4.2 - rxjs: 7.8.1 - typescript: 5.2.2 - vite: 5.0.13(@types/node@22.5.2)(terser@5.31.6) - yup: 0.32.9 - transitivePeerDependencies: - - '@swc/helpers' - - '@types/node' - - debug - - less - - lightningcss - - sass - - stylus - - sugarss - - supports-color - - terser - - '@strapi/permissions@4.25.9': - dependencies: - '@casl/ability': 6.5.0 - '@strapi/utils': 4.25.9 - lodash: 4.17.21 - qs: 6.11.1 - sift: 16.0.1 - - '@strapi/plugin-cloud@4.25.9(vz4izpmlclysylnr54rpbz6mea)': - dependencies: - '@strapi/design-system': 1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)) - '@strapi/helper-plugin': 4.25.9(@strapi/design-system@1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)))(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(debug@4.3.4)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(typescript@5.2.2) - '@strapi/icons': 1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@strapi/strapi': 4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0) - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - react-intl: 6.4.1(react@18.3.1)(typescript@5.2.2) - react-router-dom: 6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - styled-components: 6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - transitivePeerDependencies: - - '@babel/runtime' - - '@codemirror/autocomplete' - - '@codemirror/language' - - '@codemirror/lint' - - '@codemirror/search' - - '@codemirror/state' - - '@codemirror/theme-one-dark' - - '@codemirror/view' - - '@types/react' - - '@types/react-dom' - - codemirror - - debug - - react-native - - supports-color - - typescript - - '@strapi/plugin-content-manager@4.25.9(@types/node@22.5.2)(pg@8.12.0)': - dependencies: - '@sindresorhus/slugify': 1.1.0 - '@strapi/types': 4.25.9(@types/node@22.5.2)(pg@8.12.0) - '@strapi/utils': 4.25.9 - koa: 2.13.4 - koa-bodyparser: 4.4.1 - lodash: 4.17.21 - qs: 6.11.1 - transitivePeerDependencies: - - '@types/node' - - better-sqlite3 - - encoding - - mysql - - mysql2 - - pg - - pg-native - - sqlite3 - - supports-color - - tedious - - '@strapi/plugin-content-type-builder@4.25.9(z44l57356hx7plbwj2tww36tbi)': - dependencies: - '@reduxjs/toolkit': 1.9.7(react-redux@8.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(redux@4.2.1))(react@18.3.1) - '@sindresorhus/slugify': 1.1.0 - '@strapi/design-system': 1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)) - '@strapi/generators': 4.25.9 - '@strapi/helper-plugin': 4.25.9(@strapi/design-system@1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)))(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(debug@4.3.4)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(typescript@5.2.2) - '@strapi/icons': 1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@strapi/strapi': 4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0) - '@strapi/utils': 4.25.9 - fs-extra: 10.0.0 - immer: 9.0.19 - koa-bodyparser: 4.4.1 - lodash: 4.17.21 - pluralize: 8.0.0 - prop-types: 15.8.1 - qs: 6.11.1 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - react-helmet: 6.1.0(react@18.3.1) - react-intl: 6.4.1(react@18.3.1)(typescript@5.2.2) - react-redux: 8.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(redux@4.2.1) - react-router-dom: 6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - styled-components: 6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - yup: 0.32.9 - transitivePeerDependencies: - - '@babel/runtime' - - '@codemirror/autocomplete' - - '@codemirror/language' - - '@codemirror/lint' - - '@codemirror/search' - - '@codemirror/state' - - '@codemirror/theme-one-dark' - - '@codemirror/view' - - '@types/react' - - '@types/react-dom' - - codemirror - - debug - - react-native - - redux - - supports-color - - typescript - - '@strapi/plugin-email@4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@types/react@18.3.5)(codemirror@5.65.17)(debug@4.3.4)(koa@2.13.4)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(typescript@5.2.2)': - dependencies: - '@strapi/design-system': 1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)) - '@strapi/helper-plugin': 4.25.9(@strapi/design-system@1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)))(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(debug@4.3.4)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(typescript@5.2.2) - '@strapi/icons': 1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@strapi/provider-email-sendmail': 4.25.9 - '@strapi/utils': 4.25.9 - koa: 2.13.4 - lodash: 4.17.21 - prop-types: 15.8.1 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - react-intl: 6.4.1(react@18.3.1)(typescript@5.2.2) - react-query: 3.39.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - react-router-dom: 6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - styled-components: 6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - yup: 0.32.9 - transitivePeerDependencies: - - '@babel/runtime' - - '@codemirror/autocomplete' - - '@codemirror/language' - - '@codemirror/lint' - - '@codemirror/search' - - '@codemirror/state' - - '@codemirror/theme-one-dark' - - '@codemirror/view' - - '@types/react' - - '@types/react-dom' - - codemirror - - debug - - react-native - - supports-color - - typescript - - '@strapi/plugin-i18n@4.25.9(22se43zwxmtuqqjkykcc76uoqe)': - dependencies: - '@reduxjs/toolkit': 1.9.7(react-redux@8.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(redux@4.2.1))(react@18.3.1) - '@strapi/design-system': 1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)) - '@strapi/helper-plugin': 4.25.9(@strapi/design-system@1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)))(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(debug@4.3.4)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(typescript@5.2.2) - '@strapi/icons': 1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@strapi/strapi': 4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0) - '@strapi/utils': 4.25.9 - axios: 1.7.4(debug@4.3.4) - formik: 2.4.0(react@18.3.1) - immer: 9.0.19 - lodash: 4.17.21 - prop-types: 15.8.1 - qs: 6.11.1 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - react-intl: 6.4.1(react@18.3.1)(typescript@5.2.2) - react-query: 3.39.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - react-redux: 8.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(redux@4.2.1) - react-router-dom: 6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - styled-components: 6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - yup: 0.32.9 - transitivePeerDependencies: - - '@babel/runtime' - - '@codemirror/autocomplete' - - '@codemirror/language' - - '@codemirror/lint' - - '@codemirror/search' - - '@codemirror/state' - - '@codemirror/theme-one-dark' - - '@codemirror/view' - - '@types/react' - - '@types/react-dom' - - codemirror - - debug - - react-native - - redux - - supports-color - - typescript - - '@strapi/plugin-upload@4.25.9(wsugwobvzvzlrujvaffvc4jxhe)': - dependencies: - '@strapi/design-system': 1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)) - '@strapi/helper-plugin': 4.25.9(@strapi/design-system@1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)))(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(debug@4.3.4)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(typescript@5.2.2) - '@strapi/icons': 1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@strapi/provider-upload-local': 4.25.9 - '@strapi/strapi': 4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0) - '@strapi/utils': 4.25.9 - axios: 1.7.4(debug@4.3.4) - byte-size: 7.0.1 - cropperjs: 1.6.0 - date-fns: 2.30.0 - formik: 2.4.0(react@18.3.1) - fs-extra: 10.0.0 - immer: 9.0.19 - koa-range: 0.3.0 - koa-static: 5.0.0 - lodash: 4.17.21 - mime-types: 2.1.35 - prop-types: 15.8.1 - qs: 6.11.1 - react: 18.3.1 - react-dnd: 16.0.1(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(react@18.3.1) - react-dom: 18.3.1(react@18.3.1) - react-helmet: 6.1.0(react@18.3.1) - react-intl: 6.4.1(react@18.3.1)(typescript@5.2.2) - react-query: 3.39.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - react-redux: 8.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(redux@4.2.1) - react-router-dom: 6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - react-select: 5.7.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - sharp: 0.32.6 - styled-components: 6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - yup: 0.32.9 - transitivePeerDependencies: - - '@babel/runtime' - - '@codemirror/autocomplete' - - '@codemirror/language' - - '@codemirror/lint' - - '@codemirror/search' - - '@codemirror/state' - - '@codemirror/theme-one-dark' - - '@codemirror/view' - - '@types/hoist-non-react-statics' - - '@types/node' - - '@types/react' - - '@types/react-dom' - - codemirror - - debug - - react-native - - redux - - supports-color - - typescript - - '@strapi/plugin-users-permissions@4.25.9(22se43zwxmtuqqjkykcc76uoqe)': - dependencies: - '@strapi/design-system': 1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)) - '@strapi/helper-plugin': 4.25.9(@strapi/design-system@1.19.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1)))(@strapi/icons@1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@types/react@18.3.5)(debug@4.3.4)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(typescript@5.2.2) - '@strapi/icons': 1.19.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@strapi/strapi': 4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0) - '@strapi/utils': 4.25.9 - bcryptjs: 2.4.3 - formik: 2.4.0(react@18.3.1) - grant-koa: 5.4.8(koa@2.13.4) - immer: 9.0.19 - jsonwebtoken: 9.0.0 - jwk-to-pem: 2.0.5 - koa: 2.13.4 - koa2-ratelimit: 1.1.3 - lodash: 4.17.21 - prop-types: 15.8.1 - purest: 4.0.2 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - react-intl: 6.4.1(react@18.3.1)(typescript@5.2.2) - react-query: 3.39.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - react-redux: 8.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(redux@4.2.1) - react-router-dom: 6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - styled-components: 6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - url-join: 4.0.1 - yup: 0.32.9 - transitivePeerDependencies: - - '@babel/runtime' - - '@codemirror/autocomplete' - - '@codemirror/language' - - '@codemirror/lint' - - '@codemirror/search' - - '@codemirror/state' - - '@codemirror/theme-one-dark' - - '@codemirror/view' - - '@types/react' - - '@types/react-dom' - - codemirror - - debug - - mongoose - - react-native - - redis - - redux - - sequelize - - supports-color - - typescript - - '@strapi/provider-audit-logs-local@4.25.9': {} - - '@strapi/provider-email-sendmail@4.25.9': - dependencies: - '@strapi/utils': 4.25.9 - sendmail: 1.6.1 - - '@strapi/provider-upload-local@4.25.9': - dependencies: - '@strapi/utils': 4.25.9 - fs-extra: 10.0.0 - - '@strapi/strapi@4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0)': - dependencies: - '@koa/cors': 5.0.0 - '@koa/router': 10.1.1 - '@strapi/admin': 4.25.9(kgi7qayoxmh7jdvj26fzdxk4ry) - '@strapi/cloud-cli': 4.25.9(debug@4.3.4) - '@strapi/content-releases': 4.25.9(q7vptuqps424wczizevu7bwk2q) - '@strapi/data-transfer': 4.25.9(@strapi/strapi@4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0))(@types/node@22.5.2)(pg@8.12.0) - '@strapi/database': 4.25.9(@types/node@22.5.2)(pg@8.12.0) - '@strapi/generate-new': 4.25.9 - '@strapi/generators': 4.25.9 - '@strapi/logger': 4.25.9 - '@strapi/pack-up': 4.23.0(@swc/helpers@0.5.13)(@types/node@22.5.2)(debug@4.3.4)(terser@5.31.6) - '@strapi/permissions': 4.25.9 - '@strapi/plugin-content-manager': 4.25.9(@types/node@22.5.2)(pg@8.12.0) - '@strapi/plugin-content-type-builder': 4.25.9(z44l57356hx7plbwj2tww36tbi) - '@strapi/plugin-email': 4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@types/react@18.3.5)(codemirror@5.65.17)(debug@4.3.4)(koa@2.13.4)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(typescript@5.2.2) - '@strapi/plugin-upload': 4.25.9(wsugwobvzvzlrujvaffvc4jxhe) - '@strapi/types': 4.25.9(@types/node@22.5.2)(pg@8.12.0) - '@strapi/typescript-utils': 4.25.9 - '@strapi/utils': 4.25.9 - bcryptjs: 2.4.3 - boxen: 5.1.2 - chalk: 4.1.2 - ci-info: 3.8.0 - cli-progress: 3.12.0 - cli-table3: 0.6.5 - commander: 8.3.0 - concurrently: 8.2.2 - configstore: 5.0.1 - copyfiles: 2.4.1 - debug: 4.3.4(supports-color@5.5.0) - delegates: 1.0.0 - dotenv: 14.2.0 - execa: 5.1.1 - fs-extra: 10.0.0 - get-latest-version: 5.1.0(debug@4.3.4) - git-url-parse: 13.1.0 - glob: 10.4.2 - http-errors: 1.8.1 - https-proxy-agent: 5.0.1 - inquirer: 8.2.5 - is-docker: 2.2.1 - koa: 2.13.4 - koa-body: 4.2.0 - koa-compose: 4.1.0 - koa-compress: 5.1.0 - koa-favicon: 2.1.0 - koa-helmet: 7.0.2 - koa-ip: 2.1.3 - koa-session: 6.4.0 - koa-static: 5.0.0 - lodash: 4.17.21 - mime-types: 2.1.35 - node-fetch: 2.7.0 - node-machine-id: 1.1.12 - node-schedule: 2.1.1 - nodemon: 3.0.2 - open: 8.4.0 - ora: 5.4.1 - outdent: 0.8.0 - package-json: 7.0.0 - pkg-up: 3.1.0 - qs: 6.11.1 - semver: 7.5.4 - statuses: 2.0.1 - typescript: 5.2.2 - yalc: 1.0.0-pre.53 - yup: 0.32.9 - transitivePeerDependencies: - - '@babel/runtime' - - '@codemirror/autocomplete' - - '@codemirror/language' - - '@codemirror/lint' - - '@codemirror/search' - - '@codemirror/state' - - '@codemirror/theme-one-dark' - - '@codemirror/view' - - '@rspack/core' - - '@swc/core' - - '@swc/helpers' - - '@types/hoist-non-react-statics' - - '@types/node' - - '@types/react' - - '@types/react-dom' - - '@types/webpack' - - better-sqlite3 - - bufferutil - - codemirror - - encoding - - less - - lightningcss - - mongoose - - mysql - - mysql2 - - pg - - pg-native - - react - - react-dom - - react-native - - react-router-dom - - redis - - redux - - sass - - sequelize - - sockjs-client - - sqlite3 - - styled-components - - stylus - - sugarss - - supports-color - - tedious - - terser - - type-fest - - uglify-js - - utf-8-validate - - webpack-cli - - webpack-dev-server - - webpack-plugin-serve - - '@strapi/types@4.25.9(@types/node@22.5.2)(pg@8.12.0)': - dependencies: - '@casl/ability': 6.5.0 - '@koa/cors': 5.0.0 - '@koa/router': 10.1.1 - '@strapi/database': 4.25.9(@types/node@22.5.2)(pg@8.12.0) - '@strapi/logger': 4.25.9 - '@strapi/permissions': 4.25.9 - '@strapi/utils': 4.25.9 - commander: 8.3.0 - https-proxy-agent: 5.0.1 - koa: 2.13.4 - node-fetch: 2.7.0 - node-schedule: 2.1.1 - transitivePeerDependencies: - - '@types/node' - - better-sqlite3 - - encoding - - mysql - - mysql2 - - pg - - pg-native - - sqlite3 - - supports-color - - tedious - - '@strapi/typescript-utils@4.25.9': - dependencies: - chalk: 4.1.2 - cli-table3: 0.6.5 - fs-extra: 10.0.0 - lodash: 4.17.21 - prettier: 2.8.4 - typescript: 5.2.2 - - '@strapi/ui-primitives@1.19.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@radix-ui/number': 1.1.0 - '@radix-ui/primitive': 1.1.0 - '@radix-ui/react-collection': 1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-context': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-direction': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-dismissable-layer': 1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-focus-guards': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-focus-scope': 1.0.4(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-id': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-popper': 1.2.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-portal': 1.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-primitive': 1.0.3(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - '@radix-ui/react-slot': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-use-controllable-state': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-use-layout-effect': 1.0.1(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-use-previous': 1.1.0(@types/react@18.3.5)(react@18.3.1) - '@radix-ui/react-visually-hidden': 1.1.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - aria-hidden: 1.2.4 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - react-remove-scroll: 2.5.10(@types/react@18.3.5)(react@18.3.1) - transitivePeerDependencies: - - '@types/react' - - '@types/react-dom' - - '@strapi/utils@4.25.9': - dependencies: - '@sindresorhus/slugify': 1.1.0 - date-fns: 2.30.0 - http-errors: 1.8.1 - lodash: 4.17.21 - p-map: 4.0.0 - yup: 0.32.9 - - '@swc/core-darwin-arm64@1.7.23': - optional: true - - '@swc/core-darwin-x64@1.7.23': - optional: true - - '@swc/core-linux-arm-gnueabihf@1.7.23': - optional: true - - '@swc/core-linux-arm64-gnu@1.7.23': - optional: true - - '@swc/core-linux-arm64-musl@1.7.23': - optional: true - - '@swc/core-linux-x64-gnu@1.7.23': - optional: true - - '@swc/core-linux-x64-musl@1.7.23': - optional: true - - '@swc/core-win32-arm64-msvc@1.7.23': - optional: true - - '@swc/core-win32-ia32-msvc@1.7.23': - optional: true - - '@swc/core-win32-x64-msvc@1.7.23': - optional: true - - '@swc/core@1.7.23(@swc/helpers@0.5.13)': - dependencies: - '@swc/counter': 0.1.3 - '@swc/types': 0.1.12 - optionalDependencies: - '@swc/core-darwin-arm64': 1.7.23 - '@swc/core-darwin-x64': 1.7.23 - '@swc/core-linux-arm-gnueabihf': 1.7.23 - '@swc/core-linux-arm64-gnu': 1.7.23 - '@swc/core-linux-arm64-musl': 1.7.23 - '@swc/core-linux-x64-gnu': 1.7.23 - '@swc/core-linux-x64-musl': 1.7.23 - '@swc/core-win32-arm64-msvc': 1.7.23 - '@swc/core-win32-ia32-msvc': 1.7.23 - '@swc/core-win32-x64-msvc': 1.7.23 - '@swc/helpers': 0.5.13 - - '@swc/counter@0.1.3': {} - - '@swc/helpers@0.5.13': - dependencies: - tslib: 2.7.0 - - '@swc/types@0.1.12': - dependencies: - '@swc/counter': 0.1.3 - - '@szmarczak/http-timer@4.0.6': - dependencies: - defer-to-connect: 2.0.1 - - '@types/argparse@1.0.38': {} - - '@types/body-parser@1.19.5': - dependencies: - '@types/connect': 3.4.38 - '@types/node': 22.5.2 - - '@types/cacheable-request@6.0.3': - dependencies: - '@types/http-cache-semantics': 4.0.4 - '@types/keyv': 3.1.4 - '@types/node': 22.5.2 - '@types/responselike': 1.0.3 - - '@types/connect@3.4.38': - dependencies: - '@types/node': 22.5.2 - - '@types/estree@1.0.5': {} - - '@types/express-serve-static-core@4.19.5': - dependencies: - '@types/node': 22.5.2 - '@types/qs': 6.9.15 - '@types/range-parser': 1.2.7 - '@types/send': 0.17.4 - - '@types/express@4.17.21': - dependencies: - '@types/body-parser': 1.19.5 - '@types/express-serve-static-core': 4.19.5 - '@types/qs': 6.9.15 - '@types/serve-static': 1.15.7 - - '@types/fined@1.1.5': {} - - '@types/follow-redirects@1.14.4': - dependencies: - '@types/node': 22.5.2 - - '@types/formidable@1.2.8': - dependencies: - '@types/node': 22.5.2 - - '@types/glob@7.2.0': - dependencies: - '@types/minimatch': 5.1.2 - '@types/node': 22.5.2 - - '@types/hoist-non-react-statics@3.3.5': - dependencies: - '@types/react': 18.3.5 - hoist-non-react-statics: 3.3.2 - - '@types/html-minifier-terser@6.1.0': {} - - '@types/http-cache-semantics@4.0.4': {} - - '@types/http-errors@2.0.4': {} - - '@types/inquirer@6.5.0': - dependencies: - '@types/through': 0.0.33 - rxjs: 6.6.7 - - '@types/interpret@1.1.3': - dependencies: - '@types/node': 22.5.2 - - '@types/is-hotkey@0.1.10': {} - - '@types/json-schema@7.0.15': {} - - '@types/jsonwebtoken@9.0.6': - dependencies: - '@types/node': 22.5.2 - - '@types/keyv@3.1.4': - dependencies: - '@types/node': 22.5.2 - - '@types/liftoff@2.5.1': - dependencies: - '@types/fined': 1.1.5 - '@types/interpret': 1.1.3 - '@types/node': 22.5.2 - - '@types/lodash@4.17.7': {} - - '@types/mime@1.3.5': {} - - '@types/minimatch@5.1.2': {} - - '@types/node-fetch@2.6.11': - dependencies: - '@types/node': 18.19.48 - form-data: 4.0.0 - - '@types/node@18.19.48': - dependencies: - undici-types: 5.26.5 - - '@types/node@22.5.2': - dependencies: - undici-types: 6.19.8 - - '@types/normalize-package-data@2.4.4': {} - - '@types/parse-json@4.0.2': {} - - '@types/progress-stream@2.0.5': - dependencies: - '@types/node': 22.5.2 - - '@types/prop-types@15.7.12': {} - - '@types/qs@6.9.15': {} - - '@types/range-parser@1.2.7': {} - - '@types/react-transition-group@4.4.11': - dependencies: - '@types/react': 18.3.5 - - '@types/react@18.3.5': - dependencies: - '@types/prop-types': 15.7.12 - csstype: 3.1.3 - - '@types/responselike@1.0.3': - dependencies: - '@types/node': 22.5.2 - - '@types/send@0.17.4': - dependencies: - '@types/mime': 1.3.5 - '@types/node': 22.5.2 - - '@types/serve-static@1.15.7': - dependencies: - '@types/http-errors': 2.0.4 - '@types/node': 22.5.2 - '@types/send': 0.17.4 - - '@types/stylis@4.2.5': {} - - '@types/through@0.0.33': - dependencies: - '@types/node': 22.5.2 - - '@types/triple-beam@1.3.5': {} - - '@types/use-sync-external-store@0.0.3': {} - - '@ucast/core@1.10.2': {} - - '@ucast/js@3.0.4': - dependencies: - '@ucast/core': 1.10.2 - - '@ucast/mongo2js@1.3.4': - dependencies: - '@ucast/core': 1.10.2 - '@ucast/js': 3.0.4 - '@ucast/mongo': 2.4.3 - - '@ucast/mongo@2.4.3': - dependencies: - '@ucast/core': 1.10.2 - - '@uiw/codemirror-extensions-basic-setup@4.23.0(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/commands@6.6.1)(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)': - dependencies: - '@codemirror/autocomplete': 6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1) - '@codemirror/commands': 6.6.1 - '@codemirror/language': 6.10.2 - '@codemirror/lint': 6.8.1 - '@codemirror/search': 6.5.6 - '@codemirror/state': 6.4.1 - '@codemirror/view': 6.33.0 - - '@uiw/react-codemirror@4.23.0(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(codemirror@5.65.17)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': - dependencies: - '@babel/runtime': 7.25.6 - '@codemirror/commands': 6.6.1 - '@codemirror/state': 6.4.1 - '@codemirror/theme-one-dark': 6.1.2 - '@codemirror/view': 6.33.0 - '@uiw/codemirror-extensions-basic-setup': 4.23.0(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/commands@6.6.1)(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0) - codemirror: 5.65.17 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - transitivePeerDependencies: - - '@codemirror/autocomplete' - - '@codemirror/language' - - '@codemirror/lint' - - '@codemirror/search' - - '@vitejs/plugin-react-swc@3.5.0(@swc/helpers@0.5.13)(vite@5.0.13(@types/node@22.5.2)(terser@5.31.6))': - dependencies: - '@swc/core': 1.7.23(@swc/helpers@0.5.13) - vite: 5.0.13(@types/node@22.5.2)(terser@5.31.6) - transitivePeerDependencies: - - '@swc/helpers' - - '@webassemblyjs/ast@1.12.1': - dependencies: - '@webassemblyjs/helper-numbers': 1.11.6 - '@webassemblyjs/helper-wasm-bytecode': 1.11.6 - - '@webassemblyjs/floating-point-hex-parser@1.11.6': {} - - '@webassemblyjs/helper-api-error@1.11.6': {} - - '@webassemblyjs/helper-buffer@1.12.1': {} - - '@webassemblyjs/helper-numbers@1.11.6': - dependencies: - '@webassemblyjs/floating-point-hex-parser': 1.11.6 - '@webassemblyjs/helper-api-error': 1.11.6 - '@xtuc/long': 4.2.2 - - '@webassemblyjs/helper-wasm-bytecode@1.11.6': {} - - '@webassemblyjs/helper-wasm-section@1.12.1': - dependencies: - '@webassemblyjs/ast': 1.12.1 - '@webassemblyjs/helper-buffer': 1.12.1 - '@webassemblyjs/helper-wasm-bytecode': 1.11.6 - '@webassemblyjs/wasm-gen': 1.12.1 - - '@webassemblyjs/ieee754@1.11.6': - dependencies: - '@xtuc/ieee754': 1.2.0 - - '@webassemblyjs/leb128@1.11.6': - dependencies: - '@xtuc/long': 4.2.2 - - '@webassemblyjs/utf8@1.11.6': {} - - '@webassemblyjs/wasm-edit@1.12.1': - dependencies: - '@webassemblyjs/ast': 1.12.1 - '@webassemblyjs/helper-buffer': 1.12.1 - '@webassemblyjs/helper-wasm-bytecode': 1.11.6 - '@webassemblyjs/helper-wasm-section': 1.12.1 - '@webassemblyjs/wasm-gen': 1.12.1 - '@webassemblyjs/wasm-opt': 1.12.1 - '@webassemblyjs/wasm-parser': 1.12.1 - '@webassemblyjs/wast-printer': 1.12.1 - - '@webassemblyjs/wasm-gen@1.12.1': - dependencies: - '@webassemblyjs/ast': 1.12.1 - '@webassemblyjs/helper-wasm-bytecode': 1.11.6 - '@webassemblyjs/ieee754': 1.11.6 - '@webassemblyjs/leb128': 1.11.6 - '@webassemblyjs/utf8': 1.11.6 - - '@webassemblyjs/wasm-opt@1.12.1': - dependencies: - '@webassemblyjs/ast': 1.12.1 - '@webassemblyjs/helper-buffer': 1.12.1 - '@webassemblyjs/wasm-gen': 1.12.1 - '@webassemblyjs/wasm-parser': 1.12.1 - - '@webassemblyjs/wasm-parser@1.12.1': - dependencies: - '@webassemblyjs/ast': 1.12.1 - '@webassemblyjs/helper-api-error': 1.11.6 - '@webassemblyjs/helper-wasm-bytecode': 1.11.6 - '@webassemblyjs/ieee754': 1.11.6 - '@webassemblyjs/leb128': 1.11.6 - '@webassemblyjs/utf8': 1.11.6 - - '@webassemblyjs/wast-printer@1.12.1': - dependencies: - '@webassemblyjs/ast': 1.12.1 - '@xtuc/long': 4.2.2 - - '@xtuc/ieee754@1.2.0': {} - - '@xtuc/long@4.2.2': {} - - abbrev@1.1.1: {} - - abort-controller@3.0.0: - dependencies: - event-target-shim: 5.0.1 - - accepts@1.3.8: - dependencies: - mime-types: 2.1.35 - negotiator: 0.6.3 - - acorn-import-attributes@1.9.5(acorn@8.12.1): - dependencies: - acorn: 8.12.1 - - acorn-walk@8.3.3: - dependencies: - acorn: 8.12.1 - - acorn@8.12.1: {} - - addressparser@1.0.1: {} - - agent-base@6.0.2: - dependencies: - debug: 4.3.4(supports-color@5.5.0) - transitivePeerDependencies: - - supports-color - - agentkeepalive@4.5.0: - dependencies: - humanize-ms: 1.2.1 - - aggregate-error@3.1.0: - dependencies: - clean-stack: 2.2.0 - indent-string: 4.0.0 - - ajv-draft-04@1.0.0(ajv@8.13.0): - optionalDependencies: - ajv: 8.13.0 - - ajv-formats@2.1.1(ajv@8.17.1): - optionalDependencies: - ajv: 8.17.1 - - ajv-formats@3.0.1(ajv@8.13.0): - optionalDependencies: - ajv: 8.13.0 - - ajv-keywords@3.5.2(ajv@6.12.6): - dependencies: - ajv: 6.12.6 - - ajv-keywords@5.1.0(ajv@8.17.1): - dependencies: - ajv: 8.17.1 - fast-deep-equal: 3.1.3 - - ajv@6.12.6: - dependencies: - fast-deep-equal: 3.1.3 - fast-json-stable-stringify: 2.1.0 - json-schema-traverse: 0.4.1 - uri-js: 4.4.1 - - ajv@8.13.0: - dependencies: - fast-deep-equal: 3.1.3 - json-schema-traverse: 1.0.0 - require-from-string: 2.0.2 - uri-js: 4.4.1 - - ajv@8.17.1: - dependencies: - fast-deep-equal: 3.1.3 - fast-uri: 3.0.1 - json-schema-traverse: 1.0.0 - require-from-string: 2.0.2 - - ansi-align@3.0.1: - dependencies: - string-width: 4.2.3 - - ansi-escapes@4.3.2: - dependencies: - type-fest: 0.21.3 - - ansi-html-community@0.0.8: {} - - ansi-regex@2.1.1: {} - - ansi-regex@4.1.1: {} - - ansi-regex@5.0.1: {} - - ansi-regex@6.0.1: {} - - ansi-styles@2.2.1: {} - - ansi-styles@3.2.1: - dependencies: - color-convert: 1.9.3 - - ansi-styles@4.3.0: - dependencies: - color-convert: 2.0.1 - - ansi-styles@6.2.1: {} - - any-promise@1.3.0: {} - - anymatch@3.1.3: - dependencies: - normalize-path: 3.0.0 - picomatch: 2.3.1 - - aproba@2.0.0: {} - - are-we-there-yet@2.0.0: - dependencies: - delegates: 1.0.0 - readable-stream: 3.6.2 - - argparse@1.0.10: - dependencies: - sprintf-js: 1.0.3 - - argparse@2.0.1: {} - - aria-hidden@1.2.4: - dependencies: - tslib: 2.7.0 - - arr-diff@4.0.0: {} - - arr-flatten@1.1.0: {} - - arr-union@3.1.0: {} - - array-each@1.0.1: {} - - array-slice@1.1.0: {} - - array-union@2.1.0: {} - - array-unique@0.3.2: {} - - asn1.js@5.4.1: - dependencies: - bn.js: 4.12.0 - inherits: 2.0.4 - minimalistic-assert: 1.0.1 - safer-buffer: 2.1.2 - - assign-symbols@1.0.0: {} - - async@3.2.6: {} - - asynckit@0.4.0: {} - - atob@2.1.2: {} - - axios@1.7.4(debug@4.3.4): - dependencies: - follow-redirects: 1.15.6(debug@4.3.4) - form-data: 4.0.0 - proxy-from-env: 1.1.0 - transitivePeerDependencies: - - debug - - b4a@1.6.6: {} - - babel-plugin-macros@3.1.0: - dependencies: - '@babel/runtime': 7.25.6 - cosmiconfig: 7.1.0 - resolve: 1.22.8 - - balanced-match@1.0.2: {} - - bare-events@2.4.2: - optional: true - - bare-fs@2.3.3: - dependencies: - bare-events: 2.4.2 - bare-path: 2.1.3 - bare-stream: 2.2.0 - optional: true - - bare-os@2.4.2: - optional: true - - bare-path@2.1.3: - dependencies: - bare-os: 2.4.2 - optional: true - - bare-stream@2.2.0: - dependencies: - streamx: 2.20.0 - optional: true - - base64-js@1.5.1: {} - - base@0.11.2: - dependencies: - cache-base: 1.0.1 - class-utils: 0.3.6 - component-emitter: 1.3.1 - define-property: 1.0.0 - isobject: 3.0.1 - mixin-deep: 1.3.2 - pascalcase: 0.1.1 - - bcryptjs@2.4.3: {} - - big-integer@1.6.52: {} - - big.js@5.2.2: {} - - binary-extensions@2.3.0: {} - - bl@4.1.0: - dependencies: - buffer: 5.7.1 - inherits: 2.0.4 - readable-stream: 3.6.2 - - bn.js@4.12.0: {} - - boolbase@1.0.0: {} - - bowser@2.11.0: {} - - boxen@5.1.2: - dependencies: - ansi-align: 3.0.1 - camelcase: 6.3.0 - chalk: 4.1.2 - cli-boxes: 2.2.1 - string-width: 4.2.3 - type-fest: 0.20.2 - widest-line: 3.1.0 - wrap-ansi: 7.0.0 - - bplist-parser@0.2.0: - dependencies: - big-integer: 1.6.52 - - brace-expansion@1.1.11: - dependencies: - balanced-match: 1.0.2 - concat-map: 0.0.1 - - brace-expansion@2.0.1: - dependencies: - balanced-match: 1.0.2 - - braces@2.3.2: - dependencies: - arr-flatten: 1.1.0 - array-unique: 0.3.2 - extend-shallow: 2.0.1 - fill-range: 4.0.0 - isobject: 3.0.1 - repeat-element: 1.1.4 - snapdragon: 0.8.2 - snapdragon-node: 2.1.1 - split-string: 3.1.0 - to-regex: 3.0.2 - transitivePeerDependencies: - - supports-color - - braces@3.0.3: - dependencies: - fill-range: 7.1.1 - - broadcast-channel@3.7.0: - dependencies: - '@babel/runtime': 7.25.6 - detect-node: 2.1.0 - js-sha3: 0.8.0 - microseconds: 0.2.0 - nano-time: 1.0.0 - oblivious-set: 1.0.0 - rimraf: 3.0.2 - unload: 2.2.0 - - brorand@1.1.0: {} - - browserslist-to-esbuild@1.2.0: - dependencies: - browserslist: 4.23.3 - - browserslist@4.23.3: - dependencies: - caniuse-lite: 1.0.30001655 - electron-to-chromium: 1.5.13 - node-releases: 2.0.18 - update-browserslist-db: 1.1.0(browserslist@4.23.3) - - buffer-equal-constant-time@1.0.1: {} - - buffer-from@1.1.2: {} - - buffer@5.7.1: - dependencies: - base64-js: 1.5.1 - ieee754: 1.2.1 - - buildmail@3.10.0: - dependencies: - addressparser: 1.0.1 - libbase64: 0.1.0 - libmime: 2.1.0 - libqp: 1.1.0 - nodemailer-fetch: 1.6.0 - nodemailer-shared: 1.1.0 - - bundle-name@3.0.0: - dependencies: - run-applescript: 5.0.0 - - byte-size@7.0.1: {} - - bytes@3.1.2: {} - - cache-base@1.0.1: - dependencies: - collection-visit: 1.0.0 - component-emitter: 1.3.1 - get-value: 2.0.6 - has-value: 1.0.0 - isobject: 3.0.1 - set-value: 2.0.1 - to-object-path: 0.3.0 - union-value: 1.0.1 - unset-value: 1.0.0 - - cache-content-type@1.0.1: - dependencies: - mime-types: 2.1.35 - ylru: 1.4.0 - - cacheable-lookup@5.0.4: {} - - cacheable-request@7.0.4: - dependencies: - clone-response: 1.0.3 - get-stream: 5.2.0 - http-cache-semantics: 4.1.1 - keyv: 4.5.4 - lowercase-keys: 2.0.0 - normalize-url: 6.1.0 - responselike: 2.0.1 - - call-bind@1.0.7: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - set-function-length: 1.2.2 - - callsites@3.1.0: {} - - camel-case@3.0.0: - dependencies: - no-case: 2.3.2 - upper-case: 1.1.3 - - camel-case@4.1.2: - dependencies: - pascal-case: 3.1.2 - tslib: 2.7.0 - - camelcase@6.3.0: {} - - camelize@1.0.1: {} - - caniuse-lite@1.0.30001655: {} - - canvas@2.11.2: - dependencies: - '@mapbox/node-pre-gyp': 1.0.11 - nan: 2.20.0 - simple-get: 3.1.1 - transitivePeerDependencies: - - encoding - - supports-color - - chalk@1.1.3: - dependencies: - ansi-styles: 2.2.1 - escape-string-regexp: 1.0.5 - has-ansi: 2.0.0 - strip-ansi: 3.0.1 - supports-color: 2.0.0 - - chalk@2.4.2: - dependencies: - ansi-styles: 3.2.1 - escape-string-regexp: 1.0.5 - supports-color: 5.5.0 - - chalk@4.1.2: - dependencies: - ansi-styles: 4.3.0 - supports-color: 7.2.0 - - change-case@3.1.0: - dependencies: - camel-case: 3.0.0 - constant-case: 2.0.0 - dot-case: 2.1.1 - header-case: 1.0.1 - is-lower-case: 1.1.3 - is-upper-case: 1.1.2 - lower-case: 1.1.4 - lower-case-first: 1.0.2 - no-case: 2.3.2 - param-case: 2.1.1 - pascal-case: 2.0.1 - path-case: 2.1.1 - sentence-case: 2.1.1 - snake-case: 2.1.0 - swap-case: 1.1.2 - title-case: 2.1.1 - upper-case: 1.1.3 - upper-case-first: 1.1.2 - - chardet@0.7.0: {} - - chokidar@3.5.3: - dependencies: - anymatch: 3.1.3 - braces: 3.0.3 - glob-parent: 5.1.2 - is-binary-path: 2.1.0 - is-glob: 4.0.3 - normalize-path: 3.0.0 - readdirp: 3.6.0 - optionalDependencies: - fsevents: 2.3.3 - - chokidar@3.6.0: - dependencies: - anymatch: 3.1.3 - braces: 3.0.3 - glob-parent: 5.1.2 - is-binary-path: 2.1.0 - is-glob: 4.0.3 - normalize-path: 3.0.0 - readdirp: 3.6.0 - optionalDependencies: - fsevents: 2.3.3 - - chownr@1.1.4: {} - - chownr@2.0.0: {} - - chrome-trace-event@1.0.4: {} - - ci-info@3.8.0: {} - - class-utils@0.3.6: - dependencies: - arr-union: 3.1.0 - define-property: 0.2.5 - isobject: 3.0.1 - static-extend: 0.1.2 - - clean-css@5.3.3: - dependencies: - source-map: 0.6.1 - - clean-stack@2.2.0: {} - - cli-boxes@2.2.1: {} - - cli-cursor@2.1.0: - dependencies: - restore-cursor: 2.0.0 - - cli-cursor@3.1.0: - dependencies: - restore-cursor: 3.1.0 - - cli-progress@3.12.0: - dependencies: - string-width: 4.2.3 - - cli-spinners@2.9.2: {} - - cli-table3@0.6.5: - dependencies: - string-width: 4.2.3 - optionalDependencies: - '@colors/colors': 1.5.0 - - cli-width@3.0.0: {} - - cliui@7.0.4: - dependencies: - string-width: 4.2.3 - strip-ansi: 6.0.1 - wrap-ansi: 7.0.0 - - cliui@8.0.1: - dependencies: - string-width: 4.2.3 - strip-ansi: 6.0.1 - wrap-ansi: 7.0.0 - - clone-response@1.0.3: - dependencies: - mimic-response: 1.0.1 - - clone@1.0.4: {} - - co-body@5.2.0: - dependencies: - inflation: 2.1.0 - qs: 6.11.1 - raw-body: 2.5.2 - type-is: 1.6.18 - - co-body@6.2.0: - dependencies: - '@hapi/bourne': 3.0.0 - inflation: 2.1.0 - qs: 6.11.1 - raw-body: 2.5.2 - type-is: 1.6.18 - - co@4.6.0: {} - - codemirror@5.65.17: {} - - collection-visit@1.0.0: - dependencies: - map-visit: 1.0.0 - object-visit: 1.0.1 - - color-convert@1.9.3: - dependencies: - color-name: 1.1.3 - - color-convert@2.0.1: - dependencies: - color-name: 1.1.4 - - color-name@1.1.3: {} - - color-name@1.1.4: {} - - color-string@1.9.1: - dependencies: - color-name: 1.1.4 - simple-swizzle: 0.2.2 - - color-support@1.1.3: {} - - color@3.2.1: - dependencies: - color-convert: 1.9.3 - color-string: 1.9.1 - - color@4.2.3: - dependencies: - color-convert: 2.0.1 - color-string: 1.9.1 - - colorette@2.0.19: {} - - colorette@2.0.20: {} - - colorspace@1.1.4: - dependencies: - color: 3.2.1 - text-hex: 1.0.0 - - combined-stream@1.0.8: - dependencies: - delayed-stream: 1.0.0 - - commander@10.0.1: {} - - commander@2.20.3: {} - - commander@7.2.0: {} - - commander@8.3.0: {} - - common-path-prefix@3.0.0: {} - - component-emitter@1.3.1: {} - - compressible@2.0.18: - dependencies: - mime-db: 1.53.0 - - compute-scroll-into-view@1.0.20: {} - - compute-scroll-into-view@3.1.0: {} - - concat-map@0.0.1: {} - - concurrently@8.2.2: - dependencies: - chalk: 4.1.2 - date-fns: 2.30.0 - lodash: 4.17.21 - rxjs: 7.8.1 - shell-quote: 1.8.1 - spawn-command: 0.0.2 - supports-color: 8.1.1 - tree-kill: 1.2.2 - yargs: 17.7.2 - - config-chain@1.1.13: - dependencies: - ini: 1.3.8 - proto-list: 1.2.4 - - configstore@5.0.1: - dependencies: - dot-prop: 5.3.0 - graceful-fs: 4.2.11 - make-dir: 3.1.0 - unique-string: 2.0.0 - write-file-atomic: 3.0.3 - xdg-basedir: 4.0.0 - - console-control-strings@1.1.0: {} - - constant-case@2.0.0: - dependencies: - snake-case: 2.1.0 - upper-case: 1.1.3 - - content-disposition@0.5.4: - dependencies: - safe-buffer: 5.2.1 - - content-type@1.0.5: {} - - convert-source-map@1.9.0: {} - - cookie-signature@1.2.1: - optional: true - - cookie@0.4.2: {} - - cookie@0.5.0: - optional: true - - cookies@0.8.0: - dependencies: - depd: 2.0.0 - keygrip: 1.1.0 - - copy-descriptor@0.1.1: {} - - copy-to@2.0.1: {} - - copyfiles@2.4.1: - dependencies: - glob: 7.2.3 - minimatch: 3.1.2 - mkdirp: 1.0.4 - noms: 0.0.0 - through2: 2.0.5 - untildify: 4.0.0 - yargs: 16.2.0 - - core-js-pure@3.38.1: {} - - core-util-is@1.0.3: {} - - cosmiconfig@7.1.0: - dependencies: - '@types/parse-json': 4.0.2 - import-fresh: 3.3.0 - parse-json: 5.2.0 - path-type: 4.0.0 - yaml: 1.10.2 - - cosmiconfig@8.3.6(typescript@5.2.2): - dependencies: - import-fresh: 3.3.0 - js-yaml: 4.1.0 - parse-json: 5.2.0 - path-type: 4.0.0 - optionalDependencies: - typescript: 5.2.2 - - crc@3.8.0: - dependencies: - buffer: 5.7.1 - - crelt@1.0.6: {} - - cron-parser@4.9.0: - dependencies: - luxon: 3.5.0 - - cropperjs@1.6.0: {} - - cross-env@7.0.3: - dependencies: - cross-spawn: 7.0.3 - - cross-spawn@7.0.3: - dependencies: - path-key: 3.1.1 - shebang-command: 2.0.0 - which: 2.0.2 - - crypto-random-string@2.0.0: {} - - css-color-keywords@1.0.0: {} - - css-loader@6.11.0(webpack@5.94.0(esbuild@0.19.11)): - dependencies: - icss-utils: 5.1.0(postcss@8.4.44) - postcss: 8.4.44 - postcss-modules-extract-imports: 3.1.0(postcss@8.4.44) - postcss-modules-local-by-default: 4.0.5(postcss@8.4.44) - postcss-modules-scope: 3.2.0(postcss@8.4.44) - postcss-modules-values: 4.0.0(postcss@8.4.44) - postcss-value-parser: 4.2.0 - semver: 7.5.4 - optionalDependencies: - webpack: 5.94.0(esbuild@0.19.11) - - css-select@4.3.0: - dependencies: - boolbase: 1.0.0 - css-what: 6.1.0 - domhandler: 4.3.1 - domutils: 2.8.0 - nth-check: 2.1.1 - - css-to-react-native@3.2.0: - dependencies: - camelize: 1.0.1 - css-color-keywords: 1.0.0 - postcss-value-parser: 4.2.0 - - css-what@6.1.0: {} - - cssesc@3.0.0: {} - - csstype@3.1.3: {} - - data-uri-to-buffer@4.0.1: {} - - date-fns-tz@2.0.0(date-fns@2.30.0): - dependencies: - date-fns: 2.30.0 - - date-fns@2.30.0: - dependencies: - '@babel/runtime': 7.25.6 - - date-fns@3.6.0: {} - - debounce@1.2.1: {} - - debug@2.6.9: - dependencies: - ms: 2.0.0 - - debug@3.2.7: - dependencies: - ms: 2.1.3 - - debug@4.3.4(supports-color@5.5.0): - dependencies: - ms: 2.1.2 - optionalDependencies: - supports-color: 5.5.0 - - debug@4.3.6: - dependencies: - ms: 2.1.2 - - decode-uri-component@0.2.2: {} - - decompress-response@4.2.1: - dependencies: - mimic-response: 2.1.0 - - decompress-response@6.0.0: - dependencies: - mimic-response: 3.1.0 - - decompress-response@7.0.0: - dependencies: - mimic-response: 3.1.0 - - deep-equal@1.0.1: {} - - deep-extend@0.6.0: {} - - deepmerge@2.2.1: {} - - deepmerge@4.3.1: {} - - default-browser-id@3.0.0: - dependencies: - bplist-parser: 0.2.0 - untildify: 4.0.0 - - default-browser@4.0.0: - dependencies: - bundle-name: 3.0.0 - default-browser-id: 3.0.0 - execa: 7.2.0 - titleize: 3.0.0 - - defaults@1.0.4: - dependencies: - clone: 1.0.4 - - defer-to-connect@2.0.1: {} - - define-data-property@1.1.4: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - gopd: 1.0.1 - - define-lazy-prop@2.0.0: {} - - define-lazy-prop@3.0.0: {} - - define-property@0.2.5: - dependencies: - is-descriptor: 0.1.7 - - define-property@1.0.0: - dependencies: - is-descriptor: 1.0.3 - - define-property@2.0.2: - dependencies: - is-descriptor: 1.0.3 - isobject: 3.0.1 - - del@5.1.0: - dependencies: - globby: 10.0.2 - graceful-fs: 4.2.11 - is-glob: 4.0.3 - is-path-cwd: 2.2.0 - is-path-inside: 3.0.3 - p-map: 3.0.0 - rimraf: 3.0.2 - slash: 3.0.0 - - delayed-stream@1.0.0: {} - - delegates@1.0.0: {} - - depd@1.1.2: {} - - depd@2.0.0: {} - - destroy@1.2.0: {} - - detect-file@1.0.0: {} - - detect-indent@6.1.0: {} - - detect-indent@7.0.1: {} - - detect-libc@2.0.3: {} - - detect-newline@4.0.1: {} - - detect-node-es@1.1.0: {} - - detect-node@2.1.0: {} - - dir-glob@3.0.1: - dependencies: - path-type: 4.0.0 - - direction@1.0.4: {} - - dkim-signer@0.2.2: - dependencies: - libmime: 2.1.3 - - dnd-core@16.0.1: - dependencies: - '@react-dnd/asap': 5.0.2 - '@react-dnd/invariant': 4.0.2 - redux: 4.2.1 - - dom-converter@0.2.0: - dependencies: - utila: 0.4.0 - - dom-helpers@5.2.1: - dependencies: - '@babel/runtime': 7.25.6 - csstype: 3.1.3 - - dom-serializer@1.4.1: - dependencies: - domelementtype: 2.3.0 - domhandler: 4.3.1 - entities: 2.2.0 - - dom-serializer@2.0.0: - dependencies: - domelementtype: 2.3.0 - domhandler: 5.0.3 - entities: 4.5.0 - - domelementtype@2.3.0: {} - - domhandler@4.3.1: - dependencies: - domelementtype: 2.3.0 - - domhandler@5.0.3: - dependencies: - domelementtype: 2.3.0 - - domutils@2.8.0: - dependencies: - dom-serializer: 1.4.1 - domelementtype: 2.3.0 - domhandler: 4.3.1 - - domutils@3.1.0: - dependencies: - dom-serializer: 2.0.0 - domelementtype: 2.3.0 - domhandler: 5.0.3 - - dot-case@2.1.1: - dependencies: - no-case: 2.3.2 - - dot-case@3.0.4: - dependencies: - no-case: 3.0.4 - tslib: 2.7.0 - - dot-prop@5.3.0: - dependencies: - is-obj: 2.0.0 - - dotenv@14.2.0: {} - - dotenv@16.4.5: {} - - duplexer@0.1.2: {} - - eastasianwidth@0.2.0: {} - - ecdsa-sig-formatter@1.0.11: - dependencies: - safe-buffer: 5.2.1 - - ee-first@1.1.1: {} - - electron-to-chromium@1.5.13: {} - - elliptic@6.5.7: - dependencies: - bn.js: 4.12.0 - brorand: 1.1.0 - hash.js: 1.1.7 - hmac-drbg: 1.0.1 - inherits: 2.0.4 - minimalistic-assert: 1.0.1 - minimalistic-crypto-utils: 1.0.1 - - emittery@0.12.1: {} - - emoji-regex@8.0.0: {} - - emoji-regex@9.2.2: {} - - emojis-list@3.0.0: {} - - enabled@2.0.0: {} - - encodeurl@1.0.2: {} - - end-of-stream@1.4.4: - dependencies: - once: 1.4.0 - - enhanced-resolve@5.17.1: - dependencies: - graceful-fs: 4.2.11 - tapable: 2.2.1 - - entities@2.1.0: {} - - entities@2.2.0: {} - - entities@4.5.0: {} - - error-ex@1.3.2: - dependencies: - is-arrayish: 0.2.1 - - error-stack-parser@2.1.4: - dependencies: - stackframe: 1.3.4 - - es-define-property@1.0.0: - dependencies: - get-intrinsic: 1.2.4 - - es-errors@1.3.0: {} - - es-module-lexer@1.5.4: {} - - esbuild-loader@2.21.0(webpack@5.94.0(esbuild@0.19.11)): - dependencies: - esbuild: 0.16.17 - joycon: 3.1.1 - json5: 2.2.3 - loader-utils: 2.0.4 - tapable: 2.2.1 - webpack: 5.94.0(esbuild@0.19.11) - webpack-sources: 1.4.3 - - esbuild-register@3.5.0(esbuild@0.19.11): - dependencies: - debug: 4.3.4(supports-color@5.5.0) - esbuild: 0.19.11 - transitivePeerDependencies: - - supports-color - - esbuild@0.16.17: - optionalDependencies: - '@esbuild/android-arm': 0.16.17 - '@esbuild/android-arm64': 0.16.17 - '@esbuild/android-x64': 0.16.17 - '@esbuild/darwin-arm64': 0.16.17 - '@esbuild/darwin-x64': 0.16.17 - '@esbuild/freebsd-arm64': 0.16.17 - '@esbuild/freebsd-x64': 0.16.17 - '@esbuild/linux-arm': 0.16.17 - '@esbuild/linux-arm64': 0.16.17 - '@esbuild/linux-ia32': 0.16.17 - '@esbuild/linux-loong64': 0.16.17 - '@esbuild/linux-mips64el': 0.16.17 - '@esbuild/linux-ppc64': 0.16.17 - '@esbuild/linux-riscv64': 0.16.17 - '@esbuild/linux-s390x': 0.16.17 - '@esbuild/linux-x64': 0.16.17 - '@esbuild/netbsd-x64': 0.16.17 - '@esbuild/openbsd-x64': 0.16.17 - '@esbuild/sunos-x64': 0.16.17 - '@esbuild/win32-arm64': 0.16.17 - '@esbuild/win32-ia32': 0.16.17 - '@esbuild/win32-x64': 0.16.17 - - esbuild@0.19.11: - optionalDependencies: - '@esbuild/aix-ppc64': 0.19.11 - '@esbuild/android-arm': 0.19.11 - '@esbuild/android-arm64': 0.19.11 - '@esbuild/android-x64': 0.19.11 - '@esbuild/darwin-arm64': 0.19.11 - '@esbuild/darwin-x64': 0.19.11 - '@esbuild/freebsd-arm64': 0.19.11 - '@esbuild/freebsd-x64': 0.19.11 - '@esbuild/linux-arm': 0.19.11 - '@esbuild/linux-arm64': 0.19.11 - '@esbuild/linux-ia32': 0.19.11 - '@esbuild/linux-loong64': 0.19.11 - '@esbuild/linux-mips64el': 0.19.11 - '@esbuild/linux-ppc64': 0.19.11 - '@esbuild/linux-riscv64': 0.19.11 - '@esbuild/linux-s390x': 0.19.11 - '@esbuild/linux-x64': 0.19.11 - '@esbuild/netbsd-x64': 0.19.11 - '@esbuild/openbsd-x64': 0.19.11 - '@esbuild/sunos-x64': 0.19.11 - '@esbuild/win32-arm64': 0.19.11 - '@esbuild/win32-ia32': 0.19.11 - '@esbuild/win32-x64': 0.19.11 - - escalade@3.2.0: {} - - escape-html@1.0.3: {} - - escape-string-regexp@1.0.5: {} - - escape-string-regexp@2.0.0: {} - - escape-string-regexp@4.0.0: {} - - eslint-scope@5.1.1: - dependencies: - esrecurse: 4.3.0 - estraverse: 4.3.0 - - esm@3.2.25: {} - - esrecurse@4.3.0: - dependencies: - estraverse: 5.3.0 - - estraverse@4.3.0: {} - - estraverse@5.3.0: {} - - event-target-shim@5.0.1: {} - - eventemitter3@4.0.7: {} - - events@3.3.0: {} - - eventsource@2.0.2: {} - - execa@5.1.1: - dependencies: - cross-spawn: 7.0.3 - get-stream: 6.0.1 - human-signals: 2.1.0 - is-stream: 2.0.1 - merge-stream: 2.0.0 - npm-run-path: 4.0.1 - onetime: 5.1.2 - signal-exit: 3.0.7 - strip-final-newline: 2.0.0 - - execa@7.2.0: - dependencies: - cross-spawn: 7.0.3 - get-stream: 6.0.1 - human-signals: 4.3.1 - is-stream: 3.0.0 - merge-stream: 2.0.0 - npm-run-path: 5.3.0 - onetime: 6.0.0 - signal-exit: 3.0.7 - strip-final-newline: 3.0.0 - - expand-brackets@2.1.4: - dependencies: - debug: 2.6.9 - define-property: 0.2.5 - extend-shallow: 2.0.1 - posix-character-classes: 0.1.1 - regex-not: 1.0.2 - snapdragon: 0.8.2 - to-regex: 3.0.2 - transitivePeerDependencies: - - supports-color - - expand-template@2.0.3: {} - - expand-tilde@2.0.2: - dependencies: - homedir-polyfill: 1.0.3 - - extend-shallow@2.0.1: - dependencies: - is-extendable: 0.1.1 - - extend-shallow@3.0.2: - dependencies: - assign-symbols: 1.0.0 - is-extendable: 1.0.1 - - extend@3.0.2: {} - - external-editor@3.1.0: - dependencies: - chardet: 0.7.0 - iconv-lite: 0.4.24 - tmp: 0.0.33 - - extglob@2.0.4: - dependencies: - array-unique: 0.3.2 - define-property: 1.0.0 - expand-brackets: 2.1.4 - extend-shallow: 2.0.1 - fragment-cache: 0.2.1 - regex-not: 1.0.2 - snapdragon: 0.8.2 - to-regex: 3.0.2 - transitivePeerDependencies: - - supports-color - - fast-deep-equal@3.1.3: {} - - fast-fifo@1.3.2: {} - - fast-glob@3.3.2: - dependencies: - '@nodelib/fs.stat': 2.0.5 - '@nodelib/fs.walk': 1.2.8 - glob-parent: 5.1.2 - merge2: 1.4.1 - micromatch: 4.0.8 - - fast-json-stable-stringify@2.1.0: {} - - fast-safe-stringify@2.1.1: {} - - fast-uri@3.0.1: {} - - fast-xml-parser@4.4.1: - dependencies: - strnum: 1.0.5 - - fastq@1.17.1: - dependencies: - reusify: 1.0.4 - - fecha@4.2.3: {} - - fetch-blob@3.2.0: - dependencies: - node-domexception: 1.0.0 - web-streams-polyfill: 3.3.3 - - figures@3.2.0: - dependencies: - escape-string-regexp: 1.0.5 - - fill-range@4.0.0: - dependencies: - extend-shallow: 2.0.1 - is-number: 3.0.0 - repeat-string: 1.6.1 - to-regex-range: 2.1.1 - - fill-range@7.1.1: - dependencies: - to-regex-range: 5.0.1 - - find-root@1.1.0: {} - - find-up@3.0.0: - dependencies: - locate-path: 3.0.0 - - find-up@4.1.0: - dependencies: - locate-path: 5.0.0 - path-exists: 4.0.0 - - find-up@5.0.0: - dependencies: - locate-path: 6.0.0 - path-exists: 4.0.0 - - findup-sync@2.0.0: - dependencies: - detect-file: 1.0.0 - is-glob: 3.1.0 - micromatch: 3.1.10 - resolve-dir: 1.0.1 - transitivePeerDependencies: - - supports-color - - fined@1.2.0: - dependencies: - expand-tilde: 2.0.2 - is-plain-object: 2.0.4 - object.defaults: 1.1.0 - object.pick: 1.3.0 - parse-filepath: 1.0.2 - - flagged-respawn@1.0.1: {} - - flat-cache@3.2.0: - dependencies: - flatted: 3.3.1 - keyv: 4.5.4 - rimraf: 3.0.2 - - flatted@3.3.1: {} - - fn.name@1.1.0: {} - - follow-redirects@1.15.6(debug@4.3.4): - optionalDependencies: - debug: 4.3.4(supports-color@5.5.0) - - for-in@1.0.2: {} - - for-own@1.0.0: - dependencies: - for-in: 1.0.2 - - foreground-child@3.3.0: - dependencies: - cross-spawn: 7.0.3 - signal-exit: 4.1.0 - - fork-ts-checker-webpack-plugin@9.0.2(typescript@5.2.2)(webpack@5.94.0(esbuild@0.19.11)): - dependencies: - '@babel/code-frame': 7.24.7 - chalk: 4.1.2 - chokidar: 3.5.3 - cosmiconfig: 8.3.6(typescript@5.2.2) - deepmerge: 4.3.1 - fs-extra: 10.0.0 - memfs: 3.5.3 - minimatch: 3.1.2 - node-abort-controller: 3.1.1 - schema-utils: 3.3.0 - semver: 7.5.4 - tapable: 2.2.1 - typescript: 5.2.2 - webpack: 5.94.0(esbuild@0.19.11) - - form-data-encoder@1.7.2: {} - - form-data@4.0.0: - dependencies: - asynckit: 0.4.0 - combined-stream: 1.0.8 - mime-types: 2.1.35 - - formdata-node@4.4.1: - dependencies: - node-domexception: 1.0.0 - web-streams-polyfill: 4.0.0-beta.3 - - formdata-polyfill@4.0.10: - dependencies: - fetch-blob: 3.2.0 - - formidable@1.2.6: {} - - formik@2.4.0(react@18.3.1): - dependencies: - deepmerge: 2.2.1 - hoist-non-react-statics: 3.3.2 - lodash: 4.17.21 - lodash-es: 4.17.21 - react: 18.3.1 - react-fast-compare: 2.0.4 - tiny-warning: 1.0.3 - tslib: 1.14.1 - - fractional-indexing@3.2.0: {} - - fragment-cache@0.2.1: - dependencies: - map-cache: 0.2.2 - - fresh@0.5.2: {} - - fs-constants@1.0.0: {} - - fs-extra@10.0.0: - dependencies: - graceful-fs: 4.2.11 - jsonfile: 6.1.0 - universalify: 2.0.1 - - fs-extra@7.0.1: - dependencies: - graceful-fs: 4.2.11 - jsonfile: 4.0.0 - universalify: 0.1.2 - - fs-extra@8.1.0: - dependencies: - graceful-fs: 4.2.11 - jsonfile: 4.0.0 - universalify: 0.1.2 - - fs-jetpack@4.3.1: - dependencies: - minimatch: 3.1.2 - rimraf: 2.7.1 - - fs-minipass@2.1.0: - dependencies: - minipass: 3.3.6 - - fs-monkey@1.0.6: {} - - fs.realpath@1.0.0: {} - - fsevents@2.3.3: - optional: true - - function-bind@1.1.2: {} - - fuzzysort@3.0.2: {} - - gauge@3.0.2: - dependencies: - aproba: 2.0.0 - color-support: 1.1.3 - console-control-strings: 1.1.0 - has-unicode: 2.0.1 - object-assign: 4.1.1 - signal-exit: 3.0.7 - string-width: 4.2.3 - strip-ansi: 6.0.1 - wide-align: 1.1.5 - - get-caller-file@2.0.5: {} - - get-intrinsic@1.2.4: - dependencies: - es-errors: 1.3.0 - function-bind: 1.1.2 - has-proto: 1.0.3 - has-symbols: 1.0.3 - hasown: 2.0.2 - - get-it@8.6.5(debug@4.3.4): - dependencies: - '@types/follow-redirects': 1.14.4 - '@types/progress-stream': 2.0.5 - decompress-response: 7.0.0 - follow-redirects: 1.15.6(debug@4.3.4) - is-retry-allowed: 2.2.0 - progress-stream: 2.0.0 - tunnel-agent: 0.6.0 - transitivePeerDependencies: - - debug - - get-latest-version@5.1.0(debug@4.3.4): - dependencies: - get-it: 8.6.5(debug@4.3.4) - registry-auth-token: 5.0.2 - registry-url: 5.1.0 - semver: 7.5.4 - transitivePeerDependencies: - - debug - - get-nonce@1.0.1: {} - - get-package-type@0.1.0: {} - - get-stdin@9.0.0: {} - - get-stream@5.2.0: - dependencies: - pump: 3.0.0 - - get-stream@6.0.1: {} - - get-value@2.0.6: {} - - getopts@2.3.0: {} - - git-hooks-list@3.1.0: {} - - git-up@7.0.0: - dependencies: - is-ssh: 1.4.0 - parse-url: 8.1.0 - - git-url-parse@13.1.0: - dependencies: - git-up: 7.0.0 - - github-from-package@0.0.0: {} - - glob-parent@5.1.2: - dependencies: - is-glob: 4.0.3 - - glob-to-regexp@0.4.1: {} - - glob@10.4.2: - dependencies: - foreground-child: 3.3.0 - jackspeak: 3.4.3 - minimatch: 9.0.5 - minipass: 7.1.2 - package-json-from-dist: 1.0.0 - path-scurry: 1.11.1 - - glob@7.2.3: - dependencies: - fs.realpath: 1.0.0 - inflight: 1.0.6 - inherits: 2.0.4 - minimatch: 3.1.2 - once: 1.4.0 - path-is-absolute: 1.0.1 - - glob@8.1.0: - dependencies: - fs.realpath: 1.0.0 - inflight: 1.0.6 - inherits: 2.0.4 - minimatch: 5.1.6 - once: 1.4.0 - - global-modules@1.0.0: - dependencies: - global-prefix: 1.0.2 - is-windows: 1.0.2 - resolve-dir: 1.0.1 - - global-prefix@1.0.2: - dependencies: - expand-tilde: 2.0.2 - homedir-polyfill: 1.0.3 - ini: 1.3.8 - is-windows: 1.0.2 - which: 1.3.1 - - globals@11.12.0: {} - - globby@10.0.2: - dependencies: - '@types/glob': 7.2.0 - array-union: 2.1.0 - dir-glob: 3.0.1 - fast-glob: 3.3.2 - glob: 7.2.3 - ignore: 5.3.2 - merge2: 1.4.1 - slash: 3.0.0 - - globby@13.2.2: - dependencies: - dir-glob: 3.0.1 - fast-glob: 3.3.2 - ignore: 5.3.2 - merge2: 1.4.1 - slash: 4.0.0 - - gopd@1.0.1: - dependencies: - get-intrinsic: 1.2.4 - - got@11.8.6: - dependencies: - '@sindresorhus/is': 4.6.0 - '@szmarczak/http-timer': 4.0.6 - '@types/cacheable-request': 6.0.3 - '@types/responselike': 1.0.3 - cacheable-lookup: 5.0.4 - cacheable-request: 7.0.4 - decompress-response: 6.0.0 - http2-wrapper: 1.0.3 - lowercase-keys: 2.0.0 - p-cancelable: 2.1.1 - responselike: 2.0.1 - - graceful-fs@4.2.10: {} - - graceful-fs@4.2.11: {} - - grant-koa@5.4.8(koa@2.13.4): - dependencies: - grant: 5.4.22 - koa: 2.13.4 - - grant@5.4.22: - dependencies: - qs: 6.13.0 - request-compose: 2.1.6 - request-oauth: 1.0.1 - optionalDependencies: - cookie: 0.5.0 - cookie-signature: 1.2.1 - jwk-to-pem: 2.0.5 - jws: 4.0.0 - - gzip-size@6.0.0: - dependencies: - duplexer: 0.1.2 - - handlebars@4.7.8: - dependencies: - minimist: 1.2.8 - neo-async: 2.6.2 - source-map: 0.6.1 - wordwrap: 1.0.0 - optionalDependencies: - uglify-js: 3.19.3 - - has-ansi@2.0.0: - dependencies: - ansi-regex: 2.1.1 - - has-flag@3.0.0: {} - - has-flag@4.0.0: {} - - has-property-descriptors@1.0.2: - dependencies: - es-define-property: 1.0.0 - - has-proto@1.0.3: {} - - has-symbols@1.0.3: {} - - has-tostringtag@1.0.2: - dependencies: - has-symbols: 1.0.3 - - has-unicode@2.0.1: {} - - has-value@0.3.1: - dependencies: - get-value: 2.0.6 - has-values: 0.1.4 - isobject: 2.1.0 - - has-value@1.0.0: - dependencies: - get-value: 2.0.6 - has-values: 1.0.0 - isobject: 3.0.1 - - has-values@0.1.4: {} - - has-values@1.0.0: - dependencies: - is-number: 3.0.0 - kind-of: 4.0.0 - - hash.js@1.1.7: - dependencies: - inherits: 2.0.4 - minimalistic-assert: 1.0.1 - - hasown@2.0.2: - dependencies: - function-bind: 1.1.2 - - he@1.2.0: {} - - header-case@1.0.1: - dependencies: - no-case: 2.3.2 - upper-case: 1.1.3 - - helmet@6.2.0: {} - - highlight.js@10.7.3: {} - - history@4.10.1: - dependencies: - '@babel/runtime': 7.25.6 - loose-envify: 1.4.0 - resolve-pathname: 3.0.0 - tiny-invariant: 1.3.3 - tiny-warning: 1.0.3 - value-equal: 1.0.1 - - hmac-drbg@1.0.1: - dependencies: - hash.js: 1.1.7 - minimalistic-assert: 1.0.1 - minimalistic-crypto-utils: 1.0.1 - - hoist-non-react-statics@3.3.2: - dependencies: - react-is: 16.13.1 - - homedir-polyfill@1.0.3: - dependencies: - parse-passwd: 1.0.0 - - hosted-git-info@2.8.9: {} - - html-entities@2.5.2: {} - - html-escaper@2.0.2: {} - - html-minifier-terser@6.1.0: - dependencies: - camel-case: 4.1.2 - clean-css: 5.3.3 - commander: 8.3.0 - he: 1.2.0 - param-case: 3.0.4 - relateurl: 0.2.7 - terser: 5.31.6 - - html-webpack-plugin@5.6.0(webpack@5.94.0(esbuild@0.19.11)): - dependencies: - '@types/html-minifier-terser': 6.1.0 - html-minifier-terser: 6.1.0 - lodash: 4.17.21 - pretty-error: 4.0.0 - tapable: 2.2.1 - optionalDependencies: - webpack: 5.94.0(esbuild@0.19.11) - - htmlparser2@6.1.0: - dependencies: - domelementtype: 2.3.0 - domhandler: 4.3.1 - domutils: 2.8.0 - entities: 2.2.0 - - htmlparser2@8.0.2: - dependencies: - domelementtype: 2.3.0 - domhandler: 5.0.3 - domutils: 3.1.0 - entities: 4.5.0 - - http-assert@1.5.0: - dependencies: - deep-equal: 1.0.1 - http-errors: 1.8.1 - - http-cache-semantics@4.1.1: {} - - http-errors@1.6.3: - dependencies: - depd: 1.1.2 - inherits: 2.0.3 - setprototypeof: 1.1.0 - statuses: 1.5.0 - - http-errors@1.8.1: - dependencies: - depd: 1.1.2 - inherits: 2.0.4 - setprototypeof: 1.2.0 - statuses: 1.5.0 - toidentifier: 1.0.1 - - http-errors@2.0.0: - dependencies: - depd: 2.0.0 - inherits: 2.0.4 - setprototypeof: 1.2.0 - statuses: 2.0.1 - toidentifier: 1.0.1 - - http2-wrapper@1.0.3: - dependencies: - quick-lru: 5.1.1 - resolve-alpn: 1.2.1 - - https-proxy-agent@5.0.1: - dependencies: - agent-base: 6.0.2 - debug: 4.3.4(supports-color@5.5.0) - transitivePeerDependencies: - - supports-color - - human-signals@2.1.0: {} - - human-signals@4.3.1: {} - - humanize-ms@1.2.1: - dependencies: - ms: 2.1.3 - - iconv-lite@0.4.13: {} - - iconv-lite@0.4.15: {} - - iconv-lite@0.4.24: - dependencies: - safer-buffer: 2.1.2 - - icss-utils@5.1.0(postcss@8.4.44): - dependencies: - postcss: 8.4.44 - - ieee754@1.2.1: {} - - ignore-by-default@1.0.1: {} - - ignore-walk@3.0.4: - dependencies: - minimatch: 3.1.2 - - ignore@5.3.2: {} - - immer@9.0.19: {} - - immer@9.0.21: {} - - import-fresh@3.3.0: - dependencies: - parent-module: 1.0.1 - resolve-from: 4.0.0 - - import-lazy@4.0.0: {} - - imurmurhash@0.1.4: {} - - indent-string@4.0.0: {} - - inflation@2.1.0: {} - - inflight@1.0.6: - dependencies: - once: 1.4.0 - wrappy: 1.0.2 - - inherits@2.0.3: {} - - inherits@2.0.4: {} - - ini@1.3.8: {} - - ini@2.0.0: {} - - ini@4.1.1: {} - - inquirer@7.3.3: - dependencies: - ansi-escapes: 4.3.2 - chalk: 4.1.2 - cli-cursor: 3.1.0 - cli-width: 3.0.0 - external-editor: 3.1.0 - figures: 3.2.0 - lodash: 4.17.21 - mute-stream: 0.0.8 - run-async: 2.4.1 - rxjs: 6.6.7 - string-width: 4.2.3 - strip-ansi: 6.0.1 - through: 2.3.8 - - inquirer@8.2.5: - dependencies: - ansi-escapes: 4.3.2 - chalk: 4.1.2 - cli-cursor: 3.1.0 - cli-width: 3.0.0 - external-editor: 3.1.0 - figures: 3.2.0 - lodash: 4.17.21 - mute-stream: 0.0.8 - ora: 5.4.1 - run-async: 2.4.1 - rxjs: 7.8.1 - string-width: 4.2.3 - strip-ansi: 6.0.1 - through: 2.3.8 - wrap-ansi: 7.0.0 - - interpret@1.4.0: {} - - interpret@2.2.0: {} - - intl-messageformat@10.3.4: - dependencies: - '@formatjs/ecma402-abstract': 1.14.3 - '@formatjs/fast-memoize': 2.0.1 - '@formatjs/icu-messageformat-parser': 2.3.1 - tslib: 2.7.0 - - invariant@2.2.4: - dependencies: - loose-envify: 1.4.0 - - is-absolute@1.0.0: - dependencies: - is-relative: 1.0.0 - is-windows: 1.0.2 - - is-accessor-descriptor@1.0.1: - dependencies: - hasown: 2.0.2 - - is-arrayish@0.2.1: {} - - is-arrayish@0.3.2: {} - - is-binary-path@2.1.0: - dependencies: - binary-extensions: 2.3.0 - - is-buffer@1.1.6: {} - - is-class-hotfix@0.0.6: {} - - is-core-module@2.15.1: - dependencies: - hasown: 2.0.2 - - is-data-descriptor@1.0.1: - dependencies: - hasown: 2.0.2 - - is-descriptor@0.1.7: - dependencies: - is-accessor-descriptor: 1.0.1 - is-data-descriptor: 1.0.1 - - is-descriptor@1.0.3: - dependencies: - is-accessor-descriptor: 1.0.1 - is-data-descriptor: 1.0.1 - - is-docker@2.2.1: {} - - is-docker@3.0.0: {} - - is-extendable@0.1.1: {} - - is-extendable@1.0.1: - dependencies: - is-plain-object: 2.0.4 - - is-extglob@2.1.1: {} - - is-fullwidth-code-point@3.0.0: {} - - is-generator-function@1.0.10: - dependencies: - has-tostringtag: 1.0.2 - - is-glob@3.1.0: - dependencies: - is-extglob: 2.1.1 - - is-glob@4.0.3: - dependencies: - is-extglob: 2.1.1 - - is-hotkey@0.1.8: {} - - is-inside-container@1.0.0: - dependencies: - is-docker: 3.0.0 - - is-interactive@1.0.0: {} - - is-localhost-ip@2.0.0: {} - - is-lower-case@1.1.3: - dependencies: - lower-case: 1.1.4 - - is-number@3.0.0: - dependencies: - kind-of: 3.2.2 - - is-number@7.0.0: {} - - is-obj@2.0.0: {} - - is-path-cwd@2.2.0: {} - - is-path-inside@3.0.3: {} - - is-plain-obj@4.1.0: {} - - is-plain-object@2.0.4: - dependencies: - isobject: 3.0.1 - - is-plain-object@5.0.0: {} - - is-relative@1.0.0: - dependencies: - is-unc-path: 1.0.0 - - is-retry-allowed@2.2.0: {} - - is-ssh@1.4.0: - dependencies: - protocols: 2.0.1 - - is-stream@2.0.1: {} - - is-stream@3.0.0: {} - - is-type-of@1.4.0: - dependencies: - core-util-is: 1.0.3 - is-class-hotfix: 0.0.6 - isstream: 0.1.2 - - is-typedarray@1.0.0: {} - - is-unc-path@1.0.0: - dependencies: - unc-path-regex: 0.1.2 - - is-unicode-supported@0.1.0: {} - - is-upper-case@1.1.2: - dependencies: - upper-case: 1.1.3 - - is-windows@1.0.2: {} - - is-wsl@2.2.0: - dependencies: - is-docker: 2.2.1 - - isarray@0.0.1: {} - - isarray@1.0.0: {} - - isbinaryfile@4.0.10: {} - - isexe@2.0.0: {} - - isobject@2.1.0: - dependencies: - isarray: 1.0.0 - - isobject@3.0.1: {} - - isstream@0.1.2: {} - - jackspeak@3.4.3: - dependencies: - '@isaacs/cliui': 8.0.2 - optionalDependencies: - '@pkgjs/parseargs': 0.11.0 - - jest-worker@27.5.1: - dependencies: - '@types/node': 22.5.2 - merge-stream: 2.0.0 - supports-color: 8.1.1 - - jju@1.4.0: {} - - jose@4.15.9: {} - - joycon@3.1.1: {} - - js-cookie@2.2.1: {} - - js-sha3@0.8.0: {} - - js-tokens@4.0.0: {} - - js-yaml@4.1.0: - dependencies: - argparse: 2.0.1 - - jsesc@2.5.2: {} - - json-buffer@3.0.1: {} - - json-parse-even-better-errors@2.3.1: {} - - json-schema-traverse@0.4.1: {} - - json-schema-traverse@1.0.0: {} - - json5@2.2.3: {} - - jsonfile@4.0.0: - optionalDependencies: - graceful-fs: 4.2.11 - - jsonfile@6.1.0: - dependencies: - universalify: 2.0.1 - optionalDependencies: - graceful-fs: 4.2.11 - - jsonwebtoken@9.0.0: - dependencies: - jws: 3.2.2 - lodash: 4.17.21 - ms: 2.1.3 - semver: 7.6.3 - - jwa@1.4.1: - dependencies: - buffer-equal-constant-time: 1.0.1 - ecdsa-sig-formatter: 1.0.11 - safe-buffer: 5.2.1 - - jwa@2.0.0: - dependencies: - buffer-equal-constant-time: 1.0.1 - ecdsa-sig-formatter: 1.0.11 - safe-buffer: 5.2.1 - optional: true - - jwk-to-pem@2.0.5: - dependencies: - asn1.js: 5.4.1 - elliptic: 6.5.7 - safe-buffer: 5.2.1 - - jwks-rsa@3.1.0: - dependencies: - '@types/express': 4.17.21 - '@types/jsonwebtoken': 9.0.6 - debug: 4.3.4(supports-color@5.5.0) - jose: 4.15.9 - limiter: 1.1.5 - lru-memoizer: 2.3.0 - transitivePeerDependencies: - - supports-color - - jws@3.2.2: - dependencies: - jwa: 1.4.1 - safe-buffer: 5.2.1 - - jws@4.0.0: - dependencies: - jwa: 2.0.0 - safe-buffer: 5.2.1 - optional: true - - keygrip@1.1.0: - dependencies: - tsscmp: 1.0.6 - - keyv@4.5.4: - dependencies: - json-buffer: 3.0.1 - - kind-of@3.2.2: - dependencies: - is-buffer: 1.1.6 - - kind-of@4.0.0: - dependencies: - is-buffer: 1.1.6 - - kind-of@6.0.3: {} - - kleur@3.0.3: {} - - knex@2.5.0(pg@8.12.0): - dependencies: - colorette: 2.0.19 - commander: 10.0.1 - debug: 4.3.4(supports-color@5.5.0) - escalade: 3.2.0 - esm: 3.2.25 - get-package-type: 0.1.0 - getopts: 2.3.0 - interpret: 2.2.0 - lodash: 4.17.21 - pg-connection-string: 2.6.1 - rechoir: 0.8.0 - resolve-from: 5.0.0 - tarn: 3.0.2 - tildify: 2.0.0 - optionalDependencies: - pg: 8.12.0 - transitivePeerDependencies: - - supports-color - - koa-body@4.2.0: - dependencies: - '@types/formidable': 1.2.8 - co-body: 5.2.0 - formidable: 1.2.6 - - koa-bodyparser@4.4.1: - dependencies: - co-body: 6.2.0 - copy-to: 2.0.1 - type-is: 1.6.18 - - koa-compose@4.1.0: {} - - koa-compress@5.1.0: - dependencies: - bytes: 3.1.2 - compressible: 2.0.18 - http-errors: 1.8.1 - koa-is-json: 1.0.0 - statuses: 2.0.1 - - koa-convert@2.0.0: - dependencies: - co: 4.6.0 - koa-compose: 4.1.0 - - koa-favicon@2.1.0: - dependencies: - mz: 2.7.0 - - koa-helmet@7.0.2: - dependencies: - helmet: 6.2.0 - - koa-ip@2.1.3: - dependencies: - debug: 4.3.4(supports-color@5.5.0) - lodash.isplainobject: 4.0.6 - request-ip: 3.3.0 - transitivePeerDependencies: - - supports-color - - koa-is-json@1.0.0: {} - - koa-passport@5.0.0: - dependencies: - passport: 0.6.0 - - koa-range@0.3.0: - dependencies: - stream-slice: 0.1.2 - - koa-send@5.0.1: - dependencies: - debug: 4.3.4(supports-color@5.5.0) - http-errors: 1.8.1 - resolve-path: 1.4.0 - transitivePeerDependencies: - - supports-color - - koa-session@6.4.0: - dependencies: - crc: 3.8.0 - debug: 4.3.4(supports-color@5.5.0) - is-type-of: 1.4.0 - uuid: 8.3.2 - transitivePeerDependencies: - - supports-color - - koa-static@5.0.0: - dependencies: - debug: 3.2.7 - koa-send: 5.0.1 - transitivePeerDependencies: - - supports-color - - koa2-ratelimit@1.1.3: {} - - koa@2.13.4: - dependencies: - accepts: 1.3.8 - cache-content-type: 1.0.1 - content-disposition: 0.5.4 - content-type: 1.0.5 - cookies: 0.8.0 - debug: 4.3.6 - delegates: 1.0.0 - depd: 2.0.0 - destroy: 1.2.0 - encodeurl: 1.0.2 - escape-html: 1.0.3 - fresh: 0.5.2 - http-assert: 1.5.0 - http-errors: 1.8.1 - is-generator-function: 1.0.10 - koa-compose: 4.1.0 - koa-convert: 2.0.0 - on-finished: 2.4.1 - only: 0.0.2 - parseurl: 1.3.3 - statuses: 1.5.0 - type-is: 1.6.18 - vary: 1.1.2 - transitivePeerDependencies: - - supports-color - - kuler@2.0.0: {} - - libbase64@0.1.0: {} - - libmime@2.1.0: - dependencies: - iconv-lite: 0.4.13 - libbase64: 0.1.0 - libqp: 1.1.0 - - libmime@2.1.3: - dependencies: - iconv-lite: 0.4.15 - libbase64: 0.1.0 - libqp: 1.1.0 - - libqp@1.1.0: {} - - liftoff@2.5.0: - dependencies: - extend: 3.0.2 - findup-sync: 2.0.0 - fined: 1.2.0 - flagged-respawn: 1.0.1 - is-plain-object: 2.0.4 - object.map: 1.0.1 - rechoir: 0.6.2 - resolve: 1.22.8 - transitivePeerDependencies: - - supports-color - - limiter@1.1.5: {} - - lines-and-columns@1.2.4: {} - - linkify-it@3.0.3: - dependencies: - uc.micro: 1.0.6 - - loader-runner@4.3.0: {} - - loader-utils@2.0.4: - dependencies: - big.js: 5.2.2 - emojis-list: 3.0.0 - json5: 2.2.3 - - locate-path@3.0.0: - dependencies: - p-locate: 3.0.0 - path-exists: 3.0.0 - - locate-path@5.0.0: - dependencies: - p-locate: 4.1.0 - - locate-path@6.0.0: - dependencies: - p-locate: 5.0.0 - - lodash-es@4.17.21: {} - - lodash.clonedeep@4.5.0: {} - - lodash.deburr@4.1.0: {} - - lodash.get@4.4.2: {} - - lodash.isplainobject@4.0.6: {} - - lodash@4.17.21: {} - - log-symbols@2.2.0: - dependencies: - chalk: 2.4.2 - - log-symbols@4.1.0: - dependencies: - chalk: 4.1.2 - is-unicode-supported: 0.1.0 - - logform@2.6.1: - dependencies: - '@colors/colors': 1.6.0 - '@types/triple-beam': 1.3.5 - fecha: 4.2.3 - ms: 2.1.3 - safe-stable-stringify: 2.5.0 - triple-beam: 1.4.1 - - long-timeout@0.1.1: {} - - loose-envify@1.4.0: - dependencies: - js-tokens: 4.0.0 - - lower-case-first@1.0.2: - dependencies: - lower-case: 1.1.4 - - lower-case@1.1.4: {} - - lower-case@2.0.2: - dependencies: - tslib: 2.7.0 - - lowercase-keys@2.0.0: {} - - lru-cache@10.4.3: {} - - lru-cache@6.0.0: - dependencies: - yallist: 4.0.0 - - lru-memoizer@2.3.0: - dependencies: - lodash.clonedeep: 4.5.0 - lru-cache: 6.0.0 - - lru_map@0.3.3: {} - - luxon@3.5.0: {} - - mailcomposer@3.12.0: - dependencies: - buildmail: 3.10.0 - libmime: 2.1.0 - - make-dir@3.1.0: - dependencies: - semver: 6.3.1 - - make-iterator@1.0.1: - dependencies: - kind-of: 6.0.3 - - map-cache@0.2.2: {} - - map-visit@1.0.0: - dependencies: - object-visit: 1.0.1 - - markdown-it-abbr@1.0.4: {} - - markdown-it-container@3.0.0: {} - - markdown-it-deflist@2.1.0: {} - - markdown-it-emoji@2.0.2: {} - - markdown-it-footnote@3.0.3: {} - - markdown-it-ins@3.0.1: {} - - markdown-it-mark@3.0.1: {} - - markdown-it-sub@1.0.0: {} - - markdown-it-sup@1.0.0: {} - - markdown-it@12.3.2: - dependencies: - argparse: 2.0.1 - entities: 2.1.0 - linkify-it: 3.0.3 - mdurl: 1.0.1 - uc.micro: 1.0.6 - - match-sorter@6.3.4: - dependencies: - '@babel/runtime': 7.25.6 - remove-accents: 0.5.0 - - mdurl@1.0.1: {} - - media-typer@0.3.0: {} - - memfs@3.5.3: - dependencies: - fs-monkey: 1.0.6 - - memoize-one@5.2.1: {} - - memoize-one@6.0.0: {} - - merge-stream@2.0.0: {} - - merge2@1.4.1: {} - - methods@1.1.2: {} - - micromatch@3.1.10: - dependencies: - arr-diff: 4.0.0 - array-unique: 0.3.2 - braces: 2.3.2 - define-property: 2.0.2 - extend-shallow: 3.0.2 - extglob: 2.0.4 - fragment-cache: 0.2.1 - kind-of: 6.0.3 - nanomatch: 1.2.13 - object.pick: 1.3.0 - regex-not: 1.0.2 - snapdragon: 0.8.2 - to-regex: 3.0.2 - transitivePeerDependencies: - - supports-color - - micromatch@4.0.8: - dependencies: - braces: 3.0.3 - picomatch: 2.3.1 - - microseconds@0.2.0: {} - - mime-db@1.52.0: {} - - mime-db@1.53.0: {} - - mime-types@2.1.35: - dependencies: - mime-db: 1.52.0 - - mimic-fn@1.2.0: {} - - mimic-fn@2.1.0: {} - - mimic-fn@4.0.0: {} - - mimic-response@1.0.1: {} - - mimic-response@2.1.0: {} - - mimic-response@3.1.0: {} - - mini-css-extract-plugin@2.7.7(webpack@5.94.0(esbuild@0.19.11)): - dependencies: - schema-utils: 4.2.0 - webpack: 5.94.0(esbuild@0.19.11) - - minimalistic-assert@1.0.1: {} - - minimalistic-crypto-utils@1.0.1: {} - - minimatch@3.1.2: - dependencies: - brace-expansion: 1.1.11 - - minimatch@5.1.6: - dependencies: - brace-expansion: 2.0.1 - - minimatch@9.0.3: - dependencies: - brace-expansion: 2.0.1 - - minimatch@9.0.5: - dependencies: - brace-expansion: 2.0.1 - - minimist@1.2.8: {} - - minipass@3.3.6: - dependencies: - yallist: 4.0.0 - - minipass@4.2.8: {} - - minipass@5.0.0: {} - - minipass@7.1.2: {} - - minizlib@2.1.2: - dependencies: - minipass: 3.3.6 - yallist: 4.0.0 - - mixin-deep@1.3.2: - dependencies: - for-in: 1.0.2 - is-extendable: 1.0.1 - - mkdirp-classic@0.5.3: {} - - mkdirp@0.5.6: - dependencies: - minimist: 1.2.8 - - mkdirp@1.0.4: {} - - mrmime@2.0.0: {} - - ms@2.0.0: {} - - ms@2.1.2: {} - - ms@2.1.3: {} - - multistream@4.1.0: - dependencies: - once: 1.4.0 - readable-stream: 3.6.2 - - mute-stream@0.0.8: {} - - mz@2.7.0: - dependencies: - any-promise: 1.3.0 - object-assign: 4.1.1 - thenify-all: 1.6.0 - - nan@2.20.0: {} - - nano-time@1.0.0: - dependencies: - big-integer: 1.6.52 - - nanoclone@0.2.1: {} - - nanoid@3.3.7: {} - - nanomatch@1.2.13: - dependencies: - arr-diff: 4.0.0 - array-unique: 0.3.2 - define-property: 2.0.2 - extend-shallow: 3.0.2 - fragment-cache: 0.2.1 - is-windows: 1.0.2 - kind-of: 6.0.3 - object.pick: 1.3.0 - regex-not: 1.0.2 - snapdragon: 0.8.2 - to-regex: 3.0.2 - transitivePeerDependencies: - - supports-color - - napi-build-utils@1.0.2: {} - - negotiator@0.6.3: {} - - neo-async@2.6.2: {} - - no-case@2.3.2: - dependencies: - lower-case: 1.1.4 - - no-case@3.0.4: - dependencies: - lower-case: 2.0.2 - tslib: 2.7.0 - - node-abi@3.67.0: - dependencies: - semver: 7.5.4 - - node-abort-controller@3.1.1: {} - - node-addon-api@6.1.0: {} - - node-domexception@1.0.0: {} - - node-fetch@2.7.0: - dependencies: - whatwg-url: 5.0.0 - - node-fetch@3.3.2: - dependencies: - data-uri-to-buffer: 4.0.1 - fetch-blob: 3.2.0 - formdata-polyfill: 4.0.10 - - node-machine-id@1.1.12: {} - - node-plop@0.26.3: - dependencies: - '@babel/runtime-corejs3': 7.25.6 - '@types/inquirer': 6.5.0 - change-case: 3.1.0 - del: 5.1.0 - globby: 10.0.2 - handlebars: 4.7.8 - inquirer: 7.3.3 - isbinaryfile: 4.0.10 - lodash.get: 4.4.2 - mkdirp: 0.5.6 - resolve: 1.22.8 - - node-releases@2.0.18: {} - - node-schedule@2.1.1: - dependencies: - cron-parser: 4.9.0 - long-timeout: 0.1.1 - sorted-array-functions: 1.3.0 - - nodemailer-fetch@1.6.0: {} - - nodemailer-shared@1.1.0: - dependencies: - nodemailer-fetch: 1.6.0 - - nodemon@3.0.2: - dependencies: - chokidar: 3.6.0 - debug: 4.3.4(supports-color@5.5.0) - ignore-by-default: 1.0.1 - minimatch: 3.1.2 - pstree.remy: 1.1.8 - semver: 7.5.4 - simple-update-notifier: 2.0.0 - supports-color: 5.5.0 - touch: 3.1.1 - undefsafe: 2.0.5 - - noms@0.0.0: - dependencies: - inherits: 2.0.4 - readable-stream: 1.0.34 - - nopt@5.0.0: - dependencies: - abbrev: 1.1.1 - - normalize-package-data@2.5.0: - dependencies: - hosted-git-info: 2.8.9 - resolve: 1.22.8 - semver: 5.7.2 - validate-npm-package-license: 3.0.4 - - normalize-path@3.0.0: {} - - normalize-url@6.1.0: {} - - npm-bundled@1.1.2: - dependencies: - npm-normalize-package-bin: 1.0.1 - - npm-normalize-package-bin@1.0.1: {} - - npm-packlist@2.2.2: - dependencies: - glob: 7.2.3 - ignore-walk: 3.0.4 - npm-bundled: 1.1.2 - npm-normalize-package-bin: 1.0.1 - - npm-run-path@4.0.1: - dependencies: - path-key: 3.1.1 - - npm-run-path@5.3.0: - dependencies: - path-key: 4.0.0 - - npmlog@5.0.1: - dependencies: - are-we-there-yet: 2.0.0 - console-control-strings: 1.1.0 - gauge: 3.0.2 - set-blocking: 2.0.0 - - nth-check@2.1.1: - dependencies: - boolbase: 1.0.0 - - oauth-sign@0.9.0: {} - - object-assign@4.1.1: {} - - object-copy@0.1.0: - dependencies: - copy-descriptor: 0.1.1 - define-property: 0.2.5 - kind-of: 3.2.2 - - object-inspect@1.13.2: {} - - object-visit@1.0.1: - dependencies: - isobject: 3.0.1 - - object.defaults@1.1.0: - dependencies: - array-each: 1.0.1 - array-slice: 1.1.0 - for-own: 1.0.0 - isobject: 3.0.1 - - object.map@1.0.1: - dependencies: - for-own: 1.0.0 - make-iterator: 1.0.1 - - object.pick@1.3.0: - dependencies: - isobject: 3.0.1 - - oblivious-set@1.0.0: {} - - on-finished@2.4.1: - dependencies: - ee-first: 1.1.1 - - once@1.4.0: - dependencies: - wrappy: 1.0.2 - - one-time@1.0.0: - dependencies: - fn.name: 1.1.0 - - onetime@2.0.1: - dependencies: - mimic-fn: 1.2.0 - - onetime@5.1.2: - dependencies: - mimic-fn: 2.1.0 - - onetime@6.0.0: - dependencies: - mimic-fn: 4.0.0 - - only@0.0.2: {} - - open@8.4.0: - dependencies: - define-lazy-prop: 2.0.0 - is-docker: 2.2.1 - is-wsl: 2.2.0 - - open@9.1.0: - dependencies: - default-browser: 4.0.0 - define-lazy-prop: 3.0.0 - is-inside-container: 1.0.0 - is-wsl: 2.2.0 - - opener@1.5.2: {} - - ora@3.4.0: - dependencies: - chalk: 2.4.2 - cli-cursor: 2.1.0 - cli-spinners: 2.9.2 - log-symbols: 2.2.0 - strip-ansi: 5.2.0 - wcwidth: 1.0.1 - - ora@5.4.1: - dependencies: - bl: 4.1.0 - chalk: 4.1.2 - cli-cursor: 3.1.0 - cli-spinners: 2.9.2 - is-interactive: 1.0.0 - is-unicode-supported: 0.1.0 - log-symbols: 4.1.0 - strip-ansi: 6.0.1 - wcwidth: 1.0.1 - - os-paths@7.4.0: - optionalDependencies: - fsevents: 2.3.3 - - os-tmpdir@1.0.2: {} - - outdent@0.8.0: {} - - p-cancelable@2.1.1: {} - - p-finally@1.0.0: {} - - p-limit@2.3.0: - dependencies: - p-try: 2.2.0 - - p-limit@3.1.0: - dependencies: - yocto-queue: 0.1.0 - - p-locate@3.0.0: - dependencies: - p-limit: 2.3.0 - - p-locate@4.1.0: - dependencies: - p-limit: 2.3.0 - - p-locate@5.0.0: - dependencies: - p-limit: 3.1.0 - - p-map@3.0.0: - dependencies: - aggregate-error: 3.1.0 - - p-map@4.0.0: - dependencies: - aggregate-error: 3.1.0 - - p-queue@6.6.2: - dependencies: - eventemitter3: 4.0.7 - p-timeout: 3.2.0 - - p-timeout@3.2.0: - dependencies: - p-finally: 1.0.0 - - p-try@2.2.0: {} - - package-json-from-dist@1.0.0: {} - - package-json@7.0.0: - dependencies: - got: 11.8.6 - registry-auth-token: 4.2.2 - registry-url: 5.1.0 - semver: 7.5.4 - - param-case@2.1.1: - dependencies: - no-case: 2.3.2 - - param-case@3.0.4: - dependencies: - dot-case: 3.0.4 - tslib: 2.7.0 - - parent-module@1.0.1: - dependencies: - callsites: 3.1.0 - - parse-filepath@1.0.2: - dependencies: - is-absolute: 1.0.0 - map-cache: 0.2.2 - path-root: 0.1.1 - - parse-json@5.2.0: - dependencies: - '@babel/code-frame': 7.24.7 - error-ex: 1.3.2 - json-parse-even-better-errors: 2.3.1 - lines-and-columns: 1.2.4 - - parse-passwd@1.0.0: {} - - parse-path@7.0.0: - dependencies: - protocols: 2.0.1 - - parse-srcset@1.0.2: {} - - parse-url@8.1.0: - dependencies: - parse-path: 7.0.0 - - parseurl@1.3.3: {} - - pascal-case@2.0.1: - dependencies: - camel-case: 3.0.0 - upper-case-first: 1.1.2 - - pascal-case@3.1.2: - dependencies: - no-case: 3.0.4 - tslib: 2.7.0 - - pascalcase@0.1.1: {} - - passport-local@1.0.0: - dependencies: - passport-strategy: 1.0.0 - - passport-strategy@1.0.0: {} - - passport@0.6.0: - dependencies: - passport-strategy: 1.0.0 - pause: 0.0.1 - utils-merge: 1.0.1 - - path-case@2.1.1: - dependencies: - no-case: 2.3.2 - - path-exists@3.0.0: {} - - path-exists@4.0.0: {} - - path-is-absolute@1.0.1: {} - - path-key@3.1.1: {} - - path-key@4.0.0: {} - - path-parse@1.0.7: {} - - path-root-regex@0.1.2: {} - - path-root@0.1.1: - dependencies: - path-root-regex: 0.1.2 - - path-scurry@1.11.1: - dependencies: - lru-cache: 10.4.3 - minipass: 7.1.2 - - path-to-regexp@6.2.2: {} - - path-type@4.0.0: {} - - pause@0.0.1: {} - - pg-cloudflare@1.1.1: - optional: true - - pg-connection-string@2.6.1: {} - - pg-connection-string@2.6.4: {} - - pg-int8@1.0.1: {} - - pg-pool@3.6.2(pg@8.12.0): - dependencies: - pg: 8.12.0 - - pg-protocol@1.6.1: {} - - pg-types@2.2.0: - dependencies: - pg-int8: 1.0.1 - postgres-array: 2.0.0 - postgres-bytea: 1.0.0 - postgres-date: 1.0.7 - postgres-interval: 1.2.0 - - pg@8.12.0: - dependencies: - pg-connection-string: 2.6.4 - pg-pool: 3.6.2(pg@8.12.0) - pg-protocol: 1.6.1 - pg-types: 2.2.0 - pgpass: 1.0.5 - optionalDependencies: - pg-cloudflare: 1.1.1 - - pgpass@1.0.5: - dependencies: - split2: 4.2.0 - - picocolors@1.1.0: {} - - picomatch@2.3.1: {} - - pkg-up@3.1.0: - dependencies: - find-up: 3.0.0 - - plop@2.7.6: - dependencies: - '@types/liftoff': 2.5.1 - chalk: 1.1.3 - interpret: 1.4.0 - liftoff: 2.5.0 - minimist: 1.2.8 - node-plop: 0.26.3 - ora: 3.4.0 - v8flags: 2.1.1 - transitivePeerDependencies: - - supports-color - - pluralize@8.0.0: {} - - pony-cause@2.1.11: {} - - posix-character-classes@0.1.1: {} - - postcss-modules-extract-imports@3.1.0(postcss@8.4.44): - dependencies: - postcss: 8.4.44 - - postcss-modules-local-by-default@4.0.5(postcss@8.4.44): - dependencies: - icss-utils: 5.1.0(postcss@8.4.44) - postcss: 8.4.44 - postcss-selector-parser: 6.1.2 - postcss-value-parser: 4.2.0 - - postcss-modules-scope@3.2.0(postcss@8.4.44): - dependencies: - postcss: 8.4.44 - postcss-selector-parser: 6.1.2 - - postcss-modules-values@4.0.0(postcss@8.4.44): - dependencies: - icss-utils: 5.1.0(postcss@8.4.44) - postcss: 8.4.44 - - postcss-selector-parser@6.1.2: - dependencies: - cssesc: 3.0.0 - util-deprecate: 1.0.2 - - postcss-value-parser@4.2.0: {} - - postcss@8.4.38: - dependencies: - nanoid: 3.3.7 - picocolors: 1.1.0 - source-map-js: 1.2.0 - - postcss@8.4.44: - dependencies: - nanoid: 3.3.7 - picocolors: 1.1.0 - source-map-js: 1.2.0 - - postgres-array@2.0.0: {} - - postgres-bytea@1.0.0: {} - - postgres-date@1.0.7: {} - - postgres-interval@1.2.0: - dependencies: - xtend: 4.0.2 - - prebuild-install@7.1.2: - dependencies: - detect-libc: 2.0.3 - expand-template: 2.0.3 - github-from-package: 0.0.0 - minimist: 1.2.8 - mkdirp-classic: 0.5.3 - napi-build-utils: 1.0.2 - node-abi: 3.67.0 - pump: 3.0.0 - rc: 1.2.8 - simple-get: 4.0.1 - tar-fs: 2.1.1 - tunnel-agent: 0.6.0 - - prettier-plugin-packagejson@2.4.5(prettier@2.8.4): - dependencies: - sort-package-json: 2.5.1 - synckit: 0.8.5 - optionalDependencies: - prettier: 2.8.4 - - prettier@2.8.4: {} - - pretty-error@4.0.0: - dependencies: - lodash: 4.17.21 - renderkid: 3.0.0 - - process-nextick-args@2.0.1: {} - - progress-stream@2.0.0: - dependencies: - speedometer: 1.0.0 - through2: 2.0.5 - - prompts@2.4.2: - dependencies: - kleur: 3.0.3 - sisteransi: 1.0.5 - - prop-types@15.8.1: - dependencies: - loose-envify: 1.4.0 - object-assign: 4.1.1 - react-is: 16.13.1 - - property-expr@2.0.6: {} - - proto-list@1.2.4: {} - - protocols@2.0.1: {} - - proxy-from-env@1.1.0: {} - - pstree.remy@1.1.8: {} - - pump@3.0.0: - dependencies: - end-of-stream: 1.4.4 - once: 1.4.0 - - punycode@2.3.1: {} - - purest@4.0.2: - dependencies: - '@simov/deep-extend': 1.0.0 - qs: 6.13.0 - request-compose: 2.1.6 - request-multipart: 1.0.0 - request-oauth: 1.0.1 - - qs@6.11.1: - dependencies: - side-channel: 1.0.6 - - qs@6.13.0: - dependencies: - side-channel: 1.0.6 - - queue-microtask@1.2.3: {} - - queue-tick@1.0.1: {} - - quick-lru@5.1.1: {} - - randombytes@2.1.0: - dependencies: - safe-buffer: 5.2.1 - - range-parser@1.2.1: {} - - raw-body@2.5.2: - dependencies: - bytes: 3.1.2 - http-errors: 2.0.0 - iconv-lite: 0.4.24 - unpipe: 1.0.0 - - rc@1.2.8: - dependencies: - deep-extend: 0.6.0 - ini: 1.3.8 - minimist: 1.2.8 - strip-json-comments: 2.0.1 - - react-dnd-html5-backend@16.0.1: - dependencies: - dnd-core: 16.0.1 - - react-dnd@16.0.1(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(react@18.3.1): - dependencies: - '@react-dnd/invariant': 4.0.2 - '@react-dnd/shallowequal': 4.0.2 - dnd-core: 16.0.1 - fast-deep-equal: 3.1.3 - hoist-non-react-statics: 3.3.2 - react: 18.3.1 - optionalDependencies: - '@types/hoist-non-react-statics': 3.3.5 - '@types/node': 22.5.2 - '@types/react': 18.3.5 - - react-dom@18.3.1(react@18.3.1): - dependencies: - loose-envify: 1.4.0 - react: 18.3.1 - scheduler: 0.23.2 - - react-error-boundary@3.1.4(react@18.3.1): - dependencies: - '@babel/runtime': 7.25.6 - react: 18.3.1 - - react-fast-compare@2.0.4: {} - - react-fast-compare@3.2.2: {} - - react-helmet@6.1.0(react@18.3.1): - dependencies: - object-assign: 4.1.1 - prop-types: 15.8.1 - react: 18.3.1 - react-fast-compare: 3.2.2 - react-side-effect: 2.1.2(react@18.3.1) - - react-intl@6.4.1(react@18.3.1)(typescript@5.2.2): - dependencies: - '@formatjs/ecma402-abstract': 1.14.3 - '@formatjs/icu-messageformat-parser': 2.3.1 - '@formatjs/intl': 2.7.1(typescript@5.2.2) - '@formatjs/intl-displaynames': 6.3.1 - '@formatjs/intl-listformat': 7.2.1 - '@types/hoist-non-react-statics': 3.3.5 - '@types/react': 18.3.5 - hoist-non-react-statics: 3.3.2 - intl-messageformat: 10.3.4 - react: 18.3.1 - tslib: 2.7.0 - optionalDependencies: - typescript: 5.2.2 - - react-is@16.13.1: {} - - react-is@18.3.1: {} - - react-query@3.39.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1): - dependencies: - '@babel/runtime': 7.25.6 - broadcast-channel: 3.7.0 - match-sorter: 6.3.4 - react: 18.3.1 - optionalDependencies: - react-dom: 18.3.1(react@18.3.1) - - react-redux@8.1.1(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(redux@4.2.1): - dependencies: - '@babel/runtime': 7.25.6 - '@types/hoist-non-react-statics': 3.3.5 - '@types/use-sync-external-store': 0.0.3 - hoist-non-react-statics: 3.3.2 - react: 18.3.1 - react-is: 18.3.1 - use-sync-external-store: 1.2.2(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - react-dom: 18.3.1(react@18.3.1) - redux: 4.2.1 - - react-refresh@0.14.0: {} - - react-remove-scroll-bar@2.3.6(@types/react@18.3.5)(react@18.3.1): - dependencies: - react: 18.3.1 - react-style-singleton: 2.2.1(@types/react@18.3.5)(react@18.3.1) - tslib: 2.7.0 - optionalDependencies: - '@types/react': 18.3.5 - - react-remove-scroll@2.5.10(@types/react@18.3.5)(react@18.3.1): - dependencies: - react: 18.3.1 - react-remove-scroll-bar: 2.3.6(@types/react@18.3.5)(react@18.3.1) - react-style-singleton: 2.2.1(@types/react@18.3.5)(react@18.3.1) - tslib: 2.7.0 - use-callback-ref: 1.3.2(@types/react@18.3.5)(react@18.3.1) - use-sidecar: 1.1.2(@types/react@18.3.5)(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - react-remove-scroll@2.5.7(@types/react@18.3.5)(react@18.3.1): - dependencies: - react: 18.3.1 - react-remove-scroll-bar: 2.3.6(@types/react@18.3.5)(react@18.3.1) - react-style-singleton: 2.2.1(@types/react@18.3.5)(react@18.3.1) - tslib: 2.7.0 - use-callback-ref: 1.3.2(@types/react@18.3.5)(react@18.3.1) - use-sidecar: 1.1.2(@types/react@18.3.5)(react@18.3.1) - optionalDependencies: - '@types/react': 18.3.5 - - react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1): - dependencies: - '@remix-run/router': 1.19.1 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - react-router: 6.26.1(react@18.3.1) - - react-router@6.26.1(react@18.3.1): - dependencies: - '@remix-run/router': 1.19.1 - react: 18.3.1 - - react-select@5.7.0(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1): - dependencies: - '@babel/runtime': 7.25.6 - '@emotion/cache': 11.13.1 - '@emotion/react': 11.13.3(@types/react@18.3.5)(react@18.3.1) - '@floating-ui/dom': 1.6.10 - '@types/react-transition-group': 4.4.11 - memoize-one: 6.0.0 - prop-types: 15.8.1 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - react-transition-group: 4.4.5(react-dom@18.3.1(react@18.3.1))(react@18.3.1) - use-isomorphic-layout-effect: 1.1.2(@types/react@18.3.5)(react@18.3.1) - transitivePeerDependencies: - - '@types/react' - - supports-color - - react-side-effect@2.1.2(react@18.3.1): - dependencies: - react: 18.3.1 - - react-style-singleton@2.2.1(@types/react@18.3.5)(react@18.3.1): - dependencies: - get-nonce: 1.0.1 - invariant: 2.2.4 - react: 18.3.1 - tslib: 2.7.0 - optionalDependencies: - '@types/react': 18.3.5 - - react-transition-group@4.4.5(react-dom@18.3.1(react@18.3.1))(react@18.3.1): - dependencies: - '@babel/runtime': 7.25.6 - dom-helpers: 5.2.1 - loose-envify: 1.4.0 - prop-types: 15.8.1 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - - react-window@1.8.8(react-dom@18.3.1(react@18.3.1))(react@18.3.1): - dependencies: - '@babel/runtime': 7.25.6 - memoize-one: 5.2.1 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - - react@18.3.1: - dependencies: - loose-envify: 1.4.0 - - read-pkg-up@7.0.1: - dependencies: - find-up: 4.1.0 - read-pkg: 5.2.0 - type-fest: 0.8.1 - - read-pkg@5.2.0: - dependencies: - '@types/normalize-package-data': 2.4.4 - normalize-package-data: 2.5.0 - parse-json: 5.2.0 - type-fest: 0.6.0 - - readable-stream@1.0.34: - dependencies: - core-util-is: 1.0.3 - inherits: 2.0.4 - isarray: 0.0.1 - string_decoder: 0.10.31 - - readable-stream@2.3.8: - dependencies: - core-util-is: 1.0.3 - inherits: 2.0.4 - isarray: 1.0.0 - process-nextick-args: 2.0.1 - safe-buffer: 5.1.2 - string_decoder: 1.1.1 - util-deprecate: 1.0.2 - - readable-stream@3.6.2: - dependencies: - inherits: 2.0.4 - string_decoder: 1.3.0 - util-deprecate: 1.0.2 - - readdirp@3.6.0: - dependencies: - picomatch: 2.3.1 - - rechoir@0.6.2: - dependencies: - resolve: 1.22.8 - - rechoir@0.8.0: - dependencies: - resolve: 1.22.8 - - redux-thunk@2.4.2(redux@4.2.1): - dependencies: - redux: 4.2.1 - - redux@4.2.1: - dependencies: - '@babel/runtime': 7.25.6 - - regenerator-runtime@0.14.1: {} - - regex-not@1.0.2: - dependencies: - extend-shallow: 3.0.2 - safe-regex: 1.1.0 - - registry-auth-token@4.2.2: - dependencies: - rc: 1.2.8 - - registry-auth-token@5.0.2: - dependencies: - '@pnpm/npm-conf': 2.3.1 - - registry-url@5.1.0: - dependencies: - rc: 1.2.8 - - relateurl@0.2.7: {} - - remove-accents@0.5.0: {} - - renderkid@3.0.0: - dependencies: - css-select: 4.3.0 - dom-converter: 0.2.0 - htmlparser2: 6.1.0 - lodash: 4.17.21 - strip-ansi: 6.0.1 - - repeat-element@1.1.4: {} - - repeat-string@1.6.1: {} - - request-compose@2.1.6: {} - - request-ip@3.3.0: {} - - request-multipart@1.0.0: - dependencies: - bl: 4.1.0 - isstream: 0.1.2 - mime-types: 2.1.35 - multistream: 4.1.0 - uuid: 8.3.2 - - request-oauth@1.0.1: - dependencies: - oauth-sign: 0.9.0 - qs: 6.13.0 - uuid: 8.3.2 - - require-directory@2.1.1: {} - - require-from-string@2.0.2: {} - - reselect@4.1.8: {} - - resolve-alpn@1.2.1: {} - - resolve-cwd@3.0.0: - dependencies: - resolve-from: 5.0.0 - - resolve-dir@1.0.1: - dependencies: - expand-tilde: 2.0.2 - global-modules: 1.0.0 - - resolve-from@4.0.0: {} - - resolve-from@5.0.0: {} - - resolve-path@1.4.0: - dependencies: - http-errors: 1.6.3 - path-is-absolute: 1.0.1 - - resolve-pathname@3.0.0: {} - - resolve-url@0.2.1: {} - - resolve@1.22.8: - dependencies: - is-core-module: 2.15.1 - path-parse: 1.0.7 - supports-preserve-symlinks-flag: 1.0.0 - - responselike@2.0.1: - dependencies: - lowercase-keys: 2.0.0 - - restore-cursor@2.0.0: - dependencies: - onetime: 2.0.1 - signal-exit: 3.0.7 - - restore-cursor@3.1.0: - dependencies: - onetime: 5.1.2 - signal-exit: 3.0.7 - - ret@0.1.15: {} - - reusify@1.0.4: {} - - rimraf@2.7.1: - dependencies: - glob: 7.2.3 - - rimraf@3.0.2: - dependencies: - glob: 7.2.3 - - rollup@4.21.2: - dependencies: - '@types/estree': 1.0.5 - optionalDependencies: - '@rollup/rollup-android-arm-eabi': 4.21.2 - '@rollup/rollup-android-arm64': 4.21.2 - '@rollup/rollup-darwin-arm64': 4.21.2 - '@rollup/rollup-darwin-x64': 4.21.2 - '@rollup/rollup-linux-arm-gnueabihf': 4.21.2 - '@rollup/rollup-linux-arm-musleabihf': 4.21.2 - '@rollup/rollup-linux-arm64-gnu': 4.21.2 - '@rollup/rollup-linux-arm64-musl': 4.21.2 - '@rollup/rollup-linux-powerpc64le-gnu': 4.21.2 - '@rollup/rollup-linux-riscv64-gnu': 4.21.2 - '@rollup/rollup-linux-s390x-gnu': 4.21.2 - '@rollup/rollup-linux-x64-gnu': 4.21.2 - '@rollup/rollup-linux-x64-musl': 4.21.2 - '@rollup/rollup-win32-arm64-msvc': 4.21.2 - '@rollup/rollup-win32-ia32-msvc': 4.21.2 - '@rollup/rollup-win32-x64-msvc': 4.21.2 - fsevents: 2.3.3 - - run-applescript@5.0.0: - dependencies: - execa: 5.1.1 - - run-async@2.4.1: {} - - run-parallel@1.2.0: - dependencies: - queue-microtask: 1.2.3 - - rxjs@6.6.7: - dependencies: - tslib: 1.14.1 - - rxjs@7.8.1: - dependencies: - tslib: 2.7.0 - - safe-buffer@5.1.2: {} - - safe-buffer@5.2.1: {} - - safe-regex@1.1.0: - dependencies: - ret: 0.1.15 - - safe-stable-stringify@2.5.0: {} - - safer-buffer@2.1.2: {} - - sanitize-html@2.13.0: - dependencies: - deepmerge: 4.3.1 - escape-string-regexp: 4.0.0 - htmlparser2: 8.0.2 - is-plain-object: 5.0.0 - parse-srcset: 1.0.2 - postcss: 8.4.44 - - scheduler@0.23.2: - dependencies: - loose-envify: 1.4.0 - - schema-utils@3.3.0: - dependencies: - '@types/json-schema': 7.0.15 - ajv: 6.12.6 - ajv-keywords: 3.5.2(ajv@6.12.6) - - schema-utils@4.2.0: - dependencies: - '@types/json-schema': 7.0.15 - ajv: 8.17.1 - ajv-formats: 2.1.1(ajv@8.17.1) - ajv-keywords: 5.1.0(ajv@8.17.1) - - scroll-into-view-if-needed@2.2.31: - dependencies: - compute-scroll-into-view: 1.0.20 - - semver@5.7.2: {} - - semver@6.3.1: {} - - semver@7.5.4: - dependencies: - lru-cache: 6.0.0 - - semver@7.6.3: {} - - sendmail@1.6.1: - dependencies: - dkim-signer: 0.2.2 - mailcomposer: 3.12.0 - - sentence-case@2.1.1: - dependencies: - no-case: 2.3.2 - upper-case-first: 1.1.2 - - serialize-javascript@6.0.2: - dependencies: - randombytes: 2.1.0 - - set-blocking@2.0.0: {} - - set-function-length@1.2.2: - dependencies: - define-data-property: 1.1.4 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - gopd: 1.0.1 - has-property-descriptors: 1.0.2 - - set-value@2.0.1: - dependencies: - extend-shallow: 2.0.1 - is-extendable: 0.1.1 - is-plain-object: 2.0.4 - split-string: 3.1.0 - - setprototypeof@1.1.0: {} - - setprototypeof@1.2.0: {} - - shallowequal@1.1.0: {} - - sharp@0.32.6: - dependencies: - color: 4.2.3 - detect-libc: 2.0.3 - node-addon-api: 6.1.0 - prebuild-install: 7.1.2 - semver: 7.5.4 - simple-get: 4.0.1 - tar-fs: 3.0.6 - tunnel-agent: 0.6.0 - - sharp@0.33.5: - dependencies: - color: 4.2.3 - detect-libc: 2.0.3 - semver: 7.6.3 - optionalDependencies: - '@img/sharp-darwin-arm64': 0.33.5 - '@img/sharp-darwin-x64': 0.33.5 - '@img/sharp-libvips-darwin-arm64': 1.0.4 - '@img/sharp-libvips-darwin-x64': 1.0.4 - '@img/sharp-libvips-linux-arm': 1.0.5 - '@img/sharp-libvips-linux-arm64': 1.0.4 - '@img/sharp-libvips-linux-s390x': 1.0.4 - '@img/sharp-libvips-linux-x64': 1.0.4 - '@img/sharp-libvips-linuxmusl-arm64': 1.0.4 - '@img/sharp-libvips-linuxmusl-x64': 1.0.4 - '@img/sharp-linux-arm': 0.33.5 - '@img/sharp-linux-arm64': 0.33.5 - '@img/sharp-linux-s390x': 0.33.5 - '@img/sharp-linux-x64': 0.33.5 - '@img/sharp-linuxmusl-arm64': 0.33.5 - '@img/sharp-linuxmusl-x64': 0.33.5 - '@img/sharp-wasm32': 0.33.5 - '@img/sharp-win32-ia32': 0.33.5 - '@img/sharp-win32-x64': 0.33.5 - - shebang-command@2.0.0: - dependencies: - shebang-regex: 3.0.0 - - shebang-regex@3.0.0: {} - - shell-quote@1.8.1: {} - - side-channel@1.0.6: - dependencies: - call-bind: 1.0.7 - es-errors: 1.3.0 - get-intrinsic: 1.2.4 - object-inspect: 1.13.2 - - sift@16.0.1: {} - - signal-exit@3.0.7: {} - - signal-exit@4.1.0: {} - - simple-concat@1.0.1: {} - - simple-get@3.1.1: - dependencies: - decompress-response: 4.2.1 - once: 1.4.0 - simple-concat: 1.0.1 - - simple-get@4.0.1: - dependencies: - decompress-response: 6.0.0 - once: 1.4.0 - simple-concat: 1.0.1 - - simple-swizzle@0.2.2: - dependencies: - is-arrayish: 0.3.2 - - simple-update-notifier@2.0.0: - dependencies: - semver: 7.5.4 - - sirv@2.0.4: - dependencies: - '@polka/url': 1.0.0-next.25 - mrmime: 2.0.0 - totalist: 3.0.1 - - sisteransi@1.0.5: {} - - slash@3.0.0: {} - - slash@4.0.0: {} - - slate-history@0.93.0(slate@0.94.1): - dependencies: - is-plain-object: 5.0.0 - slate: 0.94.1 - - slate-react@0.98.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(slate@0.94.1): - dependencies: - '@juggle/resize-observer': 3.4.0 - '@types/is-hotkey': 0.1.10 - '@types/lodash': 4.17.7 - direction: 1.0.4 - is-hotkey: 0.1.8 - is-plain-object: 5.0.0 - lodash: 4.17.21 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - scroll-into-view-if-needed: 2.2.31 - slate: 0.94.1 - tiny-invariant: 1.0.6 - - slate@0.94.1: - dependencies: - immer: 9.0.19 - is-plain-object: 5.0.0 - tiny-warning: 1.0.3 - - snake-case@2.1.0: - dependencies: - no-case: 2.3.2 - - snapdragon-node@2.1.1: - dependencies: - define-property: 1.0.0 - isobject: 3.0.1 - snapdragon-util: 3.0.1 - - snapdragon-util@3.0.1: - dependencies: - kind-of: 3.2.2 - - snapdragon@0.8.2: - dependencies: - base: 0.11.2 - debug: 2.6.9 - define-property: 0.2.5 - extend-shallow: 2.0.1 - map-cache: 0.2.2 - source-map: 0.5.7 - source-map-resolve: 0.5.3 - use: 3.1.1 - transitivePeerDependencies: - - supports-color - - sort-object-keys@1.1.3: {} - - sort-package-json@2.5.1: - dependencies: - detect-indent: 7.0.1 - detect-newline: 4.0.1 - get-stdin: 9.0.0 - git-hooks-list: 3.1.0 - globby: 13.2.2 - is-plain-obj: 4.1.0 - sort-object-keys: 1.1.3 - - sorted-array-functions@1.3.0: {} - - source-list-map@2.0.1: {} - - source-map-js@1.2.0: {} - - source-map-resolve@0.5.3: - dependencies: - atob: 2.1.2 - decode-uri-component: 0.2.2 - resolve-url: 0.2.1 - source-map-url: 0.4.1 - urix: 0.1.0 - - source-map-support@0.5.21: - dependencies: - buffer-from: 1.1.2 - source-map: 0.6.1 - - source-map-url@0.4.1: {} - - source-map@0.5.7: {} - - source-map@0.6.1: {} - - source-map@0.7.4: {} - - spawn-command@0.0.2: {} - - spdx-correct@3.2.0: - dependencies: - spdx-expression-parse: 3.0.1 - spdx-license-ids: 3.0.20 - - spdx-exceptions@2.5.0: {} - - spdx-expression-parse@3.0.1: - dependencies: - spdx-exceptions: 2.5.0 - spdx-license-ids: 3.0.20 - - spdx-license-ids@3.0.20: {} - - speedometer@1.0.0: {} - - split-string@3.1.0: - dependencies: - extend-shallow: 3.0.2 - - split2@4.2.0: {} - - sprintf-js@1.0.3: {} - - stack-trace@0.0.10: {} - - stackframe@1.3.4: {} - - static-extend@0.1.2: - dependencies: - define-property: 0.2.5 - object-copy: 0.1.0 - - statuses@1.5.0: {} - - statuses@2.0.1: {} - - strapi-plugin-fuzzy-search@3.0.0(@strapi/strapi@4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0))(@strapi/utils@4.25.9)(yup@0.32.9): - dependencies: - '@strapi/strapi': 4.25.9(@babel/runtime@7.25.6)(@codemirror/autocomplete@6.18.0(@codemirror/language@6.10.2)(@codemirror/state@6.4.1)(@codemirror/view@6.33.0)(@lezer/common@1.2.1))(@codemirror/language@6.10.2)(@codemirror/lint@6.8.1)(@codemirror/search@6.5.6)(@codemirror/state@6.4.1)(@codemirror/theme-one-dark@6.1.2)(@codemirror/view@6.33.0)(@swc/helpers@0.5.13)(@types/hoist-non-react-statics@3.3.5)(@types/node@22.5.2)(@types/react@18.3.5)(codemirror@5.65.17)(pg@8.12.0)(react-dom@18.3.1(react@18.3.1))(react-router-dom@6.26.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)(redux@4.2.1)(styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(terser@5.31.6)(type-fest@2.19.0) - '@strapi/utils': 4.25.9 - fuzzysort: 3.0.2 - transliteration: 2.3.5 - yup: 0.32.9 - - stream-chain@2.2.5: {} - - stream-json@1.8.0: - dependencies: - stream-chain: 2.2.5 - - stream-slice@0.1.2: {} - - streamx@2.20.0: - dependencies: - fast-fifo: 1.3.2 - queue-tick: 1.0.1 - text-decoder: 1.1.1 - optionalDependencies: - bare-events: 2.4.2 - - string-argv@0.3.2: {} - - string-width@4.2.3: - dependencies: - emoji-regex: 8.0.0 - is-fullwidth-code-point: 3.0.0 - strip-ansi: 6.0.1 - - string-width@5.1.2: - dependencies: - eastasianwidth: 0.2.0 - emoji-regex: 9.2.2 - strip-ansi: 7.1.0 - - string_decoder@0.10.31: {} - - string_decoder@1.1.1: - dependencies: - safe-buffer: 5.1.2 - - string_decoder@1.3.0: - dependencies: - safe-buffer: 5.2.1 - - strip-ansi@3.0.1: - dependencies: - ansi-regex: 2.1.1 - - strip-ansi@5.2.0: - dependencies: - ansi-regex: 4.1.1 - - strip-ansi@6.0.1: - dependencies: - ansi-regex: 5.0.1 - - strip-ansi@7.1.0: - dependencies: - ansi-regex: 6.0.1 - - strip-final-newline@2.0.0: {} - - strip-final-newline@3.0.0: {} - - strip-json-comments@2.0.1: {} - - strnum@1.0.5: {} - - style-loader@3.3.4(webpack@5.94.0(esbuild@0.19.11)): - dependencies: - webpack: 5.94.0(esbuild@0.19.11) - - style-mod@4.1.2: {} - - styled-components@6.1.13(react-dom@18.3.1(react@18.3.1))(react@18.3.1): - dependencies: - '@emotion/is-prop-valid': 1.2.2 - '@emotion/unitless': 0.8.1 - '@types/stylis': 4.2.5 - css-to-react-native: 3.2.0 - csstype: 3.1.3 - postcss: 8.4.38 - react: 18.3.1 - react-dom: 18.3.1(react@18.3.1) - shallowequal: 1.1.0 - stylis: 4.3.2 - tslib: 2.6.2 - - stylis@4.2.0: {} - - stylis@4.3.2: {} - - supports-color@2.0.0: {} - - supports-color@5.5.0: - dependencies: - has-flag: 3.0.0 - - supports-color@7.2.0: - dependencies: - has-flag: 4.0.0 - - supports-color@8.1.1: - dependencies: - has-flag: 4.0.0 - - supports-preserve-symlinks-flag@1.0.0: {} - - swap-case@1.1.2: - dependencies: - lower-case: 1.1.4 - upper-case: 1.1.3 - - synckit@0.8.5: - dependencies: - '@pkgr/utils': 2.4.2 - tslib: 2.7.0 - - tapable@2.2.1: {} - - tar-fs@2.1.1: - dependencies: - chownr: 1.1.4 - mkdirp-classic: 0.5.3 - pump: 3.0.0 - tar-stream: 2.2.0 - - tar-fs@3.0.6: - dependencies: - pump: 3.0.0 - tar-stream: 3.1.7 - optionalDependencies: - bare-fs: 2.3.3 - bare-path: 2.1.3 - - tar-stream@2.2.0: - dependencies: - bl: 4.1.0 - end-of-stream: 1.4.4 - fs-constants: 1.0.0 - inherits: 2.0.4 - readable-stream: 3.6.2 - - tar-stream@3.1.7: - dependencies: - b4a: 1.6.6 - fast-fifo: 1.3.2 - streamx: 2.20.0 - - tar@6.1.13: - dependencies: - chownr: 2.0.0 - fs-minipass: 2.1.0 - minipass: 4.2.8 - minizlib: 2.1.2 - mkdirp: 1.0.4 - yallist: 4.0.0 - - tar@6.2.1: - dependencies: - chownr: 2.0.0 - fs-minipass: 2.1.0 - minipass: 5.0.0 - minizlib: 2.1.2 - mkdirp: 1.0.4 - yallist: 4.0.0 - - tarn@3.0.2: {} - - terser-webpack-plugin@5.3.10(esbuild@0.19.11)(webpack@5.94.0(esbuild@0.19.11)): - dependencies: - '@jridgewell/trace-mapping': 0.3.25 - jest-worker: 27.5.1 - schema-utils: 3.3.0 - serialize-javascript: 6.0.2 - terser: 5.31.6 - webpack: 5.94.0(esbuild@0.19.11) - optionalDependencies: - esbuild: 0.19.11 - - terser@5.31.6: - dependencies: - '@jridgewell/source-map': 0.3.6 - acorn: 8.12.1 - commander: 2.20.3 - source-map-support: 0.5.21 - - text-decoder@1.1.1: - dependencies: - b4a: 1.6.6 - - text-hex@1.0.0: {} - - thenify-all@1.6.0: - dependencies: - thenify: 3.3.1 - - thenify@3.3.1: - dependencies: - any-promise: 1.3.0 - - through2@2.0.5: - dependencies: - readable-stream: 2.3.8 - xtend: 4.0.2 - - through@2.3.8: {} - - tildify@2.0.0: {} - - tiny-invariant@1.0.6: {} - - tiny-invariant@1.3.3: {} - - tiny-warning@1.0.3: {} - - title-case@2.1.1: - dependencies: - no-case: 2.3.2 - upper-case: 1.1.3 - - titleize@3.0.0: {} - - tmp@0.0.33: - dependencies: - os-tmpdir: 1.0.2 - - to-fast-properties@2.0.0: {} - - to-object-path@0.3.0: - dependencies: - kind-of: 3.2.2 - - to-regex-range@2.1.1: - dependencies: - is-number: 3.0.0 - repeat-string: 1.6.1 - - to-regex-range@5.0.1: - dependencies: - is-number: 7.0.0 - - to-regex@3.0.2: - dependencies: - define-property: 2.0.2 - extend-shallow: 3.0.2 - regex-not: 1.0.2 - safe-regex: 1.1.0 - - toidentifier@1.0.1: {} - - toposort@2.0.2: {} - - totalist@3.0.1: {} - - touch@3.1.1: {} - - tr46@0.0.3: {} - - transliteration@2.3.5: - dependencies: - yargs: 17.7.2 - - tree-kill@1.2.2: {} - - triple-beam@1.4.1: {} - - tslib@1.14.1: {} - - tslib@2.6.2: {} - - tslib@2.7.0: {} - - tsscmp@1.0.6: {} - - tunnel-agent@0.6.0: - dependencies: - safe-buffer: 5.2.1 - - type-fest@0.20.2: {} - - type-fest@0.21.3: {} - - type-fest@0.6.0: {} - - type-fest@0.8.1: {} - - type-fest@2.19.0: {} - - type-is@1.6.18: - dependencies: - media-typer: 0.3.0 - mime-types: 2.1.35 - - typedarray-to-buffer@3.1.5: - dependencies: - is-typedarray: 1.0.0 - - typescript@5.2.2: {} - - uc.micro@1.0.6: {} - - uglify-js@3.19.3: - optional: true - - umzug@3.2.1(@types/node@22.5.2): - dependencies: - '@rushstack/ts-command-line': 4.22.6(@types/node@22.5.2) - emittery: 0.12.1 - fs-jetpack: 4.3.1 - glob: 8.1.0 - pony-cause: 2.1.11 - type-fest: 2.19.0 - transitivePeerDependencies: - - '@types/node' - - unc-path-regex@0.1.2: {} - - undefsafe@2.0.5: {} - - undici-types@5.26.5: {} - - undici-types@6.19.8: {} - - union-value@1.0.1: - dependencies: - arr-union: 3.1.0 - get-value: 2.0.6 - is-extendable: 0.1.1 - set-value: 2.0.1 - - unique-string@2.0.0: - dependencies: - crypto-random-string: 2.0.0 - - universalify@0.1.2: {} - - universalify@2.0.1: {} - - unload@2.2.0: - dependencies: - '@babel/runtime': 7.25.6 - detect-node: 2.1.0 - - unpipe@1.0.0: {} - - unset-value@1.0.0: - dependencies: - has-value: 0.3.1 - isobject: 3.0.1 - - untildify@4.0.0: {} - - update-browserslist-db@1.1.0(browserslist@4.23.3): - dependencies: - browserslist: 4.23.3 - escalade: 3.2.0 - picocolors: 1.1.0 - - upper-case-first@1.1.2: - dependencies: - upper-case: 1.1.3 - - upper-case@1.1.3: {} - - uri-js@4.4.1: - dependencies: - punycode: 2.3.1 - - urix@0.1.0: {} - - url-join@4.0.1: {} - - use-callback-ref@1.3.2(@types/react@18.3.5)(react@18.3.1): - dependencies: - react: 18.3.1 - tslib: 2.7.0 - optionalDependencies: - '@types/react': 18.3.5 - - use-isomorphic-layout-effect@1.1.2(@types/react@18.3.5)(react@18.3.1): - dependencies: - react: 18.3.1 - optionalDependencies: - '@types/react': 18.3.5 - - use-sidecar@1.1.2(@types/react@18.3.5)(react@18.3.1): - dependencies: - detect-node-es: 1.1.0 - react: 18.3.1 - tslib: 2.7.0 - optionalDependencies: - '@types/react': 18.3.5 - - use-sync-external-store@1.2.2(react@18.3.1): - dependencies: - react: 18.3.1 - - use@3.1.1: {} - - user-home@1.1.1: {} - - util-deprecate@1.0.2: {} - - utila@0.4.0: {} - - utils-merge@1.0.1: {} - - uuid@8.3.2: {} - - uuid@9.0.1: {} - - v8flags@2.1.1: - dependencies: - user-home: 1.1.1 - - validate-npm-package-license@3.0.4: - dependencies: - spdx-correct: 3.2.0 - spdx-expression-parse: 3.0.1 - - value-equal@1.0.1: {} - - vary@1.1.2: {} - - vite@5.0.13(@types/node@22.5.2)(terser@5.31.6): - dependencies: - esbuild: 0.19.11 - postcss: 8.4.44 - rollup: 4.21.2 - optionalDependencies: - '@types/node': 22.5.2 - fsevents: 2.3.3 - terser: 5.31.6 - - w3c-keyname@2.2.8: {} - - watchpack@2.4.2: - dependencies: - glob-to-regexp: 0.4.1 - graceful-fs: 4.2.11 - - wcwidth@1.0.1: - dependencies: - defaults: 1.0.4 - - web-streams-polyfill@3.3.3: {} - - web-streams-polyfill@4.0.0-beta.3: {} - - webidl-conversions@3.0.1: {} - - webpack-bundle-analyzer@4.10.2: - dependencies: - '@discoveryjs/json-ext': 0.5.7 - acorn: 8.12.1 - acorn-walk: 8.3.3 - commander: 7.2.0 - debounce: 1.2.1 - escape-string-regexp: 4.0.0 - gzip-size: 6.0.0 - html-escaper: 2.0.2 - opener: 1.5.2 - picocolors: 1.1.0 - sirv: 2.0.4 - ws: 7.5.10 - transitivePeerDependencies: - - bufferutil - - utf-8-validate - - webpack-dev-middleware@6.1.2(webpack@5.94.0(esbuild@0.19.11)): - dependencies: - colorette: 2.0.20 - memfs: 3.5.3 - mime-types: 2.1.35 - range-parser: 1.2.1 - schema-utils: 4.2.0 - optionalDependencies: - webpack: 5.94.0(esbuild@0.19.11) - - webpack-hot-middleware@2.26.0: - dependencies: - ansi-html-community: 0.0.8 - html-entities: 2.5.2 - strip-ansi: 6.0.1 - - webpack-sources@1.4.3: - dependencies: - source-list-map: 2.0.1 - source-map: 0.6.1 - - webpack-sources@3.2.3: {} - - webpack@5.94.0(esbuild@0.19.11): - dependencies: - '@types/estree': 1.0.5 - '@webassemblyjs/ast': 1.12.1 - '@webassemblyjs/wasm-edit': 1.12.1 - '@webassemblyjs/wasm-parser': 1.12.1 - acorn: 8.12.1 - acorn-import-attributes: 1.9.5(acorn@8.12.1) - browserslist: 4.23.3 - chrome-trace-event: 1.0.4 - enhanced-resolve: 5.17.1 - es-module-lexer: 1.5.4 - eslint-scope: 5.1.1 - events: 3.3.0 - glob-to-regexp: 0.4.1 - graceful-fs: 4.2.11 - json-parse-even-better-errors: 2.3.1 - loader-runner: 4.3.0 - mime-types: 2.1.35 - neo-async: 2.6.2 - schema-utils: 3.3.0 - tapable: 2.2.1 - terser-webpack-plugin: 5.3.10(esbuild@0.19.11)(webpack@5.94.0(esbuild@0.19.11)) - watchpack: 2.4.2 - webpack-sources: 3.2.3 - transitivePeerDependencies: - - '@swc/core' - - esbuild - - uglify-js - - whatwg-url@5.0.0: - dependencies: - tr46: 0.0.3 - webidl-conversions: 3.0.1 - - which@1.3.1: - dependencies: - isexe: 2.0.0 - - which@2.0.2: - dependencies: - isexe: 2.0.0 - - wide-align@1.1.5: - dependencies: - string-width: 4.2.3 - - widest-line@3.1.0: - dependencies: - string-width: 4.2.3 - - winston-transport@4.7.1: - dependencies: - logform: 2.6.1 - readable-stream: 3.6.2 - triple-beam: 1.4.1 - - winston@3.10.0: - dependencies: - '@colors/colors': 1.5.0 - '@dabh/diagnostics': 2.0.3 - async: 3.2.6 - is-stream: 2.0.1 - logform: 2.6.1 - one-time: 1.0.0 - readable-stream: 3.6.2 - safe-stable-stringify: 2.5.0 - stack-trace: 0.0.10 - triple-beam: 1.4.1 - winston-transport: 4.7.1 - - wordwrap@1.0.0: {} - - wrap-ansi@7.0.0: - dependencies: - ansi-styles: 4.3.0 - string-width: 4.2.3 - strip-ansi: 6.0.1 - - wrap-ansi@8.1.0: - dependencies: - ansi-styles: 6.2.1 - string-width: 5.1.2 - strip-ansi: 7.1.0 - - wrappy@1.0.2: {} - - write-file-atomic@3.0.3: - dependencies: - imurmurhash: 0.1.4 - is-typedarray: 1.0.0 - signal-exit: 3.0.7 - typedarray-to-buffer: 3.1.5 - - ws@7.5.10: {} - - ws@8.13.0: {} - - xdg-app-paths@8.3.0: - dependencies: - xdg-portable: 10.6.0 - optionalDependencies: - fsevents: 2.3.3 - - xdg-basedir@4.0.0: {} - - xdg-portable@10.6.0: - dependencies: - os-paths: 7.4.0 - optionalDependencies: - fsevents: 2.3.3 - - xtend@4.0.2: {} - - y18n@5.0.8: {} - - yalc@1.0.0-pre.53: - dependencies: - chalk: 4.1.2 - detect-indent: 6.1.0 - fs-extra: 8.1.0 - glob: 7.2.3 - ignore: 5.3.2 - ini: 2.0.0 - npm-packlist: 2.2.2 - yargs: 16.2.0 - - yallist@4.0.0: {} - - yaml@1.10.2: {} - - yargs-parser@20.2.9: {} - - yargs-parser@21.1.1: {} - - yargs@16.2.0: - dependencies: - cliui: 7.0.4 - escalade: 3.2.0 - get-caller-file: 2.0.5 - require-directory: 2.1.1 - string-width: 4.2.3 - y18n: 5.0.8 - yargs-parser: 20.2.9 - - yargs@17.7.2: - dependencies: - cliui: 8.0.1 - escalade: 3.2.0 - get-caller-file: 2.0.5 - require-directory: 2.1.1 - string-width: 4.2.3 - y18n: 5.0.8 - yargs-parser: 21.1.1 - - ylru@1.4.0: {} - - yocto-queue@0.1.0: {} - - yup@0.32.9: - dependencies: - '@babel/runtime': 7.25.6 - '@types/lodash': 4.17.7 - lodash: 4.17.21 - lodash-es: 4.17.21 - nanoclone: 0.2.1 - property-expr: 2.0.6 - toposort: 2.0.2 diff --git a/services/strapi/public/robots.txt b/services/strapi/public/robots.txt deleted file mode 100644 index ff5d316..0000000 --- a/services/strapi/public/robots.txt +++ /dev/null @@ -1,3 +0,0 @@ -# To prevent search engines from seeing the site altogether, uncomment the next two lines: -# User-Agent: * -# Disallow: / diff --git a/services/strapi/public/uploads/.gitkeep b/services/strapi/public/uploads/.gitkeep deleted file mode 100644 index e69de29..0000000 diff --git a/services/strapi/src/admin/app.example.js b/services/strapi/src/admin/app.example.js deleted file mode 100644 index 45cad61..0000000 --- a/services/strapi/src/admin/app.example.js +++ /dev/null @@ -1,39 +0,0 @@ -const config = { - locales: [ - // 'ar', - // 'fr', - // 'cs', - // 'de', - // 'dk', - // 'es', - // 'he', - // 'id', - // 'it', - // 'ja', - // 'ko', - // 'ms', - // 'nl', - // 'no', - // 'pl', - // 'pt-BR', - // 'pt', - // 'ru', - // 'sk', - // 'sv', - // 'th', - // 'tr', - // 'uk', - // 'vi', - // 'zh-Hans', - // 'zh', - ], -}; - -const bootstrap = (app) => { - console.log(app); -}; - -export default { - config, - bootstrap, -}; diff --git a/services/strapi/src/admin/webpack.config.example.js b/services/strapi/src/admin/webpack.config.example.js deleted file mode 100644 index 1ca45c2..0000000 --- a/services/strapi/src/admin/webpack.config.example.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/* eslint-disable no-unused-vars */ -module.exports = (config, webpack) => { - // Note: we provide webpack above so you should not `require` it - // Perform customizations to webpack config - // Important: return the modified config - return config; -}; diff --git a/services/strapi/src/api/.gitkeep b/services/strapi/src/api/.gitkeep deleted file mode 100644 index e69de29..0000000 diff --git a/services/strapi/src/api/b2-file/content-types/b2-file/schema.json b/services/strapi/src/api/b2-file/content-types/b2-file/schema.json deleted file mode 100644 index ba68a4c..0000000 --- a/services/strapi/src/api/b2-file/content-types/b2-file/schema.json +++ /dev/null @@ -1,36 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "b2_files", - "info": { - "singularName": "b2-file", - "pluralName": "b2-files", - "displayName": "B2 File", - "description": "" - }, - "options": { - "draftAndPublish": false - }, - "pluginOptions": {}, - "attributes": { - "url": { - "type": "string", - "required": false, - "unique": false - }, - "key": { - "type": "string", - "unique": true, - "required": true - }, - "uploadId": { - "type": "string", - "unique": true, - "required": true - }, - "cdnUrl": { - "type": "string", - "unique": true, - "required": true - } - } -} diff --git a/services/strapi/src/api/b2-file/controllers/b2-file.js b/services/strapi/src/api/b2-file/controllers/b2-file.js deleted file mode 100644 index 65f936a..0000000 --- a/services/strapi/src/api/b2-file/controllers/b2-file.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * b2-file controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::b2-file.b2-file'); diff --git a/services/strapi/src/api/b2-file/routes/b2-file.js b/services/strapi/src/api/b2-file/routes/b2-file.js deleted file mode 100644 index a74a8d9..0000000 --- a/services/strapi/src/api/b2-file/routes/b2-file.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * b2-file router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -module.exports = createCoreRouter('api::b2-file.b2-file'); diff --git a/services/strapi/src/api/b2-file/services/b2-file.js b/services/strapi/src/api/b2-file/services/b2-file.js deleted file mode 100644 index 03e2bdf..0000000 --- a/services/strapi/src/api/b2-file/services/b2-file.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * b2-file service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::b2-file.b2-file'); diff --git a/services/strapi/src/api/contributor/content-types/contributor/schema.json b/services/strapi/src/api/contributor/content-types/contributor/schema.json deleted file mode 100644 index 3cc046c..0000000 --- a/services/strapi/src/api/contributor/content-types/contributor/schema.json +++ /dev/null @@ -1,30 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "contributors", - "info": { - "singularName": "contributor", - "pluralName": "contributors", - "displayName": "Contributor", - "description": "" - }, - "options": { - "draftAndPublish": false - }, - "pluginOptions": {}, - "attributes": { - "name": { - "type": "string", - "required": true - }, - "url": { - "type": "string" - }, - "isFinancialDonor": { - "type": "boolean", - "default": false - }, - "isVodProvider": { - "type": "boolean" - } - } -} diff --git a/services/strapi/src/api/contributor/controllers/contributor.js b/services/strapi/src/api/contributor/controllers/contributor.js deleted file mode 100644 index 22b9cf6..0000000 --- a/services/strapi/src/api/contributor/controllers/contributor.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * contributor controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::contributor.contributor'); diff --git a/services/strapi/src/api/contributor/routes/contributor.js b/services/strapi/src/api/contributor/routes/contributor.js deleted file mode 100644 index cf61a59..0000000 --- a/services/strapi/src/api/contributor/routes/contributor.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * contributor router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -module.exports = createCoreRouter('api::contributor.contributor'); diff --git a/services/strapi/src/api/contributor/services/contributor.js b/services/strapi/src/api/contributor/services/contributor.js deleted file mode 100644 index ca75442..0000000 --- a/services/strapi/src/api/contributor/services/contributor.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * contributor service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::contributor.contributor'); diff --git a/services/strapi/src/api/discord-interaction/content-types/discord-interaction/schema.json b/services/strapi/src/api/discord-interaction/content-types/discord-interaction/schema.json deleted file mode 100644 index f36a2ed..0000000 --- a/services/strapi/src/api/discord-interaction/content-types/discord-interaction/schema.json +++ /dev/null @@ -1,26 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "discord_interactions", - "info": { - "singularName": "discord-interaction", - "pluralName": "discord-interactions", - "displayName": "Discord Interaction", - "description": "" - }, - "options": { - "draftAndPublish": false - }, - "pluginOptions": {}, - "attributes": { - "interactionId": { - "type": "string", - "required": true - }, - "userSubmittedContent": { - "type": "relation", - "relation": "oneToOne", - "target": "api::user-submitted-content.user-submitted-content", - "inversedBy": "discordInteraction" - } - } -} diff --git a/services/strapi/src/api/discord-interaction/controllers/discord-interaction.js b/services/strapi/src/api/discord-interaction/controllers/discord-interaction.js deleted file mode 100644 index cdd8502..0000000 --- a/services/strapi/src/api/discord-interaction/controllers/discord-interaction.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * discord-interaction controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::discord-interaction.discord-interaction'); diff --git a/services/strapi/src/api/discord-interaction/routes/discord-interaction.js b/services/strapi/src/api/discord-interaction/routes/discord-interaction.js deleted file mode 100644 index 65807ee..0000000 --- a/services/strapi/src/api/discord-interaction/routes/discord-interaction.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * discord-interaction router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -module.exports = createCoreRouter('api::discord-interaction.discord-interaction'); diff --git a/services/strapi/src/api/discord-interaction/services/discord-interaction.js b/services/strapi/src/api/discord-interaction/services/discord-interaction.js deleted file mode 100644 index 59f8199..0000000 --- a/services/strapi/src/api/discord-interaction/services/discord-interaction.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * discord-interaction service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::discord-interaction.discord-interaction'); diff --git a/services/strapi/src/api/goal/content-types/goal/schema.json b/services/strapi/src/api/goal/content-types/goal/schema.json deleted file mode 100644 index 0580c6f..0000000 --- a/services/strapi/src/api/goal/content-types/goal/schema.json +++ /dev/null @@ -1,30 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "goals", - "info": { - "singularName": "goal", - "pluralName": "goals", - "displayName": "Goal", - "description": "" - }, - "options": { - "draftAndPublish": false - }, - "pluginOptions": {}, - "attributes": { - "amountCents": { - "type": "string", - "required": true, - "unique": true - }, - "description": { - "type": "string", - "required": true, - "unique": true - }, - "name": { - "type": "string", - "unique": true - } - } -} diff --git a/services/strapi/src/api/goal/controllers/goal.js b/services/strapi/src/api/goal/controllers/goal.js deleted file mode 100644 index 2b8e3a3..0000000 --- a/services/strapi/src/api/goal/controllers/goal.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * goal controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::goal.goal'); diff --git a/services/strapi/src/api/goal/routes/goal.js b/services/strapi/src/api/goal/routes/goal.js deleted file mode 100644 index e836728..0000000 --- a/services/strapi/src/api/goal/routes/goal.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * goal router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -module.exports = createCoreRouter('api::goal.goal'); diff --git a/services/strapi/src/api/goal/services/goal.js b/services/strapi/src/api/goal/services/goal.js deleted file mode 100644 index 6c047ac..0000000 --- a/services/strapi/src/api/goal/services/goal.js +++ /dev/null @@ -1,11 +0,0 @@ -'use strict'; - -/** - * goal service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::goal.goal'); - - diff --git a/services/strapi/src/api/gogs/content-types/gogs/schema.json b/services/strapi/src/api/gogs/content-types/gogs/schema.json deleted file mode 100644 index d933d5d..0000000 --- a/services/strapi/src/api/gogs/content-types/gogs/schema.json +++ /dev/null @@ -1,24 +0,0 @@ -{ - "kind": "singleType", - "collectionName": "gogss", - "info": { - "singularName": "gogs", - "pluralName": "gogss", - "displayName": "Gogs" - }, - "options": { - "draftAndPublish": false - }, - "pluginOptions": {}, - "attributes": { - "apiKey": { - "type": "string", - "required": true - }, - "url": { - "type": "string", - "required": true, - "default": "https://git.futureporn.net" - } - } -} diff --git a/services/strapi/src/api/gogs/controllers/gogs.js b/services/strapi/src/api/gogs/controllers/gogs.js deleted file mode 100644 index 5d83874..0000000 --- a/services/strapi/src/api/gogs/controllers/gogs.js +++ /dev/null @@ -1,29 +0,0 @@ -'use strict'; - -/** - * gogs controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::gogs.gogs', ({ strapi }) => ({ - issues: async (ctx) => { - try { - // Fetch the 'gogs' single type from Strapi - const gogsConfig = await strapi.query('api::gogs.gogs').findOne(); - - if (!gogsConfig) { - return ctx.badRequest('Gogs configuration not found'); - } - - const { url, apiKey } = gogsConfig; - const openIssues = await strapi.service('api::gogs.gogs').fetchAllPagesFromGogsAPI(`${url}/api/v1/repos/futureporn/pm/issues?state=open`, apiKey) - const closedIssues = await strapi.service('api::gogs.gogs').fetchAllPagesFromGogsAPI(`${url}/api/v1/repos/futureporn/pm/issues?state=closed`, apiKey) - - return { openIssues, closedIssues } - } catch (error) { - console.error('Error fetching Gogs issues:', error); - return ctx.badRequest('Failed to fetch issues from Gogs'); - } - } -})); \ No newline at end of file diff --git a/services/strapi/src/api/gogs/routes/gogs.js b/services/strapi/src/api/gogs/routes/gogs.js deleted file mode 100644 index 50c6544..0000000 --- a/services/strapi/src/api/gogs/routes/gogs.js +++ /dev/null @@ -1,34 +0,0 @@ -'use strict'; - -/** - * gogs router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -const defaultRouter = createCoreRouter('api::gogs.gogs'); - -// greets https://forum.strapi.io/t/how-to-add-custom-routes-to-core-routes-in-strapi-4/14070/7 -const customRouter = (innerRouter, extraRoutes = []) => { - let routes; - return { - get prefix() { - return innerRouter.prefix; - }, - get routes() { - if (!routes) routes = extraRoutes.concat(innerRouter.routes) - return routes; - }, - }; -}; - -const myExtraRoutes = [ - { - method: "GET", - path: "/gogs/issues", - handler: "api::gogs.gogs.issues" - } -]; - -module.exports = customRouter(defaultRouter, myExtraRoutes); - diff --git a/services/strapi/src/api/gogs/services/gogs.js b/services/strapi/src/api/gogs/services/gogs.js deleted file mode 100644 index 13afb2f..0000000 --- a/services/strapi/src/api/gogs/services/gogs.js +++ /dev/null @@ -1,42 +0,0 @@ -'use strict'; - -/** - * gogs service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - - - -module.exports = createCoreService('api::gogs.gogs', ({ strapi }) => ({ - - async fetchAllPagesFromGogsAPI(url, apiKey) { - - // Fetch the first page - const response = await fetch(url, { - headers: { - 'Authorization': `token ${apiKey}` - } - }); - - if (!response.ok) { - throw new Error(`Request failed with status: ${response.status}`); - } - - const data = await response.json(); - - // Check if there are more pages available - if (response.headers.has('link')) { - const linkHeader = response.headers.get('link'); - const nextPageMatch = /<([^>]+)>;\s*rel="next"/.exec(linkHeader); - - if (nextPageMatch) { - const nextPageUrl = nextPageMatch[1]; - const nextPageData = await this.fetchAllPagesFromGogsAPI(nextPageUrl, apiKey); - return [...data, ...nextPageData]; - } - } - - return data; - } -})) \ No newline at end of file diff --git a/services/strapi/src/api/issue/content-types/issue/schema.json b/services/strapi/src/api/issue/content-types/issue/schema.json deleted file mode 100644 index b8c6768..0000000 --- a/services/strapi/src/api/issue/content-types/issue/schema.json +++ /dev/null @@ -1,36 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "issues", - "info": { - "singularName": "issue", - "pluralName": "issues", - "displayName": "issue", - "description": "" - }, - "options": { - "draftAndPublish": false - }, - "pluginOptions": {}, - "attributes": { - "url": { - "type": "string", - "required": true - }, - "sla": { - "type": "enumeration", - "enum": [ - "public", - "patron", - "authenticated" - ], - "default": "public", - "required": true - }, - "type": { - "type": "enumeration", - "enum": [ - "stall" - ] - } - } -} diff --git a/services/strapi/src/api/issue/controllers/issue.js b/services/strapi/src/api/issue/controllers/issue.js deleted file mode 100644 index a9c83e3..0000000 --- a/services/strapi/src/api/issue/controllers/issue.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * issue controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::issue.issue'); diff --git a/services/strapi/src/api/issue/routes/issue.js b/services/strapi/src/api/issue/routes/issue.js deleted file mode 100644 index 146d9fe..0000000 --- a/services/strapi/src/api/issue/routes/issue.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * issue router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -module.exports = createCoreRouter('api::issue.issue'); diff --git a/services/strapi/src/api/issue/services/issue.js b/services/strapi/src/api/issue/services/issue.js deleted file mode 100644 index 9782a38..0000000 --- a/services/strapi/src/api/issue/services/issue.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * issue service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::issue.issue'); diff --git a/services/strapi/src/api/mux-asset/content-types/mux-asset/schema.json b/services/strapi/src/api/mux-asset/content-types/mux-asset/schema.json deleted file mode 100644 index 736ba34..0000000 --- a/services/strapi/src/api/mux-asset/content-types/mux-asset/schema.json +++ /dev/null @@ -1,28 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "mux_assets", - "info": { - "singularName": "mux-asset", - "pluralName": "mux-assets", - "displayName": "Mux Asset", - "description": "" - }, - "options": { - "draftAndPublish": false - }, - "pluginOptions": {}, - "attributes": { - "playbackId": { - "type": "string", - "required": false, - "unique": true - }, - "assetId": { - "type": "string", - "unique": true - }, - "deletionQueuedAt": { - "type": "datetime" - } - } -} diff --git a/services/strapi/src/api/mux-asset/controllers/mux-asset.js b/services/strapi/src/api/mux-asset/controllers/mux-asset.js deleted file mode 100644 index e02adbc..0000000 --- a/services/strapi/src/api/mux-asset/controllers/mux-asset.js +++ /dev/null @@ -1,56 +0,0 @@ -'use strict'; - -const { JWT } = require('@mux/mux-node'); - -const MUX_SIGNING_KEY_ID = process.env.MUX_SIGNING_KEY_ID; -const MUX_SIGNING_KEY_PRIVATE_KEY = process.env.MUX_SIGNING_KEY_PRIVATE_KEY; -const MUX_PLAYBACK_RESTRICTION_ID = process.env.MUX_PLAYBACK_RESTRICTION_ID - -if (!MUX_SIGNING_KEY_PRIVATE_KEY) throw new Error('MUX_SIGNING_KEY_PRIVATE_KEY must be defined in env'); -if (!MUX_SIGNING_KEY_ID) throw new Error('MUX_SIGNING_KEY_ID must be defined in env'); -if (!MUX_PLAYBACK_RESTRICTION_ID) throw new Error('MUX_PLAYBACK_RESTRICTION_ID must be defined in env'); - - -/** - * mux-asset controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::mux-asset.mux-asset', ({ strapi }) => ({ - - async secure (ctx, next) { - - if (ctx?.query?.id === undefined) { - ctx.throw(400, 'id query param was missing!') - return - } - - const tokens = {} - - tokens.playbackToken = JWT.signPlaybackId(ctx.query.id, { - keyId: MUX_SIGNING_KEY_ID, - keySecret: MUX_SIGNING_KEY_PRIVATE_KEY, - params: { - playback_restriction_id: MUX_PLAYBACK_RESTRICTION_ID - }, - }) - - - tokens.storyboardToken = JWT.signPlaybackId(ctx.query.id, { - keyId: MUX_SIGNING_KEY_ID, - keySecret: MUX_SIGNING_KEY_PRIVATE_KEY, - type: 'storyboard' - }) - - tokens.thumbnailToken = JWT.signPlaybackId(ctx.query.id, { - keyId: MUX_SIGNING_KEY_ID, - keySecret: MUX_SIGNING_KEY_PRIVATE_KEY, - type: 'thumbnail' - }) - - - ctx.body = tokens - } -})) - diff --git a/services/strapi/src/api/mux-asset/routes/mux-asset.js b/services/strapi/src/api/mux-asset/routes/mux-asset.js deleted file mode 100644 index 2dc38eb..0000000 --- a/services/strapi/src/api/mux-asset/routes/mux-asset.js +++ /dev/null @@ -1,33 +0,0 @@ -'use strict'; - -/** - * mux-asset router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -const defaultRouter = createCoreRouter('api::mux-asset.mux-asset'); - -// greets https://forum.strapi.io/t/how-to-add-custom-routes-to-core-routes-in-strapi-4/14070/7 -const customRouter = (innerRouter, extraRoutes = []) => { - let routes; - return { - get prefix() { - return innerRouter.prefix; - }, - get routes() { - if (!routes) routes = extraRoutes.concat(innerRouter.routes) - return routes; - }, - }; -}; - -const myExtraRoutes = [ - { - method: "GET", - path: "/mux-asset/secure", - handler: "mux-asset.secure" - } -]; - -module.exports = customRouter(defaultRouter, myExtraRoutes); \ No newline at end of file diff --git a/services/strapi/src/api/mux-asset/services/mux-asset.js b/services/strapi/src/api/mux-asset/services/mux-asset.js deleted file mode 100644 index a4cbcb5..0000000 --- a/services/strapi/src/api/mux-asset/services/mux-asset.js +++ /dev/null @@ -1,78 +0,0 @@ -'use strict'; - -const { JWT } = require('@mux/mux-node'); -const { createCoreService } = require('@strapi/strapi').factories; - -/** - * mux-asset service - */ - - -module.exports = createCoreService('api::mux-asset.mux-asset', ({strapi}) => ({ - - - /** - * reference: https://docs.mux.com/guides/video/secure-video-playback#4-generate-a-json-web-token-jwt - * reference: https://docs.mux.com/guides/video/secure-video-playback#5-sign-the-json-web-token-jwt - * - * @param {String} playbackId - signed playback ID - * @param {String} keyId - signing key ID - * @param {String} keySecret - base64 encoded private key - * @param {String} playbackRestructionId - https://docs.mux.com/guides/video/secure-video-playback#create-a-playback-restriction - * @returns {Object} jwt - - * @returns {String} jwt.token - - * @returns {String} jwt.gifToken - - * @returns {String} jwt.thumbnailToken - - */ - async signJwt (playbackId, keyId, keySecret, playbackRestrictionId) { - // Set some base options we can use for a few different signing types - // Type can be either video, thumbnail, gif, or storyboard - let baseOptions = { - keyId: keyId, // Enter your signing key id here - keySecret: keySecret, // Enter your base64 encoded private key here - expiration: '7d' // E.g 60, "2 days", "10h", "7d", numeric value interpreted as seconds - }; - - const playbackToken = JWT.signPlaybackId(playbackId, { - ...baseOptions , - type: 'video', - params: { playback_restriction_id: playbackRestrictionId } - }); - - // Now the signed playback url should look like this: - // https://stream.mux.com/${playbackId}.m3u8?token=${token} - - // If you wanted to pass in params for something like a gif, use the - // params key in the options object - // const gifToken = JWT.signPlaybackId(playbackId, { - // ...baseOptions, - // type: 'gif', - // params: { time: 10 }, - // }) - - const thumbnailToken = JWT.signPlaybackId(playbackId, { - type: 'thumbnail', - params: { playback_restriction_id: playbackRestrictionId }, - }) - - // Then, use this token in a URL like this: - // https://image.mux.com/${playbackId}/animated.gif?token=${gifToken} - - // A final example, if you wanted to sign a thumbnail url with a playback restriction - const storyboardToken = JWT.sign(playbackId, { - ...baseOptions, - type: 'storyboard', - params: { playback_restriction_id: playbackRestrictionId }, - }) - - // When used in a URL, it should look like this: - // https://image.mux.com/${playbackId}/thumbnail.png?token=${thumbnailToken} - - return { - playbackToken, - storyboardToken, - thumbnailToken - } - }, - -})); \ No newline at end of file diff --git a/services/strapi/src/api/patreon/content-types/patreon/schema.json b/services/strapi/src/api/patreon/content-types/patreon/schema.json deleted file mode 100644 index a76213c..0000000 --- a/services/strapi/src/api/patreon/content-types/patreon/schema.json +++ /dev/null @@ -1,39 +0,0 @@ -{ - "kind": "singleType", - "collectionName": "patreons", - "info": { - "singularName": "patreon", - "pluralName": "patreons", - "displayName": "Patreon", - "description": "" - }, - "options": { - "draftAndPublish": false - }, - "pluginOptions": {}, - "attributes": { - "benefitId": { - "type": "string", - "required": true, - "default": "4760169" - }, - "accessToken": { - "type": "string" - }, - "refreshToken": { - "type": "string" - }, - "campaignId": { - "type": "string", - "required": true - }, - "muxAllocationCostCents": { - "type": "integer", - "default": 50, - "required": true - }, - "membershipId": { - "type": "string" - } - } -} diff --git a/services/strapi/src/api/patreon/controllers/patreon.js b/services/strapi/src/api/patreon/controllers/patreon.js deleted file mode 100644 index c06944a..0000000 --- a/services/strapi/src/api/patreon/controllers/patreon.js +++ /dev/null @@ -1,37 +0,0 @@ -'use strict'; - -/** - * patreon controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::patreon.patreon', ({ strapi }) => ({ - async getPublicPatrons(ctx) { - const patrons = await strapi.entityService.findMany('plugin::users-permissions.user', { - fields: ['username', 'vanityLink', 'isNamePublic', 'isLinkPublic', 'patreonBenefits'], - }) - - let publicPatrons = [] - for (const patron of patrons) { - let publicPatron = {} - let benefits = (!!patron?.patreonBenefits) ? patron.patreonBenefits.split(',') : [] - if (patron.isNamePublic) publicPatron.username = patron.username; - - // if patron has "Your URL displayed on Futureporn.net" benefit, - // publically share their link if they want it shared - if (benefits.includes('10663202')) { - if (patron.isLinkPublic) publicPatron.vanityLink = patron.vanityLink; - } - - if (!!publicPatron.username || !!publicPatron.vanityLink) publicPatrons.push(publicPatron); - } - - return publicPatrons - }, - async muxAllocationCount(ctx) { - const count = await strapi.service('api::patreon.patreon').getMuxAllocationCount() - return count - } -})); - diff --git a/services/strapi/src/api/patreon/routes/patreon.js b/services/strapi/src/api/patreon/routes/patreon.js deleted file mode 100644 index 52af951..0000000 --- a/services/strapi/src/api/patreon/routes/patreon.js +++ /dev/null @@ -1,37 +0,0 @@ -'use strict'; - -/** - * patreon router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -const defaultRouter = createCoreRouter('api::patreon.patreon'); - -// greets https://forum.strapi.io/t/how-to-add-custom-routes-to-core-routes-in-strapi-4/14070/7 -const customRouter = (innerRouter, extraRoutes = []) => { - let routes; - return { - get prefix() { - return innerRouter.prefix; - }, - get routes() { - if (!routes) routes = extraRoutes.concat(innerRouter.routes) - return routes; - }, - }; -}; - -const myExtraRoutes = [ - { - method: "GET", - path: "/patreon/patrons", - handler: "api::patreon.patreon.getPublicPatrons" - }, { - method: 'GET', - path: '/patreon/muxAllocationCount', - handler: 'api::patreon.patreon.muxAllocationCount' - } -]; - -module.exports = customRouter(defaultRouter, myExtraRoutes); \ No newline at end of file diff --git a/services/strapi/src/api/patreon/services/patreon.js b/services/strapi/src/api/patreon/services/patreon.js deleted file mode 100644 index 147d2a4..0000000 --- a/services/strapi/src/api/patreon/services/patreon.js +++ /dev/null @@ -1,45 +0,0 @@ -'use strict'; - -/** - * patreon service - */ - -const EleventyFetch = require("@11ty/eleventy-fetch"); -const { createCoreService } = require('@strapi/strapi').factories; - - - -module.exports = createCoreService('api::patreon.patreon', ({strapi}) => ({ - - - async getPatreonCampaign() { - return EleventyFetch('https://www.patreon.com/api/campaigns/8012692', { - duration: "12h", - type: "json", - }) - }, - - - async getPatreonCampaignPledgeSum() { - const campaign = await this.getPatreonCampaign() - return campaign.data.attributes.pledge_sum - }, - - - /** - * Calculate how many mux allocations the site should have, based on the dollar amount of pledges from patreon - * - * @param {Number} pledgeSum - USD cents - */ - async getMuxAllocationCount() { - const patreonData = await strapi.entityService.findMany('api::patreon.patreon', { - fields: ['muxAllocationCostCents'] - }) - if (!patreonData) throw new Error('patreonData in Strapi was missing'); - const muxAllocationCostCents = patreonData.muxAllocationCostCents - const pledgeSum = await this.getPatreonCampaignPledgeSum() - const muxAllocationCount = Math.floor(pledgeSum / muxAllocationCostCents); // calculate the number of mux allocations required - return muxAllocationCount; - } -})); - diff --git a/services/strapi/src/api/platform-notification/content-types/platform-notification/schema.json b/services/strapi/src/api/platform-notification/content-types/platform-notification/schema.json deleted file mode 100644 index c5561e6..0000000 --- a/services/strapi/src/api/platform-notification/content-types/platform-notification/schema.json +++ /dev/null @@ -1,56 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "platform_notifications", - "info": { - "singularName": "platform-notification", - "pluralName": "platform-notifications", - "displayName": "Platform Notification", - "description": "" - }, - "options": { - "draftAndPublish": false - }, - "pluginOptions": {}, - "attributes": { - "source": { - "type": "enumeration", - "enum": [ - "email", - "manual" - ], - "default": "manual", - "required": true - }, - "platform": { - "type": "enumeration", - "enum": [ - "fansly", - "chaturbate" - ], - "default": "chaturbate", - "required": true - }, - "date": { - "type": "datetime", - "required": true, - "unique": false - }, - "date2": { - "type": "string", - "required": true, - "unique": false, - "regex": "\\d{4}-[01]\\d-[0-3]\\dT[0-2]\\d:[0-5]\\d:[0-5]\\d\\.\\d+([+-][0-2]\\d:[0-5]\\d|Z)" - }, - "vtuber": { - "type": "relation", - "relation": "oneToOne", - "target": "api::vtuber.vtuber" - }, - "stream": { - "type": "relation", - "relation": "manyToOne", - "target": "api::stream.stream", - "inversedBy": "platformNotifications" - } - } -} \ No newline at end of file diff --git a/services/strapi/src/api/platform-notification/controllers/platform-notification.js b/services/strapi/src/api/platform-notification/controllers/platform-notification.js deleted file mode 100644 index b845147..0000000 --- a/services/strapi/src/api/platform-notification/controllers/platform-notification.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * platform-notification controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::platform-notification.platform-notification'); diff --git a/services/strapi/src/api/platform-notification/routes/platform-notification.js b/services/strapi/src/api/platform-notification/routes/platform-notification.js deleted file mode 100644 index 1bc4837..0000000 --- a/services/strapi/src/api/platform-notification/routes/platform-notification.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * platform-notification router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -module.exports = createCoreRouter('api::platform-notification.platform-notification'); diff --git a/services/strapi/src/api/platform-notification/services/platform-notification.js b/services/strapi/src/api/platform-notification/services/platform-notification.js deleted file mode 100644 index a505fc9..0000000 --- a/services/strapi/src/api/platform-notification/services/platform-notification.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * platform-notification service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::platform-notification.platform-notification'); diff --git a/services/strapi/src/api/profile/controllers/profile.js b/services/strapi/src/api/profile/controllers/profile.js deleted file mode 100644 index 610a88d..0000000 --- a/services/strapi/src/api/profile/controllers/profile.js +++ /dev/null @@ -1,21 +0,0 @@ -'use strict'; - - - -module.exports = { - update: async (ctx, next) => { - const update = strapi.plugin('users-permissions').controllers.user.update - await update(ctx); - }, - me: async (ctx) => { - const userId = ctx?.state?.user?.id; - if (!userId) return ctx.badRequest("There was no user id in the request!"); - const user = await strapi.entityService.findOne('plugin::users-permissions.user', userId, { - populate: 'role' - }); - return user - }, - test: async (ctx) => { - return 'blah' - } -}; diff --git a/services/strapi/src/api/profile/routes/profile.js b/services/strapi/src/api/profile/routes/profile.js deleted file mode 100644 index 7a33f1c..0000000 --- a/services/strapi/src/api/profile/routes/profile.js +++ /dev/null @@ -1,23 +0,0 @@ -module.exports = { - routes: [ - { - method: 'PUT', - path: '/profile/:id', - handler: 'profile.update', - config: { - prefix: '', - policies: ['global::updateOwnerOnly'] - }, - }, - { - method: 'GET', - path: '/profile/me', - handler: 'profile.me' - }, - { - method: 'GET', - path: '/profile/test', - handler: 'profile.test' - } - ], -}; diff --git a/services/strapi/src/api/profile/services/profile.js b/services/strapi/src/api/profile/services/profile.js deleted file mode 100644 index 56ac091..0000000 --- a/services/strapi/src/api/profile/services/profile.js +++ /dev/null @@ -1,7 +0,0 @@ -'use strict'; - -/** - * profile service - */ - -module.exports = () => ({}); diff --git a/services/strapi/src/api/stream/content-types/stream/lifecycles.js b/services/strapi/src/api/stream/content-types/stream/lifecycles.js deleted file mode 100644 index 852633c..0000000 --- a/services/strapi/src/api/stream/content-types/stream/lifecycles.js +++ /dev/null @@ -1,119 +0,0 @@ -const { init } = require('@paralleldrive/cuid2'); - - - -module.exports = { - async beforeUpdate(event) { - const { data } = event.params; - if (!data.cuid) { - const length = 10; // 50% odds of collision after ~51,386,368 ids - const cuid = init({ length }); - event.params.data.cuid = cuid(); - } - /** - * Here we set the stream platform based on related platformNotifications. - * For example, if there is a related fansly platformNotification, we set isFanslyStream to true. - */ - console.log('hello my good sir, we are about to set the stream platform based on related platformNotifications.') - console.log('in order to make sure we have the data we need, let us console log the data.') - console.log(data) - - }, - async afterUpdate(event) { - - - - console.log(`>>>>>>>>>>>>>> STREAM is afterUpdate !!!!!!!!!!!!`); - const { data, where, select, populate } = event.params; - console.log(data); - const id = where.id; - - - /** - * This is where we populate the archiveStatus, based on the vods we have (or do not have.) - * We do this to display to the visitor the archival state of this stream. - * This state is what populates the any% archival speedrun on the `/vt/:slug` pages. - * - * Vods with a note are automatically considered, 'issue' - * A stream with no vods is considered, 'missing' - * At least 1 vod with no notes is considred, 'good' - * - * greets https://forum.strapi.io/t/how-to-get-previous-component-data-in-lifecycle-hook/25892/4?u=ggr247 - */ - const existingData = await strapi.entityService.findOne("api::stream.stream", id, { - populate: ['vods', 'tweet'] - }) - - // Initialize archiveStatus to a default value - let archiveStatus = 'missing'; - - // Iterate through all vods to determine archiveStatus - for (const vod of existingData.vods) { - if (!!vod.videoSrcHash) { - if (!!vod.note) { - // If a vod has both videoSrcHash and note, set archiveStatus to 'issue' - archiveStatus = 'issue'; - break; // No need to check further - } else { - // If a vod has videoSrcHash but no note, set archiveStatus to 'good' - archiveStatus = 'good'; - } - } - } - - // we can't use query engine here, because that would trigger an infinite loop - // where this - // instead we access knex instance - await strapi.db.connection("streams").where({ id: id }).update({ - archive_status: archiveStatus, - }); - - - - - /** - * This is where we populate platform, based on the related platformNotification content-types. - * We do this so the UI has the data it needs to display the platform on which the stream took place. - * - * If any platformNotification is from fansly, isFanslyStream is set to true. - * If any platformNotification is from chaturbate, isChaturbateStream is set to true. - */ - const existingData2 = await strapi.entityService.findOne("api::stream.stream", id, { - populate: {'platformNotifications': true} - }) - - let isFanslyStream = false - let isChaturbateStream = false - - console.log(`lets find the platformNotifications`) - console.log(JSON.stringify(existingData2, null, 2)) - - // Iterate through all platformNotifications to determine platform - if (existingData2?.platformNotifications) { - for (const pn of existingData2.platformNotifications) { - if (pn.platform === 'fansly') { - isFanslyStream = true - } else if (pn.platform === 'chaturbate') { - isChaturbateStream = true - } - } - } - - // we can't use query engine here, because that would trigger an infinite loop - // where this - // instead we access knex instance - await strapi.db.connection("streams").where({ id: id }).update({ - is_fansly_stream: isFanslyStream, - is_chaturbate_stream: isChaturbateStream - }); - - // Old way, @deprecated. keeping as a comment until I'm sure I don't need it - // if (!!existingData.tweet) { - // await strapi.db.connection("streams").where({ id: id }).update({ - // is_chaturbate_stream: existingData.tweet.isChaturbateInvite, - // is_fansly_stream: existingData.tweet.isFanslyInvite - // }); - // } - - } -}; \ No newline at end of file diff --git a/services/strapi/src/api/stream/content-types/stream/schema.json b/services/strapi/src/api/stream/content-types/stream/schema.json deleted file mode 100644 index 63fab1f..0000000 --- a/services/strapi/src/api/stream/content-types/stream/schema.json +++ /dev/null @@ -1,79 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "streams", - "info": { - "singularName": "stream", - "pluralName": "streams", - "displayName": "Stream", - "description": "" - }, - "options": { - "draftAndPublish": false - }, - "pluginOptions": {}, - "attributes": { - "platformNotifications": { - "type": "relation", - "relation": "oneToMany", - "target": "api::platform-notification.platform-notification", - "mappedBy": "stream" - }, - "date_str": { - "type": "string", - "required": true, - "unique": true, - "regex": "\\d{4}-[01]\\d-[0-3]\\dT[0-2]\\d:[0-5]\\d:[0-5]\\d\\.\\d+([+-][0-2]\\d:[0-5]\\d|Z)" - }, - "date2": { - "type": "string", - "required": true, - "unique": true, - "regex": "\\d{4}-[01]\\d-[0-3]\\dT[0-2]\\d:[0-5]\\d:[0-5]\\d\\.\\d+([+-][0-2]\\d:[0-5]\\d|Z)" - }, - "vods": { - "type": "relation", - "relation": "oneToMany", - "target": "api::vod.vod", - "mappedBy": "stream" - }, - "vtuber": { - "type": "relation", - "relation": "manyToOne", - "target": "api::vtuber.vtuber", - "inversedBy": "streams" - }, - "tweet": { - "type": "relation", - "relation": "oneToOne", - "target": "api::tweet.tweet" - }, - "date": { - "type": "datetime", - "required": true, - "unique": true - }, - "archiveStatus": { - "type": "enumeration", - "enum": [ - "missing", - "issue", - "good" - ], - "required": true, - "default": "missing" - }, - "cuid": { - "type": "string", - "unique": true, - "required": false - }, - "isChaturbateStream": { - "type": "boolean", - "default": false - }, - "isFanslyStream": { - "type": "boolean", - "default": false - } - } -} diff --git a/services/strapi/src/api/stream/controllers/stream.js b/services/strapi/src/api/stream/controllers/stream.js deleted file mode 100644 index aaee46e..0000000 --- a/services/strapi/src/api/stream/controllers/stream.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * stream controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::stream.stream'); diff --git a/services/strapi/src/api/stream/routes/stream.js b/services/strapi/src/api/stream/routes/stream.js deleted file mode 100644 index 6af6923..0000000 --- a/services/strapi/src/api/stream/routes/stream.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * stream router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -module.exports = createCoreRouter('api::stream.stream'); diff --git a/services/strapi/src/api/stream/services/stream.js b/services/strapi/src/api/stream/services/stream.js deleted file mode 100644 index b0311da..0000000 --- a/services/strapi/src/api/stream/services/stream.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * stream service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::stream.stream'); diff --git a/services/strapi/src/api/tag-vod-relation/content-types/tag-vod-relation/schema.json b/services/strapi/src/api/tag-vod-relation/content-types/tag-vod-relation/schema.json deleted file mode 100644 index 129ef5a..0000000 --- a/services/strapi/src/api/tag-vod-relation/content-types/tag-vod-relation/schema.json +++ /dev/null @@ -1,39 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "tag_vod_relations", - "info": { - "singularName": "tag-vod-relation", - "pluralName": "tag-vod-relations", - "displayName": "Tag Vod Relation", - "description": "" - }, - "options": { - "draftAndPublish": false - }, - "pluginOptions": {}, - "attributes": { - "votes": { - "type": "integer" - }, - "creator": { - "type": "relation", - "relation": "oneToOne", - "target": "plugin::users-permissions.user" - }, - "tag": { - "type": "relation", - "relation": "oneToOne", - "target": "api::tag.tag" - }, - "creatorId": { - "type": "integer", - "required": true - }, - "vod": { - "type": "relation", - "relation": "manyToOne", - "target": "api::vod.vod", - "inversedBy": "tagVodRelations" - } - } -} diff --git a/services/strapi/src/api/tag-vod-relation/controllers/tag-vod-relation.js b/services/strapi/src/api/tag-vod-relation/controllers/tag-vod-relation.js deleted file mode 100644 index 8face45..0000000 --- a/services/strapi/src/api/tag-vod-relation/controllers/tag-vod-relation.js +++ /dev/null @@ -1,222 +0,0 @@ -'use strict'; - -/** - * tag-vod-relation controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::tag-vod-relation.tag-vod-relation', ({ strapi }) => ({ - async relate(ctx) { - - const userId = ctx?.state?.user?.id; - if (!userId) return ctx.badRequest("There was no user id in the request!"); - if (!ctx.request.body.data) return ctx.badRequest('data was missing from body'); - if (!ctx.request.body.data.tag) return ctx.badRequest('tag was missing from data'); - if (!ctx.request.body.data.vod) return ctx.badRequest('vod was missing from data'); - - const { tag: tagId, vod: vodId } = ctx.request.body.data; - - const tagVodRelation = await strapi.entityService.create('api::tag-vod-relation.tag-vod-relation', { - data: { - vod: vodId, - tag: tagId, - creator: userId, - creatorId: userId, - publishedAt: new Date(), - votes: 2 - } - }) - - return tagVodRelation - }, - async vote(ctx) { - // @todo - }, - - // // greets https://docs.strapi.io/dev-docs/backend-customization/controllers#extending-core-controllers - // // greets https://docs.strapi.io/dev-docs/backend-customization/controllers#adding-a-new-controller - // // Method 2: Wrapping a core action (leaves core logic in place) - // async find(ctx) { - // // // some custom logic here - // // ctx.query = { ...ctx.query, local: 'en' } - - // const userId = ctx?.state?.user?.id; - // if (!userId) return ctx.badRequest("There was no user id in the request!"); - - - - // // Calling the default core action - // const { data, meta } = await super.find(ctx); - - // // add isCreator if the tvr was created by this user - // let dataWithCreator = data.map((d) => { - // if (d.data.attributes.) - // }) - - // // // some more custom logic - // // meta.date = Date.now() - - // return { data, meta }; - // }, - - // greets https://docs.strapi.io/dev-docs/backend-customization/controllers#extending-core-controllers - // greets https://docs.strapi.io/dev-docs/backend-customization/controllers#adding-a-new-controller - // Method 2: Wrapping a core action (leaves core logic in place) - async create(ctx) { - console.log('>> create a tag vod relation') - // only allow unique tag, vod combos - - const { query } = ctx.request; - - const userId = ctx?.state?.user?.id; - if (!userId) return ctx.badRequest("There was no user id in the request!"); - if (!ctx.request.body.data) return ctx.badRequest('data was missing from body'); - if (!ctx.request.body.data.tag) return ctx.badRequest('tag was missing from data'); - if (!ctx.request.body.data.vod) return ctx.badRequest('vod was missing from data'); - - const { tag: tagId, vod: vodId } = ctx.request.body.data; - - console.log(`lets make a combo entityService.findMany`) - const combo = await strapi.entityService.findMany('api::tag-vod-relation.tag-vod-relation', { - populate: ['tag', 'vod'], - filters: { - $and: [{ - tag: { - id: { - $eq: ctx.request.body.data.tag - } - } - }, { - vod: { - id: { - $eq: ctx.request.body.data.vod - } - } - }] - } - }) - - if (combo.length > 0) { - return ctx.badRequest('this vod already has that tag'); - } - - // @todo add votes and creator - ctx.request.body.data.creator = userId - ctx.request.body.data.votes = 2 - - const parseBody = (ctx) => { - if (ctx.is('multipart')) { - return parseMultipartData(ctx); - } - - const { data } = ctx.request.body || {}; - - return { data }; - }; - - - const sanitizedInputData = { - vod: vodId, - tag: tagId, - publishedAt: new Date(), - creator: userId, - creatorId: userId, - votes: 2 - } - - - - - - const entity = await strapi - .service('api::tag-vod-relation.tag-vod-relation') - .create({ - ...query, - data: sanitizedInputData, - populate: { vod: true, tag: true } - }); - - const sanitizedEntity = await this.sanitizeOutput(entity, ctx); - - return this.transformResponse({ ...sanitizedEntity }); - }, - - - async tagVod (ctx) { - - // create tag if needed - const userId = ctx?.state?.user?.id; - if (!userId) return ctx.badRequest("There was no user id in the request!"); - if (!ctx.request.body.data) return ctx.badRequest('data was missing from body'); - if (!ctx.request.body.data.tagName) return ctx.badRequest('tagName was missing from data'); - if (!ctx.request.body.data.vodId) return ctx.badRequest('vodId was missing from data'); - - const { tagName, vodId } = ctx.request.body.data; - - - const tag = await strapi.service('api::tag.tag').assertTag(tagName, userId); - - - try { - const tvr = await strapi.service('api::tag-vod-relation.tag-vod-relation').assertTvr(tag.id, vodId, userId); - - - const sanitizedEntity = await this.sanitizeOutput(tvr, ctx); - return this.transformResponse({ ...sanitizedEntity }); - } catch (e) { - console.error(e) - ctx.badRequest('Vod Tag could not be created.') - } - - }, - - async deleteMine (ctx) { - // // some custom logic here - // ctx.query = { ...ctx.query, local: 'en' } - - - - const userId = ctx?.state?.user?.id; - if (!userId) return ctx.badRequest("There was no user id in the request!"); - if (!ctx.request.params.id) return ctx.badRequest('id was missing from params'); - const { id } = ctx.request.params; - - // constraints - // only able to delete tagVodRelation if - // * creator - // * publishedAt isBefore(now-24h) - - - // get the tvr the user wants to delete - const tvrToDelete = await strapi.entityService.findOne('api::tag-vod-relation.tag-vod-relation', id, { - populate: { - tag: true, - vod: true, - creator: true, - } - }) - - if (!tvrToDelete) return ctx.badRequest('Tag to be deleted does not exist.'); - - if (tvrToDelete.creator.id !== userId) - ctx.forbidden('only the creator of the tag can delete it'); - - if ((new Date(tvrToDelete.createdAt).valueOf()+86400000) < new Date().valueOf()) - ctx.forbidden('cannot delete tags older than 24 hours') - - // Calling the default core action - const { data, meta } = await super.delete(ctx); - - // delete the related tag if it has no other vod - // @todo?? or maybe this is handled by lifecycle hook? - - // // some more custom logic - // meta.date = Date.now() - - return { data, meta }; - } - - -})); - diff --git a/services/strapi/src/api/tag-vod-relation/routes/tag-vod-relation.js b/services/strapi/src/api/tag-vod-relation/routes/tag-vod-relation.js deleted file mode 100644 index 2564f0f..0000000 --- a/services/strapi/src/api/tag-vod-relation/routes/tag-vod-relation.js +++ /dev/null @@ -1,51 +0,0 @@ -'use strict'; - -/** - * tag-vod-relation router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -const defaultRouter = createCoreRouter('api::tag-vod-relation.tag-vod-relation'); - -// greets https://forum.strapi.io/t/how-to-add-custom-routes-to-core-routes-in-strapi-4/14070/7 -const customRouter = (innerRouter, extraRoutes = []) => { - let routes; - return { - get prefix() { - return innerRouter.prefix; - }, - get routes() { - if (!routes) routes = extraRoutes.concat(innerRouter.routes) - return routes; - }, - }; -}; - -const myExtraRoutes = [ - { - method: "POST", - path: "/tag-vod-relations/relate", - handler: "api::tag-vod-relation.tag-vod-relation.relate" - }, - // { - // method: 'GET', - // path: '/tag-vod-relations', - // handler: 'api::tag-vod-relation.tag-vod-relation.find' - // }, - { - method: "PUT", - path: "/tag-vod-relations/vote", - handler: "api::tag-vod-relation.tag-vod-relation.vote" - }, { - method: 'POST', - path: '/tag-vod-relations/tag', - handler: 'api::tag-vod-relation.tag-vod-relation.tagVod' - }, { - method: 'DELETE', - path: '/tag-vod-relations/deleteMine/:id', - handler: 'api::tag-vod-relation.tag-vod-relation.deleteMine' - } -]; - -module.exports = customRouter(defaultRouter, myExtraRoutes); \ No newline at end of file diff --git a/services/strapi/src/api/tag-vod-relation/services/tag-vod-relation.js b/services/strapi/src/api/tag-vod-relation/services/tag-vod-relation.js deleted file mode 100644 index e958bbc..0000000 --- a/services/strapi/src/api/tag-vod-relation/services/tag-vod-relation.js +++ /dev/null @@ -1,69 +0,0 @@ -'use strict'; - -/** - * tag-vod-relation service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::tag-vod-relation.tag-vod-relation', ({ strapi }) => ({ - - - async assertTvr(tagId, vodId, userId) { - - if (!tagId) throw new Error('tagId was missing in request'); - if (!vodId) throw new Error('vodId was missing in request'); - if (!userId) throw new Error('userId was missing in request'); - - - - let existingTvr; - existingTvr = await strapi.entityService - .findMany('api::tag-vod-relation.tag-vod-relation', { - limit: 1, - filters: { - $and: [ - { - tag: { - id: tagId, - }, - }, - { - vod: { - id: vodId - } - } - ] - }, - populate: ['tag', 'vod'] - }) - - - if (existingTvr.length === 0) { - const newTvr = await strapi.entityService.create('api::tag-vod-relation.tag-vod-relation', { - data: { - tag: tagId, - vod: vodId, - creator: userId, - creatorId: userId, - }, - populate: { - tag: true, - vod: true - } - }) - - // trigger data revalidation in next.js server - // fetch(`${nextJsServerUrl}/`) - return newTvr; - } else { - - return existingTvr[0]; - } - - - }, - -})); - - diff --git a/services/strapi/src/api/tag/content-types/tag/schema.json b/services/strapi/src/api/tag/content-types/tag/schema.json deleted file mode 100644 index 2884619..0000000 --- a/services/strapi/src/api/tag/content-types/tag/schema.json +++ /dev/null @@ -1,38 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "tags", - "info": { - "singularName": "tag", - "pluralName": "tags", - "displayName": "Tag", - "description": "" - }, - "options": { - "draftAndPublish": true - }, - "pluginOptions": {}, - "attributes": { - "name": { - "type": "string", - "unique": true, - "required": true - }, - "toy": { - "type": "relation", - "relation": "manyToOne", - "target": "api::toy.toy", - "inversedBy": "tags" - }, - "vods": { - "type": "relation", - "relation": "manyToMany", - "target": "api::vod.vod", - "inversedBy": "tags" - }, - "creator": { - "type": "relation", - "relation": "oneToOne", - "target": "plugin::users-permissions.user" - } - } -} diff --git a/services/strapi/src/api/tag/controllers/tag.js b/services/strapi/src/api/tag/controllers/tag.js deleted file mode 100644 index 51669a9..0000000 --- a/services/strapi/src/api/tag/controllers/tag.js +++ /dev/null @@ -1,87 +0,0 @@ -'use strict'; - -/** - * tag controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; -const { sanitize } = require('@strapi/utils'); - - -module.exports = createCoreController('api::tag.tag', ({ strapi }) => ({ - - async random(ctx) { - const numberOfTags = 10; // Change this number to the desired number of random tags - const contentType = strapi.contentType('api::vod.vod'); - - // Fetch only the 'id' field of all tags - const tagIds = (await strapi.entityService.findMany( - "api::tag.tag", - { - fields: ['id'], - } - )).map(tag => tag.id); - - - const selectedTags = []; - - // Randomly select the specified number of tag IDs - for (let i = 0; i < numberOfTags; i++) { - const randomIndex = Math.floor(Math.random() * tagIds.length); - const randomTagId = tagIds[randomIndex]; - - - // Fetch the full details of the randomly selected tag using its ID - const rawTag = await strapi.entityService.findOne( - "api::tag.tag", - randomTagId, // Use the tag's ID - { - filter: { - publishedAt: { - $notNull: true, - }, - }, - fields: ['id', 'name'] - } - ); - - selectedTags.push(await sanitize.contentAPI.output(rawTag, contentType, { auth: ctx.state.auth })); - - // Remove the selected tag ID from the array to avoid duplicates - tagIds.splice(randomIndex, 1); - } - - ctx.body = selectedTags; - }, - - - - async createTagRelation(ctx) { - - // we have this controller which associates a tag with a vod - // this exists so users can indirectly update vod records which they dont have permissions to update - // first we need to get the user's request. - // they are telling us a vod ID and a tag ID - // our job is to get a reference to the vod, and add the tag relation. - - if (!ctx.request.body.data) return ctx.badRequest('data was missing from body'); - if (!ctx.request.body.data.tag) return ctx.badRequest('tag was missing from data'); - if (!ctx.request.body.data.vod) return ctx.badRequest('vod was missing from data'); - - const { tag, vod: vodId } = ctx.request.body.data; - - - - await strapi.entityService.update('api::vod.vod', vodId, { - data: { - tags: { - connect: [tag] - } - } - }) - - return 'OK' - - }, -})); - diff --git a/services/strapi/src/api/tag/routes/tag.js b/services/strapi/src/api/tag/routes/tag.js deleted file mode 100644 index db82f8f..0000000 --- a/services/strapi/src/api/tag/routes/tag.js +++ /dev/null @@ -1,37 +0,0 @@ -'use strict'; - -/** - * tag router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; -const defaultRouter = createCoreRouter('api::tag.tag'); - -// greets https://forum.strapi.io/t/how-to-add-custom-routes-to-core-routes-in-strapi-4/14070/7 -const customRouter = (innerRouter, extraRoutes = []) => { - let routes; - return { - get prefix() { - return innerRouter.prefix; - }, - get routes() { - if (!routes) routes = extraRoutes.concat(innerRouter.routes) - return routes; - }, - }; -}; - -const myExtraRoutes = [ - { - method: "POST", - path: "/tag/tagRelation", - handler: "api::tag.tag.createTagRelation" - }, - { - method: 'GET', - path: '/tag/random', - handler: 'api::tag.tag.random' - } -]; - -module.exports = customRouter(defaultRouter, myExtraRoutes) diff --git a/services/strapi/src/api/tag/services/tag.js b/services/strapi/src/api/tag/services/tag.js deleted file mode 100644 index 538bebc..0000000 --- a/services/strapi/src/api/tag/services/tag.js +++ /dev/null @@ -1,46 +0,0 @@ -'use strict'; - -/** - * tag service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::tag.tag', ({ strapi }) => ({ - async assertTag(tagName, userId) { - - if (!tagName) throw new Error('tagName was missing from request'); - if (!userId) throw new Error('userId was missing from request'); - - let tagEntry; - - // does the named tag already exist? - // tagEntry = await strapi.db.query('api::tag.tag') - // .findOne({ where: { name: tagName } }); - tagEntry = (await strapi.entityService.findMany('api::tag.tag', { - limit: 1, - filters: { - $and: [ - { - publishedAt: { $notNull: true }, - }, - { - name: tagName - } - ] - } - }))[0] - - if (!tagEntry) { - tagEntry = await strapi.entityService.create('api::tag.tag', { - data: { - name: tagName, - creator: userId, - publishedAt: new Date(), - } - }) - } - - return tagEntry; - }, -})); diff --git a/services/strapi/src/api/timestamp/content-types/timestamp/schema.json b/services/strapi/src/api/timestamp/content-types/timestamp/schema.json deleted file mode 100644 index 71b8ffb..0000000 --- a/services/strapi/src/api/timestamp/content-types/timestamp/schema.json +++ /dev/null @@ -1,47 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "timestamps", - "info": { - "singularName": "timestamp", - "pluralName": "timestamps", - "displayName": "Timestamp", - "description": "" - }, - "options": { - "draftAndPublish": false, - "populateCreatorFields": true - }, - "pluginOptions": {}, - "attributes": { - "time": { - "type": "integer", - "required": true - }, - "tag": { - "type": "relation", - "relation": "oneToOne", - "target": "api::tag.tag", - "required": true - }, - "vod": { - "type": "relation", - "relation": "manyToOne", - "target": "api::vod.vod", - "inversedBy": "timestamps" - }, - "creatorId": { - "type": "integer", - "required": true - }, - "upvoters": { - "type": "relation", - "relation": "oneToMany", - "target": "plugin::users-permissions.user" - }, - "downvoters": { - "type": "relation", - "relation": "oneToMany", - "target": "plugin::users-permissions.user" - } - } -} diff --git a/services/strapi/src/api/timestamp/controllers/timestamp.js b/services/strapi/src/api/timestamp/controllers/timestamp.js deleted file mode 100644 index a131f44..0000000 --- a/services/strapi/src/api/timestamp/controllers/timestamp.js +++ /dev/null @@ -1,166 +0,0 @@ -'use strict'; - -/** - * timestamp controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::timestamp.timestamp', ({ strapi }) => ({ - - - async find(ctx) { - const { data, meta } = await super.find(ctx); - - // Iterate over each timestamp in the data array - const timestampsWithVotes = await Promise.all(data.map(async (timestamp) => { - // Retrieve the upvoters count for the current timestamp - // Retrieve the downvoters count for the current timestamp - const entry = await strapi.db - .query('api::timestamp.timestamp') - .findOne({ - populate: ['upvoters', 'downvoters'], - where: { - id: timestamp.id - } - }); - - const upvotesCount = entry.upvoters.length - const downvotesCount = entry.downvoters.length - - // Create new properties "upvotes" and "downvotes" on the timestamp object - timestamp.attributes.upvotes = upvotesCount; - timestamp.attributes.downvotes = downvotesCount; - - return timestamp; - })); - - - return { data: timestampsWithVotes, meta }; - }, - - - async assert(ctx) { - const userId = ctx?.state?.user?.id; - if (!userId) return ctx.badRequest("There was no user id in the request!"); - if (!ctx.request.body.data) return ctx.badRequest('data was missing from body'); - if (!ctx.request.body.data.tagId) return ctx.badRequest('tagId was missing from data'); - if (ctx.request.body.data.time === undefined || ctx.request.body.data.time === null || ctx.request.body.data.time < 0) return ctx.badRequest('time was missing from data'); - if (!ctx.request.body.data.vodId) return ctx.badRequest('vodId was missing from data'); - const { time, tagId, vodId } = ctx.request.body.data; - const timestamp = await strapi.service('api::timestamp.timestamp').assertTimestamp(userId, tagId, vodId, time); - return timestamp; - }, - - - // greets https://docs.strapi.io/dev-docs/backend-customization/controllers#extending-core-controllers - // greets https://docs.strapi.io/dev-docs/backend-customization/controllers#adding-a-new-controller - // Method 2: Wrapping a core action (leaves core logic in place) - async create(ctx) { - // add creatorId to the record - const userId = ctx?.state?.user?.id; - if (!userId) return ctx.badRequest("There was no user id in the request!"); - if (!ctx.request.body.data) return ctx.badRequest('data was missing from body'); - if (!ctx.request.body.data.tag) return ctx.badRequest('tag was missing from data'); - const { time, tag } = ctx.request.body.data; - - ctx.request.body.data.creatorId = userId - - // does the timestamp already exist with same combination of time+tag? - const duplicate = await strapi.db.query('api::timestamp.timestamp') - .findOne({ where: { time, tag }}) - - if (!!duplicate) return ctx.badRequest('A duplicate timestamp already exists!'); - - // Calling the default core action - const res = await super.create(ctx); - - return res - }, - - - async vote(ctx) { - const userId = ctx?.state?.user?.id; - const { direction } = ctx.request.body.data; - if (!ctx.request.params.id) return ctx.badRequest('id was missing from params'); - const { id } = ctx.request.params; - // get the ts to be voted on - const ts = await strapi.entityService.findOne('api::timestamp.timestamp', id) - if (!ts) return ctx.badRequest('timestamp does not exist'); - const res = await strapi.entityService.update('api::timestamp.timestamp', id, { - data: { - upvoters: direction === 1 ? { connect: [userId] } : { disconnect: [userId] }, - downvoters: direction === 1 ? { disconnect: [userId] } : { connect: [userId] } - } - }); - return res; - }, - - - - async delete(ctx) { - const userId = ctx?.state?.user?.id; - const { id } = ctx.request.params; - - // get the ts to be deleted - const ts = await strapi.entityService.findOne('api::timestamp.timestamp', id) - if (!ts) return ctx.badRequest('Timestamp does not exist') - - - // Refuse to delete if not the tag creator - if (ts.creatorId !== userId) return ctx.forbidden('Only the timestamp creator can delete the timestamp.') - - - const res = await super.delete(ctx) - return res - - }, - - - async deleteMine (ctx) { - // // some custom logic here - // ctx.query = { ...ctx.query, local: 'en' } - - const userId = ctx?.state?.user?.id; - if (!userId) return ctx.badRequest("There was no user id in the request!"); - if (!ctx.request.params.id) return ctx.badRequest('id was missing from params'); - const { id } = ctx.request.params; - - // constraints - // only able to delete tagVodRelation if - // * creator - // * publishedAt isBefore(now-24h) - - - // get the tvr the user wants to delete - const timestampToDelete = await strapi.entityService.findOne('api::timestamp.timestamp', id, { - populate: { - tag: true, - vod: true - } - }) - - if (!timestampToDelete) return ctx.badRequest('Timestamp to be deleted does not exist.'); - - - if (timestampToDelete.creatorId !== userId) - ctx.forbidden('only the creator of the timestamp can delete it'); - - if ((new Date(timestampToDelete.createdAt).valueOf()+86400000) < new Date().valueOf()) - ctx.forbidden('cannot delete tags older than 24 hours') - - // Calling the default core action - const { data, meta } = await super.delete(ctx); - - // delete the related tag if it has no other vod - // @todo?? or maybe this is handled by lifecycle hook? - - // // some more custom logic - // meta.date = Date.now() - - return { data, meta }; - } - - -})) - diff --git a/services/strapi/src/api/timestamp/routes/timestamp.js b/services/strapi/src/api/timestamp/routes/timestamp.js deleted file mode 100644 index 1c8e7c6..0000000 --- a/services/strapi/src/api/timestamp/routes/timestamp.js +++ /dev/null @@ -1,53 +0,0 @@ -'use strict'; - -/** - * timestamp router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - - -const defaultRouter = createCoreRouter('api::timestamp.timestamp'); - -// greets https://forum.strapi.io/t/how-to-add-custom-routes-to-core-routes-in-strapi-4/14070/7 -const customRouter = (innerRouter, extraRoutes = []) => { - let routes; - return { - get prefix() { - return innerRouter.prefix; - }, - get routes() { - if (!routes) routes = extraRoutes.concat(innerRouter.routes) - return routes; - }, - }; -}; - - - -const myExtraRoutes = [ - { - method: 'POST', - path: '/timestamps/assert', - handler: 'api::timestamp.timestamp.assert' - }, - { - method: "PUT", - path: "/timestamps/:id/vote", - handler: "api::timestamp.timestamp.vote" - }, - { - method: 'DELETE', - path: '/timestamps/:id', - handler: 'api::timestamp.timestamp.delete' - }, - { - method: 'DELETE', - path: '/timestamps/deleteMine/:id', - handler: 'api::timestamp.timestamp.deleteMine' - } -]; - -module.exports = customRouter(defaultRouter, myExtraRoutes); - - diff --git a/services/strapi/src/api/timestamp/services/timestamp.js b/services/strapi/src/api/timestamp/services/timestamp.js deleted file mode 100644 index bfa84c2..0000000 --- a/services/strapi/src/api/timestamp/services/timestamp.js +++ /dev/null @@ -1,44 +0,0 @@ -'use strict'; - -/** - * timestamp service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::timestamp.timestamp', ({ strapi }) => ({ - async assertTimestamp(userId, tagId, vodId, time) { - const existingTimestamp = await strapi.entityService.findMany('api::timestamp.timestamp', { - populate: ['vod', 'tag'], - filters: { - $and: [ - { - tag: { - id: tagId - } - }, - { - vod: { - id: vodId - } - }, - { - time: parseInt(time) - } - ] - }, - limit: 1 - }) - if (existingTimestamp.length > 0) return existingTimestamp[0]; - const newTimestamp = await strapi.entityService.create('api::timestamp.timestamp', { - data: { - tag: tagId, - vod: vodId, - creatorId: userId, - time: time, - } - }); - - return newTimestamp; - } -})); diff --git a/services/strapi/src/api/toy/content-types/toy/schema.json b/services/strapi/src/api/toy/content-types/toy/schema.json deleted file mode 100644 index 4e2baba..0000000 --- a/services/strapi/src/api/toy/content-types/toy/schema.json +++ /dev/null @@ -1,51 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "toys", - "info": { - "singularName": "toy", - "pluralName": "toys", - "displayName": "Toy", - "description": "" - }, - "options": { - "draftAndPublish": false - }, - "pluginOptions": {}, - "attributes": { - "tags": { - "type": "relation", - "relation": "oneToMany", - "target": "api::tag.tag", - "mappedBy": "toy" - }, - "make": { - "type": "string", - "required": true - }, - "model": { - "type": "string", - "required": true - }, - "aspectRatio": { - "type": "string", - "default": "2:1", - "required": true - }, - "image2": { - "type": "string", - "default": "https://futureporn-b2.b-cdn.net/default-thumbnail.webp", - "required": true - }, - "linkTag": { - "type": "relation", - "relation": "oneToOne", - "target": "api::tag.tag" - }, - "vtubers": { - "type": "relation", - "relation": "oneToMany", - "target": "api::vtuber.vtuber", - "mappedBy": "toy" - } - } -} diff --git a/services/strapi/src/api/toy/controllers/toy.js b/services/strapi/src/api/toy/controllers/toy.js deleted file mode 100644 index 32bae69..0000000 --- a/services/strapi/src/api/toy/controllers/toy.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * toy controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::toy.toy'); diff --git a/services/strapi/src/api/toy/routes/toy.js b/services/strapi/src/api/toy/routes/toy.js deleted file mode 100644 index 21d9d6a..0000000 --- a/services/strapi/src/api/toy/routes/toy.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * toy router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -module.exports = createCoreRouter('api::toy.toy'); diff --git a/services/strapi/src/api/toy/services/toy.js b/services/strapi/src/api/toy/services/toy.js deleted file mode 100644 index cd2dd6d..0000000 --- a/services/strapi/src/api/toy/services/toy.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * toy service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::toy.toy'); diff --git a/services/strapi/src/api/tweet/content-types/tweet/lifecycles.js b/services/strapi/src/api/tweet/content-types/tweet/lifecycles.js deleted file mode 100644 index e8cc61f..0000000 --- a/services/strapi/src/api/tweet/content-types/tweet/lifecycles.js +++ /dev/null @@ -1,136 +0,0 @@ -const generateCuid = require('../../../../../misc/generateCuid.js'); - -const cbUrlRegex = /chaturbate\.com/i; -const fanslyUrlRegex = /https?:\/\/(?:www\.)?fans(?:\.ly|ly\.com)\/r\/[a-zA-Z0-9_]+/; - -const cbAlternativeUrls = [ - 'shorturl.at/tNUVY' // used by ProjektMelody in the early days -] - - -/** - * Returns true if the tweet contains a chaturbate.com link - * - * @param {Object} tweet - * @returns {Boolean} - */ -const containsCBInviteLink = (tweet) => { - const containsCbUrl = (link) => { - if (!link?.url) return false; - const isCbUrl = cbUrlRegex.test(link.url); - const isAlternativeCbUrl = cbAlternativeUrls.some(alternativeUrl => link.url.includes(alternativeUrl)); - return isCbUrl || isAlternativeCbUrl; - } - try { - if (!tweet?.links) return false; - return tweet.links.some(containsCbUrl) - } catch (e) { - logger.log({ level: 'error', message: 'ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR' }); - logger.log({ level: 'error', message: e }); - return false; - } -}; - -const containsFanslyInviteLink = (tweet) => { - const containsFanslyUrl = (link) => { - if (!link?.url) return false; - return (fanslyUrlRegex.test(link?.url)) - } - try { - if (!tweet?.links) return false; - return tweet.links.some(containsFanslyUrl) - } catch (e) { - logger.log({ level: 'error', message: 'ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR ERROR' }); - logger.log({ level: 'error', message: e }); - return false; - } -}; - - -const deriveTitle = (text) => { - // @see https://www.urlregex.com/ - const urlRegex = /((([A-Za-z]{3,9}:(?:\/\/)?)(?:[\-;:&=\+\$,\w]+@)?[A-Za-z0-9\.\-]+|(?:www\.|[\-;:&=\+\$,\w]+@)[A-Za-z0-9\.\-]+)((?:\/[\+~%\/\.\w\-_]*)?\??(?:[\-\+=&;%@\.\w_]*)#?(?:[\.\!\/\\\w]*))?)/g; - let title = text - .replace(urlRegex, '') // remove urls - .replace(/\n/g, ' ') // replace newlines with spaces - .replace(/>/g, '>') // gimme dem greater-than brackets - .replace(/</g, '<') // i want them less-thans too - .replace(/&/g, '&') // ampersands are sexy - .replace(/\s+$/, ''); // remove trailing whitespace - return title; -}; - - - - -module.exports = { - async afterCreate(event) { - // * [ ] Create Stream - const id = event.result.id; - console.log(`>>> tweet afterCreate id=${id}`); - const { data } = event.params; - - console.log(data); - - // IF this tweet was a fansly or chaturbate invite, create & associate Stream - if (data.isChaturbateInvite || data.isFanslyInvite) { - const stream = await strapi.entityService.create('api::stream.stream', { - data: { - tweet: id, - vtuber: data.vtuber, - date: data.date, - date_str: data.date, - date2: data.date, - archiveStatus: 'missing', - cuid: generateCuid() - } - }); - - // console.log(data) - console.log(`stream.id=${stream.id}`); - - // const existingData = await strapi.entityService.findOne("api::stream.stream", id, { - // populate: ['vods'] - // }) - } - }, - async beforeCreate(event) { - // * [x] Set platform to CB or Fansly - // * [x] Set vtuber - // * [x] Set date - // * [x] Set id_str - // * [x] Set url - - const { data, where, select, populate } = event.params; - console.log('>>> tweet beforeCreate!'); - - const tweet = JSON.parse(data.json); - // console.log(tweet); - console.log(`containsCBInviteLink=${containsCBInviteLink(tweet)}, containsFanslyInviteLink=${containsFanslyInviteLink(tweet)}`); - - - data.isChaturbateInvite = containsCBInviteLink(tweet); - data.isFanslyInvite = containsFanslyInviteLink(tweet); - - const tweetDate = new Date(tweet.date).toISOString(); - data.id_str = tweet.id_str; - data.date = tweetDate; - data.date2 = tweetDate; - data.url = tweet.url; - - // Set VTuber - const twitterUsername = tweet.user.username; - const vtuberRecords = await strapi.entityService.findMany("api::vtuber.vtuber", { - fields: ['displayName', 'slug', 'id'], - filters: { - twitter: { - $endsWithi: twitterUsername - } - } - }); - if (!!vtuberRecords[0]) data.vtuber = vtuberRecords[0].id; - - - - } -} diff --git a/services/strapi/src/api/tweet/content-types/tweet/schema.json b/services/strapi/src/api/tweet/content-types/tweet/schema.json deleted file mode 100644 index 3d94b7c..0000000 --- a/services/strapi/src/api/tweet/content-types/tweet/schema.json +++ /dev/null @@ -1,49 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "tweets", - "info": { - "singularName": "tweet", - "pluralName": "tweets", - "displayName": "Tweet", - "description": "" - }, - "options": { - "draftAndPublish": false - }, - "pluginOptions": {}, - "attributes": { - "id_str": { - "type": "string", - "required": false, - "unique": true - }, - "url": { - "type": "string", - "required": false - }, - "date2": { - "type": "string", - "required": false, - "unique": true - }, - "json": { - "type": "text", - "required": true - }, - "vtuber": { - "type": "relation", - "relation": "oneToOne", - "target": "api::vtuber.vtuber" - }, - "isChaturbateInvite": { - "type": "boolean" - }, - "isFanslyInvite": { - "type": "boolean" - }, - "date": { - "type": "datetime", - "unique": true - } - } -} diff --git a/services/strapi/src/api/tweet/controllers/tweet.js b/services/strapi/src/api/tweet/controllers/tweet.js deleted file mode 100644 index bbb1fed..0000000 --- a/services/strapi/src/api/tweet/controllers/tweet.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * tweet controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::tweet.tweet'); diff --git a/services/strapi/src/api/tweet/routes/tweet.js b/services/strapi/src/api/tweet/routes/tweet.js deleted file mode 100644 index efce886..0000000 --- a/services/strapi/src/api/tweet/routes/tweet.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * tweet router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -module.exports = createCoreRouter('api::tweet.tweet'); diff --git a/services/strapi/src/api/tweet/services/tweet.js b/services/strapi/src/api/tweet/services/tweet.js deleted file mode 100644 index 9be1baf..0000000 --- a/services/strapi/src/api/tweet/services/tweet.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * tweet service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::tweet.tweet'); diff --git a/services/strapi/src/api/user-submitted-content/content-types/user-submitted-content/lifecycles.js b/services/strapi/src/api/user-submitted-content/content-types/user-submitted-content/lifecycles.js deleted file mode 100644 index 0b4ed24..0000000 --- a/services/strapi/src/api/user-submitted-content/content-types/user-submitted-content/lifecycles.js +++ /dev/null @@ -1,57 +0,0 @@ -const { S3Client, DeleteObjectCommand } = require("@aws-sdk/client-s3"); -require('dotenv').config() - -if (!process.env.S3_USC_BUCKET_NAME) throw new Error('S3_USC_BUCKET_NAME must be defined in env'); -if (!process.env.S3_USC_BUCKET_ENDPOINT) throw new Error('S3_USC_BUCKET_ENDPOINT must be defined in env'); -if (!process.env.S3_USC_BUCKET_REGION) throw new Error('S3_USC_BUCKET_REGION must be defined in env'); - -// AWS.config.loadFromPath('./credentials-ehl.json'); - - - -module.exports = { - - - // when strapi deletes a USC, we delete the related files in the S3 bucket. - async afterDelete(event) { - - console.log('>>> afterDelete'); - console.log(event); - - const { result } = event; - - - - // a client can be shared by different commands. - const client = new S3Client({ - endpoint: process.env.S3_USC_BUCKET_ENDPOINT, - region: process.env.S3_USC_BUCKET_REGION - }); - // https://fp-usc-dev.s3.us-west-000.backblazeb2.com/GEB7_QcaUAAQ29O.jpg - - for (const file of result.files) { - console.log(`deleting ${file.key}`); - const res = await client.send(new DeleteObjectCommand({ - Bucket: process.env.S3_USC_BUCKET_NAME, - Key: file.key - })); - - console.log(res); - } - - - - // var s3 = new S3(); - // var params = { Bucket: process.env.S3_USC_BUCKET, Key: 'your object' }; - - // const res = await s3.deleteObject(params).promise(); - - // console.log('deletion complete.'); - // console.log(res); - - // , function(err, data) { - // if (err) console.log(err, err.stack); // error - // else console.log(); // deleted - // }); - } -} \ No newline at end of file diff --git a/services/strapi/src/api/user-submitted-content/content-types/user-submitted-content/schema.json b/services/strapi/src/api/user-submitted-content/content-types/user-submitted-content/schema.json deleted file mode 100644 index e0e5896..0000000 --- a/services/strapi/src/api/user-submitted-content/content-types/user-submitted-content/schema.json +++ /dev/null @@ -1,57 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "user_submitted_contents", - "info": { - "singularName": "user-submitted-content", - "pluralName": "user-submitted-contents", - "displayName": "User Submitted Content", - "description": "" - }, - "options": { - "draftAndPublish": false - }, - "pluginOptions": {}, - "attributes": { - "uploader": { - "type": "relation", - "relation": "oneToOne", - "target": "plugin::users-permissions.user" - }, - "attribution": { - "type": "boolean", - "default": false - }, - "date": { - "type": "string", - "required": true - }, - "notes": { - "type": "richtext" - }, - "files": { - "type": "json", - "required": true - }, - "approved": { - "type": "boolean", - "default": false - }, - "vtuber": { - "type": "relation", - "relation": "oneToOne", - "target": "api::vtuber.vtuber" - }, - "uploaderId": { - "type": "integer" - }, - "streamCuid": { - "type": "string" - }, - "discordInteraction": { - "type": "relation", - "relation": "oneToOne", - "target": "api::discord-interaction.discord-interaction", - "mappedBy": "userSubmittedContent" - } - } -} diff --git a/services/strapi/src/api/user-submitted-content/controllers/user-submitted-content.js b/services/strapi/src/api/user-submitted-content/controllers/user-submitted-content.js deleted file mode 100644 index cda33f8..0000000 --- a/services/strapi/src/api/user-submitted-content/controllers/user-submitted-content.js +++ /dev/null @@ -1,64 +0,0 @@ - -'use strict'; - -require('dotenv').config() - -if (!process.env.CDN_BUCKET_USC_URL) throw new Error('CDN_BUCKET_USC_URL environment variable is required!'); - -/** - * user-submitted-content controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -// greets https://docs.strapi.io/dev-docs/backend-customization/controllers#adding-a-new-controller -module.exports = createCoreController('api::user-submitted-content.user-submitted-content', ({ strapi }) => ({ - - async createFromUppy(ctx) { - try { - // Destructure data from the request body - const { data } = ctx.request.body; - - console.log(data); - - // Check for required fields in the data - const requiredFields = ['files', 'vtuber', 'date', 'notes', 'attribution']; - if (!data) { - return ctx.badRequest('ctx.request.body.data was missing.'); - } - for (const field of requiredFields) { - console.log(`checking field=${field} data[field]=${data[field]}`); - if (data[field] === undefined || data[field] === null) { - return ctx.badRequest(`${field} was missing from request data.`); - } - } - - // Extract relevant data - const { files, vtuber, streamCuid, date, notes, attribution } = data; - const uploader = ctx.state.user.id; - const uploaderId = ctx.state.user.id; - - console.log('Creating user-submitted content'); - const usc = await strapi.entityService.create('api::user-submitted-content.user-submitted-content', { - data: { - uploader, - uploaderId, - files: files.map((f) => ({ ...f, cdnUrl: `${process.env.CDN_BUCKET_USC_URL}/${f.key}` })), - vtuber, - streamCuid, - date, - notes, - attribution, - } - }); - - return usc; - } catch (error) { - // Handle unexpected errors - console.error(error); - return ctx.badRequest('An error occurred while processing the request'); - } - } - - })); - \ No newline at end of file diff --git a/services/strapi/src/api/user-submitted-content/routes/user-submitted-content.js b/services/strapi/src/api/user-submitted-content/routes/user-submitted-content.js deleted file mode 100644 index ed91c49..0000000 --- a/services/strapi/src/api/user-submitted-content/routes/user-submitted-content.js +++ /dev/null @@ -1,33 +0,0 @@ -'use strict'; - -/** - * user-submitted-content router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -const defaultRouter = createCoreRouter('api::user-submitted-content.user-submitted-content'); - -// greets https://forum.strapi.io/t/how-to-add-custom-routes-to-core-routes-in-strapi-4/14070/7 -const customRouter = (innerRouter, extraRoutes = []) => { - let routes; - return { - get prefix() { - return innerRouter.prefix; - }, - get routes() { - if (!routes) routes = extraRoutes.concat(innerRouter.routes) - return routes; - }, - }; -}; - -const myExtraRoutes = [ - { - method: "POST", - path: "/user-submitted-contents/createFromUppy", - handler: "api::user-submitted-content.user-submitted-content.createFromUppy" - } -]; - -module.exports = customRouter(defaultRouter, myExtraRoutes); \ No newline at end of file diff --git a/services/strapi/src/api/user-submitted-content/services/user-submitted-content.js b/services/strapi/src/api/user-submitted-content/services/user-submitted-content.js deleted file mode 100644 index 9c176aa..0000000 --- a/services/strapi/src/api/user-submitted-content/services/user-submitted-content.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * user-submitted-content service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::user-submitted-content.user-submitted-content'); diff --git a/services/strapi/src/api/vod/content-types/vod/lifecycles.js b/services/strapi/src/api/vod/content-types/vod/lifecycles.js deleted file mode 100644 index 879eeee..0000000 --- a/services/strapi/src/api/vod/content-types/vod/lifecycles.js +++ /dev/null @@ -1,27 +0,0 @@ -const { init } = require('@paralleldrive/cuid2'); - -function generateCuid(event) { - const { data } = event.params; - if (!data.cuid) { - const length = 10; // 50% odds of collision after ~51,386,368 ids - const cuid = init({ length }); - event.params.data.cuid = cuid(); - } -} - -module.exports = { - async beforeUpdate(event) { - console.log(`>> beforeUpdate! We are generating a CUID`); - generateCuid(event); - }, - async beforeCreate(event) { - console.log(`>> beforeCreate! We are generating a CUID`); - generateCuid(event); - }, - async afterCreate(event) { - console.log(`>> afterCreate! We are generating a CUID`); - generateCuid(event); - } -} - - diff --git a/services/strapi/src/api/vod/content-types/vod/schema.json b/services/strapi/src/api/vod/content-types/vod/schema.json deleted file mode 100644 index 2446a76..0000000 --- a/services/strapi/src/api/vod/content-types/vod/schema.json +++ /dev/null @@ -1,142 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "vods", - "info": { - "singularName": "vod", - "pluralName": "vods", - "displayName": "VOD", - "description": "" - }, - "options": { - "draftAndPublish": true - }, - "pluginOptions": {}, - "attributes": { - "videoSrcHash": { - "type": "string", - "regex": "Qm[1-9A-HJ-NP-Za-km-z]{44,}|b[A-Za-z2-7]{58,}|B[A-Z2-7]{58,}|z[1-9A-HJ-NP-Za-km-z]{48,}|F[0-9A-F]{50,}", - "required": false, - "unique": true - }, - "video720Hash": { - "type": "string", - "unique": true, - "regex": "Qm[1-9A-HJ-NP-Za-km-z]{44,}|b[A-Za-z2-7]{58,}|B[A-Z2-7]{58,}|z[1-9A-HJ-NP-Za-km-z]{48,}|F[0-9A-F]{50,}" - }, - "video480Hash": { - "type": "string", - "regex": "Qm[1-9A-HJ-NP-Za-km-z]{44,}|b[A-Za-z2-7]{58,}|B[A-Z2-7]{58,}|z[1-9A-HJ-NP-Za-km-z]{48,}|F[0-9A-F]{50,}", - "unique": true - }, - "video360Hash": { - "type": "string", - "regex": "Qm[1-9A-HJ-NP-Za-km-z]{44,}|b[A-Za-z2-7]{58,}|B[A-Z2-7]{58,}|z[1-9A-HJ-NP-Za-km-z]{48,}|F[0-9A-F]{50,}", - "unique": true - }, - "video240Hash": { - "type": "string", - "regex": "Qm[1-9A-HJ-NP-Za-km-z]{44,}|b[A-Za-z2-7]{58,}|B[A-Z2-7]{58,}|z[1-9A-HJ-NP-Za-km-z]{48,}|F[0-9A-F]{50,}", - "unique": true - }, - "thinHash": { - "type": "string", - "regex": "Qm[1-9A-HJ-NP-Za-km-z]{44,}|b[A-Za-z2-7]{58,}|B[A-Z2-7]{58,}|z[1-9A-HJ-NP-Za-km-z]{48,}|F[0-9A-F]{50,}", - "unique": true - }, - "thiccHash": { - "type": "string", - "regex": "Qm[1-9A-HJ-NP-Za-km-z]{44,}|b[A-Za-z2-7]{58,}|B[A-Z2-7]{58,}|z[1-9A-HJ-NP-Za-km-z]{48,}|F[0-9A-F]{50,}", - "unique": true - }, - "announceTitle": { - "type": "string" - }, - "announceUrl": { - "type": "string", - "unique": false - }, - "note": { - "type": "text" - }, - "date": { - "type": "datetime" - }, - "date2": { - "type": "string", - "required": true - }, - "spoilers": { - "type": "richtext" - }, - "title": { - "type": "string" - }, - "uploader": { - "type": "relation", - "relation": "oneToOne", - "target": "plugin::users-permissions.user" - }, - "muxAsset": { - "type": "relation", - "relation": "oneToOne", - "target": "api::mux-asset.mux-asset" - }, - "videoSrcB2": { - "type": "relation", - "relation": "oneToOne", - "target": "api::b2-file.b2-file" - }, - "thumbnail": { - "type": "relation", - "relation": "oneToOne", - "target": "api::b2-file.b2-file" - }, - "chatLog": { - "type": "richtext" - }, - "tags": { - "type": "relation", - "relation": "manyToMany", - "target": "api::tag.tag", - "mappedBy": "vods" - }, - "timestamps": { - "type": "relation", - "relation": "oneToMany", - "target": "api::timestamp.timestamp", - "mappedBy": "vod" - }, - "tagVodRelations": { - "type": "relation", - "relation": "oneToMany", - "target": "api::tag-vod-relation.tag-vod-relation", - "mappedBy": "vod" - }, - "vtuber": { - "type": "relation", - "relation": "manyToOne", - "target": "api::vtuber.vtuber", - "inversedBy": "vods" - }, - "stream": { - "type": "relation", - "relation": "manyToOne", - "target": "api::stream.stream", - "inversedBy": "vods" - }, - "archiveStatus": { - "type": "enumeration", - "enum": [ - "missing", - "issue", - "good" - ], - "required": false, - "default": "issue" - }, - "cuid": { - "type": "string", - "unique": true - } - } -} diff --git a/services/strapi/src/api/vod/controllers/vod.js b/services/strapi/src/api/vod/controllers/vod.js deleted file mode 100644 index 9dc9ff2..0000000 --- a/services/strapi/src/api/vod/controllers/vod.js +++ /dev/null @@ -1,85 +0,0 @@ -'use strict'; - - -const { sanitize } = require('@strapi/utils'); - -if (!process.env.CDN_BUCKET_USC_URL) throw new Error('CDN_BUCKET_USC_URL environment variable is required!'); - -/** - * vod controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -// greets https://docs.strapi.io/dev-docs/backend-customization/controllers#adding-a-new-controller -module.exports = createCoreController('api::vod.vod', ({ strapi }) => ({ - - async createFromUppy(ctx) { - - const uploaderId = ctx.state.user.id; - - if (!ctx.request.body.data) return ctx.badRequest("data was missing in request body"); - if (!ctx.request.body.data.date) return ctx.badRequest("date was missing"); - if (!ctx.request.body.data.b2Key) return ctx.badRequest("b2Key was missing"); - if (!ctx.request.body.data.b2UploadId) return ctx.badRequest("b2UploadId was missing"); - - - const videoSrcB2 = await strapi.entityService.create('api::b2-file.b2-file', { - data: { - url: `https://f000.backblazeb2.com/b2api/v1/b2_download_file_by_id?fileId=${ctx.request.body.data.b2UploadId}`, - key: ctx.request.body.data.b2Key, - uploadId: ctx.request.body.data.b2UploadId, - cdnUrl: `${process.env.CDN_BUCKET_USC_URL}/${ctx.request.body.data.b2Key}` - } - }); - - const vod = await strapi.entityService.create('api::vod.vod', { - data: { - notes: ctx.request.body.data.notes, - date: ctx.request.body.data.date, - videoSrcB2: videoSrcB2.id, - publishedAt: null, - uploader: uploaderId, - } - }); - - return vod; - }, - - // greets https://stackoverflow.com/a/73929966/1004931 - async random(ctx) { - const numberOfEntries = 1; - const contentType = strapi.contentType('api::vod.vod') - - // Fetch only the 'id' field of all VODs - const entries = await strapi.entityService.findMany( - "api::vod.vod", - { - fields: ['id'], - filters: { - publishedAt: { - $notNull: true, - }, - } - } - ); - - // Randomly select one entry - const randomEntry = entries[Math.floor(Math.random() * entries.length)]; - - // Fetch the full details of the randomly selected VOD - const rawVod = await strapi.entityService.findOne( - "api::vod.vod", - randomEntry.id, - { - populate: '*', - }, - ); - - const sanitizedOutput = await sanitize.contentAPI.output(rawVod, contentType, { auth: ctx.state.auth }); - - ctx.body = sanitizedOutput; - } - - }) -) \ No newline at end of file diff --git a/services/strapi/src/api/vod/routes/vod.js b/services/strapi/src/api/vod/routes/vod.js deleted file mode 100644 index 132dacf..0000000 --- a/services/strapi/src/api/vod/routes/vod.js +++ /dev/null @@ -1,38 +0,0 @@ -'use strict'; - -/** - * vod router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -const defaultRouter = createCoreRouter('api::vod.vod'); - -// greets https://forum.strapi.io/t/how-to-add-custom-routes-to-core-routes-in-strapi-4/14070/7 -const customRouter = (innerRouter, extraRoutes = []) => { - let routes; - return { - get prefix() { - return innerRouter.prefix; - }, - get routes() { - if (!routes) routes = extraRoutes.concat(innerRouter.routes) - return routes; - }, - }; -}; - -const myExtraRoutes = [ - { - method: "POST", - path: "/vods/createFromUppy", - handler: "api::vod.vod.createFromUppy" - }, - { - method: "GET", - path: "/vods/random", - handler: "api::vod.vod.random" - } -]; - -module.exports = customRouter(defaultRouter, myExtraRoutes); \ No newline at end of file diff --git a/services/strapi/src/api/vod/services/vod.js b/services/strapi/src/api/vod/services/vod.js deleted file mode 100644 index 56fcec1..0000000 --- a/services/strapi/src/api/vod/services/vod.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * vod service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::vod.vod'); diff --git a/services/strapi/src/api/vtuber/content-types/vtuber/lifecycles.js b/services/strapi/src/api/vtuber/content-types/vtuber/lifecycles.js deleted file mode 100644 index 2be365a..0000000 --- a/services/strapi/src/api/vtuber/content-types/vtuber/lifecycles.js +++ /dev/null @@ -1,23 +0,0 @@ -const { createCanvas } = require('canvas'); - -function hexColorToBase64Image(hexColor) { - const canvas = createCanvas(1, 1); // Create a canvas - const ctx = canvas.getContext('2d'); - // Draw a rectangle filled with the hex color - ctx.fillStyle = hexColor; - ctx.fillRect(0, 0, canvas.width, canvas.height); - // Convert canvas content to base64 encoded image - const base64Image = canvas.toDataURL('image/png'); - return base64Image; -} - - - -module.exports = { - beforeUpdate(event) { - const { data, where, select, populate } = event.params; - const themeColor = event.params.data.themeColor; - const imageBlur = hexColorToBase64Image(themeColor); - event.params.data.imageBlur = imageBlur; - } -}; \ No newline at end of file diff --git a/services/strapi/src/api/vtuber/content-types/vtuber/schema.json b/services/strapi/src/api/vtuber/content-types/vtuber/schema.json deleted file mode 100644 index 4c64af0..0000000 --- a/services/strapi/src/api/vtuber/content-types/vtuber/schema.json +++ /dev/null @@ -1,127 +0,0 @@ -{ - "kind": "collectionType", - "collectionName": "vtubers", - "info": { - "singularName": "vtuber", - "pluralName": "vtubers", - "displayName": "Vtuber", - "description": "" - }, - "options": { - "draftAndPublish": true - }, - "pluginOptions": {}, - "attributes": { - "displayName": { - "type": "string", - "required": true - }, - "chaturbate": { - "type": "string" - }, - "twitter": { - "type": "string" - }, - "patreon": { - "type": "string" - }, - "twitch": { - "type": "string" - }, - "tiktok": { - "type": "string" - }, - "onlyfans": { - "type": "string" - }, - "youtube": { - "type": "string" - }, - "linktree": { - "type": "string" - }, - "carrd": { - "type": "string" - }, - "fansly": { - "type": "string" - }, - "pornhub": { - "type": "string" - }, - "discord": { - "type": "string" - }, - "reddit": { - "type": "string" - }, - "throne": { - "type": "string" - }, - "instagram": { - "type": "string" - }, - "facebook": { - "type": "string" - }, - "merch": { - "type": "string" - }, - "slug": { - "type": "string", - "required": true - }, - "vods": { - "type": "relation", - "relation": "oneToMany", - "target": "api::vod.vod", - "mappedBy": "vtuber" - }, - "description1": { - "type": "text", - "required": false - }, - "description2": { - "type": "text" - }, - "image": { - "type": "string", - "required": true - }, - "themeColor": { - "type": "string", - "default": "#353FFF", - "required": true - }, - "imageBlur": { - "type": "string", - "default": "" - }, - "toys": { - "type": "relation", - "relation": "oneToMany", - "target": "api::toy.toy" - }, - "toy": { - "type": "relation", - "relation": "manyToOne", - "target": "api::toy.toy", - "inversedBy": "vtubers" - }, - "streams": { - "type": "relation", - "relation": "oneToMany", - "target": "api::stream.stream", - "mappedBy": "vtuber" - }, - "fanslyId": { - "type": "string" - }, - "chaturbateId": { - "type": "string" - }, - "twitterId": { - "type": "string" - } - } -} diff --git a/services/strapi/src/api/vtuber/controllers/vtuber.js b/services/strapi/src/api/vtuber/controllers/vtuber.js deleted file mode 100644 index e13bb5e..0000000 --- a/services/strapi/src/api/vtuber/controllers/vtuber.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * vtuber controller - */ - -const { createCoreController } = require('@strapi/strapi').factories; - -module.exports = createCoreController('api::vtuber.vtuber'); diff --git a/services/strapi/src/api/vtuber/routes/vtuber.js b/services/strapi/src/api/vtuber/routes/vtuber.js deleted file mode 100644 index 8936096..0000000 --- a/services/strapi/src/api/vtuber/routes/vtuber.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * vtuber router - */ - -const { createCoreRouter } = require('@strapi/strapi').factories; - -module.exports = createCoreRouter('api::vtuber.vtuber'); diff --git a/services/strapi/src/api/vtuber/services/vtuber.js b/services/strapi/src/api/vtuber/services/vtuber.js deleted file mode 100644 index c792042..0000000 --- a/services/strapi/src/api/vtuber/services/vtuber.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -/** - * vtuber service - */ - -const { createCoreService } = require('@strapi/strapi').factories; - -module.exports = createCoreService('api::vtuber.vtuber'); diff --git a/services/strapi/src/extensions/.gitkeep b/services/strapi/src/extensions/.gitkeep deleted file mode 100644 index e69de29..0000000 diff --git a/services/strapi/src/extensions/README.md b/services/strapi/src/extensions/README.md deleted file mode 100644 index 52db426..0000000 --- a/services/strapi/src/extensions/README.md +++ /dev/null @@ -1,13 +0,0 @@ -# strapi extensions - -note by CJ - -Reminder: Plugins don't go here. Plugins are added using pnpm. example: `pnpm add strapi-chatgpt` - - -This extensions directory is the place for plugin override code. - -@see https://docs.strapi.io/dev-docs/plugins-extension - - -We are using overrides patch in patreon support. In ./users-permissions/services/providers-registry.js we make some changes. When user logs in via Patreon, we determine if the Patreon account is a Futureporn supporter. Based on that information, we update the user role to one of 'Authenticated' or 'Patron'. \ No newline at end of file diff --git a/services/strapi/src/index.js b/services/strapi/src/index.js deleted file mode 100644 index ac5feae..0000000 --- a/services/strapi/src/index.js +++ /dev/null @@ -1,20 +0,0 @@ -'use strict'; - -module.exports = { - /** - * An asynchronous register function that runs before - * your application is initialized. - * - * This gives you an opportunity to extend code. - */ - register(/*{ strapi }*/) {}, - - /** - * An asynchronous bootstrap function that runs before - * your application gets started. - * - * This gives you an opportunity to set up your data model, - * run jobs, or perform some special logic. - */ - bootstrap(/*{ strapi }*/) {}, -}; diff --git a/services/strapi/src/policies/updateOwnerOnly.js b/services/strapi/src/policies/updateOwnerOnly.js deleted file mode 100644 index 86d2786..0000000 --- a/services/strapi/src/policies/updateOwnerOnly.js +++ /dev/null @@ -1,34 +0,0 @@ -'use strict'; - -/** - * `updateOwnerOnly` policy. - */ -module.exports = (policyContext, config, { strapi }) => { - const { PolicyError } = require("@strapi/utils").errors; - - if (policyContext.state.auth.strategy.name === "api-token") { - if (policyContext.state.auth.credentials.type === "full-access") - return true; - } else if ( - policyContext.state.auth.strategy.name === "users-permissions" - ) { - - // Skip for admins - if (policyContext.state.auth.credentials.role.type === "admin") { - return true; - } - - const currentUserId = policyContext.state.auth.credentials.id; - const userToUpdate = policyContext.params.id; - - // Unable that an user can update another user - if (currentUserId != userToUpdate) { - strapi.log.info(`WARNING: User ${currentUserId} tried to edit user ${userToUpdate}`); - throw new PolicyError('Unable to edit this user ID'); - } - - return true - } - - return false; -}; \ No newline at end of file diff --git a/services/uppy/.gitignore b/services/uppy/.gitignore deleted file mode 100644 index 7610089..0000000 --- a/services/uppy/.gitignore +++ /dev/null @@ -1,145 +0,0 @@ -# Created by https://www.toptal.com/developers/gitignore/api/node -# Edit at https://www.toptal.com/developers/gitignore?templates=node - -### Node ### -# Logs -logs -*.log -npm-debug.log* -yarn-debug.log* -yarn-error.log* -lerna-debug.log* -.pnpm-debug.log* - -# Diagnostic reports (https://nodejs.org/api/report.html) -report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json - -# Runtime data -pids -*.pid -*.seed -*.pid.lock - -# Directory for instrumented libs generated by jscoverage/JSCover -lib-cov - -# Coverage directory used by tools like istanbul -coverage -*.lcov - -# nyc test coverage -.nyc_output - -# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) -.grunt - -# Bower dependency directory (https://bower.io/) -bower_components - -# node-waf configuration -.lock-wscript - -# Compiled binary addons (https://nodejs.org/api/addons.html) -build/Release - -# Dependency directories -node_modules/ -jspm_packages/ - -# Snowpack dependency directory (https://snowpack.dev/) -web_modules/ - -# TypeScript cache -*.tsbuildinfo - -# Optional npm cache directory -.npm - -# Optional eslint cache -.eslintcache - -# Optional stylelint cache -.stylelintcache - -# Microbundle cache -.rpt2_cache/ -.rts2_cache_cjs/ -.rts2_cache_es/ -.rts2_cache_umd/ - -# Optional REPL history -.node_repl_history - -# Output of 'npm pack' -*.tgz - -# Yarn Integrity file -.yarn-integrity - -# dotenv environment variable files -.env -.env.development.local -.env.test.local -.env.production.local -.env.local - -# parcel-bundler cache (https://parceljs.org/) -.cache -.parcel-cache - -# Next.js build output -.next -out - -# Nuxt.js build / generate output -.nuxt -dist - -# Gatsby files -.cache/ -# Comment in the public line in if your project uses Gatsby and not Next.js -# https://nextjs.org/blog/next-9-1#public-directory-support -# public - -# vuepress build output -.vuepress/dist - -# vuepress v2.x temp and cache directory -.temp - -# Docusaurus cache and generated files -.docusaurus - -# Serverless directories -.serverless/ - -# FuseBox cache -.fusebox/ - -# DynamoDB Local files -.dynamodb/ - -# TernJS port file -.tern-port - -# Stores VSCode versions used for testing VSCode extensions -.vscode-test - -# yarn v2 -.yarn/cache -.yarn/unplugged -.yarn/build-state.yml -.yarn/install-state.gz -.pnp.* - -### Node Patch ### -# Serverless Webpack directories -.webpack/ - -# Optional stylelint cache - -# SvelteKit build / generate output -.svelte-kit - -# End of https://www.toptal.com/developers/gitignore/api/node - diff --git a/services/uppy/README.md b/services/uppy/README.md deleted file mode 100644 index 3df56eb..0000000 --- a/services/uppy/README.md +++ /dev/null @@ -1,16 +0,0 @@ -# uppy - -This is the server-side component, "Companion" which handles user uploads. It is especially useful for allowing users to upload directly from their Dropbox, GoogleDrive, etc. - -## B2 Bucket CORS configuration - -Apply the rules to the bucket, using b2 CLI - -See **apply-backblaze-cors-rules.sh** - -Verify using `b2-linux get-bucket futureporn-uppy-dev` - -### Reference - -https://www.backblaze.com/docs/cloud-storage-enable-cors-with-the-cli -https://uppy.io/docs/aws-s3-multipart/#setting-up-your-s3-bucket \ No newline at end of file diff --git a/services/uppy/apply-backblaze-cors-rules.sh b/services/uppy/apply-backblaze-cors-rules.sh deleted file mode 100755 index 82b56c9..0000000 --- a/services/uppy/apply-backblaze-cors-rules.sh +++ /dev/null @@ -1,8 +0,0 @@ -#/bin/bash - -source ../../.env -echo "allowedOrigin ${NEXT_PUBLIC_SITE_URL} to bucket ${UPPY_B2_BUCKET}" -b2-linux \ - update-bucket \ - --corsRules "[{\"corsRuleName\":\"allowUploads\", \"allowedOrigins\": [\"${NEXT_PUBLIC_SITE_URL}\"], \"allowedHeaders\": [\"*\"], \"allowedOperations\": [\"s3_head\", \"s3_get\", \"s3_put\"], \"exposeHeaders\": [\"x-bz-content-sha1\", \"etag\"], \"maxAgeSeconds\": 3600}]" \ - "${UPPY_B2_BUCKET}" allPrivate diff --git a/services/uppy/index.js b/services/uppy/index.js deleted file mode 100644 index f9460e8..0000000 --- a/services/uppy/index.js +++ /dev/null @@ -1,125 +0,0 @@ -#!/usr/bin/env node - -import 'dotenv/config' -import express from 'express' -import bodyParser from 'body-parser' -import session from 'express-session' -import companion from '@uppy/companion' -import $MemoryStore from 'memorystore' -import cors from 'cors' -import jwt from 'jsonwebtoken' - - -const appEnv = new Array( - 'COMPANION_DOMAINS', - 'HOST', - 'SESSION_SECRET', - 'PORT', - 'FILEPATH', - 'UPLOAD_URLS', - 'SECRET', - 'B2_KEY', - 'B2_SECRET', - 'B2_BUCKET', - 'B2_ENDPOINT', - 'DRIVE_KEY', - 'DRIVE_SECRET', - 'DROPBOX_KEY', - 'DROPBOX_SECRET', - 'JWT_SECRET', - 'NEXT_PUBLIC_SITE_URL' -) - -const appContext = { - env: appEnv.reduce((acc, ev) => { - if (typeof process.env[ev] === 'undefined') throw new Error(`${ev} is undefined in env`); - acc[ev] = process.env[ev]; - return acc; - }, {}) -}; - - -const MemoryStore = $MemoryStore(session) - -console.log(`NEXT_PUBLIC_SITE_URL=${process.env.NEXT_PUBLIC_SITE_URL}`); -const corsOptions = { - origin: process.env.NEXT_PUBLIC_SITE_URL, - methods: ['POST','OPTIONS'], - allowedHeaders: ['Authorization', 'X-Easter-Egg', 'Content-Type', 'Uppy-Versions', 'Accept'], - exposedHeaders: ['Access-Control-Allow-Headers'] -} - - -const app = express(); -app.use(cors(corsOptions)) -app.use(bodyParser.json()) -app.use( - session({ - store: new MemoryStore({ - checkPeriod: 86400000 // prune expired entries every 24h - }), - saveUninitialized: false, - secret: process.env.SESSION_SECRET, - resave: false, - }) -) - -function verifyToken(req, res, next) { - const authHeader = req.headers['authorization']; - const token = authHeader && authHeader.split(' ')[1]; - - if (!token) { - return res.sendStatus(401); - } - - jwt.verify(token, new Buffer.from(process.env.JWT_SECRET, 'base64'), (err, user) => { - if (err) { - return res.sendStatus(403); - } - req.user = user; - next(); - }); -} - - -const config = { - debug: true, - logClientVersion: true, - secret: process.env.SECRET, - filePath: process.env.FILEPATH, - server: { - host: process.env.HOST - }, - uploadUrls: process.env.UPLOAD_URLS, - s3: { - key: process.env.B2_KEY, - secret: process.env.B2_SECRET, - bucket: process.env.B2_BUCKET, - region: process.env.B2_REGION, - endpoint: process.env.B2_ENDPOINT, - getAccelerateEndpoint: false - }, - // providerOptions: { - // drive: { - // key: process.env.DRIVE_KEY, - // secret: process.env.DRIVE_SECRET, - // }, - // dropbox: { - // key: process.env.DROPBOX_KEY, - // secret: process.env.DROPBOX_SECRET, - // }, - // } -} - - -const { app: companionApp, emitter } = companion.app(config); - - -app.use(verifyToken, companionApp); - - -const server = app.listen(process.env.PORT) - -companion.socket(server) - - diff --git a/services/uppy/package.json b/services/uppy/package.json deleted file mode 100644 index cd3acf0..0000000 --- a/services/uppy/package.json +++ /dev/null @@ -1,28 +0,0 @@ -{ - "type": "module", - "name": "uppy", - "version": "1.0.5", - "description": "", - "main": "index.js", - "scripts": { - "start": "node index", - "clean": "rm -rf dist", - "superclean": "rm -rf node_modules && rm -rf pnpm-lock.yaml && rm -rf dist" - }, - "keywords": [], - "author": "", - "license": "Unlicense", - "dependencies": { - "@uppy/aws-s3-multipart": "^4.0.0", - "@uppy/companion": "5.1.0", - "body-parser": "^1.20.2", - "cors": "^2.8.5", - "dotenv": "^16.4.5", - "express": "^4.19.2", - "express-session": "^1.18.0", - "jsonwebtoken": "^9.0.2", - "memorystore": "^1.6.7", - "react-hook-form": "^7.53.0", - "yup": "link:@hookform/resolvers/yup" - } -} diff --git a/services/uppy/pnpm-lock.yaml b/services/uppy/pnpm-lock.yaml deleted file mode 100644 index 9158792..0000000 --- a/services/uppy/pnpm-lock.yaml +++ /dev/null @@ -1,3177 +0,0 @@ -lockfileVersion: '9.0' - -settings: - autoInstallPeers: true - excludeLinksFromLockfile: false - -importers: - - .: - dependencies: - '@uppy/aws-s3-multipart': - specifier: ^4.0.0 - version: 4.0.0(@uppy/core@4.2.0) - '@uppy/companion': - specifier: 5.1.0 - version: 5.1.0 - body-parser: - specifier: ^1.20.2 - version: 1.20.2(supports-color@8.1.1) - cors: - specifier: ^2.8.5 - version: 2.8.5 - dotenv: - specifier: ^16.4.5 - version: 16.4.5 - express: - specifier: ^4.19.2 - version: 4.19.2(supports-color@8.1.1) - express-session: - specifier: ^1.18.0 - version: 1.18.0(supports-color@8.1.1) - jsonwebtoken: - specifier: ^9.0.2 - version: 9.0.2 - memorystore: - specifier: ^1.6.7 - version: 1.6.7 - react-hook-form: - specifier: ^7.53.0 - version: 7.53.0(react@18.3.1) - yup: - specifier: link:@hookform/resolvers/yup - version: link:@hookform/resolvers/yup - -packages: - - '@aws-crypto/crc32@5.2.0': - resolution: {integrity: sha512-nLbCWqQNgUiwwtFsen1AdzAtvuLRsQS8rYgMuxCrdKf9kOssamGLuPwyTY9wyYblNr9+1XM8v6zoDTPPSIeANg==} - engines: {node: '>=16.0.0'} - - '@aws-crypto/crc32c@5.2.0': - resolution: {integrity: sha512-+iWb8qaHLYKrNvGRbiYRHSdKRWhto5XlZUEBwDjYNf+ly5SVYG6zEoYIdxvf5R3zyeP16w4PLBn3rH1xc74Rag==} - - '@aws-crypto/sha1-browser@5.2.0': - resolution: {integrity: sha512-OH6lveCFfcDjX4dbAvCFSYUjJZjDr/3XJ3xHtjn3Oj5b9RjojQo8npoLeA/bNwkOkrSQ0wgrHzXk4tDRxGKJeg==} - - '@aws-crypto/sha256-browser@5.2.0': - resolution: {integrity: sha512-AXfN/lGotSQwu6HNcEsIASo7kWXZ5HYWvfOmSNKDsEqC4OashTp8alTmaz+F7TC2L083SFv5RdB+qU3Vs1kZqw==} - - '@aws-crypto/sha256-js@5.2.0': - resolution: {integrity: sha512-FFQQyu7edu4ufvIZ+OadFpHHOt+eSTBaYaki44c+akjg7qZg9oOQeLlk77F6tSYqjDAFClrHJk9tMf0HdVyOvA==} - engines: {node: '>=16.0.0'} - - '@aws-crypto/supports-web-crypto@5.2.0': - resolution: {integrity: sha512-iAvUotm021kM33eCdNfwIN//F77/IADDSs58i+MDaOqFrVjZo9bAal0NK7HurRuWLLpF1iLX7gbWrjHjeo+YFg==} - - '@aws-crypto/util@5.2.0': - resolution: {integrity: sha512-4RkU9EsI6ZpBve5fseQlGNUWKMa1RLPQ1dnjnQoe07ldfIzcsGb5hC5W0Dm7u423KWzawlrpbjXBrXCEv9zazQ==} - - '@aws-sdk/client-s3@3.637.0': - resolution: {integrity: sha512-y6UC94fsMvhKbf0dzfnjVP1HePeGjplfcYfilZU1COIJLyTkMcUv4XcT4I407CGIrvgEafONHkiC09ygqUauNA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/client-sso-oidc@3.637.0': - resolution: {integrity: sha512-27bHALN6Qb6m6KZmPvRieJ/QRlj1lyac/GT2Rn5kJpre8Mpp+yxrtvp3h9PjNBty4lCeFEENfY4dGNSozBuBcw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.637.0 - - '@aws-sdk/client-sso@3.637.0': - resolution: {integrity: sha512-+KjLvgX5yJYROWo3TQuwBJlHCY0zz9PsLuEolmXQn0BVK1L/m9GteZHtd+rEdAoDGBpE0Xqjy1oz5+SmtsaRUw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/client-sts@3.637.0': - resolution: {integrity: sha512-xUi7x4qDubtA8QREtlblPuAcn91GS/09YVEY/RwU7xCY0aqGuFwgszAANlha4OUIqva8oVj2WO4gJuG+iaSnhw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/core@3.635.0': - resolution: {integrity: sha512-i1x/E/sgA+liUE1XJ7rj1dhyXpAKO1UKFUcTTHXok2ARjWTvszHnSXMOsB77aPbmn0fUp1JTx2kHUAZ1LVt5Bg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-env@3.620.1': - resolution: {integrity: sha512-ExuILJ2qLW5ZO+rgkNRj0xiAipKT16Rk77buvPP8csR7kkCflT/gXTyzRe/uzIiETTxM7tr8xuO9MP/DQXqkfg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-http@3.635.0': - resolution: {integrity: sha512-iJyRgEjOCQlBMXqtwPLIKYc7Bsc6nqjrZybdMDenPDa+kmLg7xh8LxHsu9088e+2/wtLicE34FsJJIfzu3L82g==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-ini@3.637.0': - resolution: {integrity: sha512-h+PFCWfZ0Q3Dx84SppET/TFpcQHmxFW8/oV9ArEvMilw4EBN+IlxgbL0CnHwjHW64szcmrM0mbebjEfHf4FXmw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.637.0 - - '@aws-sdk/credential-provider-node@3.637.0': - resolution: {integrity: sha512-yoEhoxJJfs7sPVQ6Is939BDQJZpZCoUgKr/ySse4YKOZ24t4VqgHA6+wV7rYh+7IW24Rd91UTvEzSuHYTlxlNA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-process@3.620.1': - resolution: {integrity: sha512-hWqFMidqLAkaV9G460+1at6qa9vySbjQKKc04p59OT7lZ5cO5VH5S4aI05e+m4j364MBROjjk2ugNvfNf/8ILg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-sso@3.637.0': - resolution: {integrity: sha512-Mvz+h+e62/tl+dVikLafhv+qkZJ9RUb8l2YN/LeKMWkxQylPT83CPk9aimVhCV89zth1zpREArl97+3xsfgQvA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/credential-provider-web-identity@3.621.0': - resolution: {integrity: sha512-w7ASSyfNvcx7+bYGep3VBgC3K6vEdLmlpjT7nSIHxxQf+WSdvy+HynwJosrpZax0sK5q0D1Jpn/5q+r5lwwW6w==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sts': ^3.621.0 - - '@aws-sdk/lib-storage@3.637.0': - resolution: {integrity: sha512-HiNGOP4a1QrCWwO1joKw4mCp19nLXoF9K52PislBaYDI35IlHC3DP6MeOg5zmElwtL1GtEHFBy5olfPWPsLyLg==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-s3': ^3.637.0 - - '@aws-sdk/middleware-bucket-endpoint@3.620.0': - resolution: {integrity: sha512-eGLL0W6L3HDb3OACyetZYOWpHJ+gLo0TehQKeQyy2G8vTYXqNTeqYhuI6up9HVjBzU9eQiULVQETmgQs7TFaRg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-expect-continue@3.620.0': - resolution: {integrity: sha512-QXeRFMLfyQ31nAHLbiTLtk0oHzG9QLMaof5jIfqcUwnOkO8YnQdeqzakrg1Alpy/VQ7aqzIi8qypkBe2KXZz0A==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-flexible-checksums@3.620.0': - resolution: {integrity: sha512-ftz+NW7qka2sVuwnnO1IzBku5ccP+s5qZGeRTPgrKB7OzRW85gthvIo1vQR2w+OwHFk7WJbbhhWwbCbktnP4UA==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-host-header@3.620.0': - resolution: {integrity: sha512-VMtPEZwqYrII/oUkffYsNWY9PZ9xpNJpMgmyU0rlDQ25O1c0Hk3fJmZRe6pEkAJ0omD7kLrqGl1DUjQVxpd/Rg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-location-constraint@3.609.0': - resolution: {integrity: sha512-xzsdoTkszGVqGVPjUmgoP7TORiByLueMHieI1fhQL888WPdqctwAx3ES6d/bA9Q/i8jnc6hs+Fjhy8UvBTkE9A==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-logger@3.609.0': - resolution: {integrity: sha512-S62U2dy4jMDhDFDK5gZ4VxFdWzCtLzwbYyFZx2uvPYTECkepLUfzLic2BHg2Qvtu4QjX+oGE3P/7fwaGIsGNuQ==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-recursion-detection@3.620.0': - resolution: {integrity: sha512-nh91S7aGK3e/o1ck64sA/CyoFw+gAYj2BDOnoNa6ouyCrVJED96ZXWbhye/fz9SgmNUZR2g7GdVpiLpMKZoI5w==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-sdk-s3@3.635.0': - resolution: {integrity: sha512-RLdYJPEV4JL/7NBoFUs7VlP90X++5FlJdxHz0DzCjmiD3qCviKy+Cym3qg1gBgHwucs5XisuClxDrGokhAdTQw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-ssec@3.609.0': - resolution: {integrity: sha512-GZSD1s7+JswWOTamVap79QiDaIV7byJFssBW68GYjyRS5EBjNfwA/8s+6uE6g39R3ojyTbYOmvcANoZEhSULXg==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/middleware-user-agent@3.637.0': - resolution: {integrity: sha512-EYo0NE9/da/OY8STDsK2LvM4kNa79DBsf4YVtaG4P5pZ615IeFsD8xOHZeuJmUrSMlVQ8ywPRX7WMucUybsKug==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/region-config-resolver@3.614.0': - resolution: {integrity: sha512-vDCeMXvic/LU0KFIUjpC3RiSTIkkvESsEfbVHiHH0YINfl8HnEqR5rj+L8+phsCeVg2+LmYwYxd5NRz4PHxt5g==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/s3-presigned-post@3.637.0': - resolution: {integrity: sha512-NlsowCavLXx5OulbD8+tqfXpyxVulI2l/4naBI1i5Vj6M+FMuQvqH1Y5OQBgN42/QZlMLJ5Ff5G3aaoZjrb1pw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/s3-request-presigner@3.637.0': - resolution: {integrity: sha512-URRiEDZEICyfAXmXcXREQCsvZrapITAymvg46p1Xjnuv7PTnUB0SF18B2omPL0E5d/X+T3O9NKdtot+BqJbIWw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/signature-v4-multi-region@3.635.0': - resolution: {integrity: sha512-J6QY4/invOkpogCHjSaDON1hF03viPpOnsrzVuCvJMmclS/iG62R4EY0wq1alYll0YmSdmKlpJwHMWwGtqK63Q==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/token-providers@3.614.0': - resolution: {integrity: sha512-okItqyY6L9IHdxqs+Z116y5/nda7rHxLvROxtAJdLavWTYDydxrZstImNgGWTeVdmc0xX2gJCI77UYUTQWnhRw==} - engines: {node: '>=16.0.0'} - peerDependencies: - '@aws-sdk/client-sso-oidc': ^3.614.0 - - '@aws-sdk/types@3.609.0': - resolution: {integrity: sha512-+Tqnh9w0h2LcrUsdXyT1F8mNhXz+tVYBtP19LpeEGntmvHwa2XzvLUCWpoIAIVsHp5+HdB2X9Sn0KAtmbFXc2Q==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-arn-parser@3.568.0': - resolution: {integrity: sha512-XUKJWWo+KOB7fbnPP0+g/o5Ulku/X53t7i/h+sPHr5xxYTJJ9CYnbToo95mzxe7xWvkLrsNtJ8L+MnNn9INs2w==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-endpoints@3.637.0': - resolution: {integrity: sha512-pAqOKUHeVWHEXXDIp/qoMk/6jyxIb6GGjnK1/f8dKHtKIEs4tKsnnL563gceEvdad53OPXIt86uoevCcCzmBnw==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-format-url@3.609.0': - resolution: {integrity: sha512-fuk29BI/oLQlJ7pfm6iJ4gkEpHdavffAALZwXh9eaY1vQ0ip0aKfRTiNudPoJjyyahnz5yJ1HkmlcDitlzsOrQ==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-locate-window@3.568.0': - resolution: {integrity: sha512-3nh4TINkXYr+H41QaPelCceEB2FXP3fxp93YZXB/kqJvX0U9j0N0Uk45gvsjmEPzG8XxkPEeLIfT2I1M7A6Lig==} - engines: {node: '>=16.0.0'} - - '@aws-sdk/util-user-agent-browser@3.609.0': - resolution: {integrity: sha512-fojPU+mNahzQ0YHYBsx0ZIhmMA96H+ZIZ665ObU9tl+SGdbLneVZVikGve+NmHTQwHzwkFsZYYnVKAkreJLAtA==} - - '@aws-sdk/util-user-agent-node@3.614.0': - resolution: {integrity: sha512-15ElZT88peoHnq5TEoEtZwoXTXRxNrk60TZNdpl/TUBJ5oNJ9Dqb5Z4ryb8ofN6nm9aFf59GVAerFDz8iUoHBA==} - engines: {node: '>=16.0.0'} - peerDependencies: - aws-crt: '>=1.0.0' - peerDependenciesMeta: - aws-crt: - optional: true - - '@aws-sdk/xml-builder@3.609.0': - resolution: {integrity: sha512-l9XxNcA4HX98rwCC2/KoiWcmEiRfZe4G+mYwDbCFT87JIMj6GBhLDkAzr/W8KAaA2IDr8Vc6J8fZPgVulxxfMA==} - engines: {node: '>=16.0.0'} - - '@ioredis/commands@1.2.0': - resolution: {integrity: sha512-Sx1pU8EM64o2BrqNpEO1CNLtKQwyhuXuqyfH7oGKCk+1a33d2r5saW8zNwm3j6BTExtjrv2BxTgzzkMwts6vGg==} - - '@opentelemetry/api@1.9.0': - resolution: {integrity: sha512-3giAOQvZiH5F9bMlMiv8+GSPMeqg0dbaeo58/0SlA9sxSqZhnUtxzX9/2FzyhS9sWQf5S0GJE0AKBrFqjpeYcg==} - engines: {node: '>=8.0.0'} - - '@sindresorhus/is@5.6.0': - resolution: {integrity: sha512-TV7t8GKYaJWsn00tFDqBw8+Uqmr8A0fRU1tvTQhyZzGv0sJCGRQL3JGMI3ucuKo3XIZdUP+Lx7/gh2t3lewy7g==} - engines: {node: '>=14.16'} - - '@smithy/abort-controller@3.1.1': - resolution: {integrity: sha512-MBJBiidoe+0cTFhyxT8g+9g7CeVccLM0IOKKUMCNQ1CNMJ/eIfoo0RTfVrXOONEI1UCN1W+zkiHSbzUNE9dZtQ==} - engines: {node: '>=16.0.0'} - - '@smithy/chunked-blob-reader-native@3.0.0': - resolution: {integrity: sha512-VDkpCYW+peSuM4zJip5WDfqvg2Mo/e8yxOv3VF1m11y7B8KKMKVFtmZWDe36Fvk8rGuWrPZHHXZ7rR7uM5yWyg==} - - '@smithy/chunked-blob-reader@3.0.0': - resolution: {integrity: sha512-sbnURCwjF0gSToGlsBiAmd1lRCmSn72nu9axfJu5lIx6RUEgHu6GwTMbqCdhQSi0Pumcm5vFxsi9XWXb2mTaoA==} - - '@smithy/config-resolver@3.0.5': - resolution: {integrity: sha512-SkW5LxfkSI1bUC74OtfBbdz+grQXYiPYolyu8VfpLIjEoN/sHVBlLeGXMQ1vX4ejkgfv6sxVbQJ32yF2cl1veA==} - engines: {node: '>=16.0.0'} - - '@smithy/core@2.4.0': - resolution: {integrity: sha512-cHXq+FneIF/KJbt4q4pjN186+Jf4ZB0ZOqEaZMBhT79srEyGDDBV31NqBRBjazz8ppQ1bJbDJMY9ba5wKFV36w==} - engines: {node: '>=16.0.0'} - - '@smithy/credential-provider-imds@3.2.0': - resolution: {integrity: sha512-0SCIzgd8LYZ9EJxUjLXBmEKSZR/P/w6l7Rz/pab9culE/RWuqelAKGJvn5qUOl8BgX8Yj5HWM50A5hiB/RzsgA==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-codec@3.1.2': - resolution: {integrity: sha512-0mBcu49JWt4MXhrhRAlxASNy0IjDRFU+aWNDRal9OtUJvJNiwDuyKMUONSOjLjSCeGwZaE0wOErdqULer8r7yw==} - - '@smithy/eventstream-serde-browser@3.0.6': - resolution: {integrity: sha512-2hM54UWQUOrki4BtsUI1WzmD13/SeaqT/AB3EUJKbcver/WgKNaiJ5y5F5XXuVe6UekffVzuUDrBZVAA3AWRpQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-config-resolver@3.0.3': - resolution: {integrity: sha512-NVTYjOuYpGfrN/VbRQgn31x73KDLfCXCsFdad8DiIc3IcdxL+dYA9zEQPyOP7Fy2QL8CPy2WE4WCUD+ZsLNfaQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-node@3.0.5': - resolution: {integrity: sha512-+upXvnHNyZP095s11jF5dhGw/Ihzqwl5G+/KtMnoQOpdfC3B5HYCcDVG9EmgkhJMXJlM64PyN5gjJl0uXFQehQ==} - engines: {node: '>=16.0.0'} - - '@smithy/eventstream-serde-universal@3.0.5': - resolution: {integrity: sha512-5u/nXbyoh1s4QxrvNre9V6vfyoLWuiVvvd5TlZjGThIikc3G+uNiG9uOTCWweSRjv1asdDIWK7nOmN7le4RYHQ==} - engines: {node: '>=16.0.0'} - - '@smithy/fetch-http-handler@3.2.4': - resolution: {integrity: sha512-kBprh5Gs5h7ug4nBWZi1FZthdqSM+T7zMmsZxx0IBvWUn7dK3diz2SHn7Bs4dQGFDk8plDv375gzenDoNwrXjg==} - - '@smithy/hash-blob-browser@3.1.2': - resolution: {integrity: sha512-hAbfqN2UbISltakCC2TP0kx4LqXBttEv2MqSPE98gVuDFMf05lU+TpC41QtqGP3Ff5A3GwZMPfKnEy0VmEUpmg==} - - '@smithy/hash-node@3.0.3': - resolution: {integrity: sha512-2ctBXpPMG+B3BtWSGNnKELJ7SH9e4TNefJS0cd2eSkOOROeBnnVBnAy9LtJ8tY4vUEoe55N4CNPxzbWvR39iBw==} - engines: {node: '>=16.0.0'} - - '@smithy/hash-stream-node@3.1.2': - resolution: {integrity: sha512-PBgDMeEdDzi6JxKwbfBtwQG9eT9cVwsf0dZzLXoJF4sHKHs5HEo/3lJWpn6jibfJwT34I1EBXpBnZE8AxAft6g==} - engines: {node: '>=16.0.0'} - - '@smithy/invalid-dependency@3.0.3': - resolution: {integrity: sha512-ID1eL/zpDULmHJbflb864k72/SNOZCADRc9i7Exq3RUNJw6raWUSlFEQ+3PX3EYs++bTxZB2dE9mEHTQLv61tw==} - - '@smithy/is-array-buffer@2.2.0': - resolution: {integrity: sha512-GGP3O9QFD24uGeAXYUjwSTXARoqpZykHadOmA8G5vfJPK0/DC67qa//0qvqrJzL1xc8WQWX7/yc7fwudjPHPhA==} - engines: {node: '>=14.0.0'} - - '@smithy/is-array-buffer@3.0.0': - resolution: {integrity: sha512-+Fsu6Q6C4RSJiy81Y8eApjEB5gVtM+oFKTffg+jSuwtvomJJrhUJBu2zS8wjXSgH/g1MKEWrzyChTBe6clb5FQ==} - engines: {node: '>=16.0.0'} - - '@smithy/md5-js@3.0.3': - resolution: {integrity: sha512-O/SAkGVwpWmelpj/8yDtsaVe6sINHLB1q8YE/+ZQbDxIw3SRLbTZuRaI10K12sVoENdnHqzPp5i3/H+BcZ3m3Q==} - - '@smithy/middleware-content-length@3.0.5': - resolution: {integrity: sha512-ILEzC2eyxx6ncej3zZSwMpB5RJ0zuqH7eMptxC4KN3f+v9bqT8ohssKbhNR78k/2tWW+KS5Spw+tbPF4Ejyqvw==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-endpoint@3.1.0': - resolution: {integrity: sha512-5y5aiKCEwg9TDPB4yFE7H6tYvGFf1OJHNczeY10/EFF8Ir8jZbNntQJxMWNfeQjC1mxPsaQ6mR9cvQbf+0YeMw==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-retry@3.0.15': - resolution: {integrity: sha512-iTMedvNt1ApdvkaoE8aSDuwaoc+BhvHqttbA/FO4Ty+y/S5hW6Ci/CTScG7vam4RYJWZxdTElc3MEfHRVH6cgQ==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-serde@3.0.3': - resolution: {integrity: sha512-puUbyJQBcg9eSErFXjKNiGILJGtiqmuuNKEYNYfUD57fUl4i9+mfmThtQhvFXU0hCVG0iEJhvQUipUf+/SsFdA==} - engines: {node: '>=16.0.0'} - - '@smithy/middleware-stack@3.0.3': - resolution: {integrity: sha512-r4klY9nFudB0r9UdSMaGSyjyQK5adUyPnQN/ZM6M75phTxOdnc/AhpvGD1fQUvgmqjQEBGCwpnPbDm8pH5PapA==} - engines: {node: '>=16.0.0'} - - '@smithy/node-config-provider@3.1.4': - resolution: {integrity: sha512-YvnElQy8HR4vDcAjoy7Xkx9YT8xZP4cBXcbJSgm/kxmiQu08DwUwj8rkGnyoJTpfl/3xYHH+d8zE+eHqoDCSdQ==} - engines: {node: '>=16.0.0'} - - '@smithy/node-http-handler@3.1.4': - resolution: {integrity: sha512-+UmxgixgOr/yLsUxcEKGH0fMNVteJFGkmRltYFHnBMlogyFdpzn2CwqWmxOrfJELhV34v0WSlaqG1UtE1uXlJg==} - engines: {node: '>=16.0.0'} - - '@smithy/property-provider@3.1.3': - resolution: {integrity: sha512-zahyOVR9Q4PEoguJ/NrFP4O7SMAfYO1HLhB18M+q+Z4KFd4V2obiMnlVoUFzFLSPeVt1POyNWneHHrZaTMoc/g==} - engines: {node: '>=16.0.0'} - - '@smithy/protocol-http@4.1.0': - resolution: {integrity: sha512-dPVoHYQ2wcHooGXg3LQisa1hH0e4y0pAddPMeeUPipI1tEOqL6A4N0/G7abeq+K8wrwSgjk4C0wnD1XZpJm5aA==} - engines: {node: '>=16.0.0'} - - '@smithy/querystring-builder@3.0.3': - resolution: {integrity: sha512-vyWckeUeesFKzCDaRwWLUA1Xym9McaA6XpFfAK5qI9DKJ4M33ooQGqvM4J+LalH4u/Dq9nFiC8U6Qn1qi0+9zw==} - engines: {node: '>=16.0.0'} - - '@smithy/querystring-parser@3.0.3': - resolution: {integrity: sha512-zahM1lQv2YjmznnfQsWbYojFe55l0SLG/988brlLv1i8z3dubloLF+75ATRsqPBboUXsW6I9CPGE5rQgLfY0vQ==} - engines: {node: '>=16.0.0'} - - '@smithy/service-error-classification@3.0.3': - resolution: {integrity: sha512-Jn39sSl8cim/VlkLsUhRFq/dKDnRUFlfRkvhOJaUbLBXUsLRLNf9WaxDv/z9BjuQ3A6k/qE8af1lsqcwm7+DaQ==} - engines: {node: '>=16.0.0'} - - '@smithy/shared-ini-file-loader@3.1.4': - resolution: {integrity: sha512-qMxS4hBGB8FY2GQqshcRUy1K6k8aBWP5vwm8qKkCT3A9K2dawUwOIJfqh9Yste/Bl0J2lzosVyrXDj68kLcHXQ==} - engines: {node: '>=16.0.0'} - - '@smithy/signature-v4@4.1.0': - resolution: {integrity: sha512-aRryp2XNZeRcOtuJoxjydO6QTaVhxx/vjaR+gx7ZjaFgrgPRyZ3HCTbfwqYj6ZWEBHkCSUfcaymKPURaByukag==} - engines: {node: '>=16.0.0'} - - '@smithy/smithy-client@3.2.0': - resolution: {integrity: sha512-pDbtxs8WOhJLJSeaF/eAbPgXg4VVYFlRcL/zoNYA5WbG3wBL06CHtBSg53ppkttDpAJ/hdiede+xApip1CwSLw==} - engines: {node: '>=16.0.0'} - - '@smithy/types@3.3.0': - resolution: {integrity: sha512-IxvBBCTFDHbVoK7zIxqA1ZOdc4QfM5HM7rGleCuHi7L1wnKv5Pn69xXJQ9hgxH60ZVygH9/JG0jRgtUncE3QUA==} - engines: {node: '>=16.0.0'} - - '@smithy/url-parser@3.0.3': - resolution: {integrity: sha512-pw3VtZtX2rg+s6HMs6/+u9+hu6oY6U7IohGhVNnjbgKy86wcIsSZwgHrFR+t67Uyxvp4Xz3p3kGXXIpTNisq8A==} - - '@smithy/util-base64@3.0.0': - resolution: {integrity: sha512-Kxvoh5Qtt0CDsfajiZOCpJxgtPHXOKwmM+Zy4waD43UoEMA+qPxxa98aE/7ZhdnBFZFXMOiBR5xbcaMhLtznQQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-body-length-browser@3.0.0': - resolution: {integrity: sha512-cbjJs2A1mLYmqmyVl80uoLTJhAcfzMOyPgjwAYusWKMdLeNtzmMz9YxNl3/jRLoxSS3wkqkf0jwNdtXWtyEBaQ==} - - '@smithy/util-body-length-node@3.0.0': - resolution: {integrity: sha512-Tj7pZ4bUloNUP6PzwhN7K386tmSmEET9QtQg0TgdNOnxhZvCssHji+oZTUIuzxECRfG8rdm2PMw2WCFs6eIYkA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-buffer-from@2.2.0': - resolution: {integrity: sha512-IJdWBbTcMQ6DA0gdNhh/BwrLkDR+ADW5Kr1aZmd4k3DIF6ezMV4R2NIAmT08wQJ3yUK82thHWmC/TnK/wpMMIA==} - engines: {node: '>=14.0.0'} - - '@smithy/util-buffer-from@3.0.0': - resolution: {integrity: sha512-aEOHCgq5RWFbP+UDPvPot26EJHjOC+bRgse5A8V3FSShqd5E5UN4qc7zkwsvJPPAVsf73QwYcHN1/gt/rtLwQA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-config-provider@3.0.0': - resolution: {integrity: sha512-pbjk4s0fwq3Di/ANL+rCvJMKM5bzAQdE5S/6RL5NXgMExFAi6UgQMPOm5yPaIWPpr+EOXKXRonJ3FoxKf4mCJQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-defaults-mode-browser@3.0.15': - resolution: {integrity: sha512-FZ4Psa3vjp8kOXcd3HJOiDPBCWtiilLl57r0cnNtq/Ga9RSDrM5ERL6xt+tO43+2af6Pn5Yp92x2n5vPuduNfg==} - engines: {node: '>= 10.0.0'} - - '@smithy/util-defaults-mode-node@3.0.15': - resolution: {integrity: sha512-KSyAAx2q6d0t6f/S4XB2+3+6aQacm3aLMhs9aLMqn18uYGUepbdssfogW5JQZpc6lXNBnp0tEnR5e9CEKmEd7A==} - engines: {node: '>= 10.0.0'} - - '@smithy/util-endpoints@2.0.5': - resolution: {integrity: sha512-ReQP0BWihIE68OAblC/WQmDD40Gx+QY1Ez8mTdFMXpmjfxSyz2fVQu3A4zXRfQU9sZXtewk3GmhfOHswvX+eNg==} - engines: {node: '>=16.0.0'} - - '@smithy/util-hex-encoding@3.0.0': - resolution: {integrity: sha512-eFndh1WEK5YMUYvy3lPlVmYY/fZcQE1D8oSf41Id2vCeIkKJXPcYDCZD+4+xViI6b1XSd7tE+s5AmXzz5ilabQ==} - engines: {node: '>=16.0.0'} - - '@smithy/util-middleware@3.0.3': - resolution: {integrity: sha512-l+StyYYK/eO3DlVPbU+4Bi06Jjal+PFLSMmlWM1BEwyLxZ3aKkf1ROnoIakfaA7mC6uw3ny7JBkau4Yc+5zfWw==} - engines: {node: '>=16.0.0'} - - '@smithy/util-retry@3.0.3': - resolution: {integrity: sha512-AFw+hjpbtVApzpNDhbjNG5NA3kyoMs7vx0gsgmlJF4s+yz1Zlepde7J58zpIRIsdjc+emhpAITxA88qLkPF26w==} - engines: {node: '>=16.0.0'} - - '@smithy/util-stream@3.1.3': - resolution: {integrity: sha512-FIv/bRhIlAxC0U7xM1BCnF2aDRPq0UaelqBHkM2lsCp26mcBbgI0tCVTv+jGdsQLUmAMybua/bjDsSu8RQHbmw==} - engines: {node: '>=16.0.0'} - - '@smithy/util-uri-escape@3.0.0': - resolution: {integrity: sha512-LqR7qYLgZTD7nWLBecUi4aqolw8Mhza9ArpNEQ881MJJIU2sE5iHCK6TdyqqzcDLy0OPe10IY4T8ctVdtynubg==} - engines: {node: '>=16.0.0'} - - '@smithy/util-utf8@2.3.0': - resolution: {integrity: sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A==} - engines: {node: '>=14.0.0'} - - '@smithy/util-utf8@3.0.0': - resolution: {integrity: sha512-rUeT12bxFnplYDe815GXbq/oixEGHfRFFtcTF3YdDi/JaENIM6aSYYLJydG83UNzLXeRI5K8abYd/8Sp/QM0kA==} - engines: {node: '>=16.0.0'} - - '@smithy/util-waiter@3.1.2': - resolution: {integrity: sha512-4pP0EV3iTsexDx+8PPGAKCQpd/6hsQBaQhqWzU4hqKPHN5epPsxKbvUTIiYIHTxaKt6/kEaqPBpu/ufvfbrRzw==} - engines: {node: '>=16.0.0'} - - '@szmarczak/http-timer@5.0.1': - resolution: {integrity: sha512-+PmQX0PiAYPMeVYe237LJAYvOMYW1j2rH5YROyS3b4CTVJum34HfRvKvAzozHAQG0TnHNdUfY9nCeUyRAs//cw==} - engines: {node: '>=14.16'} - - '@transloadit/prettier-bytes@0.3.4': - resolution: {integrity: sha512-8/SnIF9Q2k52mbjRVAYLranwkaDTLb+O9r4Z/uo8uNw//SjygKvvbF4BHSOuReufaAyum1q13602VcNud25Dfg==} - - '@types/body-parser@1.19.5': - resolution: {integrity: sha512-fB3Zu92ucau0iQ0JMCFQE7b/dv8Ot07NI3KaZIkIUNXq82k4eBAqUaneXfleGY9JWskeS9y+u0nXMyspcuQrCg==} - - '@types/connect@3.4.38': - resolution: {integrity: sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==} - - '@types/express-serve-static-core@4.19.5': - resolution: {integrity: sha512-y6W03tvrACO72aijJ5uF02FRq5cgDR9lUxddQ8vyF+GvmjJQqbzDcJngEjURc+ZsG31VI3hODNZJ2URj86pzmg==} - - '@types/express@4.17.21': - resolution: {integrity: sha512-ejlPM315qwLpaQlQDTjPdsUFSc6ZsP4AN6AlWnogPjQ7CVi7PYF3YVz+CY3jE2pwYf7E/7HlDAN0rV2GxTG0HQ==} - - '@types/http-cache-semantics@4.0.4': - resolution: {integrity: sha512-1m0bIFVc7eJWyve9S0RnuRgcQqF/Xd5QsUZAZeQFr1Q3/p9JWoQQEqmVy+DPTNpGXwhgIetAoYF8JSc33q29QA==} - - '@types/http-errors@2.0.4': - resolution: {integrity: sha512-D0CFMMtydbJAegzOyHjtiKPLlvnm3iTZyZRSZoLq2mRhDdmLfIWOCYPfQJ4cu2erKghU++QvjcUjp/5h7hESpA==} - - '@types/mime@1.3.5': - resolution: {integrity: sha512-/pyBZWSLD2n0dcHE3hq8s8ZvcETHtEuF+3E7XVt0Ig2nvsVQXdghHVcEkIWjy9A0wKfTn97a/PSDYohKIlnP/w==} - - '@types/node@22.5.2': - resolution: {integrity: sha512-acJsPTEqYqulZS/Yp/S3GgeE6GZ0qYODUR8aVr/DkhHQ8l9nd4j5x1/ZJy9/gHrRlFMqkO6i0I3E27Alu4jjPg==} - - '@types/qs@6.9.15': - resolution: {integrity: sha512-uXHQKES6DQKKCLh441Xv/dwxOq1TVS3JPUMlEqoEglvlhR6Mxnlew/Xq/LRVHpLyk7iK3zODe1qYHIMltO7XGg==} - - '@types/range-parser@1.2.7': - resolution: {integrity: sha512-hKormJbkJqzQGhziax5PItDUTMAM9uE2XXQmM37dyd4hVM+5aVl7oVxMVUiVQn2oCQFN/LKCZdvSM0pFRqbSmQ==} - - '@types/retry@0.12.2': - resolution: {integrity: sha512-XISRgDJ2Tc5q4TRqvgJtzsRkFYNJzZrhTdtMoGVBttwzzQJkPnS3WWTFc7kuDRoPtPakl+T+OfdEUjYJj7Jbow==} - - '@types/send@0.17.4': - resolution: {integrity: sha512-x2EM6TJOybec7c52BX0ZspPodMsQUd5L6PRwOunVyVUhXiBSKf3AezDL8Dgvgt5o0UfKNfuA0eMLr2wLT4AiBA==} - - '@types/serve-static@1.15.7': - resolution: {integrity: sha512-W8Ym+h8nhuRwaKPaDw34QUkwsGi6Rc4yYqvKFo5rm2FUEhCFbzVWrxXUxuKK8TASjWsysJY0nsmNCGhCOIsrOw==} - - '@uppy/aws-s3-multipart@4.0.0': - resolution: {integrity: sha512-tW7W1cBh/99PhYaYDQ4Bo5h0aCIjnCgx/n6d6h7OJMu4NRD703DV/D8AHKIRl5JgecDekiTbApy2iu3q4uKkyw==} - deprecated: Use @uppy/aws-s3 instead. - - '@uppy/aws-s3@4.1.0': - resolution: {integrity: sha512-xRip1Lo3He+3J3fP/SooEFQJKWMCVADTl8J375PzvpaeNnDFKa6W2XLEEl/fGy/K7vI4sH8Znz4+omdtSFCPSQ==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/companion-client@4.1.0': - resolution: {integrity: sha512-nQ8CQfZcYVBNtFQ6ePj7FDIq38DXlH0YpzP/91LR9gnDVISJKKUuvWfr6tPktj1lRw9FZV8jLmlMKT2ituVKiw==} - peerDependencies: - '@uppy/core': ^4.2.0 - - '@uppy/companion@5.1.0': - resolution: {integrity: sha512-8VBlmQHFfmu90eg7JbDLMJ7yjg70vd+Oo3Ywp3C8xI7Lbe4wAV6xm5Xd5wQEIC1S90abjxutGDA5JaQDGYJp7g==} - engines: {node: ^18.20.0 || ^20.15.0 || >=22.0.0} - hasBin: true - - '@uppy/core@4.2.0': - resolution: {integrity: sha512-/oQ2m/xubGfANR0UfMqYFR2mT94OpuXTp9N2cQLIQmWYZtpvfX2gyNBFtQJA3Njqpmox1RfIhOAsVFFuhYVa+Q==} - - '@uppy/store-default@4.1.0': - resolution: {integrity: sha512-z5VSc4PNXpAtrrUPg5hdKJO5Ul7u4ZYLyK+tYzvEgzgR4nLVZmpGzj/d4N90jXpUqEibWKXvevODEB5VlTLHzg==} - - '@uppy/utils@6.0.2': - resolution: {integrity: sha512-ZoNeAa1YTKSlcvXe1SP3POjzjRZ9jSojorbst03vwd1Ks9vHPGf6pne61DowTXHZ3HMj1vpcIaQ1VIEWeeADlA==} - - accepts@1.3.8: - resolution: {integrity: sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==} - engines: {node: '>= 0.6'} - - array-flatten@1.1.1: - resolution: {integrity: sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==} - - asn1.js@5.4.1: - resolution: {integrity: sha512-+I//4cYPccV8LdmBLiX8CYvf9Sp3vQsrqu2QNXRcrbiWvcx/UdlFiqUJJzxRQxgsZmvhXhn4cSKeSmoFjVdupA==} - - base64-js@1.5.1: - resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==} - - basic-auth@2.0.1: - resolution: {integrity: sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==} - engines: {node: '>= 0.8'} - - bintrees@1.0.2: - resolution: {integrity: sha512-VOMgTMwjAaUG580SXn3LacVgjurrbMme7ZZNYGSSV7mmtY6QQRh0Eg3pwIcntQ77DErK1L0NxkbetjcoXzVwKw==} - - bn.js@4.12.0: - resolution: {integrity: sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==} - - body-parser@1.20.2: - resolution: {integrity: sha512-ml9pReCu3M61kGlqoTm2umSXTlRTuGTx0bfYj+uIUKKYycG5NtSbeetV3faSU6R7ajOPw0g/J1PvK4qNy7s5bA==} - engines: {node: '>= 0.8', npm: 1.2.8000 || >= 1.4.16} - - bowser@2.11.0: - resolution: {integrity: sha512-AlcaJBi/pqqJBIQ8U9Mcpc9i8Aqxn88Skv5d+xBX006BY5u8N3mGLHa5Lgppa7L/HfwgwLgZ6NYs+Ag6uUmJRA==} - - brorand@1.1.0: - resolution: {integrity: sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w==} - - buffer-equal-constant-time@1.0.1: - resolution: {integrity: sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA==} - - buffer-from@1.1.2: - resolution: {integrity: sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==} - - buffer@5.6.0: - resolution: {integrity: sha512-/gDYp/UtU0eA1ys8bOs9J6a+E/KWIY+DZ+Q2WESNUA0jFRsJOc0SNUO6xJ5SGA1xueg3NL65W6s+NY5l9cunuw==} - - bytes@3.1.2: - resolution: {integrity: sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==} - engines: {node: '>= 0.8'} - - cacheable-lookup@7.0.0: - resolution: {integrity: sha512-+qJyx4xiKra8mZrcwhjMRMUhD5NR1R8esPkzIYxX96JiecFoxAXFuz/GpR3+ev4PE1WamHip78wV0vcmPQtp8w==} - engines: {node: '>=14.16'} - - cacheable-request@10.2.14: - resolution: {integrity: sha512-zkDT5WAF4hSSoUgyfg5tFIxz8XQK+25W/TLVojJTMKBaxevLBBtLxgqguAuVQB8PVW79FVjHcU+GJ9tVbDZ9mQ==} - engines: {node: '>=14.16'} - - call-bind@1.0.7: - resolution: {integrity: sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==} - engines: {node: '>= 0.4'} - - cluster-key-slot@1.1.2: - resolution: {integrity: sha512-RMr0FhtfXemyinomL4hrWcYJxmX6deFdCxpJzhDttxgO1+bcCnkk+9drydLVDmAMG7NE6aN/fl4F7ucU/90gAA==} - engines: {node: '>=0.10.0'} - - combine-errors@3.0.3: - resolution: {integrity: sha512-C8ikRNRMygCwaTx+Ek3Yr+OuZzgZjduCOfSQBjbM8V3MfgcjSTeto/GXP6PAwKvJz/v15b7GHZvx5rOlczFw/Q==} - - common-tags@1.8.2: - resolution: {integrity: sha512-gk/Z852D2Wtb//0I+kRFNKKE9dIIVirjoqPoA1wJU+XePVXZfGeBpk45+A1rKO4Q43prqWBNY/MiIeRLbPWUaA==} - engines: {node: '>=4.0.0'} - - connect-redis@7.1.1: - resolution: {integrity: sha512-M+z7alnCJiuzKa8/1qAYdGUXHYfDnLolOGAUjOioB07pP39qxjG+X9ibsud7qUBc4jMV5Mcy3ugGv8eFcgamJQ==} - engines: {node: '>=16'} - peerDependencies: - express-session: '>=1' - - content-disposition@0.5.4: - resolution: {integrity: sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==} - engines: {node: '>= 0.6'} - - content-type@1.0.5: - resolution: {integrity: sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==} - engines: {node: '>= 0.6'} - - cookie-parser@1.4.6: - resolution: {integrity: sha512-z3IzaNjdwUC2olLIB5/ITd0/setiaFMLYiZJle7xg5Fe9KWAceil7xszYfHHBtDFYLSgJduS2Ty0P1uJdPDJeA==} - engines: {node: '>= 0.8.0'} - - cookie-signature@1.0.6: - resolution: {integrity: sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==} - - cookie-signature@1.0.7: - resolution: {integrity: sha512-NXdYc3dLr47pBkpUCHtKSwIOQXLVn8dZEuywboCOJY/osA0wFSLlSawr3KN8qXJEyX66FcONTH8EIlVuK0yyFA==} - - cookie-signature@1.2.1: - resolution: {integrity: sha512-78KWk9T26NhzXtuL26cIJ8/qNHANyJ/ZYrmEXFzUmhZdjpBv+DlWlOANRTGBt48YcyslsLrj0bMLFTmXvLRCOw==} - engines: {node: '>=6.6.0'} - - cookie@0.4.1: - resolution: {integrity: sha512-ZwrFkGJxUR3EIoXtO+yVE69Eb7KlixbaeAWfBQB9vVsNn/o+Yw69gBWSSDK825hQNdN+wF8zELf3dFNl/kxkUA==} - engines: {node: '>= 0.6'} - - cookie@0.5.0: - resolution: {integrity: sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==} - engines: {node: '>= 0.6'} - - cookie@0.6.0: - resolution: {integrity: sha512-U71cyTamuh1CRNCfpGY6to28lxvNwPG4Guz/EVjgf3Jmzv0vlDp1atT9eS5dDjMYHucpHbWns6Lwf3BKz6svdw==} - engines: {node: '>= 0.6'} - - cors@2.8.5: - resolution: {integrity: sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==} - engines: {node: '>= 0.10'} - - cron-parser@4.9.0: - resolution: {integrity: sha512-p0SaNjrHOnQeR8/VnfGbmg9te2kfyYSQ7Sc/j/6DtPL3JQvKxmjO9TSjNFpujqV3vEYYBvNNvXSxzyksBWAx1Q==} - engines: {node: '>=12.0.0'} - - custom-error-instance@2.1.1: - resolution: {integrity: sha512-p6JFxJc3M4OTD2li2qaHkDCw9SfMw82Ldr6OC9Je1aXiGfhx2W8p3GaoeaGrPJTUN9NirTM/KTxHWMUdR1rsUg==} - - debug@2.6.9: - resolution: {integrity: sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==} - peerDependencies: - supports-color: '*' - peerDependenciesMeta: - supports-color: - optional: true - - debug@4.3.6: - resolution: {integrity: sha512-O/09Bd4Z1fBrU4VzkhFqVgpPzaGbw6Sm9FEkBT1A/YBXQFGuuSxa1dN2nxgxS34JmKXqYx8CZAwEVoJFImUXIg==} - engines: {node: '>=6.0'} - peerDependencies: - supports-color: '*' - peerDependenciesMeta: - supports-color: - optional: true - - decompress-response@6.0.0: - resolution: {integrity: sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==} - engines: {node: '>=10'} - - defer-to-connect@2.0.1: - resolution: {integrity: sha512-4tvttepXG1VaYGrRibk5EwJd1t4udunSOVMdLSAL6mId1ix438oPwPZMALY41FCijukO1L0twNcGsdzS7dHgDg==} - engines: {node: '>=10'} - - define-data-property@1.1.4: - resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==} - engines: {node: '>= 0.4'} - - denque@2.1.0: - resolution: {integrity: sha512-HVQE3AAb/pxF8fQAoiqpvg9i3evqug3hoiwakOyZAwJm+6vZehbkYXZ0l4JxS+I3QxM97v5aaRNhj8v5oBhekw==} - engines: {node: '>=0.10'} - - depd@2.0.0: - resolution: {integrity: sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==} - engines: {node: '>= 0.8'} - - destroy@1.2.0: - resolution: {integrity: sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==} - engines: {node: '>= 0.8', npm: 1.2.8000 || >= 1.4.16} - - dotenv@16.4.5: - resolution: {integrity: sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==} - engines: {node: '>=12'} - - ecdsa-sig-formatter@1.0.11: - resolution: {integrity: sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==} - - ee-first@1.1.1: - resolution: {integrity: sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==} - - elliptic@6.5.7: - resolution: {integrity: sha512-ESVCtTwiA+XhY3wyh24QqRGBoP3rEdDUl3EDUUo9tft074fi19IrdpH7hLCMMP3CIj7jb3W96rn8lt/BqIlt5Q==} - - encodeurl@1.0.2: - resolution: {integrity: sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==} - engines: {node: '>= 0.8'} - - es-define-property@1.0.0: - resolution: {integrity: sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==} - engines: {node: '>= 0.4'} - - es-errors@1.3.0: - resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==} - engines: {node: '>= 0.4'} - - escape-goat@3.0.0: - resolution: {integrity: sha512-w3PwNZJwRxlp47QGzhuEBldEqVHHhh8/tIPcl6ecf2Bou99cdAt0knihBV0Ecc7CGxYduXVBDheH1K2oADRlvw==} - engines: {node: '>=10'} - - escape-html@1.0.3: - resolution: {integrity: sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==} - - escape-string-regexp@4.0.0: - resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} - engines: {node: '>=10'} - - etag@1.8.1: - resolution: {integrity: sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==} - engines: {node: '>= 0.6'} - - events@3.3.0: - resolution: {integrity: sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==} - engines: {node: '>=0.8.x'} - - express-interceptor@1.2.0: - resolution: {integrity: sha512-fCbcJv8ZwabDg0M/3PmHUxfr/WKHGMpAicR9TfGdhANV4M1GBDSrBTenHIK3aegyRN5S6eDwlvyNFiLynnc19w==} - - express-prom-bundle@7.0.0: - resolution: {integrity: sha512-VwVaCyGBGHkHdecpTqRdW1Jm2fXK8weCUKjGjNWorc9g4M+cZ3xoj+N9uQzfRWfIPXJG5QOaiAziOIalQzMwgA==} - engines: {node: '>=18'} - peerDependencies: - prom-client: '>=15.0.0' - - express-session@1.18.0: - resolution: {integrity: sha512-m93QLWr0ju+rOwApSsyso838LQwgfs44QtOP/WBiwtAgPIo/SAh1a5c6nn2BR6mFNZehTpqKDESzP+fRHVbxwQ==} - engines: {node: '>= 0.8.0'} - - express@4.19.2: - resolution: {integrity: sha512-5T6nhjsT+EOMzuck8JjBHARTHfMht0POzlA60WV2pMD3gyXw2LZnZ+ueGdNxG+0calOJcWKbpFcuzLZ91YWq9Q==} - engines: {node: '>= 0.10.0'} - - fast-safe-stringify@2.1.1: - resolution: {integrity: sha512-W+KJc2dmILlPplD/H4K9l9LcAHAfPtP6BY84uVLXQ6Evcz9Lcg33Y2z1IVblT6xdY54PXYVHEv+0Wpq8Io6zkA==} - - fast-xml-parser@4.4.1: - resolution: {integrity: sha512-xkjOecfnKGkSsOwtZ5Pz7Us/T6mrbPQrq0nh+aCO5V9nk5NLWmasAHumTKjiPJPWANe+kAZ84Jc8ooJkzZ88Sw==} - hasBin: true - - finalhandler@1.2.0: - resolution: {integrity: sha512-5uXcUVftlQMFnWC9qu/svkWv3GTd2PfUhK/3PLkYNAe7FbqJMt3515HaxE6eRL74GdsriiwujiawdaB1BpEISg==} - engines: {node: '>= 0.8'} - - form-data-encoder@2.1.4: - resolution: {integrity: sha512-yDYSgNMraqvnxiEXO4hi88+YZxaHC6QKzb5N84iRCTDeRO7ZALpir/lVmf/uXUhnwUr2O4HU8s/n6x+yNjQkHw==} - engines: {node: '>= 14.17'} - - forwarded@0.2.0: - resolution: {integrity: sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==} - engines: {node: '>= 0.6'} - - fresh@0.5.2: - resolution: {integrity: sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==} - engines: {node: '>= 0.6'} - - function-bind@1.1.2: - resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} - - get-intrinsic@1.2.4: - resolution: {integrity: sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==} - engines: {node: '>= 0.4'} - - get-stream@6.0.1: - resolution: {integrity: sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==} - engines: {node: '>=10'} - - gopd@1.0.1: - resolution: {integrity: sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==} - - got@13.0.0: - resolution: {integrity: sha512-XfBk1CxOOScDcMr9O1yKkNaQyy865NbYs+F7dr4H0LZMVgCj2Le59k6PqbNHoL5ToeaEQUYh6c6yMfVcc6SJxA==} - engines: {node: '>=16'} - - graceful-fs@4.2.11: - resolution: {integrity: sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==} - - grant@5.4.22: - resolution: {integrity: sha512-DEi+/JjXT84mmFYhSmv+SX14v+3Z7vuCIYAMwtdPCTXHMSLhWqSYqWAMXDUQZuV7yaJv2d84AYnkCFNooLKBsA==} - engines: {node: '>=12.0.0'} - - has-flag@4.0.0: - resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} - engines: {node: '>=8'} - - has-property-descriptors@1.0.2: - resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==} - - has-proto@1.0.3: - resolution: {integrity: sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==} - engines: {node: '>= 0.4'} - - has-symbols@1.0.3: - resolution: {integrity: sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==} - engines: {node: '>= 0.4'} - - hash.js@1.1.7: - resolution: {integrity: sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==} - - hasown@2.0.2: - resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} - engines: {node: '>= 0.4'} - - helmet@7.1.0: - resolution: {integrity: sha512-g+HZqgfbpXdCkme/Cd/mZkV0aV3BZZZSugecH03kl38m/Kmdx8jKjBikpDj2cr+Iynv4KpYEviojNdTJActJAg==} - engines: {node: '>=16.0.0'} - - hmac-drbg@1.0.1: - resolution: {integrity: sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==} - - http-cache-semantics@4.1.1: - resolution: {integrity: sha512-er295DKPVsV82j5kw1Gjt+ADA/XYHsajl82cGNQG2eyoPkvgUhX+nDIyelzhIWbbsXP39EHcI6l5tYs2FYqYXQ==} - - http-errors@2.0.0: - resolution: {integrity: sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==} - engines: {node: '>= 0.8'} - - http2-wrapper@2.2.1: - resolution: {integrity: sha512-V5nVw1PAOgfI3Lmeaj2Exmeg7fenjhRUgz1lPSezy1CuhPYbgQtbQj4jZfEAEMlaL+vupsvhjqCyjzob0yxsmQ==} - engines: {node: '>=10.19.0'} - - iconv-lite@0.4.24: - resolution: {integrity: sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==} - engines: {node: '>=0.10.0'} - - ieee754@1.2.1: - resolution: {integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==} - - inherits@2.0.4: - resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} - - ioredis@5.4.1: - resolution: {integrity: sha512-2YZsvl7jopIa1gaePkeMtd9rAcSjOOjPtpcLlOeusyO+XH2SK5ZcT+UCrElPP+WVIInh2TzeI4XW9ENaSLVVHA==} - engines: {node: '>=12.22.0'} - - ipaddr.js@1.9.1: - resolution: {integrity: sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==} - engines: {node: '>= 0.10'} - - ipaddr.js@2.2.0: - resolution: {integrity: sha512-Ag3wB2o37wslZS19hZqorUnrnzSkpOVy+IiiDEiTqNubEYpYuHWIf6K4psgN2ZWKExS4xhVCrRVfb/wfW8fWJA==} - engines: {node: '>= 10'} - - is-network-error@1.1.0: - resolution: {integrity: sha512-tUdRRAnhT+OtCZR/LxZelH/C7QtjtFrTu5tXCA8pl55eTUElUHT+GPYV8MBMBvea/j+NxQqVt3LbWMRir7Gx9g==} - engines: {node: '>=16'} - - is-stream@2.0.1: - resolution: {integrity: sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==} - engines: {node: '>=8'} - - js-base64@3.7.7: - resolution: {integrity: sha512-7rCnleh0z2CkXhH67J8K1Ytz0b2Y+yxTPL+/KOJoa20hfnVQ/3/T6W/KflYI4bRHRagNeXeU2bkNGI3v1oS/lw==} - - js-tokens@4.0.0: - resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} - - json-buffer@3.0.1: - resolution: {integrity: sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==} - - jsonwebtoken@9.0.2: - resolution: {integrity: sha512-PRp66vJ865SSqOlgqS8hujT5U4AOgMfhrwYIuIhfKaoSCZcirrmASQr8CX7cUg+RMih+hgznrjp99o+W4pJLHQ==} - engines: {node: '>=12', npm: '>=6'} - - jwa@1.4.1: - resolution: {integrity: sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==} - - jwa@2.0.0: - resolution: {integrity: sha512-jrZ2Qx916EA+fq9cEAeCROWPTfCwi1IVHqT2tapuqLEVVDKFDENFw1oL+MwrTvH6msKxsd1YTDVw6uKEcsrLEA==} - - jwk-to-pem@2.0.6: - resolution: {integrity: sha512-zPC/5vjyR08TpknpTGW6Z3V3lDf9dU92oHbf0jJlG8tGOzslF9xk2UiO/seSx2llCUrNAe+AvmuGTICSXiYU7A==} - - jws@3.2.2: - resolution: {integrity: sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==} - - jws@4.0.0: - resolution: {integrity: sha512-KDncfTmOZoOMTFG4mBlG0qUIOlc03fmzH+ru6RgYVZhPkyiy/92Owlt/8UEN+a4TXR1FQetfIpJE8ApdvdVxTg==} - - keyv@4.5.4: - resolution: {integrity: sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==} - - lodash._baseiteratee@4.7.0: - resolution: {integrity: sha512-nqB9M+wITz0BX/Q2xg6fQ8mLkyfF7MU7eE+MNBNjTHFKeKaZAPEzEg+E8LWxKWf1DQVflNEn9N49yAuqKh2mWQ==} - - lodash._basetostring@4.12.0: - resolution: {integrity: sha512-SwcRIbyxnN6CFEEK4K1y+zuApvWdpQdBHM/swxP962s8HIxPO3alBH5t3m/dl+f4CMUug6sJb7Pww8d13/9WSw==} - - lodash._baseuniq@4.6.0: - resolution: {integrity: sha512-Ja1YevpHZctlI5beLA7oc5KNDhGcPixFhcqSiORHNsp/1QTv7amAXzw+gu4YOvErqVlMVyIJGgtzeepCnnur0A==} - - lodash._createset@4.0.3: - resolution: {integrity: sha512-GTkC6YMprrJZCYU3zcqZj+jkXkrXzq3IPBcF/fIPpNEAB4hZEtXU8zp/RwKOvZl43NUmwDbyRk3+ZTbeRdEBXA==} - - lodash._root@3.0.1: - resolution: {integrity: sha512-O0pWuFSK6x4EXhM1dhZ8gchNtG7JMqBtrHdoUFUWXD7dJnNSUze1GuyQr5sOs0aCvgGeI3o/OJW8f4ca7FDxmQ==} - - lodash._stringtopath@4.8.0: - resolution: {integrity: sha512-SXL66C731p0xPDC5LZg4wI5H+dJo/EO4KTqOMwLYCH3+FmmfAKJEZCm6ohGpI+T1xwsDsJCfL4OnhorllvlTPQ==} - - lodash.defaults@4.2.0: - resolution: {integrity: sha512-qjxPLHd3r5DnsdGacqOMU6pb/avJzdh9tFX2ymgoZE27BmjXrNy/y4LoaiTeAb+O3gL8AfpJGtqfX/ae2leYYQ==} - - lodash.includes@4.3.0: - resolution: {integrity: sha512-W3Bx6mdkRTGtlJISOvVD/lbqjTlPPUDTMnlXZFnVwi9NKJ6tiAk6LVdlhZMm17VZisqhKcgzpO5Wz91PCt5b0w==} - - lodash.isarguments@3.1.0: - resolution: {integrity: sha512-chi4NHZlZqZD18a0imDHnZPrDeBbTtVN7GXMwuGdRH9qotxAjYs3aVLKc7zNOG9eddR5Ksd8rvFEBc9SsggPpg==} - - lodash.isboolean@3.0.3: - resolution: {integrity: sha512-Bz5mupy2SVbPHURB98VAcw+aHh4vRV5IPNhILUCsOzRmsTmSQ17jIuqopAentWoehktxGd9e/hbIXq980/1QJg==} - - lodash.isinteger@4.0.4: - resolution: {integrity: sha512-DBwtEWN2caHQ9/imiNeEA5ys1JoRtRfY3d7V9wkqtbycnAmTvRRmbHKDV4a0EYc678/dia0jrte4tjYwVBaZUA==} - - lodash.isnumber@3.0.3: - resolution: {integrity: sha512-QYqzpfwO3/CWf3XP+Z+tkQsfaLL/EnUlXWVkIk5FUPc4sBdTehEqZONuyRt2P67PXAk+NXmTBcc97zw9t1FQrw==} - - lodash.isplainobject@4.0.6: - resolution: {integrity: sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==} - - lodash.isstring@4.0.1: - resolution: {integrity: sha512-0wJxfxH1wgO3GrbuP+dTTk7op+6L41QCXbGINEmD+ny/G/eCqGzxyCsh7159S+mgDDcoarnBw6PC1PS5+wUGgw==} - - lodash.once@4.1.1: - resolution: {integrity: sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg==} - - lodash.throttle@4.1.1: - resolution: {integrity: sha512-wIkUCfVKpVsWo3JSZlc+8MB5it+2AN5W8J7YVMST30UrvcQNZ1Okbj+rbVniijTWE6FGYy4XJq/rHkas8qJMLQ==} - - lodash.uniqby@4.5.0: - resolution: {integrity: sha512-IRt7cfTtHy6f1aRVA5n7kT8rgN3N1nH6MOWLcHfpWG2SH19E3JksLK38MktLxZDhlAjCP9jpIXkOnRXlu6oByQ==} - - lodash@4.17.21: - resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==} - - long-timeout@0.1.1: - resolution: {integrity: sha512-BFRuQUqc7x2NWxfJBCyUrN8iYUYznzL9JROmRz1gZ6KlOIgmoD+njPVbb+VNn2nGMKggMsK79iUNErillsrx7w==} - - loose-envify@1.4.0: - resolution: {integrity: sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==} - hasBin: true - - lowercase-keys@3.0.0: - resolution: {integrity: sha512-ozCC6gdQ+glXOQsveKD0YsDy8DSQFjDTz4zyzEHNV5+JP5D62LmfDZ6o1cycFx9ouG940M5dE8C8CTewdj2YWQ==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - lru-cache@4.1.5: - resolution: {integrity: sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==} - - luxon@3.5.0: - resolution: {integrity: sha512-rh+Zjr6DNfUYR3bPwJEnuwDdqMbxZW7LOQfUN4B54+Cl+0o5zaU9RJ6bcidfDtC1cWCZXQ+nvX8bf6bAji37QQ==} - engines: {node: '>=12'} - - media-typer@0.3.0: - resolution: {integrity: sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==} - engines: {node: '>= 0.6'} - - memorystore@1.6.7: - resolution: {integrity: sha512-OZnmNY/NDrKohPQ+hxp0muBcBKrzKNtHr55DbqSx9hLsYVNnomSAMRAtI7R64t3gf3ID7tHQA7mG4oL3Hu9hdw==} - engines: {node: '>=0.10'} - - merge-descriptors@1.0.1: - resolution: {integrity: sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w==} - - methods@1.1.2: - resolution: {integrity: sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==} - engines: {node: '>= 0.6'} - - mime-db@1.52.0: - resolution: {integrity: sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==} - engines: {node: '>= 0.6'} - - mime-match@1.0.2: - resolution: {integrity: sha512-VXp/ugGDVh3eCLOBCiHZMYWQaTNUHv2IJrut+yXA6+JbLPXHglHwfS/5A5L0ll+jkCY7fIzRJcH6OIunF+c6Cg==} - - mime-types@2.1.35: - resolution: {integrity: sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==} - engines: {node: '>= 0.6'} - - mime@1.6.0: - resolution: {integrity: sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==} - engines: {node: '>=4'} - hasBin: true - - mimic-response@3.1.0: - resolution: {integrity: sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==} - engines: {node: '>=10'} - - mimic-response@4.0.0: - resolution: {integrity: sha512-e5ISH9xMYU0DzrT+jl8q2ze9D6eWBto+I8CNpe+VI+K2J/F/k3PdkdTdz4wvGVH4NTpo+NRYTVIuMQEMMcsLqg==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} - - minimalistic-assert@1.0.1: - resolution: {integrity: sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==} - - minimalistic-crypto-utils@1.0.1: - resolution: {integrity: sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg==} - - moment-timezone@0.5.45: - resolution: {integrity: sha512-HIWmqA86KcmCAhnMAN0wuDOARV/525R2+lOLotuGFzn4HO+FH+/645z2wx0Dt3iDv6/p61SIvKnDstISainhLQ==} - - moment@2.30.1: - resolution: {integrity: sha512-uEmtNhbDOrWPFS+hdjFCBfy9f2YoyzRpwcl+DqpC6taX21FzsTLQVbMV/W7PzNSX6x/bhC1zA3c2UQ5NzH6how==} - - morgan@1.10.0: - resolution: {integrity: sha512-AbegBVI4sh6El+1gNwvD5YIck7nSA36weD7xvIxG4in80j/UoK8AEGaWnnz8v1GxonMCltmlNs5ZKbGvl9b1XQ==} - engines: {node: '>= 0.8.0'} - - ms@2.0.0: - resolution: {integrity: sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==} - - ms@2.1.2: - resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==} - - ms@2.1.3: - resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} - - namespace-emitter@2.0.1: - resolution: {integrity: sha512-N/sMKHniSDJBjfrkbS/tpkPj4RAbvW3mr8UAzvlMHyun93XEm83IAvhWtJVHo+RHn/oO8Job5YN4b+wRjSVp5g==} - - nanoid@5.0.7: - resolution: {integrity: sha512-oLxFY2gd2IqnjcYyOXD8XGCftpGtZP2AbHbOkthDkvRywH5ayNtPVy9YlOPcHckXzbLTCHpkb7FB+yuxKV13pQ==} - engines: {node: ^18 || >=20} - hasBin: true - - negotiator@0.6.3: - resolution: {integrity: sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==} - engines: {node: '>= 0.6'} - - node-schedule@2.1.1: - resolution: {integrity: sha512-OXdegQq03OmXEjt2hZP33W2YPs/E5BcFQks46+G2gAxs4gHOIVD1u7EqlYLYSKsaIpyKCK9Gbk0ta1/gjRSMRQ==} - engines: {node: '>=6'} - - normalize-url@8.0.1: - resolution: {integrity: sha512-IO9QvjUMWxPQQhs60oOu10CRkWCiZzSUkzbXGGV9pviYl1fXYcvkzQ5jV9z8Y6un8ARoVRl4EtC6v6jNqbaJ/w==} - engines: {node: '>=14.16'} - - oauth-sign@0.9.0: - resolution: {integrity: sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==} - - object-assign@4.1.1: - resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} - engines: {node: '>=0.10.0'} - - object-inspect@1.13.2: - resolution: {integrity: sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==} - engines: {node: '>= 0.4'} - - on-finished@2.3.0: - resolution: {integrity: sha512-ikqdkGAAyf/X/gPhXGvfgAytDZtDbr+bkNUJ0N9h5MI/dmdgCs3l6hoHrcUv41sRKew3jIwrp4qQDXiK99Utww==} - engines: {node: '>= 0.8'} - - on-finished@2.4.1: - resolution: {integrity: sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==} - engines: {node: '>= 0.8'} - - on-headers@1.0.2: - resolution: {integrity: sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==} - engines: {node: '>= 0.8'} - - p-cancelable@3.0.0: - resolution: {integrity: sha512-mlVgR3PGuzlo0MmTdk4cXqXWlwQDLnONTAg6sm62XkMJEiRxN3GL3SffkYvqwonbkJBcrI7Uvv5Zh9yjvn2iUw==} - engines: {node: '>=12.20'} - - p-retry@6.2.0: - resolution: {integrity: sha512-JA6nkq6hKyWLLasXQXUrO4z8BUZGUt/LjlJxx8Gb2+2ntodU/SS63YZ8b0LUTbQ8ZB9iwOfhEPhg4ykKnn2KsA==} - engines: {node: '>=16.17'} - - parseurl@1.3.3: - resolution: {integrity: sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==} - engines: {node: '>= 0.8'} - - path-to-regexp@0.1.7: - resolution: {integrity: sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ==} - - preact@10.23.2: - resolution: {integrity: sha512-kKYfePf9rzKnxOAKDpsWhg/ysrHPqT+yQ7UW4JjdnqjFIeNUnNcEJvhuA8fDenxAGWzUqtd51DfVg7xp/8T9NA==} - - prom-client@15.1.2: - resolution: {integrity: sha512-on3h1iXb04QFLLThrmVYg1SChBQ9N1c+nKAjebBjokBqipddH3uxmOUcEkTnzmJ8Jh/5TSUnUqS40i2QB2dJHQ==} - engines: {node: ^16 || ^18 || >=20} - - proper-lockfile@4.1.2: - resolution: {integrity: sha512-TjNPblN4BwAWMXU8s9AEz4JmQxnD1NNL7bNOY/AKUzyamc379FWASUhc/K1pL2noVb+XmZKLL68cjzLsiOAMaA==} - - proxy-addr@2.0.7: - resolution: {integrity: sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==} - engines: {node: '>= 0.10'} - - pseudomap@1.0.2: - resolution: {integrity: sha512-b/YwNhb8lk1Zz2+bXXpS/LK9OisiZZ1SNsSLxN1x2OXVEhW2Ckr/7mWE5vrC1ZTiJlD9g19jWszTmJsB+oEpFQ==} - - qs@6.11.0: - resolution: {integrity: sha512-MvjoMCJwEarSbUYk5O+nmoSzSutSsTwF85zcHPQ9OrlFoZOYIjaqBAJIqIXjptyD5vThxGq52Xu/MaJzRkIk4Q==} - engines: {node: '>=0.6'} - - qs@6.13.0: - resolution: {integrity: sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==} - engines: {node: '>=0.6'} - - querystringify@2.2.0: - resolution: {integrity: sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==} - - quick-lru@5.1.1: - resolution: {integrity: sha512-WuyALRjWPDGtt/wzJiadO5AXY+8hZ80hVpe6MyivgraREW751X3SbhRvG3eLKOYN+8VEvqLcf3wdnt44Z4S4SA==} - engines: {node: '>=10'} - - random-bytes@1.0.0: - resolution: {integrity: sha512-iv7LhNVO047HzYR3InF6pUcUsPQiHTM1Qal51DcGSuZFBil1aBBWG5eHPNek7bvILMaYJ/8RU1e8w1AMdHmLQQ==} - engines: {node: '>= 0.8'} - - randombytes@2.1.0: - resolution: {integrity: sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==} - - range-parser@1.2.1: - resolution: {integrity: sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==} - engines: {node: '>= 0.6'} - - raw-body@2.5.2: - resolution: {integrity: sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==} - engines: {node: '>= 0.8'} - - react-hook-form@7.53.0: - resolution: {integrity: sha512-M1n3HhqCww6S2hxLxciEXy2oISPnAzxY7gvwVPrtlczTM/1dDadXgUxDpHMrMTblDOcm/AXtXxHwZ3jpg1mqKQ==} - engines: {node: '>=18.0.0'} - peerDependencies: - react: ^16.8.0 || ^17 || ^18 || ^19 - - react@18.3.1: - resolution: {integrity: sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ==} - engines: {node: '>=0.10.0'} - - readable-stream@3.6.2: - resolution: {integrity: sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==} - engines: {node: '>= 6'} - - redis-errors@1.2.0: - resolution: {integrity: sha512-1qny3OExCf0UvUV/5wpYKf2YwPcOqXzkwKKSmKHiE6ZMQs5heeE/c8eXK+PNllPvmjgAbfnsbpkGZWy8cBpn9w==} - engines: {node: '>=4'} - - redis-parser@3.0.0: - resolution: {integrity: sha512-DJnGAeenTdpMEH6uAJRK/uiyEIH9WVsUmoLwzudwGJUwZPp80PDBWPHXSAGNPwNvIXAbe7MSUB1zQFugFml66A==} - engines: {node: '>=4'} - - request-compose@2.1.6: - resolution: {integrity: sha512-S07L+2VbJB32WddD/o/PnYGKym63zLVbymygVWXvt8L79VAngcjAxhHaGuFOICLxEV90EasEPzqPKKHPspXP8w==} - engines: {node: '>=12.0.0'} - - request-oauth@1.0.1: - resolution: {integrity: sha512-85THTg1RgOYtqQw42JON6AqvHLptlj1biw265Tsq4fD4cPdUvhDB2Qh9NTv17yCD322ROuO9aOmpc4GyayGVBA==} - engines: {node: '>=8.0.0'} - - requires-port@1.0.0: - resolution: {integrity: sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==} - - resolve-alpn@1.2.1: - resolution: {integrity: sha512-0a1F4l73/ZFZOakJnQ3FvkJ2+gSTQWz/r2KE5OdDY0TxPm5h4GkqkWWfM47T7HsbnOtcJVEF4epCVy6u7Q3K+g==} - - responselike@3.0.0: - resolution: {integrity: sha512-40yHxbNcl2+rzXvZuVkrYohathsSJlMTXKryG5y8uciHv1+xDLHQpgjG64JUO9nrEq2jGLH6IZ8BcZyw3wrweg==} - engines: {node: '>=14.16'} - - retry@0.12.0: - resolution: {integrity: sha512-9LkiTwjUh6rT555DtE9rTX+BKByPfrMzEAtnlEtdEwr3Nkffwiihqe2bWADg+OQRjt9gl6ICdmB/ZFDCGAtSow==} - engines: {node: '>= 4'} - - retry@0.13.1: - resolution: {integrity: sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg==} - engines: {node: '>= 4'} - - safe-buffer@5.1.2: - resolution: {integrity: sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==} - - safe-buffer@5.2.1: - resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} - - safer-buffer@2.1.2: - resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==} - - semver@7.6.3: - resolution: {integrity: sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==} - engines: {node: '>=10'} - hasBin: true - - send@0.18.0: - resolution: {integrity: sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg==} - engines: {node: '>= 0.8.0'} - - serialize-error@11.0.3: - resolution: {integrity: sha512-2G2y++21dhj2R7iHAdd0FIzjGwuKZld+7Pl/bTU6YIkrC2ZMbVUjm+luj6A6V34Rv9XfKJDKpTWu9W4Gse1D9g==} - engines: {node: '>=14.16'} - - serialize-javascript@6.0.2: - resolution: {integrity: sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g==} - - serve-static@1.15.0: - resolution: {integrity: sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g==} - engines: {node: '>= 0.8.0'} - - set-function-length@1.2.2: - resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==} - engines: {node: '>= 0.4'} - - setprototypeof@1.2.0: - resolution: {integrity: sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==} - - side-channel@1.0.6: - resolution: {integrity: sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==} - engines: {node: '>= 0.4'} - - signal-exit@3.0.7: - resolution: {integrity: sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==} - - sorted-array-functions@1.3.0: - resolution: {integrity: sha512-2sqgzeFlid6N4Z2fUQ1cvFmTOLRi/sEDzSQ0OKYchqgoPmQBVyM3959qYx3fpS6Esef80KjmpgPeEr028dP3OA==} - - standard-as-callback@2.1.0: - resolution: {integrity: sha512-qoRRSyROncaz1z0mvYqIE4lCd9p2R90i6GxW3uZv5ucSu8tU7B5HXUP1gG8pVZsYNVaXjk8ClXHPttLyxAL48A==} - - statuses@2.0.1: - resolution: {integrity: sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==} - engines: {node: '>= 0.8'} - - stream-browserify@3.0.0: - resolution: {integrity: sha512-H73RAHsVBapbim0tU2JwwOiXUj+fikfiaoYAKHF3VJfA0pe2BCzkhAHBlLG6REzE+2WNZcxOXjK7lkso+9euLA==} - - string_decoder@1.3.0: - resolution: {integrity: sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==} - - strnum@1.0.5: - resolution: {integrity: sha512-J8bbNyKKXl5qYcR36TIO8W3mVGVHrmmxsd5PAItGkmyzwJvybiw2IVq5nqd0i4LSNSkB/sx9VHllbfFdr9k1JA==} - - supports-color@8.1.1: - resolution: {integrity: sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==} - engines: {node: '>=10'} - - tdigest@0.1.2: - resolution: {integrity: sha512-+G0LLgjjo9BZX2MfdvPfH+MKLCrxlXSYec5DaPYP1fe6Iyhf0/fSmJ0bFiZ1F8BT6cGXl2LpltQptzjXKWEkKA==} - - toidentifier@1.0.1: - resolution: {integrity: sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==} - engines: {node: '>=0.6'} - - tslib@2.7.0: - resolution: {integrity: sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA==} - - tus-js-client@4.1.0: - resolution: {integrity: sha512-e/nC/kJahvNYBcnwcqzuhFIvVELMMpbVXIoOOKdUn74SdQCvJd2JjqV2jZLv2EFOVbV4qLiO0lV7BxBXF21b6Q==} - engines: {node: '>=18'} - - type-fest@2.19.0: - resolution: {integrity: sha512-RAH822pAdBgcNMAfWnCBU3CFZcfZ/i1eZjwFU/dsLKumyuuP3niueg2UAukXYF0E2AAoc82ZSSf9J0WQBinzHA==} - engines: {node: '>=12.20'} - - type-is@1.6.18: - resolution: {integrity: sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==} - engines: {node: '>= 0.6'} - - uid-safe@2.1.5: - resolution: {integrity: sha512-KPHm4VL5dDXKz01UuEd88Df+KzynaohSL9fBh096KWAxSKZQDI2uBrVqtvRM4rwrIrRRKsdLNML/lnaaVSRioA==} - engines: {node: '>= 0.8'} - - undici-types@6.19.8: - resolution: {integrity: sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==} - - unpipe@1.0.0: - resolution: {integrity: sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==} - engines: {node: '>= 0.8'} - - url-parse@1.5.10: - resolution: {integrity: sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ==} - - url-value-parser@2.2.0: - resolution: {integrity: sha512-yIQdxJpgkPamPPAPuGdS7Q548rLhny42tg8d4vyTNzFqvOnwqrgHXvgehT09U7fwrzxi3RxCiXjoNUNnNOlQ8A==} - engines: {node: '>=6.0.0'} - - util-deprecate@1.0.2: - resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==} - - utils-merge@1.0.1: - resolution: {integrity: sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==} - engines: {node: '>= 0.4.0'} - - uuid@8.3.2: - resolution: {integrity: sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==} - hasBin: true - - uuid@9.0.1: - resolution: {integrity: sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==} - hasBin: true - - validator@13.12.0: - resolution: {integrity: sha512-c1Q0mCiPlgdTVVVIJIrBuxNicYE+t/7oKeI9MWLj3fh/uq2Pxh/3eeWbVZ4OcGW1TUf53At0njHw5SMdA3tmMg==} - engines: {node: '>= 0.10'} - - vary@1.1.2: - resolution: {integrity: sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==} - engines: {node: '>= 0.8'} - - wildcard@1.1.2: - resolution: {integrity: sha512-DXukZJxpHA8LuotRwL0pP1+rS6CS7FF2qStDDE1C7DDg2rLud2PXRMuEDYIPhgEezwnlHNL4c+N6MfMTjCGTng==} - - ws@8.17.1: - resolution: {integrity: sha512-6XQFvXTkbfUOZOKKILFG1PDK2NDQs4azKQl26T0YS5CxqWLgXajbPZ+h4gZekJyRqFU8pvnbAbbs/3TgRPy+GQ==} - engines: {node: '>=10.0.0'} - peerDependencies: - bufferutil: ^4.0.1 - utf-8-validate: '>=5.0.2' - peerDependenciesMeta: - bufferutil: - optional: true - utf-8-validate: - optional: true - - yallist@2.1.2: - resolution: {integrity: sha512-ncTzHV7NvsQZkYe1DW7cbDLm0YpzHmZF5r/iyP3ZnQtMiJ+pjzisCiMNI+Sj+xQF5pXhSHxSB3uDbsBTzY/c2A==} - -snapshots: - - '@aws-crypto/crc32@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/crc32c@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/sha1-browser@5.2.0': - dependencies: - '@aws-crypto/supports-web-crypto': 5.2.0 - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-locate-window': 3.568.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-crypto/sha256-browser@5.2.0': - dependencies: - '@aws-crypto/sha256-js': 5.2.0 - '@aws-crypto/supports-web-crypto': 5.2.0 - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-locate-window': 3.568.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-crypto/sha256-js@5.2.0': - dependencies: - '@aws-crypto/util': 5.2.0 - '@aws-sdk/types': 3.609.0 - tslib: 2.7.0 - - '@aws-crypto/supports-web-crypto@5.2.0': - dependencies: - tslib: 2.7.0 - - '@aws-crypto/util@5.2.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/util-utf8': 2.3.0 - tslib: 2.7.0 - - '@aws-sdk/client-s3@3.637.0': - dependencies: - '@aws-crypto/sha1-browser': 5.2.0 - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-bucket-endpoint': 3.620.0 - '@aws-sdk/middleware-expect-continue': 3.620.0 - '@aws-sdk/middleware-flexible-checksums': 3.620.0 - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-location-constraint': 3.609.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-sdk-s3': 3.635.0 - '@aws-sdk/middleware-ssec': 3.609.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/signature-v4-multi-region': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@aws-sdk/xml-builder': 3.609.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/eventstream-serde-browser': 3.0.6 - '@smithy/eventstream-serde-config-resolver': 3.0.3 - '@smithy/eventstream-serde-node': 3.0.5 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-blob-browser': 3.1.2 - '@smithy/hash-node': 3.0.3 - '@smithy/hash-stream-node': 3.1.2 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/md5-js': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-stream': 3.1.3 - '@smithy/util-utf8': 3.0.0 - '@smithy/util-waiter': 3.1.2 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sso@3.637.0': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/core': 3.635.0 - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/client-sts@3.637.0': - dependencies: - '@aws-crypto/sha256-browser': 5.2.0 - '@aws-crypto/sha256-js': 5.2.0 - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/core': 3.635.0 - '@aws-sdk/credential-provider-node': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/middleware-host-header': 3.620.0 - '@aws-sdk/middleware-logger': 3.609.0 - '@aws-sdk/middleware-recursion-detection': 3.620.0 - '@aws-sdk/middleware-user-agent': 3.637.0 - '@aws-sdk/region-config-resolver': 3.614.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@aws-sdk/util-user-agent-browser': 3.609.0 - '@aws-sdk/util-user-agent-node': 3.614.0 - '@smithy/config-resolver': 3.0.5 - '@smithy/core': 2.4.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/hash-node': 3.0.3 - '@smithy/invalid-dependency': 3.0.3 - '@smithy/middleware-content-length': 3.0.5 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/middleware-stack': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-base64': 3.0.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-body-length-node': 3.0.0 - '@smithy/util-defaults-mode-browser': 3.0.15 - '@smithy/util-defaults-mode-node': 3.0.15 - '@smithy/util-endpoints': 2.0.5 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/core@3.635.0': - dependencies: - '@smithy/core': 2.4.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-middleware': 3.0.3 - fast-xml-parser: 4.4.1 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-env@3.620.1': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-http@3.635.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-stream': 3.1.3 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-ini@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/credential-provider-env': 3.620.1 - '@aws-sdk/credential-provider-http': 3.635.0 - '@aws-sdk/credential-provider-process': 3.620.1 - '@aws-sdk/credential-provider-sso': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/credential-provider-web-identity': 3.621.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - aws-crt - - '@aws-sdk/credential-provider-node@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/credential-provider-env': 3.620.1 - '@aws-sdk/credential-provider-http': 3.635.0 - '@aws-sdk/credential-provider-ini': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/credential-provider-process': 3.620.1 - '@aws-sdk/credential-provider-sso': 3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/credential-provider-web-identity': 3.621.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - '@aws-sdk/client-sts' - - aws-crt - - '@aws-sdk/credential-provider-process@3.620.1': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/credential-provider-sso@3.637.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))': - dependencies: - '@aws-sdk/client-sso': 3.637.0 - '@aws-sdk/token-providers': 3.614.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0)) - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - transitivePeerDependencies: - - '@aws-sdk/client-sso-oidc' - - aws-crt - - '@aws-sdk/credential-provider-web-identity@3.621.0(@aws-sdk/client-sts@3.637.0)': - dependencies: - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/lib-storage@3.637.0(@aws-sdk/client-s3@3.637.0)': - dependencies: - '@aws-sdk/client-s3': 3.637.0 - '@smithy/abort-controller': 3.1.1 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/smithy-client': 3.2.0 - buffer: 5.6.0 - events: 3.3.0 - stream-browserify: 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-bucket-endpoint@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-arn-parser': 3.568.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-expect-continue@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-flexible-checksums@3.620.0': - dependencies: - '@aws-crypto/crc32': 5.2.0 - '@aws-crypto/crc32c': 5.2.0 - '@aws-sdk/types': 3.609.0 - '@smithy/is-array-buffer': 3.0.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-host-header@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-location-constraint@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-logger@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-recursion-detection@3.620.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-sdk-s3@3.635.0': - dependencies: - '@aws-sdk/core': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-arn-parser': 3.568.0 - '@smithy/core': 2.4.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-stream': 3.1.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-ssec@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/middleware-user-agent@3.637.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-endpoints': 3.637.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/region-config-resolver@3.614.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@aws-sdk/s3-presigned-post@3.637.0': - dependencies: - '@aws-sdk/client-s3': 3.637.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-format-url': 3.609.0 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-hex-encoding': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - transitivePeerDependencies: - - aws-crt - - '@aws-sdk/s3-request-presigner@3.637.0': - dependencies: - '@aws-sdk/signature-v4-multi-region': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@aws-sdk/util-format-url': 3.609.0 - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/signature-v4-multi-region@3.635.0': - dependencies: - '@aws-sdk/middleware-sdk-s3': 3.635.0 - '@aws-sdk/types': 3.609.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/signature-v4': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/token-providers@3.614.0(@aws-sdk/client-sso-oidc@3.637.0(@aws-sdk/client-sts@3.637.0))': - dependencies: - '@aws-sdk/client-sso-oidc': 3.637.0(@aws-sdk/client-sts@3.637.0) - '@aws-sdk/types': 3.609.0 - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/types@3.609.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/util-arn-parser@3.568.0': - dependencies: - tslib: 2.7.0 - - '@aws-sdk/util-endpoints@3.637.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - '@smithy/util-endpoints': 2.0.5 - tslib: 2.7.0 - - '@aws-sdk/util-format-url@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/querystring-builder': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/util-locate-window@3.568.0': - dependencies: - tslib: 2.7.0 - - '@aws-sdk/util-user-agent-browser@3.609.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/types': 3.3.0 - bowser: 2.11.0 - tslib: 2.7.0 - - '@aws-sdk/util-user-agent-node@3.614.0': - dependencies: - '@aws-sdk/types': 3.609.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@aws-sdk/xml-builder@3.609.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@ioredis/commands@1.2.0': {} - - '@opentelemetry/api@1.9.0': {} - - '@sindresorhus/is@5.6.0': {} - - '@smithy/abort-controller@3.1.1': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/chunked-blob-reader-native@3.0.0': - dependencies: - '@smithy/util-base64': 3.0.0 - tslib: 2.7.0 - - '@smithy/chunked-blob-reader@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/config-resolver@3.0.5': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-config-provider': 3.0.0 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@smithy/core@2.4.0': - dependencies: - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-retry': 3.0.15 - '@smithy/middleware-serde': 3.0.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-body-length-browser': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/credential-provider-imds@3.2.0': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - tslib: 2.7.0 - - '@smithy/eventstream-codec@3.1.2': - dependencies: - '@aws-crypto/crc32': 5.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-hex-encoding': 3.0.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-browser@3.0.6': - dependencies: - '@smithy/eventstream-serde-universal': 3.0.5 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-config-resolver@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-node@3.0.5': - dependencies: - '@smithy/eventstream-serde-universal': 3.0.5 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/eventstream-serde-universal@3.0.5': - dependencies: - '@smithy/eventstream-codec': 3.1.2 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/fetch-http-handler@3.2.4': - dependencies: - '@smithy/protocol-http': 4.1.0 - '@smithy/querystring-builder': 3.0.3 - '@smithy/types': 3.3.0 - '@smithy/util-base64': 3.0.0 - tslib: 2.7.0 - - '@smithy/hash-blob-browser@3.1.2': - dependencies: - '@smithy/chunked-blob-reader': 3.0.0 - '@smithy/chunked-blob-reader-native': 3.0.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/hash-node@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/hash-stream-node@3.1.2': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/invalid-dependency@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/is-array-buffer@2.2.0': - dependencies: - tslib: 2.7.0 - - '@smithy/is-array-buffer@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/md5-js@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/middleware-content-length@3.0.5': - dependencies: - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/middleware-endpoint@3.1.0': - dependencies: - '@smithy/middleware-serde': 3.0.3 - '@smithy/node-config-provider': 3.1.4 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/url-parser': 3.0.3 - '@smithy/util-middleware': 3.0.3 - tslib: 2.7.0 - - '@smithy/middleware-retry@3.0.15': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/protocol-http': 4.1.0 - '@smithy/service-error-classification': 3.0.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-retry': 3.0.3 - tslib: 2.7.0 - uuid: 9.0.1 - - '@smithy/middleware-serde@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/middleware-stack@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/node-config-provider@3.1.4': - dependencies: - '@smithy/property-provider': 3.1.3 - '@smithy/shared-ini-file-loader': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/node-http-handler@3.1.4': - dependencies: - '@smithy/abort-controller': 3.1.1 - '@smithy/protocol-http': 4.1.0 - '@smithy/querystring-builder': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/property-provider@3.1.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/protocol-http@4.1.0': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/querystring-builder@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - '@smithy/util-uri-escape': 3.0.0 - tslib: 2.7.0 - - '@smithy/querystring-parser@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/service-error-classification@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - - '@smithy/shared-ini-file-loader@3.1.4': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/signature-v4@4.1.0': - dependencies: - '@smithy/is-array-buffer': 3.0.0 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-hex-encoding': 3.0.0 - '@smithy/util-middleware': 3.0.3 - '@smithy/util-uri-escape': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/smithy-client@3.2.0': - dependencies: - '@smithy/middleware-endpoint': 3.1.0 - '@smithy/middleware-stack': 3.0.3 - '@smithy/protocol-http': 4.1.0 - '@smithy/types': 3.3.0 - '@smithy/util-stream': 3.1.3 - tslib: 2.7.0 - - '@smithy/types@3.3.0': - dependencies: - tslib: 2.7.0 - - '@smithy/url-parser@3.0.3': - dependencies: - '@smithy/querystring-parser': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-base64@3.0.0': - dependencies: - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-body-length-browser@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-body-length-node@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-buffer-from@2.2.0': - dependencies: - '@smithy/is-array-buffer': 2.2.0 - tslib: 2.7.0 - - '@smithy/util-buffer-from@3.0.0': - dependencies: - '@smithy/is-array-buffer': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-config-provider@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-defaults-mode-browser@3.0.15': - dependencies: - '@smithy/property-provider': 3.1.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - bowser: 2.11.0 - tslib: 2.7.0 - - '@smithy/util-defaults-mode-node@3.0.15': - dependencies: - '@smithy/config-resolver': 3.0.5 - '@smithy/credential-provider-imds': 3.2.0 - '@smithy/node-config-provider': 3.1.4 - '@smithy/property-provider': 3.1.3 - '@smithy/smithy-client': 3.2.0 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-endpoints@2.0.5': - dependencies: - '@smithy/node-config-provider': 3.1.4 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-hex-encoding@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-middleware@3.0.3': - dependencies: - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-retry@3.0.3': - dependencies: - '@smithy/service-error-classification': 3.0.3 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@smithy/util-stream@3.1.3': - dependencies: - '@smithy/fetch-http-handler': 3.2.4 - '@smithy/node-http-handler': 3.1.4 - '@smithy/types': 3.3.0 - '@smithy/util-base64': 3.0.0 - '@smithy/util-buffer-from': 3.0.0 - '@smithy/util-hex-encoding': 3.0.0 - '@smithy/util-utf8': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-uri-escape@3.0.0': - dependencies: - tslib: 2.7.0 - - '@smithy/util-utf8@2.3.0': - dependencies: - '@smithy/util-buffer-from': 2.2.0 - tslib: 2.7.0 - - '@smithy/util-utf8@3.0.0': - dependencies: - '@smithy/util-buffer-from': 3.0.0 - tslib: 2.7.0 - - '@smithy/util-waiter@3.1.2': - dependencies: - '@smithy/abort-controller': 3.1.1 - '@smithy/types': 3.3.0 - tslib: 2.7.0 - - '@szmarczak/http-timer@5.0.1': - dependencies: - defer-to-connect: 2.0.1 - - '@transloadit/prettier-bytes@0.3.4': {} - - '@types/body-parser@1.19.5': - dependencies: - '@types/connect': 3.4.38 - '@types/node': 22.5.2 - - '@types/connect@3.4.38': - dependencies: - '@types/node': 22.5.2 - - '@types/express-serve-static-core@4.19.5': - dependencies: - '@types/node': 22.5.2 - '@types/qs': 6.9.15 - '@types/range-parser': 1.2.7 - '@types/send': 0.17.4 - - '@types/express@4.17.21': - dependencies: - '@types/body-parser': 1.19.5 - '@types/express-serve-static-core': 4.19.5 - '@types/qs': 6.9.15 - '@types/serve-static': 1.15.7 - - '@types/http-cache-semantics@4.0.4': {} - - '@types/http-errors@2.0.4': {} - - '@types/mime@1.3.5': {} - - '@types/node@22.5.2': - dependencies: - undici-types: 6.19.8 - - '@types/qs@6.9.15': {} - - '@types/range-parser@1.2.7': {} - - '@types/retry@0.12.2': {} - - '@types/send@0.17.4': - dependencies: - '@types/mime': 1.3.5 - '@types/node': 22.5.2 - - '@types/serve-static@1.15.7': - dependencies: - '@types/http-errors': 2.0.4 - '@types/node': 22.5.2 - '@types/send': 0.17.4 - - '@uppy/aws-s3-multipart@4.0.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/aws-s3': 4.1.0(@uppy/core@4.2.0) - transitivePeerDependencies: - - '@uppy/core' - - '@uppy/aws-s3@4.1.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/companion-client': 4.1.0(@uppy/core@4.2.0) - '@uppy/core': 4.2.0 - '@uppy/utils': 6.0.2 - - '@uppy/companion-client@4.1.0(@uppy/core@4.2.0)': - dependencies: - '@uppy/core': 4.2.0 - '@uppy/utils': 6.0.2 - namespace-emitter: 2.0.1 - p-retry: 6.2.0 - - '@uppy/companion@5.1.0': - dependencies: - '@aws-sdk/client-s3': 3.637.0 - '@aws-sdk/client-sts': 3.637.0 - '@aws-sdk/lib-storage': 3.637.0(@aws-sdk/client-s3@3.637.0) - '@aws-sdk/s3-presigned-post': 3.637.0 - '@aws-sdk/s3-request-presigner': 3.637.0 - body-parser: 1.20.2(supports-color@8.1.1) - common-tags: 1.8.2 - connect-redis: 7.1.1(express-session@1.18.0(supports-color@8.1.1)) - content-disposition: 0.5.4 - cookie-parser: 1.4.6 - cors: 2.8.5 - escape-goat: 3.0.0 - escape-string-regexp: 4.0.0 - express: 4.19.2(supports-color@8.1.1) - express-interceptor: 1.2.0(supports-color@8.1.1) - express-prom-bundle: 7.0.0(prom-client@15.1.2)(supports-color@8.1.1) - express-session: 1.18.0(supports-color@8.1.1) - fast-safe-stringify: 2.1.1 - got: 13.0.0 - grant: 5.4.22 - helmet: 7.1.0 - ioredis: 5.4.1(supports-color@8.1.1) - ipaddr.js: 2.2.0 - jsonwebtoken: 9.0.2 - lodash: 4.17.21 - mime-types: 2.1.35 - moment: 2.30.1 - moment-timezone: 0.5.45 - morgan: 1.10.0(supports-color@8.1.1) - ms: 2.1.3 - node-schedule: 2.1.1 - prom-client: 15.1.2 - serialize-error: 11.0.3 - serialize-javascript: 6.0.2 - supports-color: 8.1.1 - tus-js-client: 4.1.0 - validator: 13.12.0 - ws: 8.17.1 - transitivePeerDependencies: - - aws-crt - - bufferutil - - utf-8-validate - - '@uppy/core@4.2.0': - dependencies: - '@transloadit/prettier-bytes': 0.3.4 - '@uppy/store-default': 4.1.0 - '@uppy/utils': 6.0.2 - lodash: 4.17.21 - mime-match: 1.0.2 - namespace-emitter: 2.0.1 - nanoid: 5.0.7 - preact: 10.23.2 - - '@uppy/store-default@4.1.0': {} - - '@uppy/utils@6.0.2': - dependencies: - lodash: 4.17.21 - preact: 10.23.2 - - accepts@1.3.8: - dependencies: - mime-types: 2.1.35 - negotiator: 0.6.3 - - array-flatten@1.1.1: {} - - asn1.js@5.4.1: - dependencies: - bn.js: 4.12.0 - inherits: 2.0.4 - minimalistic-assert: 1.0.1 - safer-buffer: 2.1.2 - optional: true - - base64-js@1.5.1: {} - - basic-auth@2.0.1: - dependencies: - safe-buffer: 5.1.2 - - bintrees@1.0.2: {} - - bn.js@4.12.0: - optional: true - - body-parser@1.20.2(supports-color@8.1.1): - dependencies: - bytes: 3.1.2 - content-type: 1.0.5 - debug: 2.6.9(supports-color@8.1.1) - depd: 2.0.0 - destroy: 1.2.0 - http-errors: 2.0.0 - iconv-lite: 0.4.24 - on-finished: 2.4.1 - qs: 6.11.0 - raw-body: 2.5.2 - type-is: 1.6.18 - unpipe: 1.0.0 - transitivePeerDependencies: - - supports-color - - bowser@2.11.0: {} - - brorand@1.1.0: - optional: true - - buffer-equal-constant-time@1.0.1: {} - - buffer-from@1.1.2: {} - - buffer@5.6.0: - dependencies: - base64-js: 1.5.1 - ieee754: 1.2.1 - - bytes@3.1.2: {} - - cacheable-lookup@7.0.0: {} - - cacheable-request@10.2.14: - dependencies: - '@types/http-cache-semantics': 4.0.4 - get-stream: 6.0.1 - http-cache-semantics: 4.1.1 - keyv: 4.5.4 - mimic-response: 4.0.0 - normalize-url: 8.0.1 - responselike: 3.0.0 - - call-bind@1.0.7: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - set-function-length: 1.2.2 - - cluster-key-slot@1.1.2: {} - - combine-errors@3.0.3: - dependencies: - custom-error-instance: 2.1.1 - lodash.uniqby: 4.5.0 - - common-tags@1.8.2: {} - - connect-redis@7.1.1(express-session@1.18.0(supports-color@8.1.1)): - dependencies: - express-session: 1.18.0(supports-color@8.1.1) - - content-disposition@0.5.4: - dependencies: - safe-buffer: 5.2.1 - - content-type@1.0.5: {} - - cookie-parser@1.4.6: - dependencies: - cookie: 0.4.1 - cookie-signature: 1.0.6 - - cookie-signature@1.0.6: {} - - cookie-signature@1.0.7: {} - - cookie-signature@1.2.1: - optional: true - - cookie@0.4.1: {} - - cookie@0.5.0: - optional: true - - cookie@0.6.0: {} - - cors@2.8.5: - dependencies: - object-assign: 4.1.1 - vary: 1.1.2 - - cron-parser@4.9.0: - dependencies: - luxon: 3.5.0 - - custom-error-instance@2.1.1: {} - - debug@2.6.9(supports-color@8.1.1): - dependencies: - ms: 2.0.0 - optionalDependencies: - supports-color: 8.1.1 - - debug@4.3.6(supports-color@8.1.1): - dependencies: - ms: 2.1.2 - optionalDependencies: - supports-color: 8.1.1 - - decompress-response@6.0.0: - dependencies: - mimic-response: 3.1.0 - - defer-to-connect@2.0.1: {} - - define-data-property@1.1.4: - dependencies: - es-define-property: 1.0.0 - es-errors: 1.3.0 - gopd: 1.0.1 - - denque@2.1.0: {} - - depd@2.0.0: {} - - destroy@1.2.0: {} - - dotenv@16.4.5: {} - - ecdsa-sig-formatter@1.0.11: - dependencies: - safe-buffer: 5.2.1 - - ee-first@1.1.1: {} - - elliptic@6.5.7: - dependencies: - bn.js: 4.12.0 - brorand: 1.1.0 - hash.js: 1.1.7 - hmac-drbg: 1.0.1 - inherits: 2.0.4 - minimalistic-assert: 1.0.1 - minimalistic-crypto-utils: 1.0.1 - optional: true - - encodeurl@1.0.2: {} - - es-define-property@1.0.0: - dependencies: - get-intrinsic: 1.2.4 - - es-errors@1.3.0: {} - - escape-goat@3.0.0: {} - - escape-html@1.0.3: {} - - escape-string-regexp@4.0.0: {} - - etag@1.8.1: {} - - events@3.3.0: {} - - express-interceptor@1.2.0(supports-color@8.1.1): - dependencies: - debug: 2.6.9(supports-color@8.1.1) - transitivePeerDependencies: - - supports-color - - express-prom-bundle@7.0.0(prom-client@15.1.2)(supports-color@8.1.1): - dependencies: - '@types/express': 4.17.21 - express: 4.19.2(supports-color@8.1.1) - on-finished: 2.4.1 - prom-client: 15.1.2 - url-value-parser: 2.2.0 - transitivePeerDependencies: - - supports-color - - express-session@1.18.0(supports-color@8.1.1): - dependencies: - cookie: 0.6.0 - cookie-signature: 1.0.7 - debug: 2.6.9(supports-color@8.1.1) - depd: 2.0.0 - on-headers: 1.0.2 - parseurl: 1.3.3 - safe-buffer: 5.2.1 - uid-safe: 2.1.5 - transitivePeerDependencies: - - supports-color - - express@4.19.2(supports-color@8.1.1): - dependencies: - accepts: 1.3.8 - array-flatten: 1.1.1 - body-parser: 1.20.2(supports-color@8.1.1) - content-disposition: 0.5.4 - content-type: 1.0.5 - cookie: 0.6.0 - cookie-signature: 1.0.6 - debug: 2.6.9(supports-color@8.1.1) - depd: 2.0.0 - encodeurl: 1.0.2 - escape-html: 1.0.3 - etag: 1.8.1 - finalhandler: 1.2.0(supports-color@8.1.1) - fresh: 0.5.2 - http-errors: 2.0.0 - merge-descriptors: 1.0.1 - methods: 1.1.2 - on-finished: 2.4.1 - parseurl: 1.3.3 - path-to-regexp: 0.1.7 - proxy-addr: 2.0.7 - qs: 6.11.0 - range-parser: 1.2.1 - safe-buffer: 5.2.1 - send: 0.18.0(supports-color@8.1.1) - serve-static: 1.15.0(supports-color@8.1.1) - setprototypeof: 1.2.0 - statuses: 2.0.1 - type-is: 1.6.18 - utils-merge: 1.0.1 - vary: 1.1.2 - transitivePeerDependencies: - - supports-color - - fast-safe-stringify@2.1.1: {} - - fast-xml-parser@4.4.1: - dependencies: - strnum: 1.0.5 - - finalhandler@1.2.0(supports-color@8.1.1): - dependencies: - debug: 2.6.9(supports-color@8.1.1) - encodeurl: 1.0.2 - escape-html: 1.0.3 - on-finished: 2.4.1 - parseurl: 1.3.3 - statuses: 2.0.1 - unpipe: 1.0.0 - transitivePeerDependencies: - - supports-color - - form-data-encoder@2.1.4: {} - - forwarded@0.2.0: {} - - fresh@0.5.2: {} - - function-bind@1.1.2: {} - - get-intrinsic@1.2.4: - dependencies: - es-errors: 1.3.0 - function-bind: 1.1.2 - has-proto: 1.0.3 - has-symbols: 1.0.3 - hasown: 2.0.2 - - get-stream@6.0.1: {} - - gopd@1.0.1: - dependencies: - get-intrinsic: 1.2.4 - - got@13.0.0: - dependencies: - '@sindresorhus/is': 5.6.0 - '@szmarczak/http-timer': 5.0.1 - cacheable-lookup: 7.0.0 - cacheable-request: 10.2.14 - decompress-response: 6.0.0 - form-data-encoder: 2.1.4 - get-stream: 6.0.1 - http2-wrapper: 2.2.1 - lowercase-keys: 3.0.0 - p-cancelable: 3.0.0 - responselike: 3.0.0 - - graceful-fs@4.2.11: {} - - grant@5.4.22: - dependencies: - qs: 6.13.0 - request-compose: 2.1.6 - request-oauth: 1.0.1 - optionalDependencies: - cookie: 0.5.0 - cookie-signature: 1.2.1 - jwk-to-pem: 2.0.6 - jws: 4.0.0 - - has-flag@4.0.0: {} - - has-property-descriptors@1.0.2: - dependencies: - es-define-property: 1.0.0 - - has-proto@1.0.3: {} - - has-symbols@1.0.3: {} - - hash.js@1.1.7: - dependencies: - inherits: 2.0.4 - minimalistic-assert: 1.0.1 - optional: true - - hasown@2.0.2: - dependencies: - function-bind: 1.1.2 - - helmet@7.1.0: {} - - hmac-drbg@1.0.1: - dependencies: - hash.js: 1.1.7 - minimalistic-assert: 1.0.1 - minimalistic-crypto-utils: 1.0.1 - optional: true - - http-cache-semantics@4.1.1: {} - - http-errors@2.0.0: - dependencies: - depd: 2.0.0 - inherits: 2.0.4 - setprototypeof: 1.2.0 - statuses: 2.0.1 - toidentifier: 1.0.1 - - http2-wrapper@2.2.1: - dependencies: - quick-lru: 5.1.1 - resolve-alpn: 1.2.1 - - iconv-lite@0.4.24: - dependencies: - safer-buffer: 2.1.2 - - ieee754@1.2.1: {} - - inherits@2.0.4: {} - - ioredis@5.4.1(supports-color@8.1.1): - dependencies: - '@ioredis/commands': 1.2.0 - cluster-key-slot: 1.1.2 - debug: 4.3.6(supports-color@8.1.1) - denque: 2.1.0 - lodash.defaults: 4.2.0 - lodash.isarguments: 3.1.0 - redis-errors: 1.2.0 - redis-parser: 3.0.0 - standard-as-callback: 2.1.0 - transitivePeerDependencies: - - supports-color - - ipaddr.js@1.9.1: {} - - ipaddr.js@2.2.0: {} - - is-network-error@1.1.0: {} - - is-stream@2.0.1: {} - - js-base64@3.7.7: {} - - js-tokens@4.0.0: {} - - json-buffer@3.0.1: {} - - jsonwebtoken@9.0.2: - dependencies: - jws: 3.2.2 - lodash.includes: 4.3.0 - lodash.isboolean: 3.0.3 - lodash.isinteger: 4.0.4 - lodash.isnumber: 3.0.3 - lodash.isplainobject: 4.0.6 - lodash.isstring: 4.0.1 - lodash.once: 4.1.1 - ms: 2.1.3 - semver: 7.6.3 - - jwa@1.4.1: - dependencies: - buffer-equal-constant-time: 1.0.1 - ecdsa-sig-formatter: 1.0.11 - safe-buffer: 5.2.1 - - jwa@2.0.0: - dependencies: - buffer-equal-constant-time: 1.0.1 - ecdsa-sig-formatter: 1.0.11 - safe-buffer: 5.2.1 - optional: true - - jwk-to-pem@2.0.6: - dependencies: - asn1.js: 5.4.1 - elliptic: 6.5.7 - safe-buffer: 5.2.1 - optional: true - - jws@3.2.2: - dependencies: - jwa: 1.4.1 - safe-buffer: 5.2.1 - - jws@4.0.0: - dependencies: - jwa: 2.0.0 - safe-buffer: 5.2.1 - optional: true - - keyv@4.5.4: - dependencies: - json-buffer: 3.0.1 - - lodash._baseiteratee@4.7.0: - dependencies: - lodash._stringtopath: 4.8.0 - - lodash._basetostring@4.12.0: {} - - lodash._baseuniq@4.6.0: - dependencies: - lodash._createset: 4.0.3 - lodash._root: 3.0.1 - - lodash._createset@4.0.3: {} - - lodash._root@3.0.1: {} - - lodash._stringtopath@4.8.0: - dependencies: - lodash._basetostring: 4.12.0 - - lodash.defaults@4.2.0: {} - - lodash.includes@4.3.0: {} - - lodash.isarguments@3.1.0: {} - - lodash.isboolean@3.0.3: {} - - lodash.isinteger@4.0.4: {} - - lodash.isnumber@3.0.3: {} - - lodash.isplainobject@4.0.6: {} - - lodash.isstring@4.0.1: {} - - lodash.once@4.1.1: {} - - lodash.throttle@4.1.1: {} - - lodash.uniqby@4.5.0: - dependencies: - lodash._baseiteratee: 4.7.0 - lodash._baseuniq: 4.6.0 - - lodash@4.17.21: {} - - long-timeout@0.1.1: {} - - loose-envify@1.4.0: - dependencies: - js-tokens: 4.0.0 - - lowercase-keys@3.0.0: {} - - lru-cache@4.1.5: - dependencies: - pseudomap: 1.0.2 - yallist: 2.1.2 - - luxon@3.5.0: {} - - media-typer@0.3.0: {} - - memorystore@1.6.7: - dependencies: - debug: 4.3.6(supports-color@8.1.1) - lru-cache: 4.1.5 - transitivePeerDependencies: - - supports-color - - merge-descriptors@1.0.1: {} - - methods@1.1.2: {} - - mime-db@1.52.0: {} - - mime-match@1.0.2: - dependencies: - wildcard: 1.1.2 - - mime-types@2.1.35: - dependencies: - mime-db: 1.52.0 - - mime@1.6.0: {} - - mimic-response@3.1.0: {} - - mimic-response@4.0.0: {} - - minimalistic-assert@1.0.1: - optional: true - - minimalistic-crypto-utils@1.0.1: - optional: true - - moment-timezone@0.5.45: - dependencies: - moment: 2.30.1 - - moment@2.30.1: {} - - morgan@1.10.0(supports-color@8.1.1): - dependencies: - basic-auth: 2.0.1 - debug: 2.6.9(supports-color@8.1.1) - depd: 2.0.0 - on-finished: 2.3.0 - on-headers: 1.0.2 - transitivePeerDependencies: - - supports-color - - ms@2.0.0: {} - - ms@2.1.2: {} - - ms@2.1.3: {} - - namespace-emitter@2.0.1: {} - - nanoid@5.0.7: {} - - negotiator@0.6.3: {} - - node-schedule@2.1.1: - dependencies: - cron-parser: 4.9.0 - long-timeout: 0.1.1 - sorted-array-functions: 1.3.0 - - normalize-url@8.0.1: {} - - oauth-sign@0.9.0: {} - - object-assign@4.1.1: {} - - object-inspect@1.13.2: {} - - on-finished@2.3.0: - dependencies: - ee-first: 1.1.1 - - on-finished@2.4.1: - dependencies: - ee-first: 1.1.1 - - on-headers@1.0.2: {} - - p-cancelable@3.0.0: {} - - p-retry@6.2.0: - dependencies: - '@types/retry': 0.12.2 - is-network-error: 1.1.0 - retry: 0.13.1 - - parseurl@1.3.3: {} - - path-to-regexp@0.1.7: {} - - preact@10.23.2: {} - - prom-client@15.1.2: - dependencies: - '@opentelemetry/api': 1.9.0 - tdigest: 0.1.2 - - proper-lockfile@4.1.2: - dependencies: - graceful-fs: 4.2.11 - retry: 0.12.0 - signal-exit: 3.0.7 - - proxy-addr@2.0.7: - dependencies: - forwarded: 0.2.0 - ipaddr.js: 1.9.1 - - pseudomap@1.0.2: {} - - qs@6.11.0: - dependencies: - side-channel: 1.0.6 - - qs@6.13.0: - dependencies: - side-channel: 1.0.6 - - querystringify@2.2.0: {} - - quick-lru@5.1.1: {} - - random-bytes@1.0.0: {} - - randombytes@2.1.0: - dependencies: - safe-buffer: 5.2.1 - - range-parser@1.2.1: {} - - raw-body@2.5.2: - dependencies: - bytes: 3.1.2 - http-errors: 2.0.0 - iconv-lite: 0.4.24 - unpipe: 1.0.0 - - react-hook-form@7.53.0(react@18.3.1): - dependencies: - react: 18.3.1 - - react@18.3.1: - dependencies: - loose-envify: 1.4.0 - - readable-stream@3.6.2: - dependencies: - inherits: 2.0.4 - string_decoder: 1.3.0 - util-deprecate: 1.0.2 - - redis-errors@1.2.0: {} - - redis-parser@3.0.0: - dependencies: - redis-errors: 1.2.0 - - request-compose@2.1.6: {} - - request-oauth@1.0.1: - dependencies: - oauth-sign: 0.9.0 - qs: 6.13.0 - uuid: 8.3.2 - - requires-port@1.0.0: {} - - resolve-alpn@1.2.1: {} - - responselike@3.0.0: - dependencies: - lowercase-keys: 3.0.0 - - retry@0.12.0: {} - - retry@0.13.1: {} - - safe-buffer@5.1.2: {} - - safe-buffer@5.2.1: {} - - safer-buffer@2.1.2: {} - - semver@7.6.3: {} - - send@0.18.0(supports-color@8.1.1): - dependencies: - debug: 2.6.9(supports-color@8.1.1) - depd: 2.0.0 - destroy: 1.2.0 - encodeurl: 1.0.2 - escape-html: 1.0.3 - etag: 1.8.1 - fresh: 0.5.2 - http-errors: 2.0.0 - mime: 1.6.0 - ms: 2.1.3 - on-finished: 2.4.1 - range-parser: 1.2.1 - statuses: 2.0.1 - transitivePeerDependencies: - - supports-color - - serialize-error@11.0.3: - dependencies: - type-fest: 2.19.0 - - serialize-javascript@6.0.2: - dependencies: - randombytes: 2.1.0 - - serve-static@1.15.0(supports-color@8.1.1): - dependencies: - encodeurl: 1.0.2 - escape-html: 1.0.3 - parseurl: 1.3.3 - send: 0.18.0(supports-color@8.1.1) - transitivePeerDependencies: - - supports-color - - set-function-length@1.2.2: - dependencies: - define-data-property: 1.1.4 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.2.4 - gopd: 1.0.1 - has-property-descriptors: 1.0.2 - - setprototypeof@1.2.0: {} - - side-channel@1.0.6: - dependencies: - call-bind: 1.0.7 - es-errors: 1.3.0 - get-intrinsic: 1.2.4 - object-inspect: 1.13.2 - - signal-exit@3.0.7: {} - - sorted-array-functions@1.3.0: {} - - standard-as-callback@2.1.0: {} - - statuses@2.0.1: {} - - stream-browserify@3.0.0: - dependencies: - inherits: 2.0.4 - readable-stream: 3.6.2 - - string_decoder@1.3.0: - dependencies: - safe-buffer: 5.2.1 - - strnum@1.0.5: {} - - supports-color@8.1.1: - dependencies: - has-flag: 4.0.0 - - tdigest@0.1.2: - dependencies: - bintrees: 1.0.2 - - toidentifier@1.0.1: {} - - tslib@2.7.0: {} - - tus-js-client@4.1.0: - dependencies: - buffer-from: 1.1.2 - combine-errors: 3.0.3 - is-stream: 2.0.1 - js-base64: 3.7.7 - lodash.throttle: 4.1.1 - proper-lockfile: 4.1.2 - url-parse: 1.5.10 - - type-fest@2.19.0: {} - - type-is@1.6.18: - dependencies: - media-typer: 0.3.0 - mime-types: 2.1.35 - - uid-safe@2.1.5: - dependencies: - random-bytes: 1.0.0 - - undici-types@6.19.8: {} - - unpipe@1.0.0: {} - - url-parse@1.5.10: - dependencies: - querystringify: 2.2.0 - requires-port: 1.0.0 - - url-value-parser@2.2.0: {} - - util-deprecate@1.0.2: {} - - utils-merge@1.0.1: {} - - uuid@8.3.2: {} - - uuid@9.0.1: {} - - validator@13.12.0: {} - - vary@1.1.2: {} - - wildcard@1.1.2: {} - - ws@8.17.1: {} - - yallist@2.1.2: {}
    - - -
    -
    - -
    -
    -
    - -
    - -