kinto 20.6.1__tar.gz → 21.1.0__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.

Files changed (394) hide show
  1. {kinto-20.6.1 → kinto-21.1.0}/PKG-INFO +1 -1
  2. {kinto-20.6.1 → kinto-21.1.0}/app.wsgi +2 -0
  3. {kinto-20.6.1 → kinto-21.1.0}/constraints.txt +1 -1
  4. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/utilities.rst +5 -0
  5. {kinto-20.6.1 → kinto-21.1.0}/docs/api/index.rst +6 -0
  6. {kinto-20.6.1 → kinto-21.1.0}/docs/configuration/settings.rst +6 -0
  7. {kinto-20.6.1 → kinto-21.1.0}/kinto/__init__.py +1 -1
  8. {kinto-20.6.1 → kinto-21.1.0}/kinto/config/__init__.py +23 -0
  9. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/initialization.py +24 -20
  10. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/metrics.py +6 -5
  11. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/views/hello.py +4 -0
  12. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/prometheus.py +65 -12
  13. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/statsd.py +8 -1
  14. {kinto-20.6.1 → kinto-21.1.0}/kinto.egg-info/PKG-INFO +1 -1
  15. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_initialization.py +7 -13
  16. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_metrics.py +3 -1
  17. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_views_hello.py +18 -0
  18. {kinto-20.6.1 → kinto-21.1.0}/tests/plugins/test_prometheus.py +49 -3
  19. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_metrics.py +22 -4
  20. {kinto-20.6.1 → kinto-21.1.0}/.dockerignore +0 -0
  21. {kinto-20.6.1 → kinto-21.1.0}/.github/CODE_OF_CONDUCT.md +0 -0
  22. {kinto-20.6.1 → kinto-21.1.0}/.github/CONTRIBUTING.md +0 -0
  23. {kinto-20.6.1 → kinto-21.1.0}/.github/PULL_REQUEST_TEMPLATE.md +0 -0
  24. {kinto-20.6.1 → kinto-21.1.0}/.github/dependabot.yml +0 -0
  25. {kinto-20.6.1 → kinto-21.1.0}/.github/release.yml +0 -0
  26. {kinto-20.6.1 → kinto-21.1.0}/.github/workflows/labels.yaml +0 -0
  27. {kinto-20.6.1 → kinto-21.1.0}/.github/workflows/publish.yml +0 -0
  28. {kinto-20.6.1 → kinto-21.1.0}/.github/workflows/scheduled.yml +0 -0
  29. {kinto-20.6.1 → kinto-21.1.0}/.github/workflows/test.yml +0 -0
  30. {kinto-20.6.1 → kinto-21.1.0}/.gitignore +0 -0
  31. {kinto-20.6.1 → kinto-21.1.0}/.readthedocs.yaml +0 -0
  32. {kinto-20.6.1 → kinto-21.1.0}/CHANGELOG.rst +0 -0
  33. {kinto-20.6.1 → kinto-21.1.0}/CONTRIBUTORS.rst +0 -0
  34. {kinto-20.6.1 → kinto-21.1.0}/Dockerfile +0 -0
  35. {kinto-20.6.1 → kinto-21.1.0}/LICENSE +0 -0
  36. {kinto-20.6.1 → kinto-21.1.0}/Makefile +0 -0
  37. {kinto-20.6.1 → kinto-21.1.0}/README.rst +0 -0
  38. {kinto-20.6.1 → kinto-21.1.0}/SECURITY.md +0 -0
  39. {kinto-20.6.1 → kinto-21.1.0}/SUPPORT.md +0 -0
  40. {kinto-20.6.1 → kinto-21.1.0}/constraints.in +0 -0
  41. {kinto-20.6.1 → kinto-21.1.0}/docker-compose.yml +0 -0
  42. {kinto-20.6.1 → kinto-21.1.0}/docs/_static/piwik.js +0 -0
  43. {kinto-20.6.1 → kinto-21.1.0}/docs/_static/theme_overrides.css +0 -0
  44. {kinto-20.6.1 → kinto-21.1.0}/docs/_templates/footer.html +0 -0
  45. {kinto-20.6.1 → kinto-21.1.0}/docs/_templates/indexcontent.html +0 -0
  46. {kinto-20.6.1 → kinto-21.1.0}/docs/_templates/layout.html +0 -0
  47. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_details-delete-list.rst +0 -0
  48. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_details-delete-object.rst +0 -0
  49. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_details-get-list.rst +0 -0
  50. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_details-get-object.rst +0 -0
  51. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_details-head-list.rst +0 -0
  52. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_details-patch-object.rst +0 -0
  53. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_details-post-list.rst +0 -0
  54. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_details-put-object.rst +0 -0
  55. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_status-delete-list.rst +0 -0
  56. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_status-delete-object.rst +0 -0
  57. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_status-get-list.rst +0 -0
  58. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_status-get-object.rst +0 -0
  59. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_status-patch-object.rst +0 -0
  60. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_status-post-list.rst +0 -0
  61. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/_status-put-object.rst +0 -0
  62. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/accounts.rst +0 -0
  63. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/admin.rst +0 -0
  64. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/authentication.rst +0 -0
  65. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/backoff.rst +0 -0
  66. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/batch.rst +0 -0
  67. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/buckets.rst +0 -0
  68. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/collections.rst +0 -0
  69. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/deprecation.rst +0 -0
  70. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/errors.rst +0 -0
  71. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/filtering.rst +0 -0
  72. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/flush.rst +0 -0
  73. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/groups.rst +0 -0
  74. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/history.rst +0 -0
  75. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/index.rst +0 -0
  76. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/openapi.rst +0 -0
  77. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/openid.rst +0 -0
  78. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/pagination.rst +0 -0
  79. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/permissions.rst +0 -0
  80. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/records.rst +0 -0
  81. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/selecting_fields.rst +0 -0
  82. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/sorting.rst +0 -0
  83. {kinto-20.6.1 → kinto-21.1.0}/docs/api/1.x/timestamps.rst +0 -0
  84. {kinto-20.6.1 → kinto-21.1.0}/docs/api/versioning.rst +0 -0
  85. {kinto-20.6.1 → kinto-21.1.0}/docs/changelog.rst +0 -0
  86. {kinto-20.6.1 → kinto-21.1.0}/docs/commandline.rst +0 -0
  87. {kinto-20.6.1 → kinto-21.1.0}/docs/community.rst +0 -0
  88. {kinto-20.6.1 → kinto-21.1.0}/docs/concepts.rst +0 -0
  89. {kinto-20.6.1 → kinto-21.1.0}/docs/conf.py +0 -0
  90. {kinto-20.6.1 → kinto-21.1.0}/docs/configuration/good-practices.rst +0 -0
  91. {kinto-20.6.1 → kinto-21.1.0}/docs/configuration/index.rst +0 -0
  92. {kinto-20.6.1 → kinto-21.1.0}/docs/configuration/production.rst +0 -0
  93. {kinto-20.6.1 → kinto-21.1.0}/docs/core/_static/theme_overrides.css +0 -0
  94. {kinto-20.6.1 → kinto-21.1.0}/docs/core/api.rst +0 -0
  95. {kinto-20.6.1 → kinto-21.1.0}/docs/core/cache.rst +0 -0
  96. {kinto-20.6.1 → kinto-21.1.0}/docs/core/decorators.rst +0 -0
  97. {kinto-20.6.1 → kinto-21.1.0}/docs/core/errors.rst +0 -0
  98. {kinto-20.6.1 → kinto-21.1.0}/docs/core/glossary.rst +0 -0
  99. {kinto-20.6.1 → kinto-21.1.0}/docs/core/images/cliquet-base.png +0 -0
  100. {kinto-20.6.1 → kinto-21.1.0}/docs/core/images/cliquet-mozilla.png +0 -0
  101. {kinto-20.6.1 → kinto-21.1.0}/docs/core/index.rst +0 -0
  102. {kinto-20.6.1 → kinto-21.1.0}/docs/core/notifications.rst +0 -0
  103. {kinto-20.6.1 → kinto-21.1.0}/docs/core/permission.rst +0 -0
  104. {kinto-20.6.1 → kinto-21.1.0}/docs/core/quickstart.rst +0 -0
  105. {kinto-20.6.1 → kinto-21.1.0}/docs/core/rationale.rst +0 -0
  106. {kinto-20.6.1 → kinto-21.1.0}/docs/core/resource.rst +0 -0
  107. {kinto-20.6.1 → kinto-21.1.0}/docs/core/storage.rst +0 -0
  108. {kinto-20.6.1 → kinto-21.1.0}/docs/core/testing.rst +0 -0
  109. {kinto-20.6.1 → kinto-21.1.0}/docs/core/utils.rst +0 -0
  110. {kinto-20.6.1 → kinto-21.1.0}/docs/core/viewsets.rst +0 -0
  111. {kinto-20.6.1 → kinto-21.1.0}/docs/faq.rst +0 -0
  112. {kinto-20.6.1 → kinto-21.1.0}/docs/images/alwaysdata-button.svg +0 -0
  113. {kinto-20.6.1 → kinto-21.1.0}/docs/images/architecture.svg +0 -0
  114. {kinto-20.6.1 → kinto-21.1.0}/docs/images/color-formatter.png +0 -0
  115. {kinto-20.6.1 → kinto-21.1.0}/docs/images/concepts-general.png +0 -0
  116. {kinto-20.6.1 → kinto-21.1.0}/docs/images/concepts-permissions.png +0 -0
  117. {kinto-20.6.1 → kinto-21.1.0}/docs/images/heroku-button.png +0 -0
  118. {kinto-20.6.1 → kinto-21.1.0}/docs/images/kinto-logo.png +0 -0
  119. {kinto-20.6.1 → kinto-21.1.0}/docs/images/kinto-logo.svg +0 -0
  120. {kinto-20.6.1 → kinto-21.1.0}/docs/images/logo-admin.svg +0 -0
  121. {kinto-20.6.1 → kinto-21.1.0}/docs/images/logo-attachment.svg +0 -0
  122. {kinto-20.6.1 → kinto-21.1.0}/docs/images/logo-community.svg +0 -0
  123. {kinto-20.6.1 → kinto-21.1.0}/docs/images/logo-history.svg +0 -0
  124. {kinto-20.6.1 → kinto-21.1.0}/docs/images/logo-javascript.svg +0 -0
  125. {kinto-20.6.1 → kinto-21.1.0}/docs/images/logo-jsonschema.svg +0 -0
  126. {kinto-20.6.1 → kinto-21.1.0}/docs/images/logo-multiapps.svg +0 -0
  127. {kinto-20.6.1 → kinto-21.1.0}/docs/images/logo-permissions.svg +0 -0
  128. {kinto-20.6.1 → kinto-21.1.0}/docs/images/logo-python.svg +0 -0
  129. {kinto-20.6.1 → kinto-21.1.0}/docs/images/logo-selfhostable.svg +0 -0
  130. {kinto-20.6.1 → kinto-21.1.0}/docs/images/logo-synchronisation.svg +0 -0
  131. {kinto-20.6.1 → kinto-21.1.0}/docs/images/overview-deployonce-selfhost.png +0 -0
  132. {kinto-20.6.1 → kinto-21.1.0}/docs/images/overview-features.png +0 -0
  133. {kinto-20.6.1 → kinto-21.1.0}/docs/images/overview-synchronisation.png +0 -0
  134. {kinto-20.6.1 → kinto-21.1.0}/docs/images/overview-use-cases.png +0 -0
  135. {kinto-20.6.1 → kinto-21.1.0}/docs/images/scalingo-button.svg +0 -0
  136. {kinto-20.6.1 → kinto-21.1.0}/docs/images/screenshot-kinto-admin-1.png +0 -0
  137. {kinto-20.6.1 → kinto-21.1.0}/docs/images/screenshot-kinto-admin-2.png +0 -0
  138. {kinto-20.6.1 → kinto-21.1.0}/docs/images/screenshot-kinto-admin-3.png +0 -0
  139. {kinto-20.6.1 → kinto-21.1.0}/docs/images/screenshot-kinto-admin-4.png +0 -0
  140. {kinto-20.6.1 → kinto-21.1.0}/docs/images/sequence-storage.png +0 -0
  141. {kinto-20.6.1 → kinto-21.1.0}/docs/images/sync-both.svg +0 -0
  142. {kinto-20.6.1 → kinto-21.1.0}/docs/images/sync-newest.svg +0 -0
  143. {kinto-20.6.1 → kinto-21.1.0}/docs/images/sync-oldest.svg +0 -0
  144. {kinto-20.6.1 → kinto-21.1.0}/docs/index.rst +0 -0
  145. {kinto-20.6.1 → kinto-21.1.0}/docs/kinto-admin.rst +0 -0
  146. {kinto-20.6.1 → kinto-21.1.0}/docs/overview.rst +0 -0
  147. {kinto-20.6.1 → kinto-21.1.0}/docs/requirements.txt +0 -0
  148. {kinto-20.6.1 → kinto-21.1.0}/docs/troubleshooting.rst +0 -0
  149. {kinto-20.6.1 → kinto-21.1.0}/docs/tutorials/custom-id-generator.rst +0 -0
  150. {kinto-20.6.1 → kinto-21.1.0}/docs/tutorials/first-steps.rst +0 -0
  151. {kinto-20.6.1 → kinto-21.1.0}/docs/tutorials/index.rst +0 -0
  152. {kinto-20.6.1 → kinto-21.1.0}/docs/tutorials/install.rst +0 -0
  153. {kinto-20.6.1 → kinto-21.1.0}/docs/tutorials/notifications-custom.rst +0 -0
  154. {kinto-20.6.1 → kinto-21.1.0}/docs/tutorials/permission-setups.rst +0 -0
  155. {kinto-20.6.1 → kinto-21.1.0}/docs/tutorials/permissions.rst +0 -0
  156. {kinto-20.6.1 → kinto-21.1.0}/docs/tutorials/synchronisation.rst +0 -0
  157. {kinto-20.6.1 → kinto-21.1.0}/docs/tutorials/write-plugin.rst +0 -0
  158. {kinto-20.6.1 → kinto-21.1.0}/kinto/__main__.py +0 -0
  159. {kinto-20.6.1 → kinto-21.1.0}/kinto/authorization.py +0 -0
  160. {kinto-20.6.1 → kinto-21.1.0}/kinto/config/kinto.tpl +0 -0
  161. {kinto-20.6.1 → kinto-21.1.0}/kinto/contribute.json +0 -0
  162. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/__init__.py +0 -0
  163. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/authentication.py +0 -0
  164. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/authorization.py +0 -0
  165. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cache/__init__.py +0 -0
  166. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cache/memcached.py +0 -0
  167. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cache/memory.py +0 -0
  168. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cache/postgresql/__init__.py +0 -0
  169. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cache/postgresql/schema.sql +0 -0
  170. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cache/testing.py +0 -0
  171. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice/__init__.py +0 -0
  172. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice/cors.py +0 -0
  173. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice/errors.py +0 -0
  174. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice/pyramidhook.py +0 -0
  175. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice/renderer.py +0 -0
  176. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice/resource.py +0 -0
  177. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice/service.py +0 -0
  178. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice/util.py +0 -0
  179. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice/validators/__init__.py +0 -0
  180. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice/validators/_colander.py +0 -0
  181. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice/validators/_marshmallow.py +0 -0
  182. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice_swagger/__init__.py +0 -0
  183. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice_swagger/converters/__init__.py +0 -0
  184. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice_swagger/converters/exceptions.py +0 -0
  185. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice_swagger/converters/parameters.py +0 -0
  186. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice_swagger/converters/schema.py +0 -0
  187. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice_swagger/swagger.py +0 -0
  188. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice_swagger/templates/index.html +0 -0
  189. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice_swagger/templates/index_script_template.html +0 -0
  190. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice_swagger/util.py +0 -0
  191. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/cornice_swagger/views.py +0 -0
  192. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/decorators.py +0 -0
  193. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/errors.py +0 -0
  194. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/events.py +0 -0
  195. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/listeners/__init__.py +0 -0
  196. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/openapi.py +0 -0
  197. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/permission/__init__.py +0 -0
  198. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/permission/memory.py +0 -0
  199. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/permission/postgresql/__init__.py +0 -0
  200. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/permission/postgresql/migrations/migration_001_002.sql +0 -0
  201. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/permission/postgresql/schema.sql +0 -0
  202. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/permission/testing.py +0 -0
  203. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/resource/__init__.py +0 -0
  204. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/resource/model.py +0 -0
  205. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/resource/schema.py +0 -0
  206. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/resource/viewset.py +0 -0
  207. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/schema.py +0 -0
  208. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/scripts.py +0 -0
  209. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/statsd.py +0 -0
  210. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/__init__.py +0 -0
  211. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/exceptions.py +0 -0
  212. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/generators.py +0 -0
  213. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/memory.py +0 -0
  214. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/__init__.py +0 -0
  215. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/client.py +0 -0
  216. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_001_002.sql +0 -0
  217. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_002_003.sql +0 -0
  218. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_003_004.sql +0 -0
  219. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_004_005.sql +0 -0
  220. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_005_006.sql +0 -0
  221. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_006_007.sql +0 -0
  222. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_007_008.sql +0 -0
  223. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_008_009.sql +0 -0
  224. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_009_010.sql +0 -0
  225. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_010_011.sql +0 -0
  226. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_011_012.sql +0 -0
  227. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_012_013.sql +0 -0
  228. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_013_014.sql +0 -0
  229. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_014_015.sql +0 -0
  230. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_015_016.sql +0 -0
  231. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_016_017.sql +0 -0
  232. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_017_018.sql +0 -0
  233. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_018_019.sql +0 -0
  234. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_019_020.sql +0 -0
  235. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_020_021.sql +0 -0
  236. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_021_022.sql +0 -0
  237. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrations/migration_022_023.sql +0 -0
  238. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/migrator.py +0 -0
  239. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/pool.py +0 -0
  240. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/postgresql/schema.sql +0 -0
  241. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/testing.py +0 -0
  242. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/storage/utils.py +0 -0
  243. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/testing.py +0 -0
  244. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/utils.py +0 -0
  245. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/views/__init__.py +0 -0
  246. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/views/batch.py +0 -0
  247. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/views/errors.py +0 -0
  248. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/views/heartbeat.py +0 -0
  249. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/views/openapi.py +0 -0
  250. {kinto-20.6.1 → kinto-21.1.0}/kinto/core/views/version.py +0 -0
  251. {kinto-20.6.1 → kinto-21.1.0}/kinto/events.py +0 -0
  252. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/__init__.py +0 -0
  253. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/accounts/__init__.py +0 -0
  254. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/accounts/authentication.py +0 -0
  255. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/accounts/scripts.py +0 -0
  256. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/accounts/utils.py +0 -0
  257. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/accounts/views.py +0 -0
  258. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/README.md +0 -0
  259. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/VERSION +0 -0
  260. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/__init__.py +0 -0
  261. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/build/VERSION +0 -0
  262. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/build/assets/asn1-EdZsLKOL.js +0 -0
  263. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/build/assets/clojure-BMjYHr_A.js +0 -0
  264. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/build/assets/css-BnMrqG3P.js +0 -0
  265. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/build/assets/index-Cs7JVwIg.css +0 -0
  266. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/build/assets/index-CylsivYB.js +0 -0
  267. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/build/assets/javascript-qCveANmP.js +0 -0
  268. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/build/assets/logo-VBRiKSPX.png +0 -0
  269. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/build/assets/mllike-CXdrOF99.js +0 -0
  270. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/build/assets/python-BuPzkPfP.js +0 -0
  271. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/build/assets/rpm-CTu-6PCP.js +0 -0
  272. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/build/assets/sql-D0XecflT.js +0 -0
  273. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/build/assets/ttcn-cfg-B9xdYoR4.js +0 -0
  274. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/build/index.html +0 -0
  275. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/public/help.html +0 -0
  276. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/admin/views.py +0 -0
  277. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/default_bucket/__init__.py +0 -0
  278. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/flush.py +0 -0
  279. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/history/__init__.py +0 -0
  280. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/history/listener.py +0 -0
  281. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/history/views.py +0 -0
  282. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/openid/__init__.py +0 -0
  283. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/openid/utils.py +0 -0
  284. {kinto-20.6.1 → kinto-21.1.0}/kinto/plugins/openid/views.py +0 -0
  285. {kinto-20.6.1 → kinto-21.1.0}/kinto/schema_validation.py +0 -0
  286. {kinto-20.6.1 → kinto-21.1.0}/kinto/views/__init__.py +0 -0
  287. {kinto-20.6.1 → kinto-21.1.0}/kinto/views/admin.py +0 -0
  288. {kinto-20.6.1 → kinto-21.1.0}/kinto/views/buckets.py +0 -0
  289. {kinto-20.6.1 → kinto-21.1.0}/kinto/views/collections.py +0 -0
  290. {kinto-20.6.1 → kinto-21.1.0}/kinto/views/contribute.py +0 -0
  291. {kinto-20.6.1 → kinto-21.1.0}/kinto/views/groups.py +0 -0
  292. {kinto-20.6.1 → kinto-21.1.0}/kinto/views/permissions.py +0 -0
  293. {kinto-20.6.1 → kinto-21.1.0}/kinto/views/records.py +0 -0
  294. {kinto-20.6.1 → kinto-21.1.0}/kinto.egg-info/SOURCES.txt +0 -0
  295. {kinto-20.6.1 → kinto-21.1.0}/kinto.egg-info/dependency_links.txt +0 -0
  296. {kinto-20.6.1 → kinto-21.1.0}/kinto.egg-info/entry_points.txt +0 -0
  297. {kinto-20.6.1 → kinto-21.1.0}/kinto.egg-info/requires.txt +0 -0
  298. {kinto-20.6.1 → kinto-21.1.0}/kinto.egg-info/top_level.txt +0 -0
  299. {kinto-20.6.1 → kinto-21.1.0}/pyproject.toml +0 -0
  300. {kinto-20.6.1 → kinto-21.1.0}/scripts/pull-kinto-admin.sh +0 -0
  301. {kinto-20.6.1 → kinto-21.1.0}/setup.cfg +0 -0
  302. {kinto-20.6.1 → kinto-21.1.0}/tests/__init__.py +0 -0
  303. {kinto-20.6.1 → kinto-21.1.0}/tests/browser.ini +0 -0
  304. {kinto-20.6.1 → kinto-21.1.0}/tests/browser.py +0 -0
  305. {kinto-20.6.1 → kinto-21.1.0}/tests/core/__init__.py +0 -0
  306. {kinto-20.6.1 → kinto-21.1.0}/tests/core/listeners.py +0 -0
  307. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/__init__.py +0 -0
  308. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_base.py +0 -0
  309. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_cache_expires.py +0 -0
  310. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_events.py +0 -0
  311. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_filter.py +0 -0
  312. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_model.py +0 -0
  313. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_object.py +0 -0
  314. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_object_permissions.py +0 -0
  315. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_pagination.py +0 -0
  316. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_partial_response.py +0 -0
  317. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_preconditions.py +0 -0
  318. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_schema.py +0 -0
  319. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_sort.py +0 -0
  320. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_sync.py +0 -0
  321. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_views.py +0 -0
  322. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_views_cors.py +0 -0
  323. {kinto-20.6.1 → kinto-21.1.0}/tests/core/resource/test_viewset.py +0 -0
  324. {kinto-20.6.1 → kinto-21.1.0}/tests/core/schema/postgresql-permission-1.sql +0 -0
  325. {kinto-20.6.1 → kinto-21.1.0}/tests/core/schema/postgresql-storage-1.6.sql +0 -0
  326. {kinto-20.6.1 → kinto-21.1.0}/tests/core/schema/postgresql-storage-11.sql +0 -0
  327. {kinto-20.6.1 → kinto-21.1.0}/tests/core/support.py +0 -0
  328. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_authentication.py +0 -0
  329. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_authorization.py +0 -0
  330. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_cache.py +0 -0
  331. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_decorators.py +0 -0
  332. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_deprecation.py +0 -0
  333. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_errors.py +0 -0
  334. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_listeners.py +0 -0
  335. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_logging.py +0 -0
  336. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_openapi.py +0 -0
  337. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_permission.py +0 -0
  338. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_schema.py +0 -0
  339. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_scripts.py +0 -0
  340. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_storage.py +0 -0
  341. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_storage_migrations.py +0 -0
  342. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_storage_pool.py +0 -0
  343. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_utils.py +0 -0
  344. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_views_batch.py +0 -0
  345. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_views_errors.py +0 -0
  346. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_views_heartbeat.py +0 -0
  347. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_views_openapi.py +0 -0
  348. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_views_postgresql.py +0 -0
  349. {kinto-20.6.1 → kinto-21.1.0}/tests/core/test_views_transaction.py +0 -0
  350. {kinto-20.6.1 → kinto-21.1.0}/tests/core/testapp/__init__.py +0 -0
  351. {kinto-20.6.1 → kinto-21.1.0}/tests/core/testapp/static/index.html +0 -0
  352. {kinto-20.6.1 → kinto-21.1.0}/tests/core/testapp/views.py +0 -0
  353. {kinto-20.6.1 → kinto-21.1.0}/tests/core/testplugin/__init__.py +0 -0
  354. {kinto-20.6.1 → kinto-21.1.0}/tests/functional.ini +0 -0
  355. {kinto-20.6.1 → kinto-21.1.0}/tests/functional.py +0 -0
  356. {kinto-20.6.1 → kinto-21.1.0}/tests/openapi/__init__.py +0 -0
  357. {kinto-20.6.1 → kinto-21.1.0}/tests/openapi/support.py +0 -0
  358. {kinto-20.6.1 → kinto-21.1.0}/tests/openapi/test_plugins.py +0 -0
  359. {kinto-20.6.1 → kinto-21.1.0}/tests/openapi/test_resources.py +0 -0
  360. {kinto-20.6.1 → kinto-21.1.0}/tests/openapi/test_responses_buckets.py +0 -0
  361. {kinto-20.6.1 → kinto-21.1.0}/tests/openapi/test_responses_collections.py +0 -0
  362. {kinto-20.6.1 → kinto-21.1.0}/tests/openapi/test_responses_errors.py +0 -0
  363. {kinto-20.6.1 → kinto-21.1.0}/tests/openapi/test_responses_groups.py +0 -0
  364. {kinto-20.6.1 → kinto-21.1.0}/tests/openapi/test_responses_records.py +0 -0
  365. {kinto-20.6.1 → kinto-21.1.0}/tests/openapi/test_validation.py +0 -0
  366. {kinto-20.6.1 → kinto-21.1.0}/tests/plugins/__init__.py +0 -0
  367. {kinto-20.6.1 → kinto-21.1.0}/tests/plugins/test_accounts.py +0 -0
  368. {kinto-20.6.1 → kinto-21.1.0}/tests/plugins/test_admin.py +0 -0
  369. {kinto-20.6.1 → kinto-21.1.0}/tests/plugins/test_default_bucket.py +0 -0
  370. {kinto-20.6.1 → kinto-21.1.0}/tests/plugins/test_flush.py +0 -0
  371. {kinto-20.6.1 → kinto-21.1.0}/tests/plugins/test_history.py +0 -0
  372. {kinto-20.6.1 → kinto-21.1.0}/tests/plugins/test_openid.py +0 -0
  373. {kinto-20.6.1 → kinto-21.1.0}/tests/plugins/test_statsd.py +0 -0
  374. {kinto-20.6.1 → kinto-21.1.0}/tests/support.py +0 -0
  375. {kinto-20.6.1 → kinto-21.1.0}/tests/test_authorization.py +0 -0
  376. {kinto-20.6.1 → kinto-21.1.0}/tests/test_config.py +0 -0
  377. {kinto-20.6.1 → kinto-21.1.0}/tests/test_configuration/test.ini +0 -0
  378. {kinto-20.6.1 → kinto-21.1.0}/tests/test_init.py +0 -0
  379. {kinto-20.6.1 → kinto-21.1.0}/tests/test_main.py +0 -0
  380. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_admin.py +0 -0
  381. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_buckets.py +0 -0
  382. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_collections.py +0 -0
  383. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_collections_cache.py +0 -0
  384. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_contribute.py +0 -0
  385. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_disable_default.py +0 -0
  386. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_groups.py +0 -0
  387. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_hello.py +0 -0
  388. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_objects_permissions.py +0 -0
  389. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_permissions.py +0 -0
  390. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_records.py +0 -0
  391. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_schema_collection.py +0 -0
  392. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_schema_group.py +0 -0
  393. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_schema_record.py +0 -0
  394. {kinto-20.6.1 → kinto-21.1.0}/tests/test_views_version.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: kinto
