nene2-python 1.8.5__tar.gz → 1.8.6__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.
Files changed (231) hide show
  1. {nene2_python-1.8.5 → nene2_python-1.8.6}/CHANGELOG.md +13 -0
  2. {nene2_python-1.8.5 → nene2_python-1.8.6}/PKG-INFO +1 -1
  3. nene2_python-1.8.6/docs/field-trials/2026-05-field-trial-41.md +126 -0
  4. nene2_python-1.8.6/docs/field-trials/2026-05-field-trial-42.md +137 -0
  5. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/how-to/run-tests.md +21 -0
  6. {nene2_python-1.8.5 → nene2_python-1.8.6}/pyproject.toml +1 -1
  7. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/http/__init__.py +6 -1
  8. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/http/problem_details.py +21 -0
  9. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/http/test_problem_details.py +17 -4
  10. {nene2_python-1.8.5 → nene2_python-1.8.6}/uv.lock +1 -1
  11. {nene2_python-1.8.5 → nene2_python-1.8.6}/.env.example +0 -0
  12. {nene2_python-1.8.5 → nene2_python-1.8.6}/.github/workflows/ci.yml +0 -0
  13. {nene2_python-1.8.5 → nene2_python-1.8.6}/.github/workflows/docs.yml +0 -0
  14. {nene2_python-1.8.5 → nene2_python-1.8.6}/.github/workflows/publish.yml +0 -0
  15. {nene2_python-1.8.5 → nene2_python-1.8.6}/.gitignore +0 -0
  16. {nene2_python-1.8.5 → nene2_python-1.8.6}/.vitepress/config.mts +0 -0
  17. {nene2_python-1.8.5 → nene2_python-1.8.6}/.vitepress/theme/custom.css +0 -0
  18. {nene2_python-1.8.5 → nene2_python-1.8.6}/.vitepress/theme/index.ts +0 -0
  19. {nene2_python-1.8.5 → nene2_python-1.8.6}/AGENTS.md +0 -0
  20. {nene2_python-1.8.5 → nene2_python-1.8.6}/CLAUDE.md +0 -0
  21. {nene2_python-1.8.5 → nene2_python-1.8.6}/Dockerfile +0 -0
  22. {nene2_python-1.8.5 → nene2_python-1.8.6}/LICENSE +0 -0
  23. {nene2_python-1.8.5 → nene2_python-1.8.6}/README.md +0 -0
  24. {nene2_python-1.8.5 → nene2_python-1.8.6}/alembic/README +0 -0
  25. {nene2_python-1.8.5 → nene2_python-1.8.6}/alembic/env.py +0 -0
  26. {nene2_python-1.8.5 → nene2_python-1.8.6}/alembic/script.py.mako +0 -0
  27. {nene2_python-1.8.5 → nene2_python-1.8.6}/alembic/versions/001_create_notes_and_tags_tables.py +0 -0
  28. {nene2_python-1.8.5 → nene2_python-1.8.6}/alembic.ini +0 -0
  29. {nene2_python-1.8.5 → nene2_python-1.8.6}/compose.yaml +0 -0
  30. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/adr/0001-toolchain.md +0 -0
  31. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/adr/0002-clean-architecture.md +0 -0
  32. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/adr/0003-security-first.md +0 -0
  33. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/adr/0004-ai-first-design.md +0 -0
  34. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/adr/0005-logging.md +0 -0
  35. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/adr/0006-rate-limiting.md +0 -0
  36. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/adr/0009-mcp-design.md +0 -0
  37. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/adr/0010-async-use-case.md +0 -0
  38. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/adr/0011-mcp-as-core-dependency.md +0 -0
  39. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/de/index.md +0 -0
  40. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/de/tutorials/getting-started.md +0 -0
  41. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/explanation/architecture.md +0 -0
  42. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/explanation/design-philosophy.md +0 -0
  43. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-1.md +0 -0
  44. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-10.md +0 -0
  45. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-11.md +0 -0
  46. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-12.md +0 -0
  47. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-13.md +0 -0
  48. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-14.md +0 -0
  49. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-15.md +0 -0
  50. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-16.md +0 -0
  51. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-17.md +0 -0
  52. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-18.md +0 -0
  53. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-19.md +0 -0
  54. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-2.md +0 -0
  55. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-20.md +0 -0
  56. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-21.md +0 -0
  57. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-22.md +0 -0
  58. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-23.md +0 -0
  59. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-24.md +0 -0
  60. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-25.md +0 -0
  61. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-26.md +0 -0
  62. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-27.md +0 -0
  63. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-28.md +0 -0
  64. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-29.md +0 -0
  65. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-3.md +0 -0
  66. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-30.md +0 -0
  67. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-31.md +0 -0
  68. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-32.md +0 -0
  69. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-33.md +0 -0
  70. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-34.md +0 -0
  71. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-35.md +0 -0
  72. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-36.md +0 -0
  73. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-37.md +0 -0
  74. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-38.md +0 -0
  75. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-39.md +0 -0
  76. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-4.md +0 -0
  77. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-40.md +0 -0
  78. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-5.md +0 -0
  79. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-6.md +0 -0
  80. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-7.md +0 -0
  81. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-8.md +0 -0
  82. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/field-trials/2026-05-field-trial-9.md +0 -0
  83. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/fr/index.md +0 -0
  84. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/fr/tutorials/getting-started.md +0 -0
  85. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/how-to/add-new-domain.md +0 -0
  86. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/how-to/async-use-case.md +0 -0
  87. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/how-to/configure-auth.md +0 -0
  88. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/how-to/new-project.md +0 -0
  89. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/how-to/problem-details.md +0 -0
  90. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/how-to/sqlalchemy-repository.md +0 -0
  91. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/how-to/validation.md +0 -0
  92. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/howto/mcp-setup.md +0 -0
  93. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/index.md +0 -0
  94. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/ja/explanation/architecture.md +0 -0
  95. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/ja/explanation/design-philosophy.md +0 -0
  96. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/ja/how-to/add-new-domain.md +0 -0
  97. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/ja/how-to/configure-auth.md +0 -0
  98. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/ja/how-to/new-project.md +0 -0
  99. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/ja/how-to/run-tests.md +0 -0
  100. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/ja/how-to/sqlalchemy-repository.md +0 -0
  101. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/ja/howto/mcp-setup.md +0 -0
  102. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/ja/index.md +0 -0
  103. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/ja/reference/api.md +0 -0
  104. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/ja/reference/configuration.md +0 -0
  105. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/ja/reference/framework-modules.md +0 -0
  106. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/ja/tutorials/first-domain.md +0 -0
  107. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/ja/tutorials/getting-started.md +0 -0
  108. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/pt-br/index.md +0 -0
  109. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/pt-br/tutorials/getting-started.md +0 -0
  110. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/reference/api.md +0 -0
  111. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/reference/configuration.md +0 -0
  112. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/reference/framework-modules.md +0 -0
  113. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/roadmap.md +0 -0
  114. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/todo/current.md +0 -0
  115. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/tutorials/first-domain.md +0 -0
  116. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/tutorials/getting-started.md +0 -0
  117. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/zh/index.md +0 -0
  118. {nene2_python-1.8.5 → nene2_python-1.8.6}/docs/zh/tutorials/getting-started.md +0 -0
  119. {nene2_python-1.8.5 → nene2_python-1.8.6}/package-lock.json +0 -0
  120. {nene2_python-1.8.5 → nene2_python-1.8.6}/package.json +0 -0
  121. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/__init__.py +0 -0
  122. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/__main__.py +0 -0
  123. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/app.py +0 -0
  124. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/comment/__init__.py +0 -0
  125. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/comment/entity.py +0 -0
  126. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/comment/exceptions.py +0 -0
  127. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/comment/handler.py +0 -0
  128. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/comment/repository.py +0 -0
  129. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/comment/sqlalchemy_repository.py +0 -0
  130. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/comment/use_case.py +0 -0
  131. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/mcp.py +0 -0
  132. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/note/__init__.py +0 -0
  133. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/note/async_use_case.py +0 -0
  134. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/note/entity.py +0 -0
  135. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/note/exceptions.py +0 -0
  136. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/note/handler.py +0 -0
  137. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/note/repository.py +0 -0
  138. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/note/sqlalchemy_repository.py +0 -0
  139. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/note/use_case.py +0 -0
  140. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/schema.py +0 -0
  141. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/tag/__init__.py +0 -0
  142. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/tag/entity.py +0 -0
  143. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/tag/exceptions.py +0 -0
  144. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/tag/handler.py +0 -0
  145. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/tag/repository.py +0 -0
  146. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/tag/sqlalchemy_repository.py +0 -0
  147. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/example/tag/use_case.py +0 -0
  148. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/__init__.py +0 -0
  149. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/auth/__init__.py +0 -0
  150. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/auth/api_key.py +0 -0
  151. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/auth/bearer_token.py +0 -0
  152. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/auth/exceptions.py +0 -0
  153. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/auth/interfaces.py +0 -0
  154. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/auth/local_verifier.py +0 -0
  155. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/config/__init__.py +0 -0
  156. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/config/settings.py +0 -0
  157. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/database/__init__.py +0 -0
  158. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/database/exceptions.py +0 -0
  159. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/database/health.py +0 -0
  160. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/database/interfaces.py +0 -0
  161. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/database/sqlalchemy_executor.py +0 -0
  162. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/database/utils.py +0 -0
  163. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/http/health.py +0 -0
  164. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/http/pagination.py +0 -0
  165. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/log/__init__.py +0 -0
  166. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/log/setup.py +0 -0
  167. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/mcp/__init__.py +0 -0
  168. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/mcp/http_client.py +0 -0
  169. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/mcp/server.py +0 -0
  170. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/middleware/__init__.py +0 -0
  171. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/middleware/domain_exception.py +0 -0
  172. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/middleware/error_handler.py +0 -0
  173. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/middleware/request_id.py +0 -0
  174. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/middleware/request_logging.py +0 -0
  175. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/middleware/request_size_limit.py +0 -0
  176. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/middleware/security_headers.py +0 -0
  177. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/middleware/throttle.py +0 -0
  178. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/py.typed +0 -0
  179. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/use_case/__init__.py +0 -0
  180. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/use_case/protocols.py +0 -0
  181. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/validation/__init__.py +0 -0
  182. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/nene2/validation/exceptions.py +0 -0
  183. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/scripts/__init__.py +0 -0
  184. {nene2_python-1.8.5 → nene2_python-1.8.6}/src/scripts/export_openapi.py +0 -0
  185. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/__init__.py +0 -0
  186. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/__init__.py +0 -0
  187. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/comment/__init__.py +0 -0
  188. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/comment/test_comment_http.py +0 -0
  189. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/comment/test_comment_repository.py +0 -0
  190. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/comment/test_comment_use_case.py +0 -0
  191. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/conftest.py +0 -0
  192. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/note/__init__.py +0 -0
  193. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/note/test_async_note_use_case.py +0 -0
  194. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/note/test_list_notes.py +0 -0
  195. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/note/test_note_repository.py +0 -0
  196. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/tag/__init__.py +0 -0
  197. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/tag/test_tag_repository.py +0 -0
  198. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/tag/test_tags.py +0 -0
  199. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/test_cors.py +0 -0
  200. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/example/test_mcp.py +0 -0
  201. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/__init__.py +0 -0
  202. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/auth/__init__.py +0 -0
  203. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/auth/test_api_key.py +0 -0
  204. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/auth/test_bearer_token.py +0 -0
  205. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/auth/test_token_issuer.py +0 -0
  206. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/config/__init__.py +0 -0
  207. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/config/test_settings.py +0 -0
  208. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/database/__init__.py +0 -0
  209. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/database/test_transaction.py +0 -0
  210. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/database/test_utils.py +0 -0
  211. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/http/__init__.py +0 -0
  212. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/http/test_health.py +0 -0
  213. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/http/test_pagination.py +0 -0
  214. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/log/__init__.py +0 -0
  215. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/log/test_setup.py +0 -0
  216. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/mcp/__init__.py +0 -0
  217. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/mcp/test_http_client.py +0 -0
  218. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/middleware/__init__.py +0 -0
  219. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/middleware/test_error_handler.py +0 -0
  220. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/middleware/test_request_id.py +0 -0
  221. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/middleware/test_request_logging.py +0 -0
  222. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/middleware/test_request_size_limit.py +0 -0
  223. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/middleware/test_security_headers.py +0 -0
  224. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/middleware/test_simple_domain_handler.py +0 -0
  225. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/middleware/test_throttle.py +0 -0
  226. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/use_case/__init__.py +0 -0
  227. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/use_case/test_protocols.py +0 -0
  228. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/validation/__init__.py +0 -0
  229. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/nene2/validation/test_exceptions.py +0 -0
  230. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/scripts/__init__.py +0 -0
  231. {nene2_python-1.8.5 → nene2_python-1.8.6}/tests/scripts/test_export_openapi.py +0 -0
