kinto 21.0.0__tar.gz → 21.1.1__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of kinto might be problematic. Click here for more details.
- {kinto-21.0.0 → kinto-21.1.1}/PKG-INFO +1 -1
- {kinto-21.0.0 → kinto-21.1.1}/app.wsgi +2 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/utilities.rst +5 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/index.rst +6 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/configuration/settings.rst +6 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/__init__.py +1 -1
- {kinto-21.0.0 → kinto-21.1.1}/kinto/config/__init__.py +23 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/initialization.py +18 -17
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/views/hello.py +4 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/prometheus.py +26 -3
- {kinto-21.0.0 → kinto-21.1.1}/kinto.egg-info/PKG-INFO +1 -1
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_initialization.py +2 -2
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_views_hello.py +18 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/plugins/test_prometheus.py +46 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_metrics.py +14 -4
- {kinto-21.0.0 → kinto-21.1.1}/.dockerignore +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/.github/CODE_OF_CONDUCT.md +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/.github/CONTRIBUTING.md +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/.github/PULL_REQUEST_TEMPLATE.md +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/.github/dependabot.yml +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/.github/release.yml +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/.github/workflows/labels.yaml +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/.github/workflows/publish.yml +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/.github/workflows/scheduled.yml +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/.github/workflows/test.yml +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/.gitignore +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/.readthedocs.yaml +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/CHANGELOG.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/CONTRIBUTORS.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/Dockerfile +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/LICENSE +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/Makefile +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/README.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/SECURITY.md +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/SUPPORT.md +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/constraints.in +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/constraints.txt +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docker-compose.yml +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/_static/piwik.js +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/_static/theme_overrides.css +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/_templates/footer.html +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/_templates/indexcontent.html +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/_templates/layout.html +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_details-delete-list.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_details-delete-object.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_details-get-list.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_details-get-object.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_details-head-list.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_details-patch-object.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_details-post-list.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_details-put-object.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_status-delete-list.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_status-delete-object.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_status-get-list.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_status-get-object.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_status-patch-object.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_status-post-list.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/_status-put-object.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/accounts.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/admin.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/authentication.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/backoff.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/batch.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/buckets.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/collections.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/deprecation.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/errors.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/filtering.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/flush.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/groups.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/history.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/index.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/openapi.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/openid.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/pagination.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/permissions.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/records.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/selecting_fields.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/sorting.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/1.x/timestamps.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/api/versioning.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/changelog.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/commandline.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/community.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/concepts.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/conf.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/configuration/good-practices.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/configuration/index.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/configuration/production.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/_static/theme_overrides.css +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/api.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/cache.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/decorators.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/errors.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/glossary.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/images/cliquet-base.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/images/cliquet-mozilla.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/index.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/notifications.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/permission.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/quickstart.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/rationale.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/resource.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/storage.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/testing.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/utils.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/core/viewsets.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/faq.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/alwaysdata-button.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/architecture.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/color-formatter.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/concepts-general.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/concepts-permissions.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/heroku-button.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/kinto-logo.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/kinto-logo.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/logo-admin.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/logo-attachment.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/logo-community.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/logo-history.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/logo-javascript.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/logo-jsonschema.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/logo-multiapps.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/logo-permissions.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/logo-python.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/logo-selfhostable.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/logo-synchronisation.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/overview-deployonce-selfhost.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/overview-features.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/overview-synchronisation.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/overview-use-cases.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/scalingo-button.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/screenshot-kinto-admin-1.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/screenshot-kinto-admin-2.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/screenshot-kinto-admin-3.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/screenshot-kinto-admin-4.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/sequence-storage.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/sync-both.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/sync-newest.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/images/sync-oldest.svg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/index.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/kinto-admin.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/overview.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/requirements.txt +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/troubleshooting.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/tutorials/custom-id-generator.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/tutorials/first-steps.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/tutorials/index.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/tutorials/install.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/tutorials/notifications-custom.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/tutorials/permission-setups.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/tutorials/permissions.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/tutorials/synchronisation.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/docs/tutorials/write-plugin.rst +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/__main__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/authorization.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/config/kinto.tpl +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/contribute.json +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/authentication.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/authorization.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cache/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cache/memcached.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cache/memory.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cache/postgresql/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cache/postgresql/schema.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cache/testing.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice/cors.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice/errors.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice/pyramidhook.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice/renderer.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice/resource.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice/service.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice/util.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice/validators/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice/validators/_colander.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice/validators/_marshmallow.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice_swagger/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice_swagger/converters/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice_swagger/converters/exceptions.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice_swagger/converters/parameters.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice_swagger/converters/schema.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice_swagger/swagger.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice_swagger/templates/index.html +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice_swagger/templates/index_script_template.html +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice_swagger/util.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/cornice_swagger/views.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/decorators.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/errors.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/events.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/listeners/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/metrics.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/openapi.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/permission/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/permission/memory.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/permission/postgresql/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/permission/postgresql/migrations/migration_001_002.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/permission/postgresql/schema.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/permission/testing.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/resource/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/resource/model.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/resource/schema.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/resource/viewset.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/schema.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/scripts.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/statsd.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/exceptions.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/generators.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/memory.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/client.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_001_002.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_002_003.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_003_004.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_004_005.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_005_006.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_006_007.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_007_008.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_008_009.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_009_010.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_010_011.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_011_012.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_012_013.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_013_014.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_014_015.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_015_016.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_016_017.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_017_018.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_018_019.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_019_020.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_020_021.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_021_022.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrations/migration_022_023.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/migrator.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/pool.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/postgresql/schema.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/testing.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/storage/utils.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/testing.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/utils.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/views/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/views/batch.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/views/errors.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/views/heartbeat.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/views/openapi.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/core/views/version.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/events.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/accounts/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/accounts/authentication.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/accounts/scripts.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/accounts/utils.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/accounts/views.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/README.md +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/VERSION +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/build/VERSION +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/build/assets/asn1-EdZsLKOL.js +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/build/assets/clojure-BMjYHr_A.js +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/build/assets/css-BnMrqG3P.js +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/build/assets/index-Cs7JVwIg.css +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/build/assets/index-CylsivYB.js +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/build/assets/javascript-qCveANmP.js +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/build/assets/logo-VBRiKSPX.png +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/build/assets/mllike-CXdrOF99.js +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/build/assets/python-BuPzkPfP.js +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/build/assets/rpm-CTu-6PCP.js +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/build/assets/sql-D0XecflT.js +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/build/assets/ttcn-cfg-B9xdYoR4.js +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/build/index.html +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/public/help.html +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/admin/views.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/default_bucket/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/flush.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/history/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/history/listener.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/history/views.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/openid/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/openid/utils.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/openid/views.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/plugins/statsd.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/schema_validation.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/views/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/views/admin.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/views/buckets.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/views/collections.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/views/contribute.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/views/groups.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/views/permissions.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto/views/records.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto.egg-info/SOURCES.txt +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto.egg-info/dependency_links.txt +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto.egg-info/entry_points.txt +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto.egg-info/requires.txt +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/kinto.egg-info/top_level.txt +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/pyproject.toml +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/scripts/pull-kinto-admin.sh +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/setup.cfg +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/browser.ini +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/browser.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/listeners.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_base.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_cache_expires.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_events.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_filter.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_model.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_object.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_object_permissions.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_pagination.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_partial_response.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_preconditions.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_schema.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_sort.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_sync.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_views.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_views_cors.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/resource/test_viewset.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/schema/postgresql-permission-1.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/schema/postgresql-storage-1.6.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/schema/postgresql-storage-11.sql +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/support.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_authentication.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_authorization.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_cache.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_decorators.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_deprecation.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_errors.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_listeners.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_logging.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_metrics.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_openapi.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_permission.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_schema.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_scripts.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_storage.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_storage_migrations.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_storage_pool.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_utils.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_views_batch.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_views_errors.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_views_heartbeat.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_views_openapi.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_views_postgresql.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/test_views_transaction.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/testapp/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/testapp/static/index.html +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/testapp/views.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/core/testplugin/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/functional.ini +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/functional.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/openapi/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/openapi/support.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/openapi/test_plugins.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/openapi/test_resources.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/openapi/test_responses_buckets.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/openapi/test_responses_collections.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/openapi/test_responses_errors.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/openapi/test_responses_groups.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/openapi/test_responses_records.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/openapi/test_validation.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/plugins/__init__.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/plugins/test_accounts.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/plugins/test_admin.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/plugins/test_default_bucket.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/plugins/test_flush.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/plugins/test_history.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/plugins/test_openid.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/plugins/test_statsd.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/support.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_authorization.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_config.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_configuration/test.ini +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_init.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_main.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_admin.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_buckets.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_collections.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_collections_cache.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_contribute.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_disable_default.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_groups.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_hello.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_objects_permissions.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_permissions.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_records.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_schema_collection.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_schema_group.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_schema_record.py +0 -0
- {kinto-21.0.0 → kinto-21.1.1}/tests/test_views_version.py +0 -0
|
@@ -8,6 +8,8 @@ here = os.path.dirname(__file__)
|
|
|
8
8
|
|
|
9
9
|
ini_path = os.environ.get('KINTO_INI')
|
|
10
10
|
if ini_path is None:
|
|
11
|
+
# WARNING: if you modify this default value, you should
|
|
12
|
+
# also change the default in `kinto.config.config_attributes()`
|
|
11
13
|
ini_path = os.path.join(here, 'config', 'kinto.ini')
|
|
12
14
|
|
|
13
15
|
# Set up logging
|
|
@@ -25,6 +25,11 @@ The returned value is a JSON mapping containing:
|
|
|
25
25
|
- ``readonly``: Only requests with read operations are allowed.
|
|
26
26
|
|
|
27
27
|
- ``capabilities``: a mapping used by clients to detect optional features of the API.
|
|
28
|
+
- ``config``: attributes of the configuration file used by the server.
|
|
29
|
+
|
|
30
|
+
- ``path``: path on the server
|
|
31
|
+
- ``hash``: SHA256 hash of the file
|
|
32
|
+
- ``modified``: datetime of the file
|
|
28
33
|
|
|
29
34
|
- Example:
|
|
30
35
|
|
|
@@ -513,6 +513,12 @@ Prometheus metrics can be enabled with (disabled by default):
|
|
|
513
513
|
|
|
514
514
|
# kinto.prometheus_prefix = kinto-prod
|
|
515
515
|
|
|
516
|
+
# Expose metrics created time (default: true)
|
|
517
|
+
# kinto.prometheus_created_metrics_enabled = false
|
|
518
|
+
|
|
519
|
+
# Exclude certain labels to reduce cardinality (default: none)
|
|
520
|
+
# kinto.prometheus_exclude_labels = record_id group_id
|
|
521
|
+
|
|
516
522
|
Metrics can then be crawled from the ``/__metrics__`` endpoint.
|
|
517
523
|
|
|
518
524
|
|
|
@@ -1,6 +1,9 @@
|
|
|
1
1
|
import codecs
|
|
2
2
|
import logging
|
|
3
3
|
import os
|
|
4
|
+
from datetime import datetime
|
|
5
|
+
from functools import lru_cache
|
|
6
|
+
from hashlib import sha256
|
|
4
7
|
from time import strftime
|
|
5
8
|
|
|
6
9
|
from kinto import __version__
|
|
@@ -69,3 +72,23 @@ def init(config_file, backend, cache_backend, host="127.0.0.1"):
|
|
|
69
72
|
values.update(cache_backend_to_values[cache_backend])
|
|
70
73
|
|
|
71
74
|
render_template("kinto.tpl", config_file, **values)
|
|
75
|
+
|
|
76
|
+
|
|
77
|
+
@lru_cache(maxsize=1)
|
|
78
|
+
def config_attributes():
|
|
79
|
+
"""
|
|
80
|
+
Returns a hash of the config `.ini` file content.
|
|
81
|
+
The path is only known from `app.wsgi`, so we have to read
|
|
82
|
+
the environment variable again. Since tests are not run through
|
|
83
|
+
WSGI, then the variable is not set.
|
|
84
|
+
"""
|
|
85
|
+
# WARNING: this default value should be the same as `app.wsgi`
|
|
86
|
+
ini_path = os.environ.get("KINTO_INI", os.path.join(".", "config", "kinto.ini"))
|
|
87
|
+
if not os.path.exists(ini_path):
|
|
88
|
+
logger.error(f"Could not find config file at {ini_path}")
|
|
89
|
+
return None
|
|
90
|
+
return {
|
|
91
|
+
"path": ini_path,
|
|
92
|
+
"hash": sha256(open(ini_path, "rb").read()).hexdigest(),
|
|
93
|
+
"modified": datetime.fromtimestamp(os.path.getmtime(ini_path)).isoformat(),
|
|
94
|
+
}
|
|
@@ -472,10 +472,18 @@ def setup_metrics(config):
|
|
|
472
472
|
auth, user_id = user_id.split(":")
|
|
473
473
|
metrics_service.count("users", unique=[("auth", auth), ("userid", user_id)])
|
|
474
474
|
|
|
475
|
+
status = event.response.status_code
|
|
476
|
+
|
|
477
|
+
if status >= 400:
|
|
478
|
+
# Prevent random values of 404 responses to become label values.
|
|
479
|
+
request_matchdict = {}
|
|
480
|
+
else:
|
|
481
|
+
request_matchdict = dict(request.matchdict or {})
|
|
482
|
+
|
|
475
483
|
# Add extra labels to metrics, based on fields extracted from the request matchdict.
|
|
476
484
|
metrics_matchdict_fields = aslist(settings["metrics_matchdict_fields"])
|
|
477
485
|
# Turn the `id` field of object endpoints into `{resource}_id` (eg. `mushroom_id`, `bucket_id`)
|
|
478
|
-
enhanced_matchdict =
|
|
486
|
+
enhanced_matchdict = request_matchdict
|
|
479
487
|
try:
|
|
480
488
|
enhanced_matchdict[request.current_resource_name + "_id"] = enhanced_matchdict.get(
|
|
481
489
|
"id", ""
|
|
@@ -487,8 +495,6 @@ def setup_metrics(config):
|
|
|
487
495
|
(field, enhanced_matchdict.get(field, "")) for field in metrics_matchdict_fields
|
|
488
496
|
]
|
|
489
497
|
|
|
490
|
-
status = event.response.status_code
|
|
491
|
-
|
|
492
498
|
service = request.current_service
|
|
493
499
|
if service:
|
|
494
500
|
# Use the service name as endpoint if available.
|
|
@@ -501,16 +507,14 @@ def setup_metrics(config):
|
|
|
501
507
|
"unnamed" if status != 404 else "unknown"
|
|
502
508
|
) # Do not multiply cardinality for unknown endpoints.
|
|
503
509
|
|
|
510
|
+
request_labels = [
|
|
511
|
+
("method", request.method.lower()),
|
|
512
|
+
("endpoint", endpoint),
|
|
513
|
+
("status", str(status)),
|
|
514
|
+
] + metrics_matchdict_labels
|
|
515
|
+
|
|
504
516
|
# Count served requests.
|
|
505
|
-
metrics_service.count(
|
|
506
|
-
"request_summary",
|
|
507
|
-
unique=[
|
|
508
|
-
("method", request.method.lower()),
|
|
509
|
-
("endpoint", endpoint),
|
|
510
|
-
("status", str(status)),
|
|
511
|
-
]
|
|
512
|
-
+ metrics_matchdict_labels,
|
|
513
|
-
)
|
|
517
|
+
metrics_service.count("request_summary", unique=request_labels)
|
|
514
518
|
|
|
515
519
|
try:
|
|
516
520
|
current = utils.msec_time()
|
|
@@ -518,8 +522,7 @@ def setup_metrics(config):
|
|
|
518
522
|
metrics_service.timer(
|
|
519
523
|
"request_duration",
|
|
520
524
|
value=duration,
|
|
521
|
-
labels=
|
|
522
|
-
+ metrics_matchdict_labels,
|
|
525
|
+
labels=request_labels,
|
|
523
526
|
)
|
|
524
527
|
except AttributeError: # pragma: no cover
|
|
525
528
|
# Logging was not setup in this Kinto app (unlikely but possible)
|
|
@@ -527,9 +530,7 @@ def setup_metrics(config):
|
|
|
527
530
|
|
|
528
531
|
# Observe response size.
|
|
529
532
|
metrics_service.observe(
|
|
530
|
-
"request_size",
|
|
531
|
-
len(event.response.body or b""),
|
|
532
|
-
labels=[("endpoint", endpoint)] + metrics_matchdict_labels,
|
|
533
|
+
"request_size", len(event.response.body or b""), labels=request_labels
|
|
533
534
|
)
|
|
534
535
|
|
|
535
536
|
# Count authentication verifications.
|
|
@@ -2,6 +2,7 @@ import colander
|
|
|
2
2
|
from pyramid.authorization import Authenticated
|
|
3
3
|
from pyramid.security import NO_PERMISSION_REQUIRED
|
|
4
4
|
|
|
5
|
+
from kinto.config import config_attributes
|
|
5
6
|
from kinto.core import Service
|
|
6
7
|
|
|
7
8
|
|
|
@@ -26,14 +27,17 @@ hello_response_schemas = {
|
|
|
26
27
|
def get_hello(request):
|
|
27
28
|
"""Return information regarding the current instance."""
|
|
28
29
|
settings = request.registry.settings
|
|
30
|
+
|
|
29
31
|
project_name = settings["project_name"]
|
|
30
32
|
project_version = settings["project_version"]
|
|
33
|
+
|
|
31
34
|
data = dict(
|
|
32
35
|
project_name=project_name,
|
|
33
36
|
project_version=project_version,
|
|
34
37
|
http_api_version=settings["http_api_version"],
|
|
35
38
|
project_docs=settings["project_docs"],
|
|
36
39
|
url=request.route_url(hello.name),
|
|
40
|
+
config=config_attributes(),
|
|
37
41
|
)
|
|
38
42
|
|
|
39
43
|
eos = get_eos(request)
|
|
@@ -6,6 +6,7 @@ from time import perf_counter as time_now
|
|
|
6
6
|
|
|
7
7
|
from pyramid.exceptions import ConfigurationError
|
|
8
8
|
from pyramid.response import Response
|
|
9
|
+
from pyramid.settings import asbool, aslist
|
|
9
10
|
from zope.interface import implementer
|
|
10
11
|
|
|
11
12
|
from kinto.core import metrics
|
|
@@ -102,7 +103,7 @@ class Timer:
|
|
|
102
103
|
|
|
103
104
|
@implementer(metrics.IMetricsService)
|
|
104
105
|
class PrometheusService:
|
|
105
|
-
def __init__(self, prefix=""):
|
|
106
|
+
def __init__(self, prefix="", exclude_labels=None):
|
|
106
107
|
prefix_clean = ""
|
|
107
108
|
if prefix:
|
|
108
109
|
# In GCP Console, the metrics are grouped by the first
|
|
@@ -111,10 +112,19 @@ class PrometheusService:
|
|
|
111
112
|
# (eg. `remote-settings` -> `remotesettings_`, `kinto_` -> `kinto_`)
|
|
112
113
|
prefix_clean = _fix_metric_name(prefix).replace("_", "") + "_"
|
|
113
114
|
self.prefix = prefix_clean.lower()
|
|
115
|
+
self.exclude_labels = exclude_labels or []
|
|
116
|
+
|
|
117
|
+
def _exclude_labels(self, labels):
|
|
118
|
+
return [
|
|
119
|
+
(label_name, label_value)
|
|
120
|
+
for label_name, label_value in labels
|
|
121
|
+
if label_name not in self.exclude_labels
|
|
122
|
+
]
|
|
114
123
|
|
|
115
124
|
def timer(self, key, value=None, labels=[]):
|
|
116
125
|
global _METRICS
|
|
117
126
|
key = self.prefix + key
|
|
127
|
+
labels = self._exclude_labels(labels)
|
|
118
128
|
|
|
119
129
|
if key not in _METRICS:
|
|
120
130
|
_METRICS[key] = prometheus_module.Histogram(
|
|
@@ -144,6 +154,7 @@ class PrometheusService:
|
|
|
144
154
|
def observe(self, key, value, labels=[]):
|
|
145
155
|
global _METRICS
|
|
146
156
|
key = self.prefix + key
|
|
157
|
+
labels = self._exclude_labels(labels)
|
|
147
158
|
|
|
148
159
|
if key not in _METRICS:
|
|
149
160
|
_METRICS[key] = prometheus_module.Summary(
|
|
@@ -184,6 +195,7 @@ class PrometheusService:
|
|
|
184
195
|
label_name, label_value = unique.rsplit(".", 1)
|
|
185
196
|
unique = [(label_name, label_value)]
|
|
186
197
|
|
|
198
|
+
unique = self._exclude_labels(unique)
|
|
187
199
|
labels = [
|
|
188
200
|
(_fix_metric_name(label_name), label_value) for label_name, label_value in unique
|
|
189
201
|
]
|
|
@@ -229,6 +241,11 @@ def includeme(config):
|
|
|
229
241
|
)
|
|
230
242
|
raise ConfigurationError(error_msg)
|
|
231
243
|
|
|
244
|
+
settings = config.get_settings()
|
|
245
|
+
|
|
246
|
+
if not asbool(settings.get("prometheus_created_metrics_enabled", True)):
|
|
247
|
+
prometheus_module.disable_created_metrics()
|
|
248
|
+
|
|
232
249
|
config.add_api_capability(
|
|
233
250
|
"prometheus",
|
|
234
251
|
description="Prometheus metrics.",
|
|
@@ -250,7 +267,13 @@ def includeme(config):
|
|
|
250
267
|
pass
|
|
251
268
|
_METRICS.clear()
|
|
252
269
|
|
|
253
|
-
settings = config.get_settings()
|
|
254
270
|
prefix = settings.get("prometheus_prefix", settings["project_name"])
|
|
255
271
|
|
|
256
|
-
|
|
272
|
+
# If we want to reduce the metrics cardinality, we can exclude certain
|
|
273
|
+
# labels (eg. records_id). This way all metrics will be grouped by the
|
|
274
|
+
# remaining labels.
|
|
275
|
+
exclude_labels = aslist(settings.get("prometheus_exclude_labels", ""))
|
|
276
|
+
|
|
277
|
+
config.registry.registerUtility(
|
|
278
|
+
PrometheusService(prefix=prefix, exclude_labels=exclude_labels), metrics.IMetricsService
|
|
279
|
+
)
|
|
@@ -433,7 +433,7 @@ class MetricsConfigurationTest(unittest.TestCase):
|
|
|
433
433
|
self.mocked().observe.assert_any_call(
|
|
434
434
|
"request_size",
|
|
435
435
|
len("{}"),
|
|
436
|
-
labels=[("endpoint", "heartbeat")],
|
|
436
|
+
labels=[("method", "get"), ("endpoint", "heartbeat"), ("status", "200")],
|
|
437
437
|
)
|
|
438
438
|
|
|
439
439
|
def test_statsd_observe_request_duration(self):
|
|
@@ -443,7 +443,7 @@ class MetricsConfigurationTest(unittest.TestCase):
|
|
|
443
443
|
self.mocked().timer.assert_any_call(
|
|
444
444
|
"request_duration",
|
|
445
445
|
value=mock.ANY,
|
|
446
|
-
labels=[("endpoint", "heartbeat"), ("
|
|
446
|
+
labels=[("method", "get"), ("endpoint", "heartbeat"), ("status", "200")],
|
|
447
447
|
)
|
|
448
448
|
|
|
449
449
|
def test_statsd_counts_unknown_urls(self):
|
|
@@ -1,7 +1,9 @@
|
|
|
1
|
+
import os
|
|
1
2
|
from unittest import mock
|
|
2
3
|
|
|
3
4
|
from pyramid import testing
|
|
4
5
|
|
|
6
|
+
from kinto.config import config_attributes
|
|
5
7
|
from kinto.core.testing import unittest
|
|
6
8
|
|
|
7
9
|
from .support import BaseWebTest
|
|
@@ -75,6 +77,22 @@ class HelloViewTest(BaseWebTest, unittest.TestCase):
|
|
|
75
77
|
|
|
76
78
|
self.assertTrue(response.json["http_api_version"])
|
|
77
79
|
|
|
80
|
+
def test_return_config_file_info(self):
|
|
81
|
+
config_attributes.cache_clear()
|
|
82
|
+
before = os.getenv("KINTO_INI", None)
|
|
83
|
+
os.environ["KINTO_INI"] = "tests/test_configuration/test.ini"
|
|
84
|
+
|
|
85
|
+
response = self.app.get("/")
|
|
86
|
+
|
|
87
|
+
self.assertEqual(response.json["config"]["hash"], mock.ANY)
|
|
88
|
+
self.assertEqual(response.json["config"]["modified"], mock.ANY)
|
|
89
|
+
self.assertEqual(response.json["config"]["path"], mock.ANY)
|
|
90
|
+
|
|
91
|
+
if before is None:
|
|
92
|
+
del os.environ["KINTO_INI"]
|
|
93
|
+
else:
|
|
94
|
+
os.environ["KINTO_INI"] = before
|
|
95
|
+
|
|
78
96
|
|
|
79
97
|
class APICapabilitiesTest(BaseWebTest, unittest.TestCase):
|
|
80
98
|
def test_list_of_capabilities_contains_basicauth_by_default(self):
|
|
@@ -157,3 +157,49 @@ class PrometheusNoPrefixTest(PrometheusWebTest):
|
|
|
157
157
|
|
|
158
158
|
resp = self.app.get("/__metrics__")
|
|
159
159
|
self.assertIn("TYPE price summary", resp.text)
|
|
160
|
+
|
|
161
|
+
|
|
162
|
+
@skip_if_no_prometheus
|
|
163
|
+
class PrometheusNoCreatedTest(PrometheusWebTest):
|
|
164
|
+
@classmethod
|
|
165
|
+
def get_app_settings(cls, extras=None):
|
|
166
|
+
settings = super().get_app_settings(extras)
|
|
167
|
+
settings["prometheus_created_metrics_enabled"] = "false"
|
|
168
|
+
return settings
|
|
169
|
+
|
|
170
|
+
def test_metrics_created_not_in_response(self):
|
|
171
|
+
self.app.app.registry.metrics.observe("price", 111)
|
|
172
|
+
|
|
173
|
+
resp = self.app.get("/__metrics__")
|
|
174
|
+
|
|
175
|
+
self.assertIn("TYPE kintoprod_price summary", resp.text)
|
|
176
|
+
self.assertNotIn("TYPE kintoprod_price_created summary", resp.text)
|
|
177
|
+
|
|
178
|
+
|
|
179
|
+
@skip_if_no_prometheus
|
|
180
|
+
class PrometheusExcludedLabelsTest(PrometheusWebTest):
|
|
181
|
+
@classmethod
|
|
182
|
+
def get_app_settings(cls, extras=None):
|
|
183
|
+
settings = super().get_app_settings(extras)
|
|
184
|
+
settings["prometheus_exclude_labels"] = "record_id group_id"
|
|
185
|
+
return settings
|
|
186
|
+
|
|
187
|
+
def test_metrics_excluded_labels(self):
|
|
188
|
+
headers = get_user_headers("aaa")
|
|
189
|
+
self.app.put("/buckets/bid", headers=headers)
|
|
190
|
+
self.app.put("/buckets/bid/collections/cid", headers=headers)
|
|
191
|
+
self.app.put("/buckets/bid/groups/gid", headers=headers)
|
|
192
|
+
self.app.put("/buckets/bid/collections/cid/records/rid", headers=headers)
|
|
193
|
+
|
|
194
|
+
resp = self.app.get("/__metrics__")
|
|
195
|
+
|
|
196
|
+
self.assertNotIn("group_id=", resp.text)
|
|
197
|
+
self.assertNotIn("record_id=", resp.text)
|
|
198
|
+
self.assertIn(
|
|
199
|
+
'kintoprod_request_size_count{bucket_id="bid",collection_id="",endpoint="group-object",method="put",status="201"}',
|
|
200
|
+
resp.text,
|
|
201
|
+
)
|
|
202
|
+
self.assertIn(
|
|
203
|
+
'kintoprod_request_size_count{bucket_id="bid",collection_id="cid",endpoint="record-object",method="put",status="201"}',
|
|
204
|
+
resp.text,
|
|
205
|
+
)
|
|
@@ -31,13 +31,13 @@ class ViewsMetricsTest(BaseWebTest, unittest.TestCase):
|
|
|
31
31
|
self.app.get("/buckets/beers/collections/barley/records", headers=self.headers)
|
|
32
32
|
|
|
33
33
|
resp = self.app.get("/__metrics__")
|
|
34
|
-
|
|
34
|
+
|
|
35
35
|
self.assertIn(
|
|
36
|
-
'request_size_sum{bucket_id="beers",collection_id="",endpoint="bucket-object",group_id="",record_id=""}',
|
|
36
|
+
'request_size_sum{bucket_id="beers",collection_id="",endpoint="bucket-object",group_id="",method="put",record_id="",status="201"}',
|
|
37
37
|
resp.text,
|
|
38
38
|
)
|
|
39
39
|
self.assertIn(
|
|
40
|
-
'request_size_sum{bucket_id="beers",collection_id="",endpoint="group-object",group_id="amateurs",record_id=""}',
|
|
40
|
+
'request_size_sum{bucket_id="beers",collection_id="",endpoint="group-object",group_id="amateurs",method="put",record_id="",status="201"}',
|
|
41
41
|
resp.text,
|
|
42
42
|
)
|
|
43
43
|
self.assertIn(
|
|
@@ -45,7 +45,7 @@ class ViewsMetricsTest(BaseWebTest, unittest.TestCase):
|
|
|
45
45
|
resp.text,
|
|
46
46
|
)
|
|
47
47
|
self.assertIn(
|
|
48
|
-
'request_duration_sum{bucket_id="beers",collection_id="barley",endpoint="record-object",group_id="",method="put",record_id="abc"}',
|
|
48
|
+
'request_duration_sum{bucket_id="beers",collection_id="barley",endpoint="record-object",group_id="",method="put",record_id="abc",status="201"}',
|
|
49
49
|
resp.text,
|
|
50
50
|
)
|
|
51
51
|
|
|
@@ -61,3 +61,13 @@ class ViewsMetricsTest(BaseWebTest, unittest.TestCase):
|
|
|
61
61
|
'request_summary_total{bucket_id="beers",collection_id="barley",endpoint="record-plural",group_id="",method="get",record_id="",status="200"}',
|
|
62
62
|
resp.text,
|
|
63
63
|
)
|
|
64
|
+
|
|
65
|
+
def test_4xx_do_not_have_matchdict_labels_values(self):
|
|
66
|
+
self.app.get("/buckets/water", headers=self.headers, status=403)
|
|
67
|
+
|
|
68
|
+
resp = self.app.get("/__metrics__")
|
|
69
|
+
|
|
70
|
+
self.assertIn(
|
|
71
|
+
'request_summary_total{bucket_id="",collection_id="",endpoint="bucket-object",group_id="",method="get",record_id="",status="403"}',
|
|
72
|
+
resp.text,
|
|
73
|
+
)
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|