3
- Version: 20.6.1
3
+ Version: 21.1.0
4
4
  Summary: Kinto Web Service - Store, Sync, Share, and Self-Host.
5
5
  Author-email: Mozilla Services <developers@kinto-storage.org>
6
6
  License: Copyright 2012 - Mozilla Foundation
@@ -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
@@ -149,7 +149,7 @@ rpds-py==0.17.1
149
149
  # via
150
150
  # jsonschema
151
151
  # referencing
152
- ruff==0.11.5
152
+ ruff==0.11.6
153
153
  # via -r constraints.in
154
154
  sentry-sdk==2.8.0
155
155
  # via -r constraints.in
@@ -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
 
@@ -11,6 +11,12 @@ API
11
11
  Changelog
12
12
  ---------
13
13
 
14
+ 1.23 (2025-04-16)
15
+ '''''''''''''''''
16
+
17
+ - Added ``config`` field to the root URL endpoint
18
+
19
+
14
20
  1.22 (2019-03-20)
15
21
  '''''''''''''''''
16
22
 
@@ -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
 
@@ -14,7 +14,7 @@ from kinto.core import utils
14
14
  __version__ = pkg_resources.get_distribution(__package__).version
15
15
 
16
16
  # Implemented HTTP API Version
17
- HTTP_API_VERSION = "1.22"
17
+ HTTP_API_VERSION = "1.23"
18
18
 