@@ -5,6 +5,19 @@ Format follows [Keep a Changelog](https://keepachangelog.com/en/1.1.0/).
5
5
 
6
6
  ---
7
7
 
8
+ ## [1.8.6] — 2026-05-20
9
+
10
+ FT41〜FT42 フィールドトライアル — structlog テスト統合ドキュメント・configure_problem_details リセット関数。
11
+
12
+ ### Added
13
+ - `nene2.http.reset_problem_details()` — `configure_problem_details()` で設定したグローバル状態をテスト間でリセットするヘルパー関数 (FT42)
14
+ - Field trial reports: `docs/field-trials/2026-05-field-trial-41.md`、`docs/field-trials/2026-05-field-trial-42.md`
15
+
16
+ ### Changed
17
+ - `docs/how-to/run-tests.md` — `configure_for_testing()` + `caplog` による structlog ログキャプチャパターンを追記 (FT41)
18
+
19
+ ---
20
+
8
21
  ## [1.8.5] — 2026-05-20
9
22
 
10
23
  FT37〜FT40 フィールドトライアル — RequestSizeLimitMiddleware パスごとサイズ制限とドキュメント改善。
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: nene2-python
3
- Version: 1.8.5
3
+ Version: 1.8.6
4
4
  Summary: NENE2 Python — minimal API framework following NENE2's design philosophy
5
5
  Project-URL: Homepage, https://github.com/hideyukiMORI/nene2-python
6
6
  Project-URL: Repository, https://github.com/hideyukiMORI/nene2-python
@@ -0,0 +1,126 @@
1
+ # Field Trial 41: structlog テスト統合 — configure_for_testing() + caplog ログ相関
2
+
3
+ **日付**: 2026-05-20
4
+ **バージョン**: v1.8.5 時点
5
+ **テーマ**: `configure_for_testing()` + pytest `caplog` でリクエスト ID をログで追跡するパターンの実運用確認
6
+
7
+ ---
8
+
9
+ ## 概要
10
+
11
+ `nene2.log.configure_for_testing()` を `conftest.py` のモジュールレベルで呼び出し、
12
+ `RequestIdMiddleware` + `RequestLoggingMiddleware` を組み合わせたアプリのログを
13
+ pytest の `caplog` でキャプチャ・検証するパターンを実装した。
14
+ structlog と stdlib logging の橋渡し構造における制約も確認した。
15
+
16
+ ---
17
+
18
+ ## 実装内容
19
+
20
+ `/home/xi/docker/nene2-python-FT/ft41-log-testing/` に以下を作成:
21
+
22
+ - **`conftest.py`** — モジュールレベルで `configure_for_testing()` を呼び出し
23
+ - **`app.py`** — `RequestIdMiddleware` / `RequestLoggingMiddleware` / `ErrorHandlerMiddleware` を積んだ FastAPI アプリ。`extra_context` パラメータで静的フィールドを全ログに付加
24
+ - **`test_app.py`** — 正常系・ヘッダー確認・caplog キャプチャ・ミドルウェアログ確認 (5 件)
25
+ - **`test_friction.py`** — 摩擦点の確認テスト (4 件)
26
+
27
+ **テスト結果**: 9 件全通過 ✅
28
+
29
+ ---
30
+
31
+ ## 摩擦点
32
+
33
+ ### FP41-1: configure_for_testing() は conftest.py のモジュールレベルで呼ぶ必要がある
34
+
35
+ **分類**: 軽微な摩擦(設計通り・注意喚起)
36
+
37
+ `configure_for_testing()` は structlog のグローバル設定を変更するため、
38
+ テスト関数内で呼んでも機能するが、全テストに適用するには
39
+ `conftest.py` のモジュールレベルで呼ぶことが重要。
40
+ テスト関数内で呼ぶと、その関数のみに限定されず他のテストに副作用を与える可能性がある。
41
+
42
+ ```python
43
+ # conftest.py — 正しいパターン
44
+ from nene2.log import configure_for_testing
45
+ configure_for_testing()
46
+ ```
47
+
48
+ **判断**: ドキュメントに記載されたパターン通り。現行の `docs/how-to/run-tests.md` に
49
+ 明示的に記載することで摩擦を減らせる。
50
+
51
+ ---
52
+
53
+ ### FP41-2: caplog.records の LogRecord に request_id フィールドが直接ない
54
+
55
+ **分類**: 設計上の制約(許容範囲)
56
+
57
+ `RequestIdMiddleware` が `structlog.contextvars.bind_contextvars(request_id=...)` で
58
+ バインドした値は、pytest の `caplog` が返す stdlib `LogRecord` に直接属性として
59
+ アクセスできない(`record.request_id` が存在しない)。
60
+
61
+ `ProcessorFormatter` を通すとメッセージ文字列に request_id が含まれるが、
62
+ 構造化フィールドとして直接取り出すことはできない。
63
+
64
+ ```python
65
+ # NG — LogRecord に直接属性はない
66
+ for record in caplog.records:
67
+ print(record.request_id) # AttributeError
68
+
69
+ # OK — メッセージ文字列に含まれる
70
+ assert "request-id-test" not in " ".join(r.message for r in caplog.records)
71
+ # (request_id の値は UUID であり、テスト側から事前に知ることはできない)
72
+ ```
73
+
74
+ **判断**: structlog と stdlib logging の橋渡し構造による設計上の制約。
75
+ `caplog` でのログ検証はメッセージ文字列ベースで行うのが現実的なアプローチ。
76
+ `ProcessorFormatter` に対するテストを書きたい場合は structlog の `capture_logs()` を使う方法もある。
77
+
78
+ ---
79
+
80
+ ### FP41-3: caplog のキャプチャには configure_for_testing() が必須
81
+
82
+ **分類**: 注意喚起(ドキュメントで対応済み)
83
+
84
+ `configure_for_testing()` を呼ばない状態では、structlog のログは
85
+ pytest の `caplog` にキャプチャされない。
86
+ JSON レンダラーのまま stdout に出力されるため、
87
+ テストで structlog ログを検証するには必ず `configure_for_testing()` を呼ぶ必要がある。
88
+
89
+ **判断**: FT18 で実装した機能の使い方確認。`run-tests.md` に caplog との統合手順を追記する価値がある。
90
+
91
+ ---
92
+
93
+ ### FP41-4: structlog.contextvars でバインドした値はメッセージ文字列に含まれる
94
+
95
+ **分類**: 摩擦なし(設計の確認)
96
+
97
+ `structlog.contextvars.bind_contextvars(key="value")` でバインドした値は、
98
+ `configure_for_testing()` の設定下では `record.message` にキー=値形式で含まれる。
99
+ テストで構造化フィールドを検証する場合はメッセージ文字列の部分一致で確認できる。
100
+
101
+ ```python
102
+ structlog.contextvars.bind_contextvars(request_id="test-123")
103
+ log.info("hello")
104
+ # caplog.records[0].message → "hello request_id=test-123" (または類似形式)
105
+ ```
106
+
107
+ **判断**: structlog + caplog の統合パターンとして文字列検索が現実的。
108
+
109
+ ---
110
+
111
+ ## フレームワーク変更
112
+
113
+ なし(全て設計通りの挙動)
114
+
115
+ ドキュメント追記のみ検討:
116
+ - `docs/how-to/run-tests.md` に `configure_for_testing()` + caplog パターンを追記
117
+
118
+ ---
119
+
120
+ ## 関連
121
+
122
+ - `nene2.log.configure_for_testing` (FT18, v1.8.0)
123
+ - `nene2.middleware.RequestIdMiddleware`
124
+ - `nene2.middleware.RequestLoggingMiddleware`
125
+ - FT18 (configure_for_testing 実装, v1.8.0)
126
+ - FT30 (RequestLoggingMiddleware extra_context, v1.8.2)
@@ -0,0 +1,137 @@
1
+ # Field Trial 42: get_request_id() Depends + configure_problem_details() 実運用検証
2
+
3
+ **日付**: 2026-05-20
4
+ **バージョン**: v1.8.5 時点
5
+ **テーマ**: `get_request_id()` を FastAPI `Depends` で注入しレスポンスに含めるパターン、および `configure_problem_details()` のプロジェクト全体設定
6
+
7
+ ---
8
+
9
+ ## 概要
10
+
11
+ `nene2.middleware.get_request_id()` を `Annotated[str, Depends(get_request_id)]` 構文で
12
+ ハンドラーに注入し、レスポンスボディに `request_id` を含めるパターンを実装した。
13
+ `configure_problem_details()` でプロジェクト全体の Problem Details base_url を設定し、
14
+ カスタム例外を `SimpleDomainHandler` でマッピングする構成も確認した。
15
+
16
+ ---
17
+
18
+ ## 実装内容
19
+
20
+ `/home/xi/docker/nene2-python-FT/ft42-request-id-depends/` に以下を作成:
21
+
22
+ - **`app.py`** — `get_request_id()` Depends 注入・`configure_problem_details()` 設定・`SimpleDomainHandler` でカスタム例外マッピング
23
+ - **`test_app.py`** — 正常系・404・Problem Details base_url・request_id 相関 (9 件)
24
+ - **`test_friction.py`** — 摩擦点の確認テスト (4 件)
25
+
26
+ **テスト結果**: 13 件全通過 ✅
27
+
28
+ ---
29
+
30
+ ## 摩擦点
31
+
32
+ ### FP42-1: RequestIdMiddleware なしで get_request_id() を呼ぶと空文字が返る
33
+
34
+ **分類**: 注意喚起(ドキュメントに記載済み)
35
+
36
+ `get_request_id()` は `contextvars` を参照する。
37
+ テスト関数から直接呼ぶ場合や `RequestIdMiddleware` を経由しない場合は `""` を返す。
38
+ `TestClient` 経由でリクエストを送れば正しく UUID が返る。
39
+
40
+ ```python
41
+ # RequestIdMiddleware なしで直接呼ぶ → ""
42
+ from nene2.middleware import get_request_id
43
+ assert get_request_id() == ""
44
+
45
+ # TestClient 経由で呼ぶ → UUID v4
46
+ r = client.get("/debug/request-id")
47
+ assert len(r.json()["request_id"]) == 36
48
+ ```
49
+
50
+ **判断**: ドキュメントに記載済みの動作。TestClient 経由で使うことを徹底すれば問題ない。
51
+
52
+ ---
53
+
54
+ ### FP42-2: configure_problem_details() のグローバル状態がテスト間で共有される
55
+
56
+ **分類**: 軽微な摩擦(設計上の制約・テスト時の注意点)
57
+
58
+ `configure_problem_details()` はモジュールレベルのグローバル変数 `_configured_base_url` を変更する。
59
+ 異なる `base_url` で複数の `create_app()` を呼ぶと最後の設定が残り、
60
+ テスト間で state が漏れる。
61
+
62
+ ```python
63
+ create_app(base_url="https://first.example.com/problems/")
64
+ # _configured_base_url == "https://first.example.com/problems/"
65
+
66
+ create_app(base_url="https://second.example.com/problems/")
67
+ # _configured_base_url == "https://second.example.com/problems/"
68
+ # ← first の設定は上書きされている
69
+ ```
70
+
71
+ **対処**: テスト間で隔離が必要な場合、`nene2.http.problem_details._configured_base_url = None`
72
+ で手動リセットするか、全テストで同一の base_url を使う。
73
+ 運用環境では起動時に一度だけ呼ぶ設計なので実害はない。
74
+
75
+ **判断**: アプリ起動時一度だけ呼ぶ設計であり仕様通り。テスト向けに `reset_problem_details()` 関数を追加する価値があるかもしれない。
76
+
77
+ ---
78
+
79
+ ### FP42-3: SimpleDomainHandler のエラーレスポンスに request_id が自動付与されない
80
+
81
+ **分類**: 設計上の制約(許容範囲・パターン提示)
82
+
83
+ `ErrorHandlerMiddleware` + `SimpleDomainHandler` が生成する 404 レスポンスには
84
+ `request_id` フィールドが自動追加されない。
85
+ `X-Request-Id` ヘッダーは `RequestIdMiddleware` が付与するが、
86
+ レスポンスボディへの `request_id` 付与はアプリ側で明示的に行う必要がある。
87
+
88
+ エラーレスポンスに `request_id` を含めたい場合は、`problem_details_response()` を
89
+ 直接呼ぶ exception handler を登録するか、`SimpleDomainHandler` を継承して
90
+ `request_id` を `extra` に追加するカスタム実装が必要。
91
+
92
+ **判断**: `ErrorHandlerMiddleware` はドメインレイヤーに依存しない設計のため、
93
+ `request_id` のような HTTP 横断概念を自動付与しないのは正しい。
94
+ クライアントが `X-Request-Id` ヘッダーを参照すれば相関できる。
95
+
96
+ ---
97
+
98
+ ### FP42-4: Annotated[str, Depends(get_request_id)] 構文は問題なく動作する
99
+
100
+ **分類**: 摩擦なし(良い設計の確認)
101
+
102
+ Python 3.12+ 推奨の `Annotated` 構文で `get_request_id()` を注入できる。
103
+ FastAPI の型推論も正しく動作し、`str` 型として扱われる。
104
+
105
+ ```python
106
+ from typing import Annotated
107
+ from fastapi import Depends
108
+ from nene2.middleware import get_request_id
109
+
110
+ async def handler(
111
+ request_id: Annotated[str, Depends(get_request_id)],
112
+ ) -> JSONResponse:
113
+ return JSONResponse({"request_id": request_id})
114
+ ```
115
+
116
+ **判断**: FT25 で実装した `get_request_id()` は `Annotated` + `Depends` パターンと完全に互換。
117
+
118
+ ---
119
+
120
+ ## フレームワーク変更
121
+
122
+ なし(全て設計通りの挙動)
123
+
124
+ 以下のドキュメント追記を検討:
125
+ - `docs/how-to/` に `get_request_id()` Depends パターンの how-to ガイドを追加
126
+
127
+ ---
128
+
129
+ ## 関連
130
+
131
+ - `nene2.middleware.get_request_id` (FT25, v1.8.1)
132
+ - `nene2.middleware.RequestIdMiddleware`
133
+ - `nene2.http.configure_problem_details` (FT19, v1.8.0)
134
+ - `nene2.middleware.SimpleDomainHandler` (FT21, v1.8.0)
135
+ - FT19 (configure_problem_details 実装, v1.8.0)
136
+ - FT21 (SimpleDomainHandler 実装, v1.8.0)
137
+ - FT25 (get_request_id 実装, v1.8.1)
@@ -103,6 +103,27 @@ engine = create_engine(
103
103
 
104
104
  `StaticPool` guarantees all logical connections share the same underlying SQLite connection, so tables created in one operation are visible to the next.
105
105
 
106
+ ## Capturing structlog output with caplog
107
+
108
+ Call `configure_for_testing()` at module level in `conftest.py` to route structlog through stdlib logging so pytest's `caplog` fixture can capture it.
109
+
110
+ ```python
111
+ # conftest.py
112
+ from nene2.log import configure_for_testing
113
+ configure_for_testing()
114
+ ```
115
+
116
+ Then assert on message strings in tests:
117
+
118
+ ```python
119
+ def test_handler_logs(caplog: pytest.LogCaptureFixture) -> None:
120
+ client = TestClient(create_app())
121
+ client.post("/api/echo", json={"message": "hello"})
122
+ assert any("processing echo" in r.message for r in caplog.records)
123
+ ```
124
+
125
+ **Note**: `caplog.records` returns stdlib `LogRecord` objects. Fields bound with `structlog.contextvars.bind_contextvars()` (such as `request_id`) are not directly accessible as `record.request_id` — they appear as part of the formatted message string instead.
126
+
106
127
  ## Coverage requirements
107
128
 
108
129
  | Scope | Target |
@@ -1,6 +1,6 @@
1
1
  [project]
2
2
  name = "nene2-python"
3
- version = "1.8.5"
3
+ version = "1.8.6"
4
4
  description = "NENE2 Python — minimal API framework following NENE2's design philosophy"
5
5
  readme = "README.md"
6
6
  license = {text = "MIT"}
@@ -8,7 +8,11 @@ from .health import (
8
8
  HealthStatus,
9
9
  )
10
10
  from .pagination import PaginationQuery, PaginationQueryParser, PaginationResponse
11
- from .problem_details import configure_problem_details, problem_details_response
11
+ from .problem_details import (
12
+ configure_problem_details,
13
+ problem_details_response,
14
+ reset_problem_details,
15
+ )
12
16
 
13
17
  __all__ = [
14
18
  "AsyncCompositeHealthCheck",
@@ -21,4 +25,5 @@ __all__ = [
21
25
  "PaginationResponse",
22
26
  "configure_problem_details",
23
27
  "problem_details_response",
28
+ "reset_problem_details",
24
29
  ]
@@ -27,6 +27,27 @@ def configure_problem_details(base_url: str) -> None:
27
27
  _configured_base_url = base_url
28
28
 
29
29
 
30
+ def reset_problem_details() -> None:
31
+ """Reset the base_url configured by configure_problem_details().
32
+
33
+ Intended for use in tests only. Restores the default behaviour of
34
+ falling back to ``PROBLEM_DETAILS_BASE_URL``.
35
+
36
+ Example::
37
+
38
+ import pytest
39
+ from nene2.http import reset_problem_details
40
+
41
+
42
+ @pytest.fixture(autouse=True)
43
+ def _reset():
44
+ yield
45
+ reset_problem_details()
46
+ """
47
+ global _configured_base_url # noqa: PLW0603
48
+ _configured_base_url = None
49
+
50
+
30
51
  def problem_details_response(
31
52
  problem_type: str,
32
53
  title: str,
@@ -2,16 +2,15 @@
2
2
 
3
3
  import pytest
4
4
 
5
- import nene2.http.problem_details as _mod
6
- from nene2.http import configure_problem_details, problem_details_response
5
+ from nene2.http import configure_problem_details, problem_details_response, reset_problem_details
7
6
 
8
7
 
9
8
  @pytest.fixture(autouse=True)
10
9
  def reset_configured_base_url() -> None:
11
10
  """Reset module-level configured base_url between tests."""
12
- _mod._configured_base_url = None
11
+ reset_problem_details()
13
12
  yield
14
- _mod._configured_base_url = None
13
+ reset_problem_details()
15
14
 
16
15
 
17
16
  def test_problem_details_response_uses_default_base_url() -> None:
@@ -61,3 +60,17 @@ def test_problem_details_response_includes_detail_when_provided() -> None:
61
60
  def test_problem_details_response_includes_extra_fields() -> None:
62
61
  r = problem_details_response("not-found", "Not Found", 404, extra={"resource_id": 42})
63
62
  assert b'"resource_id":42' in r.body
63
+
64
+
65
+ def test_reset_problem_details_restores_default_base_url() -> None:
66
+ configure_problem_details("https://custom.example.com/problems/")
67
+ reset_problem_details()
68
+ r = problem_details_response("not-found", "Not Found", 404)
69
+ assert b"nene2.dev/problems/not-found" in r.body
70
+
71
+
72
+ def test_reset_problem_details_is_idempotent() -> None:
73
+ reset_problem_details()
74
+ reset_problem_details()
75
+ r = problem_details_response("not-found", "Not Found", 404)
76
+ assert b"nene2.dev/problems/not-found" in r.body
@@ -925,7 +925,7 @@ wheels = [
925
925
 
926
926
  [[package]]
927
927
  name = "nene2-python"
928
- version = "1.8.5"
928
+ version = "1.8.6"
929
929
  source = { editable = "." }
930
930
  dependencies = [
931
931
  { name = "alembic" },
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