19
19
  # Main kinto logger
20
20
  logger = logging.getLogger(__name__)
@@ -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
+ }
@@ -1,7 +1,6 @@
1
1
  import logging
2
2
  import random
3
3
  import re
4
- import urllib.parse
5
4
  import warnings
6
5
  from datetime import datetime
7
6
 
@@ -466,14 +465,6 @@ def setup_metrics(config):
466
465
  request = event.request
467
466
  metrics_service = config.registry.metrics
468
467
 
469
- try:
470
- endpoint = utils.strip_uri_prefix(request.path)
471
- endpoint = urllib.parse.quote_plus(endpoint, safe="/?&=-_")
472
- except UnicodeDecodeError as e:
473
- # This `on_new_response` callback is also called when a HTTP 400
474
- # is returned because of an invalid UTF-8 path. We still want metrics.
475
- endpoint = str(e)
476
-
477
468
  # Count unique users.
478
469
  user_id = request.prefixed_userid
479
470
  if user_id:
@@ -496,13 +487,27 @@ def setup_metrics(config):
496
487
  (field, enhanced_matchdict.get(field, "")) for field in metrics_matchdict_fields
497
488
  ]
498
489
 
490
+ status = event.response.status_code
491
+
492
+ service = request.current_service
493
+ if service:
494
+ # Use the service name as endpoint if available.
495
+ endpoint = service.name
496
+ elif route := request.matched_route:
497
+ # Use the route name as endpoint if we're not on a Cornice service.
498
+ endpoint = route.name
499
+ else:
500
+ endpoint = (
501
+ "unnamed" if status != 404 else "unknown"
502
+ ) # Do not multiply cardinality for unknown endpoints.
503
+
499
504
  # Count served requests.
500
505
  metrics_service.count(
501
506
  "request_summary",
502
507
  unique=[
503
508
  ("method", request.method.lower()),
504
509
  ("endpoint", endpoint),
505
- ("status", str(event.response.status_code)),
510
+ ("status", str(status)),
506
511
  ]
507
512
  + metrics_matchdict_labels,
508
513
  )
@@ -510,10 +515,11 @@ def setup_metrics(config):
510
515
  try:
511
516
  current = utils.msec_time()
512
517
  duration = current - request._received_at
513
- metrics_service.observe(
518
+ metrics_service.timer(
514
519
  "request_duration",
515
- duration,
516
- labels=[("endpoint", endpoint)] + metrics_matchdict_labels,
520
+ value=duration,
521
+ labels=[("endpoint", endpoint), ("method", request.method.lower())]
522
+ + metrics_matchdict_labels,
517
523
  )
518
524
  except AttributeError: # pragma: no cover
519
525
  # Logging was not setup in this Kinto app (unlikely but possible)
@@ -527,13 +533,11 @@ def setup_metrics(config):
527
533
  )
528
534
 
529
535
  # Count authentication verifications.
530
- if hasattr(request, "authn_type"):
531
- metrics_service.count(f"authn_type.{request.authn_type}")
532
-
533
- # Count view calls.
534
- service = request.current_service
535
- if service:
536
- metrics_service.count(f"view.{service.name}.{request.method}")
536
+ try:
537
+ metrics_service.count("authentication", unique=[("type", request.authn_type)])
538
+ except AttributeError:
539
+ # Not authenticated
540
+ pass
537
541
 
538
542
  config.add_subscriber(on_new_response, NewResponse)
539
543
 
@@ -47,7 +47,7 @@ class NoOpTimer:
47
47
 
48
48
  @implementer(IMetricsService)
49
49
  class NoOpMetricsService:
50
- def timer(self, key):
50
+ def timer(self, key, value=None, labels=[]):
51
51
  return NoOpTimer()
52
52
 
53
53
  def observe(self, key, value, labels=[]):
@@ -65,11 +65,12 @@ def watch_execution_time(metrics_service, obj, prefix="", classname=None):
65
65
  classname = classname or utils.classname(obj)
66
66
  members = dir(obj)
67
67
  for name in members:
68
- value = getattr(obj, name)
69
- is_method = isinstance(value, types.MethodType)
68
+ method = getattr(obj, name)
69
+ is_method = isinstance(method, types.MethodType)
70
70
  if not name.startswith("_") and is_method:
71
- statsd_key = f"{prefix}.{classname}.{name}"
72
- decorated_method = metrics_service.timer(statsd_key)(value)
71
+ statsd_key = f"{prefix}.{classname}"
72
+ labels = [("method", name)]
73
+ decorated_method = metrics_service.timer(statsd_key, labels=labels)(method)
73
74
  setattr(obj, name, decorated_method)
74
75
 
75
76
 
@@ -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
@@ -49,10 +50,27 @@ def _fix_metric_name(s):
49
50
 
50
51
 
51
52
  class Timer:
52
- def __init__(self, summary):
53
- self.summary = summary
53
+ """
54
+ A decorator to time the execution of a function. It will use the
55
+ `prometheus_client.Histogram` to record the time taken by the function
56
+ in milliseconds. The histogram is passed as an argument to the
57
+ constructor.
58
+
59
+ Main limitation: it does not support `labels` on the decorator.
60
+ """
61
+
62
+ def __init__(self, histogram):
63
+ self.histogram = histogram
54
64
  self._start_time = None
55
65
 
66
+ def set_labels(self, labels):
67
+ if not labels:
68
+ return
69
+ self.histogram = self.histogram.labels(*(label_value for _, label_value in labels))
70
+
71
+ def observe(self, value):
72
+ return self.histogram.observe(value)
73
+
56
74
  def __call__(self, f):
57
75
  @safe_wraps(f)
58
76
  def _wrapped(*args, **kwargs):
@@ -61,7 +79,7 @@ class Timer:
61
79
  return f(*args, **kwargs)
62
80
  finally:
63
81
  dt_ms = 1000.0 * (time_now() - start_time)
64
- self.summary.observe(dt_ms)
82
+ self.histogram.observe(dt_ms)
65
83
 
66
84
  return _wrapped
67
85
 
@@ -79,13 +97,13 @@ class Timer:
79
97
  if self._start_time is None: # pragma: nocover
80
98
  raise RuntimeError("Timer has not started.")
81
99
  dt_ms = 1000.0 * (time_now() - self._start_time)
82
- self.summary.observe(dt_ms)
100
+ self.histogram.observe(dt_ms)
83
101
  return self
84
102
 
85
103
 
86
104
  @implementer(metrics.IMetricsService)
87
105
  class PrometheusService:
88
- def __init__(self, prefix=""):
106
+ def __init__(self, prefix="", exclude_labels=None):
89
107
  prefix_clean = ""
90
108
  if prefix:
91
109
  # In GCP Console, the metrics are grouped by the first
@@ -94,26 +112,49 @@ class PrometheusService:
94
112
  # (eg. `remote-settings` -> `remotesettings_`, `kinto_` -> `kinto_`)
95
113
  prefix_clean = _fix_metric_name(prefix).replace("_", "") + "_"
96
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
+ ]
97
123
 
98
- def timer(self, key):
124
+ def timer(self, key, value=None, labels=[]):
99
125
  global _METRICS
100
126
  key = self.prefix + key
127
+ labels = self._exclude_labels(labels)
101
128
 
102
129
  if key not in _METRICS:
103
- _METRICS[key] = prometheus_module.Summary(
104
- _fix_metric_name(key), f"Summary of {key}", registry=get_registry()
130
+ _METRICS[key] = prometheus_module.Histogram(
131
+ _fix_metric_name(key),
132
+ f"Histogram of {key}",
133
+ registry=get_registry(),
134
+ labelnames=[label_name for label_name, _ in labels],
105
135
  )
106
136
 
107
- if not isinstance(_METRICS[key], prometheus_module.Summary):
137
+ if not isinstance(_METRICS[key], prometheus_module.Histogram):
108
138
  raise RuntimeError(
109
139
  f"Metric {key} already exists with different type ({_METRICS[key]})"
110
140
  )
111
141
 
112
- return Timer(_METRICS[key])
142
+ timer = Timer(_METRICS[key])
143
+ timer.set_labels(labels)
144
+
145
+ if value is not None:
146
+ # We are timing something.
147
+ return timer.observe(value)
148
+
149
+ # We are not timing anything, just returning the timer object
150
+ # (eg. to be used as decorator or context manager).
151
+ # Note that in this case, the labels values will be the same for all calls.
152
+ return timer
113
153
 
114
154
  def observe(self, key, value, labels=[]):
115
155
  global _METRICS
116
156
  key = self.prefix + key
157
+ labels = self._exclude_labels(labels)
117
158
 
118
159
  if key not in _METRICS:
119
160
  _METRICS[key] = prometheus_module.Summary(
@@ -154,6 +195,7 @@ class PrometheusService:
154
195
  label_name, label_value = unique.rsplit(".", 1)
155
196
  unique = [(label_name, label_value)]
156
197
 
198
+ unique = self._exclude_labels(unique)
157
199
  labels = [
158
200
  (_fix_metric_name(label_name), label_value) for label_name, label_value in unique
159
201
  ]
@@ -199,6 +241,11 @@ def includeme(config):
199
241
  )
200
242
  raise ConfigurationError(error_msg)
201
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
+
202
249
  config.add_api_capability(
203
250
  "prometheus",
204
251
  description="Prometheus metrics.",
@@ -220,7 +267,13 @@ def includeme(config):
220
267
  pass
221
268
  _METRICS.clear()
222
269
 
223
- settings = config.get_settings()
224
270
  prefix = settings.get("prometheus_prefix", settings["project_name"])
225
271
 
226
- config.registry.registerUtility(PrometheusService(prefix=prefix), metrics.IMetricsService)
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
+ )
@@ -1,4 +1,5 @@
1
1
  import warnings
2
+ from datetime import timedelta
2
3
  from urllib.parse import urlparse
3
4
 
4
5
  from pyramid.exceptions import ConfigurationError
@@ -26,7 +27,13 @@ class StatsDService:
26
27
  def __init__(self, host, port, prefix):
27
28
  self._client = statsd_module.StatsClient(host, port, prefix=prefix)
28
29
 
29
- def timer(self, key):
30
+ def timer(self, key, value=None, labels=[]):
31
+ if labels:
32
+ # [("method", "get")] -> "method.get"
33
+ key = f"{key}." + ".".join(f"{label[0]}.{sanitize(label[1])}" for label in labels)
34
+ if value:
35
+ value = timedelta(seconds=value)
36
+ return self._client.timing(key, value)
30
37
  return self._client.timer(key)
31
38
 
32
39
  def observe(self, key, value, labels=[]):
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: kinto
3
- Version: 20.6.1
3
+ Version: 21.1.0
4
4
  Summary: Kinto Web Service - Store, Sync, Share, and Self-Host.
5
5
  Author-email: Mozilla Services <developers@kinto-storage.org>
6
6
  License: Copyright 2012 - Mozilla Foundation
@@ -386,7 +386,7 @@ class MetricsConfigurationTest(unittest.TestCase):
386
386
  kinto.core.initialize(self.config, "0.0.1", "settings_prefix")
387
387
  app = webtest.TestApp(self.config.make_wsgi_app())
388
388
  app.get("/v0/", headers=get_user_headers("bob"))
389
- self.mocked().count.assert_any_call("authn_type.basicauth")
389
+ self.mocked().count.assert_any_call("authentication", unique=[("type", "basicauth")])
390
390
 
391
391
  #
392
392
  # Endpoints.
@@ -404,7 +404,7 @@ class MetricsConfigurationTest(unittest.TestCase):
404
404
  app.get("/v0/__heartbeat__")
405
405
  self.mocked().count.assert_any_call(
406
406
  "request_summary",
407
- unique=[("method", "get"), ("endpoint", "/__heartbeat__"), ("status", "200")],
407
+ unique=[("method", "get"), ("endpoint", "heartbeat"), ("status", "200")],
408
408
  )
409
409
 
410
410
  def test_statsd_sanitizes_url_in_metrics(self):
@@ -420,7 +420,7 @@ class MetricsConfigurationTest(unittest.TestCase):
420
420
  ("method", "get"),
421
421
  (
422
422
  "endpoint",
423
- "/changeset%27%257C%2522%253F%253E%253C%21DOCTYPE%2522http%3A//xh3E%27%29%2C%27/l%27%29%2520from%2520dual%29%257C%27",
423
+ "unknown",
424
424
  ),
425
425
  ("status", "404"),
426
426
  ],
@@ -433,25 +433,19 @@ 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=[("endpoint", "heartbeat")],
437
437
  )
438
438
 
439
439
  def test_statsd_observe_request_duration(self):
440
440
  kinto.core.initialize(self.config, "0.0.1", "settings_prefix")
441
441
  app = webtest.TestApp(self.config.make_wsgi_app())
442
442
  app.get("/v0/__heartbeat__")
443
- self.mocked().observe.assert_any_call(
443
+ self.mocked().timer.assert_any_call(
444
444
  "request_duration",
445
- mock.ANY,
446
- labels=[("endpoint", "/__heartbeat__")],
445
+ value=mock.ANY,
446
+ labels=[("endpoint", "heartbeat"), ("method", "get")],
447
447
  )
448
448
 
449
- def test_statsd_counts_views_and_methods(self):
450
- kinto.core.initialize(self.config, "0.0.1", "settings_prefix")
451
- app = webtest.TestApp(self.config.make_wsgi_app())
452
- app.get("/v0/__heartbeat__")
453
- self.mocked().count.assert_any_call("view.heartbeat.GET")
454
-
455
449
  def test_statsd_counts_unknown_urls(self):
456
450
  kinto.core.initialize(self.config, "0.0.1", "settings_prefix")
457
451
  app = webtest.TestApp(self.config.make_wsgi_app())
@@ -24,7 +24,9 @@ class WatchExecutionTimeTest(unittest.TestCase):
24
24
 
25
25
  def test_public_methods_generates_statsd_calls(self):
26
26
  self.test_object.test_method()
27
- self.mocked.timer.assert_called_with("test.testedclass.test_method")
27
+ self.mocked.timer.assert_called_with(
28
+ "test.testedclass", labels=[("method", "test_method")]
29
+ )
28
30
 
29
31
  def test_private_methods_does_not_generates_statsd_calls(self):
30
32
  self.test_object._private_method()
@@ -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):
@@ -60,7 +60,7 @@ class ServiceTest(PrometheusWebTest):
60
60
  self.assertEqual(my_func(1, 1), 2)
61
61
 
62
62
  resp = self.app.get("/__metrics__")
63
- self.assertIn("TYPE kintoprod_func_latency_context summary", resp.text)
63
+ self.assertIn("TYPE kintoprod_func_latency_context histogram", resp.text)
64
64
 
65
65
  def test_timer_can_be_used_as_decorator(self):
66
66
  decorated = self.app.app.registry.metrics.timer("func.latency.decorator")(my_func)
@@ -68,7 +68,7 @@ class ServiceTest(PrometheusWebTest):
68
68
  self.assertEqual(decorated(1, 1), 2)
69
69
 
70
70
  resp = self.app.get("/__metrics__")
71
- self.assertIn("TYPE kintoprod_func_latency_decorator summary", resp.text)
71
+ self.assertIn("TYPE kintoprod_func_latency_decorator histogram", resp.text)
72
72
 
73
73
  def test_timer_can_be_used_as_decorator_on_partial_function(self):
74
74
  partial = functools.partial(my_func, 3)
@@ -77,7 +77,7 @@ class ServiceTest(PrometheusWebTest):
77
77
  self.assertEqual(decorated(3), 6)
78
78
 
79
79
  resp = self.app.get("/__metrics__")
80
- self.assertIn("TYPE kintoprod_func_latency_partial summary", resp.text)
80
+ self.assertIn("TYPE kintoprod_func_latency_partial histogram", resp.text)
81
81
 
82
82
  def test_observe_a_single_value(self):
83
83
  self.app.app.registry.metrics.observe("price", 111)
@@ -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"}',
200
+ resp.text,
201
+ )
202
+ self.assertIn(
203
+ 'kintoprod_request_size_count{bucket_id="bid",collection_id="cid",endpoint="record-object"}',
204
+ resp.text,
205
+ )
@@ -26,20 +26,38 @@ class ViewsMetricsTest(BaseWebTest, unittest.TestCase):
26
26
  self.app.put("/buckets/beers/collections/barley", headers=self.headers)
27
27
  self.app.put("/buckets/beers/collections/barley/records/abc", headers=self.headers)
28
28
 
29
+ self.app.get("/buckets", headers=self.headers)
30
+ self.app.get("/buckets/beers/collections", headers=self.headers)
31
+ self.app.get("/buckets/beers/collections/barley/records", headers=self.headers)
32
+
29
33
  resp = self.app.get("/__metrics__")
34
+ print(resp.text)
35
+ self.assertIn(
36
+ 'request_size_sum{bucket_id="beers",collection_id="",endpoint="bucket-object",group_id="",record_id=""}',
37
+ resp.text,
38
+ )
30
39
  self.assertIn(
31
- 'request_size_sum{bucket_id="beers",collection_id="",endpoint="/buckets/beers",group_id="",record_id=""}',
40
+ 'request_size_sum{bucket_id="beers",collection_id="",endpoint="group-object",group_id="amateurs",record_id=""}',
32
41
  resp.text,
33
42
  )
34
43
  self.assertIn(
35
- 'request_size_sum{bucket_id="beers",collection_id="",endpoint="/buckets/beers/groups/amateurs",group_id="amateurs",record_id=""}',
44
+ 'request_summary_total{bucket_id="beers",collection_id="barley",endpoint="collection-object",group_id="",method="put",record_id="",status="201"}',
45
+ resp.text,
46
+ )
47
+ self.assertIn(
48
+ 'request_duration_sum{bucket_id="beers",collection_id="barley",endpoint="record-object",group_id="",method="put",record_id="abc"}',
49
+ resp.text,
50
+ )
51
+
52
+ self.assertIn(
53
+ 'request_summary_total{bucket_id="",collection_id="",endpoint="bucket-plural",group_id="",method="get",record_id="",status="200"}',
36
54
  resp.text,
37
55
  )
38
56
  self.assertIn(
39
- 'request_summary_total{bucket_id="beers",collection_id="barley",endpoint="/buckets/beers/collections/barley",group_id="",method="put",record_id="",status="201"}',
57
+ 'request_summary_total{bucket_id="beers",collection_id="",endpoint="collection-plural",group_id="",method="get",record_id="",status="200"}',
40
58
  resp.text,
41
59
  )
42
60
  self.assertIn(
43
- 'request_duration_sum{bucket_id="beers",collection_id="barley",endpoint="/buckets/beers/collections/barley/records/abc",group_id="",record_id="abc"}',
61
+ 'request_summary_total{bucket_id="beers",collection_id="barley",endpoint="record-plural",group_id="",method="get",record_id="",status="200"}',
44
62
  resp.text,
45
63
  )
File without changes
File without changes